我正在处理一个图像查看器控件,它由一个滚动查看器内部的图像组成。 要进行缩放,我在图像的layouttransform中使用scaletransform。 Zoomin正在点击鼠标。如果用户点击图像,我希望鼠标位置的像素位于滚动查看器的可视区域的中心。 为了实现这一点,我想修改scrollviewer偏移量,但我不确定如何计算将像素放在鼠标指针中心的正确偏移量。
Here描述了如何将滚动条保持在相对位置。这很好,但不是我想要的。
如何正确计算偏移?
以下是我的一些代码:
xaml:
<ScrollViewer Margin="5" Name="scrvImagePanel" VerticalAlignment="Stretch"
CanContentScroll="True"
DockPanel.Dock="Left"
HorizontalAlignment="Stretch"
HorizontalContentAlignment="Stretch"
HorizontalScrollBarVisibility="Auto"
VerticalContentAlignment="Stretch"
VerticalScrollBarVisibility="Auto"
Background="{StaticResource {x:Static SystemColors.ControlDarkBrushKey}}"
ScrollChanged="OnScrollChanged">
<Image Name="imgPicture"
MouseUp="imgMouseUp"
Stretch="None"
HorizontalAlignment="Left"
VerticalAlignment="Top"
>
<Image.LayoutTransform>
<TransformGroup>
<ScaleTransform x:Name="imgZoomTransform" ScaleX="1" ScaleY="1"></ScaleTransform>
</TransformGroup>
</Image.LayoutTransform>
</Image>
</ScrollViewer>
和代码:
private void imgMouseUp(object sender, MouseButtonEventArgs e)
{
Point absScreenPos = e.GetPosition(imgPicture);
//calculate the new zoom factor
this.ZoomFactor = imgZoomTransform.ScaleX;
if (e.ChangedButton == MouseButton.Right)
{
this.ZoomFactor /= 2.0;
}
else if (e.ChangedButton == MouseButton.Left)
{
this.ZoomFactor *= 2.0;
}
//perform the zooming
this.DoZoom(this.ZoomFactor, absScreenPos.X, absScreenPos.Y);
}
private void DoZoom(double zoom, double posX, double posY)
{
// update the scale transform
imgZoomTransform.ScaleX = zoom;
imgZoomTransform.ScaleY = zoom;
this.AdjustScroll(new Point(posX, posY));
}
private void AdjustScroll(Point? centerPoint)
{
if(centerPoint != null)
{
var sv = this.scrvImagePanel;
double offsetX = (centerPoint.Value.X / this.imgPicture.ActualWidth) * sv.ActualWidth;
double offsetY = (centerPoint.Value.Y / this.imgPicture.ActualHeight) * sv.ActualHeight;
sv.ScrollToHorizontalOffset(offsetX);
sv.ScrollToVerticalOffset(offsetY);
}
}
答案 0 :(得分:1)
这个想法很简单,当鼠标停止时,你应该计算它与中心的距离,并将其与当前偏移量相加或相减。
请考虑以下代码:
bool left = false;
double scale = 2;
double ZoomFactor;
private void imgMouseUp(object sender, MouseButtonEventArgs e)
{
Point absScreenPos = e.GetPosition(imgPicture);
//calculate the new zoom factor
this.ZoomFactor = imgZoomTransform.ScaleX;
if (e.ChangedButton == MouseButton.Right)
{
left = false;
this.ZoomFactor /= scale;
}
else if (e.ChangedButton == MouseButton.Left)
{
left = true;
this.ZoomFactor *= scale;
}
//perform the zooming
this.DoZoom(this.ZoomFactor, absScreenPos.X, absScreenPos.Y);
}
private void AdjustScroll(Point? centerPoint)
{
if (centerPoint != null)
{
Point poi = Mouse.GetPosition(scrvImagePanel);
double Xmove = ( scrvImagePanel.ActualHeight) / 2 - poi.X;
double Ymove = (scrvImagePanel.ActualWidth) / 2 - poi.Y;
if (left)
{
scrvImagePanel.ScrollToHorizontalOffset((scrvImagePanel.HorizontalOffset - Xmove) * scale);
scrvImagePanel.ScrollToVerticalOffset((scrvImagePanel.VerticalOffset - Ymove) * scale);
}
else
{
scrvImagePanel.ScrollToHorizontalOffset((scrvImagePanel.HorizontalOffset - Xmove) / scale);
scrvImagePanel.ScrollToVerticalOffset((scrvImagePanel.VerticalOffset - Ymove) / scale);
}
}
}
我知道此代码无法正常运行,但我认为这必须为您提供路径