using System; using System.Windows; using System.Windows.Controls; using System.Windows.Controls.Primitives; using System.Windows.Media; using System.Windows.Input; namespace AIStudio.Wpf.DiagramDesigner { public class ZoomBox : Control { private Thumb zoomThumb; private Canvas zoomCanvas; private Slider zoomSlider; private ScaleTransform scaleTransform; #region DPs #region ScrollViewer public ScrollViewer ScrollViewer { get { return (ScrollViewer)GetValue(ScrollViewerProperty); } set { SetValue(ScrollViewerProperty, value); } } public static readonly DependencyProperty ScrollViewerProperty = DependencyProperty.Register(nameof(ScrollViewer), typeof(ScrollViewer), typeof(ZoomBox)); #endregion #region DesignerCanvas public static readonly DependencyProperty DesignerCanvasProperty = DependencyProperty.Register(nameof(DesignerCanvas), typeof(DesignerCanvas), typeof(ZoomBox), new FrameworkPropertyMetadata(null, new PropertyChangedCallback(OnDesignerCanvasChanged))); public DesignerCanvas DesignerCanvas { get { return (DesignerCanvas)GetValue(DesignerCanvasProperty); } set { SetValue(DesignerCanvasProperty, value); } } private static void OnDesignerCanvasChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) { ZoomBox target = (ZoomBox)d; DesignerCanvas oldDesignerCanvas = (DesignerCanvas)e.OldValue; DesignerCanvas newDesignerCanvas = target.DesignerCanvas; target.OnDesignerCanvasChanged(oldDesignerCanvas, newDesignerCanvas); } protected virtual void OnDesignerCanvasChanged(DesignerCanvas oldDesignerCanvas, DesignerCanvas newDesignerCanvas) { if (oldDesignerCanvas != null) { newDesignerCanvas.LayoutUpdated -= new EventHandler(this.DesignerCanvas_LayoutUpdated); newDesignerCanvas.MouseWheel -= new MouseWheelEventHandler(this.DesignerCanvas_MouseWheel); } if (newDesignerCanvas != null) { newDesignerCanvas.LayoutUpdated += new EventHandler(this.DesignerCanvas_LayoutUpdated); newDesignerCanvas.MouseWheel += new MouseWheelEventHandler(this.DesignerCanvas_MouseWheel); newDesignerCanvas.LayoutTransform = this.scaleTransform; } } #endregion #endregion public override void OnApplyTemplate() { base.OnApplyTemplate(); if (this.ScrollViewer == null) return; this.zoomThumb = Template.FindName("PART_ZoomThumb", this) as Thumb; if (this.zoomThumb == null) throw new Exception("PART_ZoomThumb template is missing!"); this.zoomCanvas = Template.FindName("PART_ZoomCanvas", this) as Canvas; if (this.zoomCanvas == null) throw new Exception("PART_ZoomCanvas template is missing!"); this.zoomSlider = Template.FindName("PART_ZoomSlider", this) as Slider; if (this.zoomSlider == null) throw new Exception("PART_ZoomSlider template is missing!"); this.zoomThumb.DragDelta += new DragDeltaEventHandler(this.Thumb_DragDelta); this.zoomSlider.ValueChanged += new RoutedPropertyChangedEventHandler<double>(this.ZoomSlider_ValueChanged); this.scaleTransform = new ScaleTransform(); } private void ZoomSlider_ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e) { double scale = e.NewValue / e.OldValue; double halfViewportHeight = this.ScrollViewer.ViewportHeight / 2; double newVerticalOffset = ((this.ScrollViewer.VerticalOffset + halfViewportHeight) * scale - halfViewportHeight); double halfViewportWidth = this.ScrollViewer.ViewportWidth / 2; double newHorizontalOffset = ((this.ScrollViewer.HorizontalOffset + halfViewportWidth) * scale - halfViewportWidth); this.scaleTransform.ScaleX *= scale; this.scaleTransform.ScaleY *= scale; this.ScrollViewer.ScrollToHorizontalOffset(newHorizontalOffset); this.ScrollViewer.ScrollToVerticalOffset(newVerticalOffset); } private void Thumb_DragDelta(object sender, DragDeltaEventArgs e) { double scale, xOffset, yOffset; this.InvalidateScale(out scale, out xOffset, out yOffset); this.ScrollViewer.ScrollToHorizontalOffset(this.ScrollViewer.HorizontalOffset + e.HorizontalChange / scale); this.ScrollViewer.ScrollToVerticalOffset(this.ScrollViewer.VerticalOffset + e.VerticalChange / scale); } private void DesignerCanvas_LayoutUpdated(object sender, EventArgs e) { //htzk try { double scale, xOffset, yOffset; this.InvalidateScale(out scale, out xOffset, out yOffset); this.zoomThumb.Width = this.ScrollViewer.ViewportWidth * scale; this.zoomThumb.Height = this.ScrollViewer.ViewportHeight * scale; Canvas.SetLeft(this.zoomThumb, xOffset + this.ScrollViewer.HorizontalOffset * scale); Canvas.SetTop(this.zoomThumb, yOffset + this.ScrollViewer.VerticalOffset * scale); } catch { } } private void DesignerCanvas_MouseWheel(object sender, EventArgs e) { MouseWheelEventArgs wheel = (MouseWheelEventArgs)e; //divide the value by 10 so that it is more smooth double value = Math.Max(0, wheel.Delta / 10); value = Math.Min(wheel.Delta, 10); this.zoomSlider.Value += value; } private void InvalidateScale(out double scale, out double xOffset, out double yOffset) { double w = DesignerCanvas.ActualWidth * this.scaleTransform.ScaleX; double h = DesignerCanvas.ActualHeight * this.scaleTransform.ScaleY; // zoom canvas size double x = this.zoomCanvas.ActualWidth; double y = this.zoomCanvas.ActualHeight; double scaleX = x / w; double scaleY = y / h; scale = (scaleX < scaleY) ? scaleX : scaleY; xOffset = (x - scale * w) / 2; yOffset = (y - scale * h) / 2; } } }