Since almost 5 years, Windows 10 is bringing innovations and powerful usages of devices' capabilities; The Windows Inking platform ✏ is a one great example of natural user interface: With the use of the pen, your computer or tablet becomes a intelligent digital paper! You can free your imagination and draw/paint thanks to apps like Fresh Paint. Also with AI services your written words are understood and the corresponding text is produced; your drawings are recognized and converted into geometric shapes!
In the Enterprise space, writing and drawing can be a vector for innovation and productivity for users. Let's first go deeper in some usages of the Windows Inking capabilities. We will then explore how Azure AI Services can unlock impressive scenarios.
The starting point and docs
The Microsoft docs website gives you all the APIs and controls details to bring you into the Digital Inking journey. Let me give you some pieces of advise for starting:
- Just open/compile/look at the source code of the SimpleInk Sample which is part of the Universal Windows official samples. The XAML is straightforward. The C# code is light and easily understandable. It will give you the ability to give a try to inking code in your project with only some copy/paste. The initial controls to use are
- Keep an eye on the Windows Ink UX guidelines - https://docs.microsoft.com/en-us/windows/uwp/design/controls-and-patterns/inking-controls
- Bookmark the landing page for Inking interactions - https://docs.microsoft.com/en-us/windows/uwp/design/input/pen-and-stylus-interactions
- Refer to the following 2 namespaces
- Windows.UI.Input.Inking - https://docs.microsoft.com/en-us/uwp/api/Windows.UI.Input.Inking
- Windows.UI.Xaml.Controls - https://docs.microsoft.com/en-us/uwp/api/windows.ui.xaml.controls (For Ink* controls)
 
--
With all the documentation mentioned before, you would be easily able to
- Use the InkCanvas for allowing pen/mouse/touch drawing
- Add a InkToolbar to change the pencil type/size/color
- Use the ruler to draw perfect lines
- Select Inkstrokes, copy/cut/paste them
- Save and load serialized Inkstrokes on disk
The idea: Go further of the InkStrokes
The purpose of this blog post is to go deeper on one scenario: Accept Inkstrokes and convert them to shapes! What? Inkstrokes are not good? Do not misunderstand: Strokes are perfect. They are the digital objects representing the strokes that the user created by using the pen/mouse/touch. The idea is to leverage on the drawings to create digital formalized shapes. This way we can produce high fidelity floor plan, measure lengths, areas; we can create vectorial shapes and figures for printing, creating user interfaces, etc.
The initial project
The simplest way to be able to accept inking is to use the InkCanvas control. Create a new empty UWP blank projet and just add the following XAML:
<InkCanvas x:Name="inkCanvas" />
In the C# code, we just say what type of input we accept for this InkCanvas. Make your choice depending of the requirements of your app (You can change it at anytime by code):
// Initialize the InkCanvas
inkCanvas.InkPresenter.InputDeviceTypes =
    Windows.UI.Core.CoreInputDeviceTypes.Mouse |
    Windows.UI.Core.CoreInputDeviceTypes.Pen | 
    Windows.UI.Core.CoreInputDeviceTypes.Touch;
We are done, you can draw on the whole surface of the app with pen/touch/mouse.
Sample drawing by hands on a UWP Window
The source code is available on GitHub - https://github.com/sbovo/UWP-Advanced-Inking/tree/master/StrokesToShapes
Getting the last InkStroke drawn ✏
In order to be able to do some actions after each draw, the Ink platform gives us the StrokesCollected event on the InkPresenter object associated with the InkCanvas.
Note: The InkPresenter provides properties, methods, and events for managing the input. That includes the processing and the rendering of the Ink input. It allows us to gain a fined control on the input.
Let's add a event handler for the StrokesCollected event:
inkCanvas.InkPresenter.StrokesCollected += InkPresenter_StrokesCollected;
Then, in this event handler, we just get the latest stroke from the InkPresenter:
private void InkPresenter_StrokesCollected(
    Windows.UI.Input.Inking.InkPresenter sender, 
    Windows.UI.Input.Inking.InkStrokesCollectedEventArgs args)
 {
     InkStroke stroke = inkCanvas.InkPresenter.StrokeContainer.GetStrokes().Last();
 }
Working with XAML Shapes
XAML Shapes were first introduced in WPF. If you discover XAML Shapes, the following page gives you a perfect startup: Draw shapes - https://docs.microsoft.com/en-us/windows/uwp/design/controls-and-patterns/shapes.
XAML Shapes are basically vector-based regions in the XAML UI. They provide you the ability to draw not only simple forms like rectangles, circles, ellipses but also any path, Bezier curves or geometric shapes. All are vector's shapes displayed on a XAML Canvas.
Let's go back to our goal: We have the last InkStroke and we would like to "replace" it by a nice and friendly XAML Shapes that we could manipulate/move/modify easily by APIs.
First, we modify the XAML page in order to add a new Canvas. The one that will contains all XAML Shapes:
<Canvas x:Name="ShapesCanvas" />
The entire XAML page is like:
<Grid>
   <InkCanvas x:Name="inkCanvas" />
   <!-- Canvas for displaying the "recognized" XAML Shapes -->
   <Canvas x:Name="ShapesCanvas" />
</Grid>
It now like we have two layers, on for drawing on the surface and the other one for displaying/manipulating all the final XAML shapes.
We can now do two actions:
- Create the new XAML Line based on the properties of the last InkStroke and add this line to the ShapesCanvas
- Remove the last InkStroke from the inkCanvas in order to have a clean display made only by XAML Shapes
In order to be able to perform these two actions when the user has just finished to draw a line, we use the StrokesCollected event of the InkPresenter object attached to the InkCanvas. We add the following event handler declaration in the constructor of the page:
// When the user finished to draw something on the InkCanvas
inkCanvas.InkPresenter.StrokesCollected += InkPresenter_StrokesCollected;
Let's code the 2 actions:
private void InkPresenter_StrokesCollected(
     Windows.UI.Input.Inking.InkPresenter sender,
     Windows.UI.Input.Inking.InkStrokesCollectedEventArgs args)
 {
     InkStroke stroke = inkCanvas.InkPresenter.StrokeContainer.GetStrokes().Last();
     
     // Action 1 = We use a function that we will implement just after to create the XAML Line
     Line line = ConvertStrokeToXAMLLine(stroke); 
     // Action 2 = We add the Line in the second Canvas
     ShapesCanvas.Children.Add(line);
     
     // We delete the InkStroke from the InkCanvas
     stroke.Selected = true;
     inkCanvas.InkPresenter.StrokeContainer.DeleteSelected();
 }
THE "conversion" ⚙: InkStroke → XAML Line
It is just about creating a Line (Windows.UI.Xaml.Shapes.Line) object with the initial and final positions of the InkStroke 😒😀:
private Line ConvertStrokeToXAMLLine(InkStroke stroke)
{
	var line = new Line();
	line.Stroke = new SolidColorBrush(Windows.UI.Colors.Green);
	line.StrokeThickness = 6;
	// The origin = (X1, Y1)
	line.X1 = stroke.GetInkPoints().First().Position.X;
	line.Y1 = stroke.GetInkPoints().First().Position.Y;
	// The end = (X2, Y2)
	line.X2 = stroke.GetInkPoints().Last().Position.X;
	line.Y2 = stroke.GetInkPoints().Last().Position.Y;
	return line;
}
Here we go. We did it. We can draw on the surface and the final line will only be the XAML Line we added to the second Canvas.
XAML Shapes (Line) on a UWP Window
This scenario "converting an Ink stroke to a XAML shape" is the basis for being able to achieve great and complex manipulations. Stay tuned for upcoming blog posts about XAML Shapes manipulations and modifications.
As usual, all the source code is available on GitHub - https://github.com/microsoft/Windows-AppConsult-Samples-UWP/
Feedback is welcome 😉
Happy coding on the Windows Platform!
@sbovo for the AppConsult team.
Inking series' articles
This article is part of a series exploring concepts about inking and XAML Shapes. Here are all links:
- 
Use the UWP Inking platform as input for advanced scenarios ⇐ You are here 
- Free your mind: Start manipulating XAML Shapes
References
- 
SimpleInk sample: Find examples of customization and extensibility possibilities provided by the Ink controls and APIs - https://github.com/Microsoft/Windows-universal-samples/tree/master/Samples/SimpleInk 
- 
XAML Controls Gallery sample: Explore the controls in action - https://github.com/Microsoft/Xaml-Controls-Gallery 
- 
Windows Ink UX guidelines - https://docs.microsoft.com/en-us/windows/uwp/design/controls-and-patterns/inking-controls 
- 
Pen interactions and Windows Ink in UWP apps - https://docs.microsoft.com/en-us/windows/uwp/design/input/pen-and-stylus-interactions 
- 
Windows.UI.Input.Inking - https://docs.microsoft.com/en-us/uwp/api/Windows.UI.Input.Inking 
- 
Windows.UI.Xaml.Controls - https://docs.microsoft.com/en-us/uwp/api/windows.ui.xaml.controls (For Ink* controls) 
- 
Windows.UI.Xaml.Shapes - https://docs.microsoft.com/en-us/uwp/api/Windows.UI.Xaml.Shapes 
- 
Draw shapes - https://docs.microsoft.com/en-us/windows/uwp/design/controls-and-patterns/shapes 
- 
XAML Line - https://docs.microsoft.com/en-us/uwp/api/Windows.UI.Xaml.Shapes.Line