欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解WPF的InkCanvas選擇模式

 更新時間:2021年04月02日 10:09:11   作者:louzi  
這篇文章主要介紹了WPF InkCanvas選擇模式的相關資料,幫助大家更好的理解和學習使用c# wpf,感興趣的朋友可以了解下

InkCanvas是WPF中進行墨跡繪制的控件,本文介紹下InkCanvas控件是如何進行選擇操作的。文中有誤的地方希望大家進行批評指正。

InkCanvas的選擇效果

使用WPF可以輕松實現白板功能,只需要添加一個InkCanvas控件。修改InkCanvas的EditingMode屬性可以控制InkCanvas的操作模式,如書寫、選擇、擦除等模式。
如下demo在窗口中添加一個InkCanvas,然后添加一個Button實現書寫與選擇模式的切換。

// xaml
<Grid>
  <InkCanvas x:Name="inkCanvas"/>
  <Button x:Name="btnChangeMode" Content="select" Click="Button_Click"
      Width="50" Height="30" HorizontalAlignment="Left" VerticalAlignment="Bottom" Margin="10"/>
</Grid>

// cs
private void Button_Click(object sender, RoutedEventArgs e)
{
	if(btnChangeMode.Content == "select")
	{
    inkCanvas.EditingMode = InkCanvasEditingMode.Select;
    btnChangeMode.Content = "write";
	}
	else
	{
    inkCanvas.EditingMode = InkCanvasEditingMode.Ink;
    btnChangeMode.Content = "select";
	}
}

運行demo,書寫后點擊按鈕進行選擇,可以看到InkCanvas的選擇操作如下圖所示:

從圖中可以看出,InkCanvas的選擇效果有如下特點:

  1. 選中后筆跡高亮;
  2. 選中后顯示選擇框;
  3. 拖動選擇框,選擇框隨著鼠標移動,但選擇的筆跡并未移動。

接下來看下WPF是如何實現這種選擇操作的。

InkCanvas選擇模式的實現

首先,InkCanvas的編輯功能(書寫、擦除、選擇等)是通過EditingCoordinator管理的,該類包含一系列的EditingBehavior,實現選擇過程的為LassoSelectionBehavior類,實現選擇后對選擇框操作的為SelectionEditor與SelectionEditingBehavior。本文主要介紹選擇后對選擇框的操作過程,選擇過程以及筆跡的高亮顯示打算單獨寫一篇文章進行介紹。

在InkCanvas中,與選擇功能相關的對象有InkCanvasSelection、InkCanvasSelectionAdorner及InkCanvasFeedbackAdorner。后兩者為裝飾器,裝飾器的介紹可參考官方文檔。

先看InkCanvasSelectionAdorner類,直接看其OnRender方法,代碼如下。首先繪制了選擇框的背景,然后繪制了選擇框(矩形虛線效果),最后繪制了選擇框上的9個小矩形按鈕。按鈕可以進行拖動調節(jié),具體實現邏輯可以看代碼,本文不贅述。

protected override void OnRender(DrawingContext drawingContext)
{
  DrawBackground(drawingContext);

  Rect rectWireFrame = GetWireFrameRect()
  if(!rectWireFrame.IsEmpty)
  {
    drawingContext.DrawRectangle(null, _adornerBorderPen, rectWireFrame);

    DrawHandles(drawingContext, rectWireFrame);
  }
}

再看InkCanvasFeedbackAdorner類,同樣看OnRender方法,代碼如下。其僅繪制了矩形虛線選擇框,通過這兩個類的OnRender方法,結合上文中的動畫,可以知道選中后使用InkCanvasSelectionAdorner進行裝飾,對選擇框的操作(拖動)使用InkCanvasFeedbackAdorner進行裝飾。

protected override void OnRender(DrawingContext drawingContext)
{
  drawingContext.DrawRectangle(null, _adornerBorderPen,
    new Rect(CornerResizeHandleSize / 2, CornerResizeHandleSize / 2,
    _frameSize.Width - CornerResizeHandleSize, _frameSize.Height - CornerResizeHandleSize));
}

接下來看下這兩個Adorner是對誰進行裝飾的,首先看InkCanvas的OnPreApplyTemplate方法,代碼如下。注釋部分是InkCanvas的Visual Tree,可以了解到InkCanvas的內部結構。再看下SelectionAdorner的初始化,可以看出是對InnerCanvas進行裝飾,InnerCanvas是InkCanvas的內部容器,放置筆跡及其它UIElement。SelectionAdorner添加了對ActiveEditingMode的綁定,當Mode為None時,隱藏,否則顯示。FeedbackAdorner的裝飾對象通過其構造函數可以看出,也是裝飾的InnerCanvas。

internal override void OnPreApplyTemplate()
{
  base.OnPreApplyTemplate();

  // Build our visual tree here.
  // <InkCanvas>
  //   <AdornerDecorator>
  //     <InkPresenter>
  //       <InnerCanvas/>
  //       <ContainerVisual/>
  //       <HostVisual/>
  //     </InkPresenter>
  //     <AdornerLayer>
  //       <InkCanvasSelectionAdorner/>
  //       <InkCanvasFeedbackAdorner/>
  //     </AdornerLayer>
  //   </AdornerDecorator>
  // </InkCanvas>

  if(_localAdornerDecorator == null)
  {
    _localAdornerDecorator = new AdornerDecorator();
    InkPresenter inkPresenter = InkPresenter;

    AddVisualChild(_localAdornerDecorator);
    _localAdornerDecorator.Child = inkPresenter;
    inkPresenter.Child = InnerCanvas;

    _localAdornerDecorator.AdornerLayer.Add(SelectionAdorner);
  }
}

internal InkCanvasSelectionAdorner SelectionAdorner
{
  get
  {
    if(_selectionAdorner == null)
    {
      _selectionAdorner = new InkCanvasSelectionAdorner(InnerCanvas);

      Binding activedEditingModeBinding = new Binding();
      activedEditingModeBinding.Path = new PropertyPath(InkCanvas.ActiveEditingModeProperty);
      activedEditingModeBinding.Mode = BindingMode.OneWay;
      activedEditingModeBinding.Source = this;
      activedEditingModeBinding.Converter = new ActiveEditingMode2VisibilityConverter();
      _selectionAdorner.SetBinding(UIElement.VisibilityProperty, activedEditingModeBinding);
    }

    return _selectionAdorner;
  }
}

// InkCanvasFeedbackAdorner
internal InkCanvasFeedbackAdorner(InkCanvas inkCanvas)
  : base((inkCanvas != null ? inkCanvas.InnerCanvas : null))
  {...}

最后,我們看下對選擇框進行的操作是如何實現的。選擇后會激活SelectionEditingBehavior,在其OnActivate方法中,綁定了SelectionAdorner的MouseMove/MouseUp/LostMouseCapture事件,并調用InkCanvasSelection.StartFeedbackAdorner()方法對FeedbackAdorner進行初始化,將其添加到AdornerLayer中。然后通過響應MouseMove,調用InkSelection.UpdateFeedbackAdorner()方法更新FeedbackAdorner的位置。最后在MouseUp響應中釋放FeedbackAdorner。刪減代碼如下,具體的實現邏輯可以看WPF源碼。

protected override void OnActive()
{
  // ...
  InkCanvas.InkCanvasSelection.StartFeedbackAdorner(_selectionRect, _hitResult);

  InkCanvas.SelectionAdorner.AddHandler(Mouse.MouseUpEvent, new MouseButtonEventHandler(OnMouseUp));
  InkCanvas.SelectionAdorner.AddHandler(Mouse.MouseMoveEvent, new MouseEventHandler(OnMouseMove));
  InkCanvas.SelectionAdorner.AddHandler(Mouse.LostMouseCaptureEvent, new MouseEventHandler(OnLostMouseCapture));
}

private void OnMouseMove(object sender, MouseEventArgs args)
{
  // ...
  InkCanvas.InkCanvasSelection.UpdateFeedbackAdorner(newRect);
  // ...
}

以上就是詳解WPF的InkCanvas選擇模式的詳細內容,更多關于WPF的InkCanvas選擇模式的資料請關注腳本之家其它相關文章!

相關文章

  • C# ManualResetEvent使用方法詳解

    C# ManualResetEvent使用方法詳解

    這篇文章主要為大家詳細介紹了ManualResetEvent使用方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • C#和JavaScript實現交互的方法

    C#和JavaScript實現交互的方法

    最近做一個小項目不可避免的需要前端腳本與后臺進行交互。由于是在asp.net中實現,故問題演化成asp.net中jiavascript與后臺c#如何進行交互。
    2015-05-05
  • C#中TreeView節(jié)點的自定義繪制方法

    C#中TreeView節(jié)點的自定義繪制方法

    這篇文章主要介紹了C#中TreeView節(jié)點的自定義繪制方法,實例展示了TreeView節(jié)點的操作技巧,需要的朋友可以參考下
    2015-02-02
  • c#文件的復制,移動,創(chuàng)建(實例代碼)

    c#文件的復制,移動,創(chuàng)建(實例代碼)

    c#文件的復制,移動,創(chuàng)建(實例代碼),需要的朋友可以參考一下
    2013-04-04
  • c#多線程中Lock()關鍵字的用法小結

    c#多線程中Lock()關鍵字的用法小結

    本篇文章主要是對c#多線程中Lock()關鍵字的用法進行了詳細的總結介紹,需要的朋友可以過來參考下,希望對大家有所幫助
    2014-01-01
  • winform實現創(chuàng)建最前端窗體的方法

    winform實現創(chuàng)建最前端窗體的方法

    這篇文章主要介紹了winform實現創(chuàng)建最前端窗體的方法,涉及C#窗體屬性設置的相關技巧,非常具有實用價值,需要的朋友可以參考下
    2015-08-08
  • Unity實現畫線條功能

    Unity實現畫線條功能

    這篇文章主要為大家詳細介紹了Unity實現畫線條功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-03-03
  • 開源一個c# 新的雪花算法

    開源一個c# 新的雪花算法

    這篇文章主要介紹了開源一個c# 新的雪花算法的相關資料,幫助大家更好的理解和學習使用c#,感興趣的朋友可以了解下
    2021-03-03
  • C#中的Hashtable?類使用詳解

    C#中的Hashtable?類使用詳解

    這篇文章主要介紹了C#中的Hashtable?類使用詳解,文章圍繞主題展開詳細的內容介紹,具有一定的參考價值,需要的小伙伴可以參考一下
    2022-09-09
  • C# 獲取動態(tài)key的json對象的值案例

    C# 獲取動態(tài)key的json對象的值案例

    這篇文章主要介紹了C# 獲取動態(tài)key的json對象的值案例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-01-01

最新評論