背水一戰 Windows 10 (61) - 控制項(媒體類): InkCanvas 塗鴉編輯

来源:http://www.cnblogs.com/webabcd/archive/2017/07/27/7242825.html
-Advertisement-
Play Games

背水一戰 Windows 10 之 控制項(媒體類): InkCanvas 塗鴉編輯 ...


[源碼下載]


背水一戰 Windows 10 (61) - 控制項(媒體類): InkCanvas 塗鴉編輯



作者:webabcd


介紹
背水一戰 Windows 10 之 控制項(媒體類)

  • InkCanvas 塗鴉編輯



示例
演示 InkCanvas 塗鴉板編輯相關的操作
Controls/MediaControl/InkCanvasDemo2.xaml

<Page
    x:Class="Windows10.Controls.MediaControl.InkCanvasDemo2"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:local="using:Windows10.Controls.MediaControl"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    mc:Ignorable="d">

    <Grid Background="Transparent">
        <StackPanel Margin="10 0 10 10">

            <Grid Background="White" Width="480" Height="320" Margin="5" HorizontalAlignment="Left">
                <!--
                    Canvas - 在本例中用於繪製選中框(本例中選中框通過右鍵繪製)
                -->
                <Canvas Name="selectionCanvas"/>
                <!--
                    InkCanvas - 塗鴉板控制項
                -->
                <InkCanvas Name="inkCanvas" />
            </Grid>

            <Button Name="clear" Content="清除全部塗鴉" Margin="5" Click="clear_Click" />
            <Button Name="cut" Content="剪切選中塗鴉到剪切板" Margin="5" Click="cut_Click" />
            <Button Name="copy" Content="複製選中塗鴉到剪切板" Margin="5" Click="copy_Click" />
            <Button Name="paste" Content="從剪切板粘貼塗鴉" Margin="5" Click="paste_Click" />
            <Button Name="move" Content="移動選中塗鴉" Margin="5" Click="move_Click" />
            <Button Name="cloneAll" Content="克隆全部塗鴉" Margin="5" Click="cloneAll_Click" />
            <Button Name="selectAll" Content="選中全部塗鴉" Margin="5" Click="selectAll_Click" />
            <Button Name="changeColor" Content="改變全部塗鴉的顏色" Margin="5" Click="changeColor_Click" />

        </StackPanel>
    </Grid>
</Page>

Controls/MediaControl/InkCanvasDemo2.xaml.cs

/*
 * InkCanvas - 塗鴉板控制項(繼承自 FrameworkElement, 請參見 /Controls/BaseControl/FrameworkElementDemo/)
 *     InkPresenter - 獲取 InkPresenter 對象
 *        
 * InkPresenter - 塗鴉板
 *     StrokeContainer - 返回 InkStrokeContainer 類型的對象
 *     InputProcessingConfiguration.RightDragAction - 塗鴉的輸入按鍵
 *         AllowProcessing - 所有按鍵均可用於塗鴉,比如滑鼠的左鍵和右鍵均可用於塗鴉。預設值
 *         LeaveUnprocessed - 特殊按鍵不可用於塗鴉,比如滑鼠的右鍵不可用於塗鴉
 *     UnprocessedInput - 用於監聽指針的一些事件
 *         PointerEntered, PointerExited, PointerHovered, PointerLost, PointerMoved, PointerPressed, PointerReleased - 顧名思義的一些事件
 *         註:當 InputProcessingConfiguration.RightDragAction 為 AllowProcessing 時,則只會觸發 PointerEntered, PointerExited, PointerHovered 事件
 *     SetPredefinedConfiguration(InkPresenterPredefinedConfiguration value) - 設置塗鴉行為(SimpleSinglePointer - 單接觸點塗鴉; SimpleMultiplePointer - 多接觸點塗鴉)
 *     StrokesCollected - 當一個或多個 InkStroke 被畫進來時觸發的事件
 *     StrokesErased - 當一個或多個 InkStroke 被擦除時觸發的事件
 *     StrokeInput - 用於監聽一些塗鴉事件
 *         StrokeStarted, StrokeEnded, StrokeContinued, StrokeCanceled - 顧名思義的一些事件
 *     
 * InkStrokeContainer - 用於管理塗鴉
 *     GetStrokes() - 獲取當前塗鴉板的所有 InkStroke 對象集合
 *     BoundingRect - 獲取當前塗鴉板的所有 InkStroke 對象的 Rect 區域
 *     Clear() - 清除所有塗鴉
 *     SelectWithLine(Point from, Point to) - 返回指定對角線內的塗鴉所在的 Rect 區域
 *     SelectWithPolyLine(IEnumerable<Point> polyline) - 返回指定 Polyline 內的塗鴉所在的 Rect 區域
 *     CopySelectedToClipboard() - 複製選中的塗鴉到剪切板
 *     CanPasteFromClipboard() - 剪切板中是否有塗鴉數據
 *     PasteFromClipboard(Point position) - 從剪切板粘貼塗鴉到指定的位置,並返回粘貼塗鴉的 Rect 區域
 *     MoveSelected(Point translation) - 指定偏移量,並移動選中的塗鴉,返回的是移動後的塗鴉的 Rect 區域
 *     DeleteSelected() - 刪除選中的塗鴉,並返回刪除塗鴉的 Rect 區域
 *     AddStroke(InkStroke stroke) - 在塗鴉板上繪製指定的 InkStroke 對象
 *     AddStrokes(IEnumerable<InkStroke> strokes) - 在塗鴉板上繪製指定的 InkStroke 對象集合
 *     
 * InkStroke - 塗鴉對象(這是一次的塗鴉對象,即滑鼠按下後移動然後再抬起後所繪製出的塗鴉)
 *     BoundingRect - 獲取當前 InkStroke 的 Rect 區域
 *     DrawingAttributes - 塗鴉筆尖屬性(參見 InkCanvasDemo1.xaml.cs)
 *     PointTransform - 用於轉換 InkStroke 的 Matrix3x2 仿射轉換矩陣(Matrix3x2 提供了一些簡便的方法:CreateRotation, CreateScale, CreateSkew, CreateTranslation 等)
 *     Selected - 是否被選中
 *     Clone() - 克隆一份,返回克隆後的 InkStroke 對象
 *     GetInkPoints(), GetRenderingSegments() - 用於獲取組成 InkStroke 的一堆貝塞爾曲線的數據
 *     
 * 
 * 註:在 InkCanvas 有濕(wet)和乾(dry)的感念
 * 所謂的濕就是指滑鼠按下後移動,在滑鼠抬起來之前繪製出的塗鴉
 * 所謂的乾就是指滑鼠抬起後,真正顯示在 InkCanvas 上的塗鴉(滑鼠抬起後可以將 wet 做一些自定義處理,然後再 dry 到 InkCanvas 上)
 * 如何在 wet 和 dry 之間做自定義處理呢?需要通過 InkPresenter 的 ActivateCustomDrying() 方法獲取到 InkSynchronizer 對象來實現,具體的示例在之後介紹 Win2D 的時候再寫
 */

using Windows.Foundation;
using Windows.UI;
using Windows.UI.Core;
using Windows.UI.Input.Inking;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Media;
using Windows.UI.Xaml.Shapes;
using System.Linq;
using System.Collections.Generic;
using System;

namespace Windows10.Controls.MediaControl
{
    public sealed partial class InkCanvasDemo2 : Page
    {
        // 右鍵軌跡(用於選擇塗鴉)
        private Polyline _polyline;
        // 被選中的塗鴉所在的 Rect 區域
        private Rect _rect;

        public InkCanvasDemo2()
        {
            this.InitializeComponent();

            inkCanvas.InkPresenter.InputDeviceTypes = CoreInputDeviceTypes.Mouse | CoreInputDeviceTypes.Pen | CoreInputDeviceTypes.Touch;

            InkDrawingAttributes drawingAttributes = inkCanvas.InkPresenter.CopyDefaultDrawingAttributes();
            drawingAttributes.IgnorePressure = true;
            drawingAttributes.Color = Colors.Red;
            drawingAttributes.Size = new Size(4, 4);
            inkCanvas.InkPresenter.UpdateDefaultDrawingAttributes(drawingAttributes);

            // 此設置用於禁用滑鼠右鍵的塗鴉功能,從而在後面實現滑鼠右鍵的選擇塗鴉的功能
            inkCanvas.InkPresenter.InputProcessingConfiguration.RightDragAction = InkInputRightDragAction.LeaveUnprocessed;
            inkCanvas.InkPresenter.UnprocessedInput.PointerPressed += UnprocessedInput_PointerPressed;
            inkCanvas.InkPresenter.UnprocessedInput.PointerMoved += UnprocessedInput_PointerMoved;
            inkCanvas.InkPresenter.UnprocessedInput.PointerReleased += UnprocessedInput_PointerReleased;

            inkCanvas.InkPresenter.StrokeInput.StrokeStarted += StrokeInput_StrokeStarted;
            inkCanvas.InkPresenter.StrokesErased += InkPresenter_StrokesErased;
        }
       
        // 右鍵按下
        private void UnprocessedInput_PointerPressed(InkUnprocessedInput sender, PointerEventArgs args)
        {
            // 右鍵軌跡
            _polyline = new Polyline()
            {
                Stroke = new SolidColorBrush(Colors.Blue),
                StrokeThickness = 1,
                StrokeDashArray = new DoubleCollection() { 5, 2 },
            };
            _polyline.Points.Add(args.CurrentPoint.RawPosition);

            // 繪製右鍵軌跡
            selectionCanvas.Children.Add(_polyline);
        }

        // 右鍵按下後在 InkCanvas 中移動
        private void UnprocessedInput_PointerMoved(InkUnprocessedInput sender, PointerEventArgs args)
        {
            // 更新右鍵軌跡
            _polyline.Points.Add(args.CurrentPoint.RawPosition);
        }

        // 右鍵抬起
        private void UnprocessedInput_PointerReleased(InkUnprocessedInput sender, PointerEventArgs args)
        {
            // 更新右鍵軌跡
            _polyline.Points.Add(args.CurrentPoint.RawPosition);

            // 獲取右鍵圈起來的區域內的塗鴉所在的 Rect 區域
            _rect = inkCanvas.InkPresenter.StrokeContainer.SelectWithPolyLine(_polyline.Points);

            DrawBoundingRect();
        }

        // 繪製 _rect 區域(即被選中的塗鴉所在的 Rect 區域)
        private void DrawBoundingRect()
        {
            selectionCanvas.Children.Clear();

            if ((_rect.Width == 0) || (_rect.Height == 0) || _rect.IsEmpty)
            {
                return;
            }

            var rectangle = new Rectangle()
            {
                Stroke = new SolidColorBrush(Colors.Blue),
                StrokeThickness = 1,
                StrokeDashArray = new DoubleCollection() { 5, 2 },
                Width = _rect.Width,
                Height = _rect.Height
            };

            Canvas.SetLeft(rectangle, _rect.X);
            Canvas.SetTop(rectangle, _rect.Y);

            selectionCanvas.Children.Add(rectangle);
        }


        // 開始此次塗鴉後
        private void StrokeInput_StrokeStarted(InkStrokeInput sender, PointerEventArgs args)
        {
            ClearSelection();
        }

        // 擦除某些塗鴉後
        private void InkPresenter_StrokesErased(InkPresenter sender, InkStrokesErasedEventArgs args)
        {
            ClearSelection();
        }
        
        // 清除塗鴉的選中狀態,以及選中框
        private void ClearSelection()
        {
            IReadOnlyList<InkStroke> strokes = inkCanvas.InkPresenter.StrokeContainer.GetStrokes();
            foreach (var stroke in strokes)
            {
                stroke.Selected = false;
            }
            ClearDrawnBoundingRect();
        }

        // 清除塗鴉的選中框
        private void ClearDrawnBoundingRect()
        {
            if (selectionCanvas.Children.Any())
            {
                selectionCanvas.Children.Clear();
                _rect = Rect.Empty;
            }
        }


        // 清除全部塗鴉
        private void clear_Click(object sender, RoutedEventArgs e)
        {
            ClearDrawnBoundingRect();

            inkCanvas.InkPresenter.StrokeContainer.Clear();
        }

        // 剪切選中塗鴉到剪切板
        private void cut_Click(object sender, RoutedEventArgs e)
        {
            ClearDrawnBoundingRect();

            inkCanvas.InkPresenter.StrokeContainer.CopySelectedToClipboard();
            Rect rect = inkCanvas.InkPresenter.StrokeContainer.DeleteSelected();
        }

        // 複製選中塗鴉到剪切板
        private void copy_Click(object sender, RoutedEventArgs e)
        {
            inkCanvas.InkPresenter.StrokeContainer.CopySelectedToClipboard();
        }

        // 從剪切板粘貼塗鴉
        private void paste_Click(object sender, RoutedEventArgs e)
        {
            if (inkCanvas.InkPresenter.StrokeContainer.CanPasteFromClipboard())
            {
                Rect rect = inkCanvas.InkPresenter.StrokeContainer.PasteFromClipboard(new Point(0, 0));
            }
        }

        // 移動選中塗鴉
        private void move_Click(object sender, RoutedEventArgs e)
        {
            Rect rect = inkCanvas.InkPresenter.StrokeContainer.MoveSelected(new Point(10, 10));

            _rect = rect;
            DrawBoundingRect();
        }

        // 克隆全部塗鴉
        private void cloneAll_Click(object sender, RoutedEventArgs e)
        {
            Random random = new Random();

            IReadOnlyList<InkStroke> oldStrokes = inkCanvas.InkPresenter.StrokeContainer.GetStrokes();
            List<InkStroke> newStrokes = new List<InkStroke>();
            foreach (InkStroke oldStroke in oldStrokes)
            {
                InkStroke newStroke = oldStroke.Clone();
                newStroke.Selected = true;
                newStrokes.Add(newStroke);
            }
            inkCanvas.InkPresenter.StrokeContainer.AddStrokes(newStrokes);
            inkCanvas.InkPresenter.StrokeContainer.MoveSelected(new Point(random.Next(5, 30), random.Next(5, 30)));

            ClearSelection();
        }

        // 選中全部塗鴉
        private void selectAll_Click(object sender, RoutedEventArgs e)
        {
            Rect rect = inkCanvas.InkPresenter.StrokeContainer.BoundingRect;
            _rect = rect;

            IReadOnlyList<InkStroke> strokes = inkCanvas.InkPresenter.StrokeContainer.GetStrokes();
            foreach (var stroke in strokes)
            {
                stroke.Selected = true;
            }

            DrawBoundingRect();
        }

        // 改變全部塗鴉的顏色
        private void changeColor_Click(object sender, RoutedEventArgs e)
        {
            Random random = new Random();
            Color color = Color.FromArgb(255, (byte)random.Next(0, 256), (byte)random.Next(0, 256), (byte)random.Next(0, 256));

            IReadOnlyList<InkStroke> strokes = inkCanvas.InkPresenter.StrokeContainer.GetStrokes();
            foreach (var stroke in strokes)
            {
                InkDrawingAttributes drawingAttributes = stroke.DrawingAttributes;
                drawingAttributes.Color = color;
                stroke.DrawingAttributes = drawingAttributes;
            }
        }
    }
}



OK
[源碼下載]


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 作為一名合格的程式員,我們必須要有一個良好的合作伙伴——個人電腦,因此作為一個程式員,我們要從認識電腦和操作系統開始!如何選擇適合自己的電腦配置?如何選擇和安裝適合自己的操作系統?這是一個程式員應該有的基本素質! 電腦的組成: [ CPU + 記憶體 + 顯卡 + 主板 + 硬碟 + 散熱風扇 ...
  • iostat——I/O statistics(輸入輸出統計) 1. 作用 實時列出整個CPU與介面設備的I/O狀態。 2. 安裝 iostat屬於sysstat軟體包,可以用yum直接安裝: yum install sysstat 3. 用法 iostat 【參數】【間隔秒數】【檢測次數】 參數:[ ...
  • .tar.gz 解壓為 tar -zxvf xx.tar.gz 壓縮為 tar -zcvf target.tar.gz ./src_dir zip 解壓為 unzip target.zip .tar.bz2 解壓為 tar -jxvf xx.tar.bz2 壓縮為 tar -zcvf target. ...
  • python 中 time 有三種格式: float, struct tuple(time.struct_time 或 datetime.datetime), str 常用的: float --> struct tuple: time.localtime( float ) struct time t ...
  • 1,查看ip 地址 ifconifg; 2,查看是否安裝samba伺服器,rpm -qa | grep samba; 3,如果有該伺服器,啟動 service smb start,否則進行安裝 yum install samba; samba-3.0.21b-2 // 如果有這行輸出,則表示安裝了  ...
  • 網上C#USB通信的資料比較少, 基本上都是基於LibUsbDotNet 和 CyUsb, 關於印表機設備的還有一個OPOS。 本篇文章基於LibUsbDotNet。 1. 下載並安裝 LibUsbDotNet 安裝文件。 2. 運行Filter Wizard, Install a device f ...
  • MiniProfiler ,一個簡單而有效的迷你剖析器,可以有效的實時監控頁面。通過直接引用、Ajax、Iframe形式訪問的其它頁面進行監控,監控內容包括資料庫內容,並可以顯示資料庫訪問的SQL。 1.安裝 首先新建一個 asp.net mvc 項目 右鍵項目,管理NuGet程式包。 安裝 Min ...
  • 前言 很多同學可能對於CAP這個項目想有更一步的瞭解,包括她的工作原理以及適用的場景,所以博主就準備了一場直播給大家講解了一下,這個視頻是直播的一個錄像。 由於我這次直播本來是沒有打算對外的,所以也是沒有怎麼準備的,所以在直播的過程中出現了一些講解不到位或者是意外的情況,還請大家能夠諒解~。 最後, ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...