feat: extract common button style.
This commit is contained in:
@@ -7,18 +7,29 @@
|
|||||||
<Setter Property="Template">
|
<Setter Property="Template">
|
||||||
<ControlTemplate TargetType="u:DialogControl">
|
<ControlTemplate TargetType="u:DialogControl">
|
||||||
<Border
|
<Border
|
||||||
Classes="Shadow"
|
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
|
Classes="Hover"
|
||||||
IsHitTestVisible="True"
|
IsHitTestVisible="True"
|
||||||
|
Padding="2"
|
||||||
Theme="{DynamicResource CardBorder}">
|
Theme="{DynamicResource CardBorder}">
|
||||||
<Grid RowDefinitions="Auto, *, Auto">
|
<Grid RowDefinitions="Auto, *, Auto">
|
||||||
<StackPanel Grid.Row="0" HorizontalAlignment="Right">
|
<DockPanel
|
||||||
<Button Name="{x:Static u:DialogControl.PART_CloseButton}" >Close</Button>
|
Name="{x:Static u:DialogControl.PART_TitleArea}"
|
||||||
</StackPanel>
|
Grid.Row="0"
|
||||||
|
LastChildFill="False"
|
||||||
|
Background="Transparent">
|
||||||
|
<TextBlock DockPanel.Dock="Left" Text="Title" Margin="8 8 0 0" />
|
||||||
|
<Button
|
||||||
|
Name="{x:Static u:MessageBoxWindow.PART_CloseButton}"
|
||||||
|
DockPanel.Dock="Right"
|
||||||
|
Margin="0,4,4,0"
|
||||||
|
Theme="{DynamicResource CloseButton}">
|
||||||
|
</Button>
|
||||||
|
</DockPanel>
|
||||||
<ContentPresenter
|
<ContentPresenter
|
||||||
Grid.Row="0"
|
Grid.Row="0"
|
||||||
Grid.RowSpan="2"
|
Grid.RowSpan="3"
|
||||||
Content="{TemplateBinding Content}" />
|
Content="{TemplateBinding Content}" />
|
||||||
</Grid>
|
</Grid>
|
||||||
</Border>
|
</Border>
|
||||||
@@ -58,8 +69,16 @@
|
|||||||
<Panel Margin="{TemplateBinding WindowDecorationMargin}" Background="Transparent" />
|
<Panel Margin="{TemplateBinding WindowDecorationMargin}" Background="Transparent" />
|
||||||
<ChromeOverlayLayer />
|
<ChromeOverlayLayer />
|
||||||
<Grid RowDefinitions="Auto, *, Auto">
|
<Grid RowDefinitions="Auto, *, Auto">
|
||||||
<Button Name="{x:Static u:DialogWindow.PART_CloseButton}" VerticalAlignment="Top" HorizontalAlignment="Right">Close</Button>
|
<Button
|
||||||
<ContentPresenter Grid.Row="0" Grid.RowSpan="2" Content="{TemplateBinding Content}" />
|
Name="{x:Static u:DialogWindow.PART_CloseButton}"
|
||||||
|
HorizontalAlignment="Right"
|
||||||
|
VerticalAlignment="Top">
|
||||||
|
Close
|
||||||
|
</Button>
|
||||||
|
<ContentPresenter
|
||||||
|
Grid.Row="0"
|
||||||
|
Grid.RowSpan="2"
|
||||||
|
Content="{TemplateBinding Content}" />
|
||||||
</Grid>
|
</Grid>
|
||||||
</Panel>
|
</Panel>
|
||||||
</ControlTemplate>
|
</ControlTemplate>
|
||||||
|
|||||||
42
src/Ursa.Themes.Semi/Controls/DialogShared.axaml
Normal file
42
src/Ursa.Themes.Semi/Controls/DialogShared.axaml
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
<ResourceDictionary xmlns="https://github.com/avaloniaui" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
|
||||||
|
<!-- Add Resources Here -->
|
||||||
|
|
||||||
|
<ControlTheme x:Key="CloseButton" TargetType="Button">
|
||||||
|
<Setter Property="CornerRadius" Value="6" />
|
||||||
|
<Setter Property="Margin" Value="0, 4" />
|
||||||
|
<Setter Property="Padding" Value="4" />
|
||||||
|
<Setter Property="Height" Value="28" />
|
||||||
|
<Setter Property="Width" Value="28" />
|
||||||
|
<Setter Property="Cursor" Value="Hand" />
|
||||||
|
<Setter Property="VerticalAlignment" Value="Stretch" />
|
||||||
|
<Setter Property="Template">
|
||||||
|
<ControlTemplate TargetType="Button">
|
||||||
|
<Border
|
||||||
|
Name="PART_Border"
|
||||||
|
Padding="{TemplateBinding Padding}"
|
||||||
|
Background="Transparent"
|
||||||
|
CornerRadius="{TemplateBinding CornerRadius}">
|
||||||
|
<PathIcon
|
||||||
|
Width="12"
|
||||||
|
Height="12"
|
||||||
|
Data="{DynamicResource WindowCloseIconGlyph}"/>
|
||||||
|
</Border>
|
||||||
|
</ControlTemplate>
|
||||||
|
</Setter>
|
||||||
|
|
||||||
|
<Style Selector="^:pointerover /template/ Border">
|
||||||
|
<Setter Property="Background" Value="{DynamicResource CaptionButtonClosePointeroverBackground}" />
|
||||||
|
</Style>
|
||||||
|
<Style Selector="^:pointerover /template/ PathIcon">
|
||||||
|
<Setter Property="Foreground" Value="White" />
|
||||||
|
</Style>
|
||||||
|
|
||||||
|
<Style Selector="^:pressed /template/ Border">
|
||||||
|
<Setter Property="Background" Value="{DynamicResource CaptionButtonClosePressedBackground}" />
|
||||||
|
</Style>
|
||||||
|
<Style Selector="^:pressed /template/ PathIcon">
|
||||||
|
<Setter Property="Foreground" Value="White" />
|
||||||
|
</Style>
|
||||||
|
</ControlTheme>
|
||||||
|
|
||||||
|
</ResourceDictionary>
|
||||||
@@ -45,23 +45,7 @@
|
|||||||
Name="{x:Static u:MessageBoxWindow.PART_CloseButton}"
|
Name="{x:Static u:MessageBoxWindow.PART_CloseButton}"
|
||||||
Grid.Column="1"
|
Grid.Column="1"
|
||||||
Margin="0,4,4,0"
|
Margin="0,4,4,0"
|
||||||
Background="{DynamicResource CaptionButtonClosePointeroverBackground}"
|
Theme="{DynamicResource CloseButton}"/>
|
||||||
BorderBrush="{DynamicResource CaptionButtonClosePressedBackground}"
|
|
||||||
Theme="{DynamicResource CaptionButton}">
|
|
||||||
<Button.Styles>
|
|
||||||
<Style Selector="Button:pointerover">
|
|
||||||
<Setter Property="Foreground" Value="White" />
|
|
||||||
</Style>
|
|
||||||
<Style Selector="Button:pressed">
|
|
||||||
<Setter Property="Foreground" Value="White" />
|
|
||||||
</Style>
|
|
||||||
</Button.Styles>
|
|
||||||
<PathIcon
|
|
||||||
Width="12"
|
|
||||||
Height="12"
|
|
||||||
Data="{DynamicResource WindowCloseIconGlyph}"
|
|
||||||
Foreground="{Binding $parent[Button].Foreground}" />
|
|
||||||
</Button>
|
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid
|
<Grid
|
||||||
Grid.Row="1"
|
Grid.Row="1"
|
||||||
|
|||||||
@@ -5,6 +5,7 @@
|
|||||||
<ResourceInclude Source="Banner.axaml" />
|
<ResourceInclude Source="Banner.axaml" />
|
||||||
<ResourceInclude Source="ButtonGroup.axaml" />
|
<ResourceInclude Source="ButtonGroup.axaml" />
|
||||||
<ResourceInclude Source="Dialog.axaml" />
|
<ResourceInclude Source="Dialog.axaml" />
|
||||||
|
<ResourceInclude Source="DialogShared.axaml" />
|
||||||
<ResourceInclude Source="Divider.axaml" />
|
<ResourceInclude Source="Divider.axaml" />
|
||||||
<ResourceInclude Source="DualBadge.axaml" />
|
<ResourceInclude Source="DualBadge.axaml" />
|
||||||
<ResourceInclude Source="IconButton.axaml" />
|
<ResourceInclude Source="IconButton.axaml" />
|
||||||
|
|||||||
11
src/Ursa/Common/DialogButton.cs
Normal file
11
src/Ursa/Common/DialogButton.cs
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
namespace Ursa.Common;
|
||||||
|
|
||||||
|
public enum DialogButton
|
||||||
|
{
|
||||||
|
None,
|
||||||
|
OK,
|
||||||
|
OKCancel,
|
||||||
|
YesNo,
|
||||||
|
YesNoCancel,
|
||||||
|
}
|
||||||
|
|
||||||
15
src/Ursa/Common/DialogIcon.cs
Normal file
15
src/Ursa/Common/DialogIcon.cs
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
namespace Ursa.Common;
|
||||||
|
|
||||||
|
public enum DialogIcon
|
||||||
|
{
|
||||||
|
Asterisk, // Same as Information
|
||||||
|
Error,
|
||||||
|
Exclamation, // Same as Warning
|
||||||
|
Hand, // Same as Error
|
||||||
|
Information,
|
||||||
|
None,
|
||||||
|
Question,
|
||||||
|
Stop, // Same as Error
|
||||||
|
Warning,
|
||||||
|
Success,
|
||||||
|
}
|
||||||
10
src/Ursa/Common/DialogResult.cs
Normal file
10
src/Ursa/Common/DialogResult.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
namespace Ursa.Common;
|
||||||
|
|
||||||
|
public enum DialogResult
|
||||||
|
{
|
||||||
|
Cancel,
|
||||||
|
No,
|
||||||
|
None,
|
||||||
|
OK,
|
||||||
|
Yes,
|
||||||
|
}
|
||||||
@@ -2,16 +2,20 @@ using Avalonia;
|
|||||||
using Avalonia.Controls;
|
using Avalonia.Controls;
|
||||||
using Avalonia.Controls.Metadata;
|
using Avalonia.Controls.Metadata;
|
||||||
using Avalonia.Controls.Primitives;
|
using Avalonia.Controls.Primitives;
|
||||||
|
using Avalonia.Input;
|
||||||
using Avalonia.Interactivity;
|
using Avalonia.Interactivity;
|
||||||
|
|
||||||
namespace Ursa.Controls;
|
namespace Ursa.Controls;
|
||||||
|
|
||||||
[TemplatePart(PART_CloseButton, typeof(Button))]
|
[TemplatePart(PART_CloseButton, typeof(Button))]
|
||||||
|
[TemplatePart(PART_TitleArea, typeof(Panel))]
|
||||||
public class DialogControl: ContentControl
|
public class DialogControl: ContentControl
|
||||||
{
|
{
|
||||||
public const string PART_CloseButton = "PART_CloseButton";
|
public const string PART_CloseButton = "PART_CloseButton";
|
||||||
|
public const string PART_TitleArea = "PART_TitleArea";
|
||||||
|
|
||||||
private Button? _closeButton;
|
private Button? _closeButton;
|
||||||
|
private Panel? _titleArea;
|
||||||
public event EventHandler<object?>? OnClose;
|
public event EventHandler<object?>? OnClose;
|
||||||
|
|
||||||
static DialogControl()
|
static DialogControl()
|
||||||
@@ -32,6 +36,7 @@ public class DialogControl: ContentControl
|
|||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected override void OnApplyTemplate(TemplateAppliedEventArgs e)
|
protected override void OnApplyTemplate(TemplateAppliedEventArgs e)
|
||||||
{
|
{
|
||||||
@@ -40,13 +45,30 @@ public class DialogControl: ContentControl
|
|||||||
{
|
{
|
||||||
_closeButton.Click -= Close;
|
_closeButton.Click -= Close;
|
||||||
}
|
}
|
||||||
|
_titleArea?.RemoveHandler(PointerMovedEvent, OnTitlePointerMove);
|
||||||
|
_titleArea?.RemoveHandler(PointerPressedEvent, OnTitlePointerPressed);
|
||||||
|
|
||||||
_closeButton = e.NameScope.Find<Button>(PART_CloseButton);
|
_closeButton = e.NameScope.Find<Button>(PART_CloseButton);
|
||||||
|
_titleArea = e.NameScope.Find<Panel>(PART_TitleArea);
|
||||||
if (_closeButton is not null)
|
if (_closeButton is not null)
|
||||||
{
|
{
|
||||||
_closeButton.Click += Close;
|
_closeButton.Click += Close;
|
||||||
}
|
}
|
||||||
|
_titleArea?.AddHandler(PointerMovedEvent, OnTitlePointerMove, RoutingStrategies.Bubble);
|
||||||
|
_titleArea?.AddHandler(PointerPressedEvent, OnTitlePointerPressed, RoutingStrategies.Bubble);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void OnTitlePointerPressed(object sender, PointerPressedEventArgs e)
|
||||||
|
{
|
||||||
|
e.Source = this;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnTitlePointerMove(object sender, PointerEventArgs e)
|
||||||
|
{
|
||||||
|
e.Source = this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public Task<T> ShowAsync<T>()
|
public Task<T> ShowAsync<T>()
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -15,12 +15,6 @@ public class OverlayDialogHost: Canvas
|
|||||||
{
|
{
|
||||||
private readonly List<DialogControl> _dialogs = new();
|
private readonly List<DialogControl> _dialogs = new();
|
||||||
private readonly List<DialogControl> _modalDialogs = new();
|
private readonly List<DialogControl> _modalDialogs = new();
|
||||||
|
|
||||||
private Rectangle _overlayMask = new()
|
|
||||||
{
|
|
||||||
Fill = new SolidColorBrush(new Color(1, 0, 0, 0)),
|
|
||||||
[Rectangle.ZIndexProperty] = 0,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static readonly StyledProperty<string> HostIdProperty = AvaloniaProperty.Register<OverlayDialogHost, string>(
|
public static readonly StyledProperty<string> HostIdProperty = AvaloniaProperty.Register<OverlayDialogHost, string>(
|
||||||
nameof(HostId));
|
nameof(HostId));
|
||||||
@@ -66,37 +60,27 @@ public class OverlayDialogHost: Canvas
|
|||||||
protected override void OnPointerMoved(PointerEventArgs e)
|
protected override void OnPointerMoved(PointerEventArgs e)
|
||||||
{
|
{
|
||||||
base.OnPointerMoved(e);
|
base.OnPointerMoved(e);
|
||||||
if (e.Source is Control item)
|
if (e.Source is DialogControl item)
|
||||||
{
|
{
|
||||||
if (e.GetCurrentPoint(this).Properties.IsLeftButtonPressed)
|
if (e.GetCurrentPoint(this).Properties.IsLeftButtonPressed)
|
||||||
{
|
{
|
||||||
var parent = item.FindAncestorOfType<DialogControl>();
|
|
||||||
if (parent is null)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
var p = e.GetPosition(this);
|
var p = e.GetPosition(this);
|
||||||
var left= p.X - _lastPoint.X;
|
var left= p.X - _lastPoint.X;
|
||||||
var top = p.Y - _lastPoint.Y;
|
var top = p.Y - _lastPoint.Y;
|
||||||
left = MathUtilities.Clamp(left, 0, Bounds.Width - parent.Bounds.Width);
|
left = MathUtilities.Clamp(left, 0, Bounds.Width - item.Bounds.Width);
|
||||||
top = MathUtilities.Clamp(top, 0, Bounds.Height - parent.Bounds.Height);
|
top = MathUtilities.Clamp(top, 0, Bounds.Height - item.Bounds.Height);
|
||||||
Canvas.SetLeft(parent, left);
|
Canvas.SetLeft(item, left);
|
||||||
Canvas.SetTop(parent, top);
|
Canvas.SetTop(item, top);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void OnPointerPressed(PointerPressedEventArgs e)
|
protected override void OnPointerPressed(PointerPressedEventArgs e)
|
||||||
{
|
{
|
||||||
base.OnPointerPressed(e);
|
// base.OnPointerPressed(e);
|
||||||
if (e.Source is Control item)
|
if (e.Source is DialogControl item)
|
||||||
{
|
{
|
||||||
var parent = item.FindAncestorOfType<DialogControl>();
|
_lastPoint = e.GetPosition(item);
|
||||||
if (parent is null)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
_lastPoint = e.GetPosition(parent);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user