InkCanvas는 완전히 투명한 오버레이이므로 잉크 스트로크 속성을 설정하기 위한 기본 제공 UI를 제공하지 않습니다. 기본 수동 입력 환경을 변경하고 사용자가 잉크 스트로크 속성을 설정하고 다른 사용자 지정 수동 입력 기능을 지원해야 하는 경우 다음 두 가지 옵션이 있습니다.
코드 숨김에서 InkCanvas에 바인딩된 기본 InkPresenter 개체를 사용합니다.
어떤 경우에는 사용자 기본 설정 또는 디바이스 상태에 따라 잉크 도구 모음의 위치와 방향을 설정하고자 할 수도 있습니다. 다음은 설정 > 디바이스 > 펜 & Windows Ink > 펜 > 글을 쓸 때 사용하는 손 선택에서 지정한 왼손 또는 오른손 쓰기 기본 설정에 따라 잉크 도구 모음의 위치와 방향을 설정하는 방법을 보여주는 예입니다.
주요 손 설정
Windows.UI.ViewManagement의 HandPreference 속성을 통해 이 설정을 쿼리하고 반환된 값에 따라 HorizontalAlignment를 설정할 수 있습니다. 이 예에서는 왼손을 주로 사용하는 사용자를 위해 앱의 왼쪽에 그리고 오른손을 주로 사용하는 사용자를 위해 오른쪽에 도구 모음을 배치합니다.
바인딩을 사용하여 사용자 기본 설정, 디바이스 설정 또는 디바이스 상태 변경 내용에 따른 UI 업데이트를 검색할 수도 있습니다. 다음 예에서는 이전 예제를 확장하여 바인딩, ViewMOdel 개체 및 INotifyPropertyChanged 인터페이스를 사용하여 디바이스 방향에 따라 잉크 도구 모음을 동적으로 위치 조정하는 방법을 보여줍니다.
ViewModels 폴더에 새 클래스를 추가합니다(이 예에서는 InkToolbarSnippetHostViewModel.cs).
참고
응용 프로그램의 수명 기간 동안 이러한 유형의 개체는 하나만 필요하므로 싱글톤 패턴을 사용했습니다.
파일에 using System.ComponentModel 네임스페이스를 추가합니다.
인스턴스라는 고정적인 멤버 변수 및 인스턴스라는 고정적인 읽기 전용 속성을 추가합니다. 인스턴스 속성을 통해서만 이 클래스에 액세스할 수 있도록 하려면 생성자를 비공개 항목으로 설정합니다.
참고
이 클래스는 INotifyPropertyChanged 인터페이스로부터 상속됩니다. 이 인터페이스는 클라이언트, 일반적으로 바인딩 클라이언트에게 속성 값이 변경되었음을 알리는 데 사용됩니다. 이를 사용하여 디바이스 방향 변경을 처리하게 됩니다(이후 단계에서 이 코드를 확장하고 추가로 설명할 예정임).
using System.ComponentModel;
namespace locationandorientation.ViewModels
{
public class InkToolbarSnippetHostViewModel : INotifyPropertyChanged
{
private static InkToolbarSnippetHostViewModel instance;
public static InkToolbarSnippetHostViewModel Instance
{
get
{
if (null == instance)
{
instance = new InkToolbarSnippetHostViewModel();
}
return instance;
}
}
}
private InkToolbarSnippetHostViewModel() { }
}
InkToolbarSnippetHostViewModel 클래스에 두 개의 부울 속성(LeftHandedLayout(이전의 XAML 전용 예제와 기능 동일) 및 PortraitLayout(디바이스의 방향))을 추가합니다.
참고
PortraitLayout 속성은 설정 가능하며 PropertyChanged 이벤트에 대한 정의를 포함합니다.
public bool LeftHandedLayout
{
get
{
bool leftHandedLayout = false;
Windows.UI.ViewManagement.UISettings settings =
new Windows.UI.ViewManagement.UISettings();
leftHandedLayout = (settings.HandPreference ==
Windows.UI.ViewManagement.HandPreference.LeftHanded);
return leftHandedLayout;
}
}
public bool portraitLayout = false;
public bool PortraitLayout
{
get
{
Windows.UI.ViewManagement.ApplicationViewOrientation winOrientation =
Windows.UI.ViewManagement.ApplicationView.GetForCurrentView().Orientation;
portraitLayout =
(winOrientation ==
Windows.UI.ViewManagement.ApplicationViewOrientation.Portrait);
return portraitLayout;
}
set
{
if (value.Equals(portraitLayout)) return;
portraitLayout = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("PortraitLayout"));
}
}
InkToolbarSnippetHostViewModel.cs 파일로 돌아가서 PortraitLayout 및 LeftHandedLayout 부울 속성을 InkToolbarSnippetHostViewModel 클래스에 추가하고 해당 속성 값이 변경되면 PortraitLayout을 다시 바인딩할 수 있도록 지원합니다.
public bool LeftHandedLayout
{
get
{
bool leftHandedLayout = false;
Windows.UI.ViewManagement.UISettings settings =
new Windows.UI.ViewManagement.UISettings();
leftHandedLayout = (settings.HandPreference ==
Windows.UI.ViewManagement.HandPreference.LeftHanded);
return leftHandedLayout;
}
}
public bool portraitLayout = false;
public bool PortraitLayout
{
get
{
Windows.UI.ViewManagement.ApplicationViewOrientation winOrientation =
Windows.UI.ViewManagement.ApplicationView.GetForCurrentView().Orientation;
portraitLayout =
(winOrientation ==
Windows.UI.ViewManagement.ApplicationViewOrientation.Portrait);
return portraitLayout;
}
set
{
if (value.Equals(portraitLayout)) return;
portraitLayout = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("PortraitLayout"));
}
}
#region INotifyPropertyChanged Members
public event PropertyChangedEventHandler PropertyChanged;
protected void OnPropertyChanged(string property)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(property));
}
#endregion
이제 사용자의 주요 손 선호도에 맞게 조정되고 사용자 디바이스의 방향에 동적으로 응답하는 수동 입력 앱이 있어야 합니다.
선택한 단추 지정
초기화 시 연필 단추가 선택된 Windows Ink 도구 모음
기본적으로 앱이 시작되고 도구 모음이 초기화될 때 첫 번째(또는 맨 왼쪽) 단추가 선택됩니다. 이것은 기본 Windows Ink 도구 모음에서 볼펜 단추입니다.
프레임워크는 기본 제공 단추의 순서를 정의하기 때문에 첫 번째 단추는 기본적으로 활성화하려는 펜이나 도구가 아닐 수 있습니다.
이 기본 동작을 재정의하고 도구 모음에서 선택한 단추를 지정할 수 있습니다.
이 예제에서는 연필 단추가 선택되고 연필이 활성화된(볼펜 대신) 상태에서 기본 도구 모음을 초기화합니다.
/// <summary>
/// An empty page that can be used on its own or navigated to within a Frame.
/// Here, we set up InkToolbar event listeners.
/// </summary>
public MainPage_CodeBehind()
{
this.InitializeComponent();
// Add handlers for InkToolbar events.
inkToolbar.Loaded += inkToolbar_Loaded;
}
/// <summary>
/// Handle the Loaded event of the InkToolbar.
/// By default, the active tool is set to the first tool on the toolbar.
/// Here, we set the active tool to the pencil button.
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void inkToolbar_Loaded(object sender, RoutedEventArgs e)
{
InkToolbarToolButton pencilButton = inkToolbar.GetToolButton(InkToolbarTool.Pencil);
inkToolbar.ActiveTool = pencilButton;
}
기본 제공 단추 지정
초기화 시 포함되는 특정 단추
언급한 바와 같이 Windows Ink 도구 모음에는 기본 제공 단추 컬렉션이 포함되어 있습니다. 이러한 단추는 다음 순서로(왼쪽에서 오른쪽으로) 표시됩니다.
단추는 여기에 지정된 순서가 아니라 프레임워크에서 정의한 순서대로 도구 모음에 추가됩니다.
<Grid Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
<Grid.RowDefinitions>
<RowDefinition Height="Auto"/>
<RowDefinition Height="*"/>
</Grid.RowDefinitions>
<StackPanel x:Name="HeaderPanel" Orientation="Horizontal" Grid.Row="0">
<TextBlock x:Name="Header"
Text="Basic ink sample"
Style="{ThemeResource HeaderTextBlockStyle}"
Margin="10,0,0,0" />
</StackPanel>
<Grid Grid.Row="1">
<Image Source="Assets\StoreLogo.png" />
<!-- Clear the default InkToolbar buttons by setting InitialControls to None. -->
<!-- Set the active tool to the pencil button. -->
<InkCanvas x:Name="inkCanvas" />
<InkToolbar x:Name="inkToolbar"
VerticalAlignment="Top"
TargetInkCanvas="{x:Bind inkCanvas}"
InitialControls="None">
<!--
Add only the ballpoint pen, pencil, and eraser.
Note that the buttons are added to the toolbar in the order
defined by the framework, not the order we specify here.
-->
<InkToolbarEraserButton />
<InkToolbarBallpointPenButton />
<InkToolbarPencilButton/>
</InkToolbar>
</Grid>
</Grid>
코드 숨김
첫 번째 예제에서 InkCanvas 및 InkToolbar에 대한 XAML 선언을 사용합니다.
/// <summary>
/// An empty page that can be used on its own or navigated to within a Frame.
/// Here, we set up InkToolbar event listeners.
/// </summary>
public MainPage_CodeBehind()
{
this.InitializeComponent();
// Add handlers for InkToolbar events.
inkToolbar.Loading += inkToolbar_Loading;
}
단추는 여기에 지정된 순서가 아니라 프레임워크에서 정의한 순서대로 도구 모음에 추가됩니다.
InkToolbar에 단추를 추가합니다.
/// <summary>
/// Handles the Loading event of the InkToolbar.
/// Here, we identify the buttons to include on the InkToolbar.
/// </summary>
/// <param name="sender">The InkToolbar</param>
/// <param name="args">The InkToolbar event data.
/// If there is no event data, this parameter is null</param>
private void inkToolbar_Loading(FrameworkElement sender, object args)
{
// Clear all built-in buttons from the InkToolbar.
inkToolbar.InitialControls = InkToolbarInitialControls.None;
// Add only the ballpoint pen, pencil, and eraser.
// Note that the buttons are added to the toolbar in the order
// defined by the framework, not the order we specify here.
InkToolbarBallpointPenButton ballpoint = new InkToolbarBallpointPenButton();
InkToolbarPencilButton pencil = new InkToolbarPencilButton();
InkToolbarEraserButton eraser = new InkToolbarEraserButton();
inkToolbar.Children.Add(eraser);
inkToolbar.Children.Add(ballpoint);
inkToolbar.Children.Add(pencil);
}
사용자 지정 단추 및 수동 입력 기능
InkToolbar를 통해 제공되는 단추(및 관련 수동 입력 기능)의 컬렉션을 사용자 지정하고 확장할 수 있습니다.
InkToolbar는 단추 유형의 두 가지 고유한 그룹으로 구성됩니다.
"도구" 버튼 그룹에는 빌트인 그리기, 지우기 및 강조 표시 버튼이 있습니다. 사용자 지정 펜 및 도구가 여기에 추가됩니다.
참고 기능 선택은 함께 사용할 수 없습니다.
기본 제공 눈금자 단추가 포함된 "토글" 단추 그룹. 사용자 지정 토글이 여기에 추가됩니다.
참고 기능은 함께 사용할 수 있으며 다른 활성 도구와 동시에 사용할 수 있습니다.
애플리케이션 및 필요한 수동 입력 기능에 따라 다음 단추(사용자 지정 잉크 기능에 바인딩됨)를 InkToolbar에 추가할 수 있습니다.
사용자 지정 펜 – 잉크 색상표 및 펜 팁 속성(예: 모양, 회전 및 크기)이 호스트 앱에서 정의되는 펜입니다.
사용자 지정 도구 – 호스트 앱에서 정의한 도구 중 펜이 아닌 도구입니다.
사용자 지정 토글 – 앱 정의 기능의 상태를 켜거나 끄도록 설정합니다. 이 기능을 켜면 기능이 활성 도구와 함께 작동합니다.
참고 기본 제공 단추의 표시 순서는 변경할 수 없습니다. 기본 표시 순서는 볼펜, 연필, 형광펜, 지우개 및 눈금자입니다. 사용자 지정 펜은 마지막 기본 펜에 추가되고, 마지막 펜 버튼과 지우개 버튼 사이에 사용자 지정 도구 단추가 추가되며 눈금자 버튼 뒤에 사용자 지정 토글 버튼이 추가됩니다. (사용자 지정 단추는 지정된 순서대로 추가됩니다.)
사용자 지정 펜
사용자 지정 펜(사용자 지정 펜 단추를 통해 활성화됨)을 만들어 잉크 색상표 및 펜 팁 속성(예: 도형, 회전 및 크기)을 정의할 수 있습니다.
사용자 지정 붓글씨 펜 단추
이 예제에서는 기본 서예 잉크 스트로크를 사용하도록 설정하는 광범위한 팁으로 사용자 지정 펜을 정의합니다. 단추 플라이아웃에 표시되는 색상표에서 브러시 컬렉션도 사용자 지정합니다.
코드 숨김
먼저, 사용자 지정 펜을 정의하고 코드 숨김에서 그리기 특성을 지정합니다. 나중에 XAML에서 이 사용자 지정 펜을 참조합니다.
솔루션 탐색기에서 프로젝트를 마우스 오른쪽 단추로 클릭하고 추가 -> 새 항목을 선택합니다.
Visual C# -> 코드에서 새 클래스 파일을 추가하고 CalligraphicPen.cs를 호출합니다.
Calligraphic.cs에서 default using 블록을 다음으로 바꿉다.
using System.Numerics;
using Windows.UI;
using Windows.UI.Input.Inking;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Media;
CalligraphicPen.cs에 정의된 사용자 지정 펜()과 사용자 지정 펜(CalligraphicPen)과 사용자 지정 펜(CalligraphicPenPalette)에서 지원하는 브러시 컬렉션에 대한 참조를 만드는 로컬 페이지 리소스 사전을 선언합니다.
<Page.Resources>
<!-- Add the custom CalligraphicPen to the page resources. -->
<local:CalligraphicPen x:Key="CalligraphicPen" />
<!-- Specify the colors for the palette of the custom pen. -->
<BrushCollection x:Key="CalligraphicPenPalette">
<SolidColorBrush Color="Blue" />
<SolidColorBrush Color="Red" />
</BrushCollection>
</Page.Resources>
이전 코드 조각에서는 터치 수동 입력(toggleButton)을 위한 사용자 지정 토글 단추에 Click 이벤트 수신기 및 처리기(Toggle_Custom)를 선언했습니다. 이 처리기는 InkPresenter의 InputDeviceTypes 속성을 통해 CoreInputDeviceTypes.Touch에 대한 지원을 간단하게 토글합니다.
그리고 SymbolIcon 요소와 코드 숨김 파일(TouchWritingIcon)에 정의된 필드에 바인딩하는 {x:Bind} 태그 확장을 사용하여 단추의 아이콘을 지정했습니다.
다음 코드 조각에는 Click 이벤트 처리기와 TouchWritingIcon 정의가 모두 포함됩니다.
namespace Ink_Basic_InkToolbar
{
/// <summary>
/// An empty page that can be used on its own or navigated to within a Frame.
/// </summary>
public sealed partial class MainPage_AddCustomToggle : Page
{
Symbol TouchWritingIcon = (Symbol)0xED5F;
public MainPage_AddCustomToggle()
{
this.InitializeComponent();
}
// Handler for the custom toggle button that enables touch inking.
private void CustomToggle_Click(object sender, RoutedEventArgs e)
{
if (toggleButton.IsChecked == true)
{
inkCanvas.InkPresenter.InputDeviceTypes |= CoreInputDeviceTypes.Touch;
}
else
{
inkCanvas.InkPresenter.InputDeviceTypes &= ~CoreInputDeviceTypes.Touch;
}
}
}
}
사용자 지정 도구
사용자 지정 도구 단추를 만들어 앱에서 정의한 펜이 아닌 도구를 호출할 수 있습니다.
기본적으로 InkPresenter는 모든 입력을 잉크 스트로크 또는 지우기 스트로크로 처리합니다. 여기에는 펜 배럴 단추, 마우스 오른쪽 단추 또는 이와 유사한 보조 하드웨어 지원 기능으로 수정된 입력이 포함됩니다. 그러나 특정 입력을 처리되지 않은 상태로 두도록 InkPresenter를 구성할 수 있으며, 이후 사용자 지정 처리를 위해 앱으로 전달할 수 있습니다.
이 예제에서는 선택할 때 후속 스트로크를 처리하고 잉크 대신 선택 올가미(파선)로 렌더링하는 사용자 지정 도구 단추를 정의합니다. 선택 영역 범위 내의 모든 잉크 스트로크가 선택됨으로 설정됩니다.
참고
InkCanvas 및 InkToolbar UX 지침 모두 Inking 컨트롤을 참조하세요. 다음 권장 사항은 이 예제와 관련이 있습니다.
스트로크 선택을 입력할 때 "선택 도구" 도구 설명과 함께 "Segoe MLD2 자산" 글꼴의 "EF20" 아이콘을 도구 단추에 사용하는 것이 좋습니다.
XAML
먼저 스트로크 선택이 구성된 이벤트 처리기(customToolButton_Click)를 지정하는 Click 이벤트 수신기를 사용하여 InkToolbarCustomToolButton 요소(customToolButton)를 선언합니다. (스트로크 선택 영역을 복사, 절단 및 붙여넣기 위한 단추 집합도 추가했습니다.)
선택 스트로크를 그리기 위한 Canvas 요소도 추가합니다. 별도의 레이어를 사용하여 선택 스트로크를 그리면 InkCanvas 및 그 콘텐츠가 그대로 유지됩니다.
그리고 SymbolIcon 요소와 코드 숨김 파일(SelectIcon)에 정의된 필드에 바인딩하는 {x:Bind} 태그 확장을 사용하여 단추의 아이콘을 지정했습니다.
다음 코드 조각에는 Click 이벤트 처리기와 SelectIcon 정의가 모두 포함됩니다.
namespace Ink_Basic_InkToolbar
{
/// <summary>
/// An empty page that can be used on its own or navigated to within a Frame.
/// </summary>
public sealed partial class MainPage_AddCustomTool : Page
{
// Icon for custom selection tool button.
Symbol SelectIcon = (Symbol)0xEF20;
// Stroke selection tool.
private Polyline lasso;
// Stroke selection area.
private Rect boundingRect;
public MainPage_AddCustomTool()
{
this.InitializeComponent();
// Listen for new ink or erase strokes to clean up selection UI.
inkCanvas.InkPresenter.StrokeInput.StrokeStarted +=
StrokeInput_StrokeStarted;
inkCanvas.InkPresenter.StrokesErased +=
InkPresenter_StrokesErased;
}
private void customToolButton_Click(object sender, RoutedEventArgs e)
{
// By default, the InkPresenter processes input modified by
// a secondary affordance (pen barrel button, right mouse
// button, or similar) as ink.
// To pass through modified input to the app for custom processing
// on the app UI thread instead of the background ink thread, set
// InputProcessingConfiguration.RightDragAction to LeaveUnprocessed.
inkCanvas.InkPresenter.InputProcessingConfiguration.RightDragAction =
InkInputRightDragAction.LeaveUnprocessed;
// Listen for unprocessed pointer events from modified input.
// The input is used to provide selection functionality.
inkCanvas.InkPresenter.UnprocessedInput.PointerPressed +=
UnprocessedInput_PointerPressed;
inkCanvas.InkPresenter.UnprocessedInput.PointerMoved +=
UnprocessedInput_PointerMoved;
inkCanvas.InkPresenter.UnprocessedInput.PointerReleased +=
UnprocessedInput_PointerReleased;
}
// Handle new ink or erase strokes to clean up selection UI.
private void StrokeInput_StrokeStarted(
InkStrokeInput sender, Windows.UI.Core.PointerEventArgs args)
{
ClearSelection();
}
private void InkPresenter_StrokesErased(
InkPresenter sender, InkStrokesErasedEventArgs args)
{
ClearSelection();
}
private void cutButton_Click(object sender, RoutedEventArgs e)
{
inkCanvas.InkPresenter.StrokeContainer.CopySelectedToClipboard();
inkCanvas.InkPresenter.StrokeContainer.DeleteSelected();
ClearSelection();
}
private void copyButton_Click(object sender, RoutedEventArgs e)
{
inkCanvas.InkPresenter.StrokeContainer.CopySelectedToClipboard();
}
private void pasteButton_Click(object sender, RoutedEventArgs e)
{
if (inkCanvas.InkPresenter.StrokeContainer.CanPasteFromClipboard())
{
inkCanvas.InkPresenter.StrokeContainer.PasteFromClipboard(
new Point(0, 0));
}
else
{
// Cannot paste from clipboard.
}
}
// Clean up selection UI.
private void ClearSelection()
{
var strokes = inkCanvas.InkPresenter.StrokeContainer.GetStrokes();
foreach (var stroke in strokes)
{
stroke.Selected = false;
}
ClearBoundingRect();
}
private void ClearBoundingRect()
{
if (selectionCanvas.Children.Any())
{
selectionCanvas.Children.Clear();
boundingRect = Rect.Empty;
}
}
// Handle unprocessed pointer events from modified input.
// The input is used to provide selection functionality.
// Selection UI is drawn on a canvas under the InkCanvas.
private void UnprocessedInput_PointerPressed(
InkUnprocessedInput sender, PointerEventArgs args)
{
// Initialize a selection lasso.
lasso = new Polyline()
{
Stroke = new SolidColorBrush(Windows.UI.Colors.Blue),
StrokeThickness = 1,
StrokeDashArray = new DoubleCollection() { 5, 2 },
};
lasso.Points.Add(args.CurrentPoint.RawPosition);
selectionCanvas.Children.Add(lasso);
}
private void UnprocessedInput_PointerMoved(
InkUnprocessedInput sender, PointerEventArgs args)
{
// Add a point to the lasso Polyline object.
lasso.Points.Add(args.CurrentPoint.RawPosition);
}
private void UnprocessedInput_PointerReleased(
InkUnprocessedInput sender, PointerEventArgs args)
{
// Add the final point to the Polyline object and
// select strokes within the lasso area.
// Draw a bounding box on the selection canvas
// around the selected ink strokes.
lasso.Points.Add(args.CurrentPoint.RawPosition);
boundingRect =
inkCanvas.InkPresenter.StrokeContainer.SelectWithPolyLine(
lasso.Points);
DrawBoundingRect();
}
// Draw a bounding rectangle, on the selection canvas, encompassing
// all ink strokes within the lasso area.
private void DrawBoundingRect()
{
// Clear all existing content from the selection canvas.
selectionCanvas.Children.Clear();
// Draw a bounding rectangle only if there are ink strokes
// within the lasso area.
if (!((boundingRect.Width == 0) ||
(boundingRect.Height == 0) ||
boundingRect.IsEmpty))
{
var rectangle = new Rectangle()
{
Stroke = new SolidColorBrush(Windows.UI.Colors.Blue),
StrokeThickness = 1,
StrokeDashArray = new DoubleCollection() { 5, 2 },
Width = boundingRect.Width,
Height = boundingRect.Height
};
Canvas.SetLeft(rectangle, boundingRect.X);
Canvas.SetTop(rectangle, boundingRect.Y);
selectionCanvas.Children.Add(rectangle);
}
}
}
}
잉크 렌더링 사용자 지정
기본적으로 잉크 입력은 대기 시간이 짧은 백그라운드 스레드에서 처리되고 그려질 때 "젖은" 상태로 렌더링됩니다. 스트로크가 완료되면(펜 또는 손가락을 떼거나 마우스 단추를 놓으면) 스트로크가 UI 스레드에서 처리되고 InkCanvas 계층(애플리케이션 콘텐츠 위 및 젖은 잉크 교체)에 "건조"하게 렌더링됩니다.
잉크 플랫폼을 사용하면 이 동작을 재정의하고 잉크 입력을 사용자 지정 건조하여 수동 입력 환경을 완전히 사용자 지정할 수 있습니다.
사용자 지정 건조 및 InkToolbar
앱이 사용자 지정 건조 구현을 사용하여 InkPresenter의 기본 잉크 렌더링 동작을 재정의하는 경우 렌더링된 잉크 스트로크는 더 이상 InkToolbar에서 사용할 수 없으며 InkToolbar의 기본 제공 지우기 명령은 예상대로 작동하지 않습니다. 지우기 기능을 제공하려면 모든 포인터 이벤트를 처리하고, 각 스트로크에서 적중 테스트를 수행하고, 기본 제공 "모든 잉크 지우기" 명령을 재정의해야 합니다.
Power Apps model-driven apps implement a consistent user experience. A key component of that experience is how users are presented and can interact with available commands. As part of designing the model-driven app, a maker can customize the bars to add commands, hide commands and modify their behavior. This learning path introduces you to how to customize the command bar. We also cover common scenarios and advanced concepts.