Merge pull request #254 from yangjieshao/main
ImageViewer 鼠标滚动最小缩放比例从原图的0.1倍改为渲染区域的0.1倍
This commit is contained in:
BIN
demo/Ursa.Demo/Assets/3x.png
Normal file
BIN
demo/Ursa.Demo/Assets/3x.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 3.1 MiB |
@@ -20,8 +20,9 @@
|
|||||||
<u:ImageViewer
|
<u:ImageViewer
|
||||||
Name="viewer"
|
Name="viewer"
|
||||||
Width="600"
|
Width="600"
|
||||||
Height="300"
|
Height="300"
|
||||||
Source="../Assets/WORLD.png">
|
MinScale="0.5"
|
||||||
|
Source="../Assets/3x.png">
|
||||||
<u:ImageViewer.Overlayer>
|
<u:ImageViewer.Overlayer>
|
||||||
<Grid
|
<Grid
|
||||||
HorizontalAlignment="Stretch"
|
HorizontalAlignment="Stretch"
|
||||||
|
|||||||
@@ -51,6 +51,16 @@ public class ImageViewer: TemplatedControl
|
|||||||
set => SetAndRaise(ScaleProperty, ref _scale, value);
|
set => SetAndRaise(ScaleProperty, ref _scale, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static readonly DirectProperty<ImageViewer, double> MinScaleProperty = AvaloniaProperty.RegisterDirect<ImageViewer, double>(
|
||||||
|
nameof(MinScale), o => o.MinScale, (o, v) => o.MinScale = v, unsetValue: 0.1);
|
||||||
|
|
||||||
|
public double MinScale
|
||||||
|
{
|
||||||
|
get => _minScale;
|
||||||
|
set => SetAndRaise(ScaleProperty, ref _minScale, value);
|
||||||
|
}
|
||||||
|
private double _minScale = 1;
|
||||||
|
|
||||||
private double _translateX;
|
private double _translateX;
|
||||||
|
|
||||||
public static readonly DirectProperty<ImageViewer, double> TranslateXProperty = AvaloniaProperty.RegisterDirect<ImageViewer, double>(
|
public static readonly DirectProperty<ImageViewer, double> TranslateXProperty = AvaloniaProperty.RegisterDirect<ImageViewer, double>(
|
||||||
@@ -101,6 +111,8 @@ public class ImageViewer: TemplatedControl
|
|||||||
set => SetValue(StretchProperty, value);
|
set => SetValue(StretchProperty, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private double _sourceMinScale = 0.1;
|
||||||
|
|
||||||
static ImageViewer()
|
static ImageViewer()
|
||||||
{
|
{
|
||||||
FocusableProperty.OverrideDefaultValue<ImageViewer>(true);
|
FocusableProperty.OverrideDefaultValue<ImageViewer>(true);
|
||||||
@@ -109,6 +121,7 @@ public class ImageViewer: TemplatedControl
|
|||||||
TranslateXProperty.Changed.AddClassHandler<ImageViewer>((o,e)=>o.OnTranslateXChanged(e));
|
TranslateXProperty.Changed.AddClassHandler<ImageViewer>((o,e)=>o.OnTranslateXChanged(e));
|
||||||
TranslateYProperty.Changed.AddClassHandler<ImageViewer>((o, e) => o.OnTranslateYChanged(e));
|
TranslateYProperty.Changed.AddClassHandler<ImageViewer>((o, e) => o.OnTranslateYChanged(e));
|
||||||
StretchProperty.Changed.AddClassHandler<ImageViewer>((o, e) => o.OnStretchChanged(e));
|
StretchProperty.Changed.AddClassHandler<ImageViewer>((o, e) => o.OnStretchChanged(e));
|
||||||
|
MinScaleProperty.Changed.AddClassHandler<ImageViewer>((o, e) => o.OnMinScaleChanged(e));
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnTranslateYChanged(AvaloniaPropertyChangedEventArgs args)
|
private void OnTranslateYChanged(AvaloniaPropertyChangedEventArgs args)
|
||||||
@@ -161,14 +174,40 @@ public class ImageViewer: TemplatedControl
|
|||||||
_image.Height = size.Height;
|
_image.Height = size.Height;
|
||||||
}
|
}
|
||||||
Scale = GetScaleRatio(width/size.Width, height/size.Height, this.Stretch);
|
Scale = GetScaleRatio(width/size.Width, height/size.Height, this.Stretch);
|
||||||
|
_sourceMinScale = Math.Min(width * MinScale / size.Width, height * MinScale / size.Height);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnStretchChanged(AvaloniaPropertyChangedEventArgs args)
|
private void OnStretchChanged(AvaloniaPropertyChangedEventArgs args)
|
||||||
{
|
{
|
||||||
var stretch = args.GetNewValue<Stretch>();
|
var stretch = args.GetNewValue<Stretch>();
|
||||||
Scale = GetScaleRatio(Width / _image!.Width, Height / _image!.Height, stretch);
|
Scale = GetScaleRatio(Width / _image!.Width, Height / _image!.Height, stretch);
|
||||||
|
if(_image is { })
|
||||||
|
{
|
||||||
|
_sourceMinScale = Math.Min(Width * MinScale / _image.Width, Height * MinScale / _image.Height);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_sourceMinScale = MinScale;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void OnMinScaleChanged(AvaloniaPropertyChangedEventArgs args)
|
||||||
|
{
|
||||||
|
if (_image is { })
|
||||||
|
{
|
||||||
|
_sourceMinScale = Math.Min(Width * MinScale / _image.Width, Height * MinScale / _image.Height);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_sourceMinScale = MinScale;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_sourceMinScale > Scale)
|
||||||
|
{
|
||||||
|
Scale = _sourceMinScale;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private double GetScaleRatio(double widthRatio, double heightRatio, Stretch stretch)
|
private double GetScaleRatio(double widthRatio, double heightRatio, Stretch stretch)
|
||||||
{
|
{
|
||||||
return stretch switch
|
return stretch switch
|
||||||
@@ -203,6 +242,11 @@ public class ImageViewer: TemplatedControl
|
|||||||
_image.Width = size.Width;
|
_image.Width = size.Width;
|
||||||
_image.Height = size.Height;
|
_image.Height = size.Height;
|
||||||
Scale = GetScaleRatio(width/size.Width, height/size.Height, this.Stretch);
|
Scale = GetScaleRatio(width/size.Width, height/size.Height, this.Stretch);
|
||||||
|
_sourceMinScale = Math.Min(width * MinScale / size.Width, height * MinScale / size.Height);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_sourceMinScale = MinScale;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -218,7 +262,7 @@ public class ImageViewer: TemplatedControl
|
|||||||
{
|
{
|
||||||
var scale = Scale;
|
var scale = Scale;
|
||||||
scale /= 1.1;
|
scale /= 1.1;
|
||||||
if (scale < 0.1) scale = 0.1;
|
if (scale < _sourceMinScale) scale = _sourceMinScale;
|
||||||
Scale = scale;
|
Scale = scale;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user