using System; using System.Windows; using System.Windows.Controls; using System.Windows.Controls.Primitives; using System.Windows.Media; using System.Windows.Input; using AIStudio.Wpf.DiagramDesigner.Helpers; namespace AIStudio.Wpf.DiagramDesigner { public class ZoomBox : Control { private Thumb zoomThumb; private Canvas zoomCanvas; private Slider zoomSlider; #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), new FrameworkPropertyMetadata(null, new PropertyChangedCallback(OnScrollViewerChanged))); private static void OnScrollViewerChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) { ZoomBox target = (ZoomBox)d; ScrollViewer oldvalue = (ScrollViewer)e.OldValue; ScrollViewer newvalue = (ScrollViewer)e.NewValue; target.OnScrollViewerChanged(oldvalue, newvalue); } protected virtual void OnScrollViewerChanged(ScrollViewer oldvalue, ScrollViewer newvalue) { if (newvalue != null) { DesignerCanvas = VisualHelper.FindChild(newvalue); } } #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) { oldDesignerCanvas.LayoutUpdated -= new EventHandler(this.DesignerCanvas_LayoutUpdated); oldDesignerCanvas.MouseWheel -= new MouseWheelEventHandler(this.DesignerCanvas_MouseWheel); } if (newDesignerCanvas != null) { newDesignerCanvas.LayoutUpdated += new EventHandler(this.DesignerCanvas_LayoutUpdated); newDesignerCanvas.MouseWheel += new MouseWheelEventHandler(this.DesignerCanvas_MouseWheel); } } #endregion public static readonly DependencyProperty ZoomValueProperty = DependencyProperty.Register(nameof(ZoomValue), typeof(double), typeof(ZoomBox), new UIPropertyMetadata(1d)); public double ZoomValue { get { return (double)GetValue(ZoomValueProperty); } set { SetValue(ZoomValueProperty, value); } } #endregion public override void OnApplyTemplate() { base.OnApplyTemplate(); if (this.ScrollViewer == null) return; DesignerCanvas = VisualHelper.FindChild(this.ScrollViewer); 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(this.ZoomSlider_ValueChanged); } private void ZoomSlider_ValueChanged(object sender, RoutedPropertyChangedEventArgs 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.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 * ZoomValue; double h = DesignerCanvas.ActualHeight * ZoomValue; // 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; } } }