'Programming/XAML'에 해당되는 글 15건
- 2018.01.02
- 2017.12.22
- 2017.12.20
- 2017.12.19
- 2017.12.16
- 2017.12.15
- 2017.12.13
- 2017.12.12
- 2017.12.12
- 2017.12.12
Copyright
이 글은 Channer9에 Andy Wigley, BobTabor, Clint Ruthas, Chavin이라는 4명의 저자가 만드신 'Windows 10 development for absolute beginners'라는 무료강의 시리즈의 정리노트(UWP Cheat Sheet.txt)로 누구나 다운받을 수 있는 파일의 내용을 욺긴 것입니다.
UWP Cheat Sheet
***************
UWP-004 - What is XAML?
=======================================
XAML - XML Syntax, create instances of Classes that define the UI.
UWP-005 - Understanding Type Converters
=======================================
Type Converters - Convert literal strings in XAML into enumerations, instances of classes, etc.
UWP-006 - Understanding Default Properties, Complex Properties and the Property Element Syntax
=======================================
Default Property ... Ex. sets Content property:
<Button>Click Me</Button>
Complex Properties - Break out a property into its own element syntax:
<Button Name="ClickMeButton"
HorizontalAlignment="Left"
Content="Click Me"
Margin="20,20,0,0"
VerticalAlignment="Top"
Click="ClickMeButton_Click"
Width="200"
Height="100"
>
<Button.Background>
<LinearGradientBrush EndPoint="0.5,1" StartPoint="0.5,0">
<GradientStop Color="Black" Offset="0"/>
<GradientStop Color="Red" Offset="1"/>
</LinearGradientBrush>
</Button.Background>
</Button>
UWP-007 - Understanding XAML Schemas and Namespace Declarations
=======================================
Don't touch the schema stuff - it's necessary!
Schemas define rules for XAML, for UWP, for designer support, etc.
Namespaces tell XAML parser where to find the definition / rules for a given element in the XAML.
UWP-008 - XAML Layout with Grids
========================================
Layout controls don't have a content property ...
they have a Chidren property of type UIElementCollection.
By embedding any control inside of a layout control, you are implicitly calling the Add method of the Children collection property.
<Grid Background="Black">
<Grid.RowDefinitions>
<RowDefinition Height="*" />
<RowDefinition Height="*" />
<RowDefinition Height="*" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="*" />
<ColumnDefinition Width="*" />
<ColumnDefinition Width="*" />
</Grid.ColumnDefinitions>
</Grid>
Sizes expressed in terms of: Explicit pixels - 100
Auto - use the largest value of elements it contains to define the width / height
* (Star Sizing) - Utilize all the available space
1* - Of all available space, give me 1 "share"
2* - Of all available space, give me 2 "shares"
3* - Of all available space, give me 3 "shares"
6 total shares ... 3* would be 50% of the available width / height.
Elements put themselves into rows and columns using attached property syntax:
...
...
<Button Grid.Row="0" />
</Grid>
- When referencing Rows and Columns ... 0-based.
- There's always one default implicit cell: Row 0, Column 0
- If not specified, element will be in the default cell
UWP-009 - XAML Layout with StackPanel
====================================
<StackPanel>
<TextBlock>Top</TextBlock>
<TextBlock>Bottom</TextBlock>
</StackPanel>
- Vertical Orientation by default.
- Left-to-right flow by default when Horizontal orientation.
- Most layouts will combine multiple layout controls.
- Grid will overlap controls. StackPanel will stack them.
UWP-017 - XAML Layout with the RelativePanel
====================================
It basically defines an area within which you can position and align child objects
-- in relation to each other
-- or in relation to the parent panel.
Controls use attached properties to position themselves.
Panel alignment relationships (AlignTopWithPanel, AlignLeftWithPanel, ? are applied first.
Sibling alignment relationships (AlignTopWith, AlignLeftWith, ? are applied second.
Sibling positional relationships (Above, Below, RightOf, LeftOf) are applied last.
<RelativePanel MinHeight="300" Grid.Row="1">
<Rectangle Name="RedRectangle" RelativePanel.AlignRightWithPanel="True" />
<Rectangle RelativePanel.LeftOf="RedRectangle" />
</RelativePanel>
UWP-018 - XAML Layout with the SplitPanel
====================================
The split view allows us to create a panel that can be displayed or hidden.
We would use the SplitView to implement hamburger navigation.
The are two parts to a SplitView:
1) The part that is hidden by default (Pane)
2) The part that is shown by default (Content)
You define other controls inside of the SplitView.Pane and SplitView.Content.
<SplitView Name="MySplitView"
CompactPaneLength="50"
IsPaneOpen="False"
DisplayMode="CompactInline"
OpenPaneLength="200" >
<SplitView.Pane>
</SplitView.Pane>
<SplitView.Content>
</SplitView.Content>
</SplitView>
Inline ?Panel completely covers content. When expanded, panel pushes content.
CompactInline ?Pane covers most of the Content. When expanded, panel pushes content.
Overlay ?Panel completely covers content. When expanded, panel covers content.
CompactOverlay ?Panel covers most of the content. When expanded, panel covers content.
Open / Close Pane in C#: MySplitView.IsPaneOpen = !MySplitView.IsPaneOpen;
UWP_019 - Working with Navigation
====================================
App > Window > Frame > MainPage
You can load pages into a child frame or into the root frame:
Frame.Navigate(typeof(Page2), additionalParameter);
You can retrieve additionalParameter on the page you navigated to:
protected override void OnNavigatedTo(NavigationEventArgs e)
{
value = (string)e.Parameter;
}
Traverse back stack (history):
if (Frame.CanGoBack) {
Frame.GoBack();
}
if (Frame.CanGoForward) {
Frame.GoForward();
}
Create a global variable by declaring a static internal field in the App class definition.
UWP-020 - Common XAML Controls - Part 1
====================================
<CheckBox Name="MyCheckBox" Content="Agree?" Tapped="MyCheckBox_Tapped" />
CheckBoxResultTextBlock.Text = MyCheckBox.IsChecked.ToString();
<RadioButton Name="YesRadioButton" Content="Yes" GroupName="MyGroup" checked="RadioButton_Checked" />
<RadioButton Name="NoRadioButton" Content="No" GroupName="MyGroup" Checked="RadioButton_Checked" />
RadioButtonTextBlock.Text = (bool)YesRadioButton.IsChecked ? "Yes" : "No";
<ComboBox SelectionChanged="ComboBox_SelectionChanged" >
<ComboBoxItem Content="Fourth" />
<ComboBoxItem Content="Fifth" />
<ComboBoxItem Content="Sixth" IsSelected="True" />
</ComboBox>
if (ComboBoxResultTextBlock == null) return;
var combo = (ComboBox)sender;
var item = (ComboBoxItem)combo.SelectedItem;
ComboBoxResultTextBlock.Text = item.Content.ToString();
<ListBox Name="MyListBox" SelectionMode="Multiple" SelectionChanged="ListBox_SelectionChanged">
<ListBoxItem Content="First" />
<ListBoxItem Content="Second" />
<ListBoxItem Content="Third" />
</ListBox>
var selectedItems = MyListBox.Items.Cast<ListBoxItem>()
.Where(p => p.IsSelected)
.Select(t => t.Content.ToString())
.ToArray();
ListBoxResultTextBlock.Text = string.Join(", ", selectedItems);
<Image Source="Assets/logo.png" Stretch="UniformToFill" />
<ToggleButton Name="MyToggleButton" Content="Premium Option" IsThreeState="True" Click="MyToggleButton_Click" />
ToggleButtonResultTextBlock.Text = MyToggleButton.IsChecked.ToString();
<ToggleSwitch>
<ToggleSwitch.OffContent>
<TextBlock Text="I'm off right now." />
</ToggleSwitch.OffContent>
<ToggleSwitch.OnContent>
<TextBlock Text="I'm on!" />
</ToggleSwitch.OnContent>
</ToggleSwitch>
UWP-021 - Implementing a Simple Hamburger Navigation Menu
====================================
Jerry Nixon's Example: http://bit.do/hamburger-nav
Use Character Map to find the code to display icons using Segoe MDL5 Assets.
Hamburger: 
Use ListBox and ListBoxItems for the navigation links inside of a SplitView.
UWP-025 - Common XAML Controls - Part 2
====================================
<TimePicker ClockIdentifier="12HourClock" />
<CalendarDatePicker PlaceholderText="choose a date" />
<CalendarView SelectionMode="Multiple"
SelectedDatesChanged="MyCalendarView_SelectedDatesChanged" />
private void MyCalendarView_SelectedDatesChanged(CalendarView sender, CalendarViewSelectedDatesChangedEventArgs args)
{
var selectedDates = sender.SelectedDates.Select(p => p.Date.Month.ToString() + "/" + p.Date.Day.ToString()).ToArray();
var values = string.Join(", ", selectedDates);
CalendarViewResultTextBlock.Text = values;
}
<Button Content="Flyout">
<Button.Flyout>
<Flyout x:Name="MyFlyout">
</Flyout>
</Button.Flyout>
</Button>
MyFlyout.Hide();
<Button Content="FlyoutMenu">
<Button.Flyout>
<MenuFlyout Placement="Bottom">
<MenuFlyoutItem Text="Item 1" />
<MenuFlyoutItem Text="Item 2" />
<MenuFlyoutSeparator />
<MenuFlyoutSubItem Text="Item 3">
<MenuFlyoutItem Text="Item 4" />
<MenuFlyoutSubItem Text="Item 5">
<MenuFlyoutItem Text="Item 6" />
<MenuFlyoutItem Text="Item 7" />
</MenuFlyoutSubItem>
</MenuFlyoutSubItem>
<MenuFlyoutSeparator />
<ToggleMenuFlyoutItem Text="Item 8" />
</MenuFlyout>
</Button.Flyout>
</Button>
<!-- You can apply this to anything ... ex. Image: -->
<!-- https://msdn.microsoft.com/en-us/library/windows/apps/xaml/dn308516.aspx -->
<AutoSuggestBox Name="MyAutoSuggestBox"
QueryIcon="Find"
PlaceholderText="Search"
TextChanged="MyAutoSuggestBox_TextChanged" />
private string[] selectionItems = new string[] { "Ferdinand", "Frank", "Frida", "Nigel", "Tag", "Tanya", "Tanner", "Todd" };
private void MyAutoSuggestBox_TextChanged(AutoSuggestBox sender, AutoSuggestBoxTextChangedEventArgs args)
{
var autoSuggestBox = (AutoSuggestBox)sender;
var filtered = selectionItems.Where(p => p.StartsWith(autoSuggestBox.Text)).ToArray();
autoSuggestBox.ItemsSource = filtered;
}
<Slider Maximum="100" Minimum="0" />
<ProgressBar Maximum="100" Value="{x:Bind MySlider.Value, Mode=OneWay}" />
<ProgressRing IsActive="True" />
UWP-026 - Working with the ScrollViewer
====================================
<ScrollViewer
HorizontalScrollBarVisibility="Auto"
VerticalScrollBarVisibility="Auto">
</ScrollViewer>
You can put anything inside of it, however, don't put it inside of a StackPanel!
UWP-027 - Canvas and Shapes
====================================
Canvas allows you to do absolute positioning via attached properties.
<Line X1="10"
X2="200"
Y1="10"
Y2="10"
Stroke="Black"
Fill="Black"
StrokeThickness="5"
StrokeEndLineCap="Triangle" />
<Polyline Canvas.Left="150"
Canvas.Top="0"
Stroke="Black"
StrokeThickness="5"
Fill="Red"
Points="50,25 0,100 100,100 50,25"
StrokeLineJoin="Round"
StrokeStartLineCap="Round"
StrokeEndLineCap="Round" />
<Rectangle />
<Ellipse />
Canvas.ZIndex="100"
The higher the ZIndex, the higher in the stack it appears (covering what is below it).
UWP-028 - XAML Styles
====================================
https://dev.windows.com/en-us/design
<Page.Resources>
<SolidColorBrush x:Key="MyBrush" Color="Brown" />
<Style TargetType="Button" x:Key="MyButtonStyle">
<Setter Property="Background" Value="Blue" />
<Setter Property="FontFamily" Value="Arial Black" />
<Setter Property="FontSize" Value="36" />
</Style>
</Page.Resources>
Binding: {StaticResource ResourceName}
<Button Content="OK" Style="{StaticResource MyButtonStyle}" />
Create Page or Application level resource dictionaries
<Application.Resources>
</Application.Resources>
Split up your styles into Resource Dictionary files:
<!-- Dictionary1.xaml -->
<ResourceDictionary>
<SolidColorBrush x:Key="brush" Color="Red"/>
</ResourceDictionary>
<Page>
<Page.Resources>
<ResourceDictionary>
<ResourceDictionary.MergedDictionaries>
<ResourceDictionary Source="Dictionary1.xaml"/>
<ResourceDictionary Source="Dictionary2.xaml"/>
</ResourceDictionary.MergedDictionaries>
</ResourceDictionary>
</Page.Resources>
<TextBlock Foreground="{StaticResource SomeStyle}" Text="Hi" />
</Page>
UWP-029 - XAML Themes
====================================
http://bit.do/theme-resources
Put your mouse on a style, hit F12 to open generic.xaml
<Grid Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
<Rectangle Width="100" Height="100" Fill="{ThemeResource SystemAccentColor}" />
<Rectangle Width="50" Height="50" Fill="{ThemeResource SystemColorWindowColor}" />
</Grid>
<App RequestedTheme="Light">
High Contrast themes override styles.
Lots of different styles of system styles defined:
<TextBlock Text="page name" Style="{StaticResource HeaderTextBlockStyle}" />
Many styles defined in generic.xaml used BasedOn attribute ... and you can too!
UWP-037 - Utilizing the VisualStateManager to Create Adaptive Triggers
=================================================
VisualStateManager manages changes to XAML element attributes based on screen size using Adaptive Triggers (MinWindowWidth, MinWindowHeight) and Setters (to change target property values).
<VisualStateManager.VisualStateGroups>
<VisualStateGroup>
<VisualState x:Name="NarrowLayout">
<VisualState.StateTriggers>
<AdaptiveTrigger MinWindowWidth="0"/>
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Target="MyImage.Width" Value="200" />
<Setter Target="LayoutGrid.Background" Value="Blue" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="WideLayout">
<VisualState.StateTriggers>
<AdaptiveTrigger MinWindowWidth="600"/>
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Target="MyImage.Width" Value="200" />
<Setter Target="LayoutGrid.Background" Value="Blue" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
UWP-038 - Working with Adaptive Layout
=======================================
http://bit.do/adaptive-ui
Use adaptive triggers to move StackPanels (filled with content) into different Grid cells.
UWP-039 - Adaptive Layout with Device Specific Views
======================================================
Choose different versions of files to use based on the device running the app.
Example:
/DeviceFamily-Mobile
-- MainPage.xaml
/DeviceFamily-Desktop
-- MainPage.xaml
... or use a different file suffix for different device specific views:
MainPage.DeviceFamily-Mobile.xaml
MainPage.DeviceFamily-Desktop.xaml
http://bit.do/device-specific-views
UWP-40 - Data Binding to the GridView and ListView Controls
==========================================================
Bind to a List<T> where T is a POCO in your app (I put mine in the Models folder).
<Page
...
xmlns:data="using:xBindDataExample.Models">
<Page.Resources>
<DataTemplate x:DataType="data:Book" x:Key="BookDataTemplate">
<StackPanel HorizontalAlignment="Center">
<Image Source="{x:Bind CoverImage}" />
<TextBlock Text="{x:Bind Title}" />
<TextBlock Text="{x:Bind Author}" />
</StackPanel>
</DataTemplate>
</Page.Resources>
...
<GridView ItemsSource="{x:Bind Books}"
IsItemClickEnabled="True"
ItemClick="GridView_ItemClick"
ItemTemplate="{StaticResource BookDataTemplate}">
</GridView>
...
Code Behind
------------
public sealed partial class MainPage : Page
{
private List<Book> Books;
public MainPage()
{
this.InitializeComponent();
Books = BookManager.GetBooks();
}
private void GridView_ItemClick(object sender, ItemClickEventArgs e)
{
var book = (Book)e.ClickedItem;
ResultTextBlock.Text = "You selected " + book.Title;
}
}
UWP-041 - Keeping Data Controls Updated with ObservableCollection
=======================================
If the contents of List<T> will change, make sure you use ObservableCollection<T> instead!
UWP-042 - Utilizing User Controls as Data Templates
========================================
If you intend to combine the VisualStateManager with data bound controls, you will need to put your Data Template code inside of a User Control, then create the VisualStateManager code inside of the User Control.
1) Create a User Control.
2) Cut the Data Template out of the MainPage.xaml and copy it into the User Control.
3) Reference the User Control from inside the Data Template:
<local:ContactTemplate HorizontalAlignment="Stretch" VerticalAlignment="Stretch" />
4) Modify the contents of the User Control changing x:Bind statements to utilize object.property notation:
<UserControl>
<StackPanel>
<Image Source="{x:Bind Contact.AvatarPath}" />
<TextBlock Text="{x:Bind Contact.FirstName}" />
<TextBlock Text="{x:Bind Contact.LastName}" />
</StackPanel>
</UserControl>
5) Add this in the User Control's Code Behind:
public Models.Contact Contact { get { return this.DataContext as Models.Contact; } }
public ContactTemplate()
{
this.InitializeComponent();
this.DataContextChanged += (s, e) => Bindings.Update();
}
출처
이 글은 Channer9에 Andy Wigley, BobTabor, Clint Ruthas, Chavin이라는 4명의 저자가 만드신 'Windows 10 development for absolute beginners'라는 무료강의 시리즈에서 강의 정리노트(UWP Cheat Sheet.txt)로 누구나 다운받을 수 있는 파일의 내용을 욺긴 것입니다. 강의는 더 많은 내용과 Demo를 포함하고 있습니다.
(XAML Layout in Depth) Advanced Topics (0) | 2017.12.22 |
---|---|
(XAML Layout in Depth) Transformations and Projections (0) | 2017.12.20 |
(XAML Layout in Depth) Panels (0) | 2017.12.19 |
(XAML Layout in Depth) Layout-properties of Element (0) | 2017.12.16 |
(XAML Layout in Depth) Layout Basics 2 (0) | 2017.12.15 |
Copyright
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 마지막 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents).
Content
3. Layout properties of Element
4. Panels
5. Transformations and Projections
Outline
The Grid : Overlay and shared size groups
Layout the content of a Control
Change the Panel of an ItemsControl
ScrollViewer and Viewbox
Animated move of elements in a Panel
The Grid as Overlay-container
If you don't specify RowDefinitions and ColumnDefinirions the Grid is one single cell.
Elements in one cell are drawn over each other in the order that defined in XAML(the last element is on top)
So the Grid is the perfect panel to overlay items.
eg. Loading-overlay
<Grid>
<!-- Content -->
<some panel> ...content... </some panel>
<!-- Loading overlay -->
<Gird //custom control로 만들어 따로 빼두는게 이용에 편리하다.
Background="#AAFFFFFF" //FFFFFF == white, alpha channel of AA == a bit transparent
d:IsHidden="True" //Hidden this element to do not overlay elements on the designer
Visibility="{x:Bind ....IsLoading}" //Data binding 로딩이 끝나면 사라질 수 있도록
>
<ProgressRing .../>
</Grid>
</Gird>
The Grid : Shared Size Groups (WPF only)
Share the same size between different ColumnDefinitions or RowDefinitions, Even across multiple Grid-instances
Set the SharedSizeGroup property on RowDefinitions or ColumnDefinitnons, its of type string(without space, not start with number).
Then Set the attached property Grid.IsSharedSizeScope on a parent to true.
eg. <StackPanel Grid.IsSharedSizeScope="True">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition SharedSizeGroup="myGroup">
<ColumnDefinition/>
</Grid.ColumnDefinitions>
</Grid>
<Grid>
<Grid.ColumnDefinitions> //Size of this column is sync with the above column with SharedSizeGroup
<ColumnDefinition SharedSizeGroup="myGroup">
<ColumnDefinition/>
</Grid.ColumnDefinitions>
</Grid>
</Grid>
# Star-sized columns or rows with a SharedSizeGroup are treated as Auto at runtime.
Layout the Content of a Control
Let's assume that the content of a Control is a FrameworkElement which contains Horizontal/VerticalAlignment and Margin
The Control class is a subclass of FrameworkElement and it defines Horizontal/VerticalContentAlignment and Padding to layout its content
When you set Horizontal/VerticalContentAlignment property, the Control internally sets the Horizontal/VerticalAlignment on its Content, and there is a Padding property that sets the Margin property on the Content. All those properties are wired together in the Control template of a Control.
eg. //To Strecth Contents of ListView
<ListView HorizontalAlignment="Stretch"
HorizontalContentAlignment="Stretch" //Have effect in WPF
Background="YellowGreen" ItemsSource="{x:Bind peaple}">
<ListView.ItemContainerStyle>
<Style TargetType="ListViewItem">
<Setter Property="HorizontalContentAlignment" Value="Stretch" /> //Have effect in UWP
</Style>
</ListView.ItemContainerStyle>
<ListView.ItemTemplate>
<DataTemplate x:DataType="model:Person">
<Border HorizontalAlignment="Stretch" Background="Orange" CornerRadius="5" Margin="1"> //No effect
<StackPanel Margin="5">
<TextBlock Text="{x:Bind Name}"/>
<TextBlock Text="{x:Bind Age}"/>
</StackPanel>
</Border>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
Change the Panel of an ItemsControl
An ItemsControl can contain many objects. Typical ItemsControls are the ListView and ComboBox
To change the panel there is two options
1. Set the ItemsPanel property of ItemsControl
Assign an ItemsPanelTemplate to this property and specify the panel in the ItemsPanelTemplate
2. Assign a new ControlTemplate to the Template property of ItemsControl
Inside of that Template, create a Panel and Set the IsItemsHost property(defined in the Panel class) to true
(This property is readonly in WinRT)
Normally change the Panel of an ItemsControl, choose option 1.
When you are defining the new ControlTemplate to create a new look for your ItemsControl, choose option 2
When There's Not Enough Space
When there is not enough space for panel, panels are just clipping their elements.
1. Put your panel into a ScrollViewer, it supports Horizontal/Vertical scroll bars
2. Or put your panel into a Viewbox, it will just shrink or scale your panel
The ScrollViewer
To display a vertical and horizontal scrollbar
ScrollViewer class inherits from ContentControl, so it has a Content property of type Object (like button)
Control the scrollbar visibility with the properties VerticalScrollBarVisibiity(default Visible) and HorizontalScrollBarVisibility(default Disabled), both properties take a value of the ScrollBarVisibility enumeration(Disabled, Auto, Hidden, Visible)
# Some ItemControls, like the ListView, have an internal ScrollViewer, it is defined in the ControlTemplate of the ListView
Use attached properties to modify the internal ScrollViewer
eg. <ListView ScrollViewer.VerticalScrollBarVisibility="Auto" />
The Viewbox
To stretch an element to the available space
The Viewbox class inherits via Decorator from FrameworkElement in WPF. In WinRT, Viewbox inherits FrameworkElement directly
The Decorator class defines a child property that takes a single UIElement. In the WinRT, the child property is directly defined on the Viewbox itself. ViewBox has a child property anyway, so in XAML you can just place any UIElement inside of Viewbox. Then the Viewbox will automatically stretch the element based on the available space.
To control the stretching, the Viewbox has a Stretch property that takes a value of the Stretch enumeration
None : no stretching (DesiredSize)
Fill : aspect ratio changes to fill the available space
Uniform(default) : aspect ratio never changed, there is leftover space
UniformToFill : aspect ratio never changed, and fill the leftover space
and StretchDirection property of type enum StretchDirection
UpOnly : only grow (DesiredSize가 최소크기이고 그 이하로 공간이 부족해지면 clip된다)
DownOnly : only shrink (DesiredSize가 최대크기)
Both(default) : can be shrink and grow
Animated Move of Elements in a Panel
In some scenarios an animated move makes the user experience much better
Instead of using a TranslateTransform, you can also use the FluidMoveBehavior that animates a change in the position of an element (high-level class), it's like an animated TranslateTransform, but it's much simpler to use
To use it open up Blend and attach the FluidMoveBehavior to a Panel (asset -> behavior -> drap&drop -> property setting)
In Blend, There's also a FluidMoveSetTagBehavior that allows you to define a starting point for the move
그러나 FluidMoveBehavior가 UWP project에 기본적으로 들어있지는 않았다, Nuget을 통해 받아 사용하는 듯 하나, 아직 Behavior와 관련한 내용을 모르겠다. Pluralsight에서 관련 강의를 찾았다(link).
Summary (생략)
출처
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 마지막 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents). 제가 정리한 것보다 더 많은 내용과 Demo를 포함하고 있으며 최종 Summary는 생략하겠습니다. Microsoft 지원을 통해 한달간 무료로 Pluralsight의 강의를 들으실 수도 있습니다.
UWP Cheat Sheet for Absolute Beginner (0) | 2018.01.02 |
---|---|
(XAML Layout in Depth) Transformations and Projections (0) | 2017.12.20 |
(XAML Layout in Depth) Panels (0) | 2017.12.19 |
(XAML Layout in Depth) Layout-properties of Element (0) | 2017.12.16 |
(XAML Layout in Depth) Layout Basics 2 (0) | 2017.12.15 |
Copyright
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 네번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents).
Content
3. Layout properties of Element
4. Panels
5. Transformations and Projections
Outline
Transformations
Get the position of an element
Projections
Transformations
Defines how to map points from one coordinate space into another one
A transformation can be used on an element to - rotate it, scale it, skew it, translate it(move it)
Transformation Classes
RotateTransform class
ScaleTransform class
SkewTransform class
TranslateTransform class
MatrixTransform class
All transformations in the two-dimensional space can be described by a 3x3 matrix
While the Rotate/Scale/Skew/TranslateTransform class is high-level classes, the MatrixTransform class is low-level. It allows you to change the 3x3 matrix directly to rotate, scale, skew, translate an element.
TransformGroup class (to combine transformations)
CompositeTransform class (not in WPF)
Single transform class that can rotate, scale, skew, and translate an element
These all Transform classes inherit from the Transform class. The Transform class is the base class for two-dimensional transformations.
Also Transform class inherits from GeneralTransform. The GeneralTransform contains the base logic for transformations between points and rectangles.
All the Transform classes are DependencyObjects and their properties have been implemented as Dependency properties. That means you can use the properties in a style, you can use them as a target for a data binding, or you can event animate them.
LayoutTransform vs RenderTransform
On a FrameworkElement, there are LayoutTransform and RenderTransform properties. Both are of type Transform.
A Transform can be assigned to an element's...
1. LayoutTransform property (WPF only)
The Transform that assigned to the LayoutTransform property is applied before the layout process. 따라서 layout에 영향을 미치므로, 예를들어 StackPanel의 child중 하나를 rotate시켰다면 그 child는 더 많은 Heigth/Width를 차지한 채 layout에 표시된다DesiredSize is bigger). So the layout process runs whenever you change the transform.
2. RenderTransform property (all XAML-based frameworks)
The Transform that assigned to the RenderTransform property is applied after the layout process. 따라서 layout에 영향을 미치지 않으므로, 에를들어 StackPanel의 child중 하나를 rotate시켰다면 그 child만 회전하고 나머지 child의 위치에 영향을 주지 않는다. Whenever you change the transformation that has been assigned to the RenderTransform property, only a new rendering will occur, but no layout process will be executed. So when you plan to change the Transform object at runtime, you should assign it to the RenderTransform property.
The Origin of a RenderTransform
Layout process가 RenderTransform과 무관하기 떄문에 Origin을 필요로 한다.
Origin is defined in the RenderTransformOrigin property(Type Point) that is defined in the UIElement class.
Default value is 0, 0. Specified origin is relative to the size of an element (center of element 0.5, 0.5, bottom right corner 1, 1)
Rotate, Scale and Skew
RotateTransform : Angle property to define clockwise rotation (default 0)
ScaleTransform : Properties ScaleX/ScaleY to define scale (both default value 1 == scale is by default 100%)
SkewTransform : Properties AngleX/AngleY to define skew angle (defualt 0)
All three Transforms have the Properties CenterX and CenterY (default 0) : Use them to define an absolute (pixel-based) origin
Element에 RenderTransformOrigin property를 지정하는 대신에, you could also define the origin by setting the properties CenterX and CenterY on your Transform object. When you group multiple transforms with the TransformGroup, the properties CenterX and CenterY allow you to define a different origin for each transform in that TransformGroup.
Get the Position of an Element
WinRT eg.
//Call TransformToVisual method on green element and pass orange element as parameter, then returns GeneralTransform object
//GeneralTransform contains base logic for transformations between points and rectangles.
GeneralTransform generalTransform = green.TransformToVisual(orange);
//Call TransformPoint method on the GeneralTransform object with a new Point that contains default 0 for X and Y
//then it returns a point that contains Horizontal/Vertical distance between default value and generalTransform
Point point = generalTransform.TransformPoint(new Point());
//In WPF Transform method instead of TransformPoint method
WPF (alternative)
Point pint = green.TranslatePoint(new Point(), orange);
Translate an Element
TranslateTransform class has the properties X and Y to define the translation(move)
The TranslateTransform has no effect when it is assigned to the LayoutTransform property of an element, because the transformation occurs before the layout process and the layout process arranges element and gives it a final position. So Assign it to the RenderTransform property
eg. private void Rectangle_Tapped(object sender, TappedRoutedEventArgs e)
{
var element = e.OriginalSource as FrameworkElement;
if (element != null)
{
//Get the position of target (where the rectangle will positioned)
var generalTransform = target.TransformToVisual(element);
var point = generalTransform.TransformPoint(new Point());
//Set TranslateTransform to element's RenderTransform
var translateTransform = element.RenderTransform as TranslateTransform;
if (translateTransform == null)
{
translateTransform = new TranslateTransform();
element.RenderTransform = translateTransform;
}
//Translate(move) the element to the position of the target
//Here can be replaced to an animation
translateTransform.X = point.X;
translateTransform.Y = point.Y;
}
}
MatrixTransform
Low-level class to create custom 2D-transformations
Matrix-Property contains the 3x3 affine transformation matrix
The Matrix can be modifed to rotate scale, skew and translate an element (need linear algebra)
Combine Transformations
Instead of a MatrixTransform, you can also use the TransformGroup to combine transformations(much simpler).
TransformGroup has a Children property of type TransformCollection and inherits from Transform, can be assign it either to the RenderTransform or to the LayoutTransform property of an element.
eg. <Rectangle.RenderTransform>
<TransformGroup>
<RotateTransform Angle="45" />
<ScaleTransform ScaleX="2" ScaleY="2"/>
</TransformGroup>
</Rectangle.RenderTransform>
There is another class to combine multiple transformations (not available in WPF) : CompositeTransform
CompositeTransform contains Rotation, ScaleX/ScaleY, SkewX/SkewY, TranslateX/TranslateY properties
Projections (Not available in WPF)
A projection is a perspective transformation of an element
The perspective transformation is a 3D-like effect that you can apply to your element.
In Silverlight, this Projection class is abstract.
In the WinRT, the constructor is protected.
you can assign PlanProjection or Matrix3DProjection to the Projection property of any UIElement.
PlaneProjection
To rotate an element around the axis X, Y and Z
It contains the properties RotationX, RotationY, and RotationZ (angle in degrees, default 0) and CenterOfRotationX, CenterOfRotationY, CenterOfRoationZ properties define the center of rotation.
CenterOfRotationX/Y contain a relative value (default 0.5)
CenterOfRoataionZ contain a absolute value in pixels (default 0, <0 means center of rotation is behind the element, >0 front of it)
Translate your element with offsets
LocalOffsetX/Y/Z : Translation before rotation
GrobalOffsetX/Y/Z : Translation after rotation (just move the center of rotation)
Matrix3DProjection
Low-level class to create custom 3D-transformations
ProjectionMatrix property contains the 4x4 transform matrix (Matrix3D) used for the projection
Summary (생략)
출처
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 네번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents). 제가 정리한 것보다 더 많은 내용과 Demo를 포함하고 있으며 최종 Summary는 생략하겠습니다. Microsoft 지원을 통해 한달간 무료로 Pluralsight의 강의를 들으실 수도 있습니다.
UWP Cheat Sheet for Absolute Beginner (0) | 2018.01.02 |
---|---|
(XAML Layout in Depth) Advanced Topics (0) | 2017.12.22 |
(XAML Layout in Depth) Panels (0) | 2017.12.19 |
(XAML Layout in Depth) Layout-properties of Element (0) | 2017.12.16 |
(XAML Layout in Depth) Layout Basics 2 (0) | 2017.12.15 |
Copyright
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 세챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents).
Content
3. Layout properties of Element
4. Panels
5. Transformations and Projections
Outline
Elements with children
Panel classes (Grid, StackPanel, Canvas, WrapPanel, DockPanel, VariableSizeWrapGrid)
Build application layouts
Elements with children
In XAML-based applications, there are four main categories of elements that can have children.
1. Single UIElement
Child Property of type UIElement.
(In the WPF, Decorator class is a base class for those elements. Decorator class가 child 속성을 정의한다. Well known subclasses of Decorator are Border and Viewbox. WinRT에는 Decorator class는 없지만 Border, Viewbox class는 여전히 존재하며 UIElement타입의 child property를 가지고 있다.)
2. Many UIElements (of type Panel, 이 글의 주된 타겟)
Children property of type UIElementCollection
3. Single object
Content property of type Object
Those elements inherit from the ContentControl class, which defines that Content property. ContentControl은 예를 들어 Button class, ScrollViewer class, Window class(WPF) 등이다.
4. Many Objects
Items property of type ItemsCollection and in addition, an ItemsSource property of type IEnumerable.
Those two properties are implemented in the ItemsControl class이며 ComboBox, ListView가 유명한 subclass이다.
ContentControl이나 ItemsControl에는 한가지 규칙이 있다. 만약 object가 UIElement라면, it is rendered in its normal way. object가 UIElement가 아니라면, the result of the ToString method is displayed in a text block. 이것을 피하기 위해 you can create a user interface for your object. You do this by specifying a data template. ContentControl class는 Content property와 함께 ContentTemplate property를 가지며, ContentTemplate property는 user interface(for the object that is stored in the Content property)를 지정할 data template를 받는다. 마찬가지로 ItemsControl class도 ItemTemplate property를 가진다. ItemsTemplate property is also of type DataTemplate unless you specify the user interface for a single object in that ItemsControl(For more about data template, see link). Internally an ItemsControl is using a panel to layout its children.
The Base Class : Panel
The Base class for all the Panels in the XAML-based frameworks is the class Panel. It directly inherits from FrameworkElement.
- Children property (of type UIElementCollection)
- Background property
Default is null, 만약 panel에 input event handler를 만들었을 때 Background가 null이면 event가 실행되지 못한다. 색상을 원하지 않는 경우라면 Transparent를 두면 된다.
- Attached property ZIndex(WPF의 Panel class만 contains the Zindex property, 다른 경우 ZIndex는 Canvas class에 정의되어 있다)
Panel-subclasses in WPF and WinRT
WPF & WinRT
Grid, Canvas, StackPanel
VirtualizingStackPanel : used inside of an ItemsControl. It does a UI virtualization for the items in that ItemsControl.
That means that the UI for items that are by far out of the view is not created by default.
The UI for those items is only created when you scroll to them. So such a VirtualizingStackPanel will gain performance when you have a lot of items in your ItemsControl.
WPF specific panel
DockPanel : allows to dock elements at the Left, Top, Right, Bottom
WrapPanel : stacks elements per default in a horizontal way, and it automatically inserts a line break if there is not enough space.
UniformGrid : simple Grid where all cells have the same size
etc....
WinRT specific panel
VariableSizedWrapGrid : create a kind of Grid layout and each cell can have a variable size
SwapChain : it is used as a hosting suffice to render direct X content in your XAML application
Many of the other panels available in the WinRT can only be used as a panel in an ItemsControl(따라서 WinRT로 layout을 만들때는 Grid, StackPanel, VariableSizedWrapGrid 등 뿐이다.)
The Grid
Arranges children in rows and columns
To define rows and columns, use the properties RowDefinitions and ColumnDefinitions.
To arrange your children, use the attached properties Grid.Row, Grid.Column, Grid.RowSpan, Grid.ColumnSpan.
(Grid.Row과 Grid.Column의 디폴트 값은 0이고, Grid.RowSpan과 Grid.ColumnSpan의 디폴트 값은 1이다.)
RowDefinition class와 ColumnDefinition class는 세개의 중요한 속성을 가진다.
- Height(of type GridLength), MinHeight(of type double), MaxHeight(of type double)
- Width(of type GridLength), 위와 동일
The GridLenght struct supports 3 different units (GridUnitType) that you can use to set the Height or Width.
Those three units are defined in the GridUnitType enumeration : Auto, Pixel, Star(default)
For XAML, a TypeConverter exists that allows you a very simple usage of those three different units.
The GridSplitter (WPF, Silverlight only, for UWP see this link)
The GridSplitter is a control that allows the user to resize the rows and columns in a Grid.
By default, the GridSplitter resizes rows or columns based on its alignment.
The Canvas
Position children explicitly with coordinates
use the attached properties (Canvas.Left, Canvas.Top, Canvas.Right, Canvas.Bottom)
For overlapping elements, either use Canvas.ZIndex (in WPF Panel.ZIndex) or change the order of the children
Canvas is not for layout, use the Canvas for graphical stuff.
The StackPanel
Stacks elements in one line
The Orientation property can be set to Vertical(default) or Horizontal
The WrapPanel (WPF only)
Stacks elements in one line and adds a linebreak when there's not enough space left
The Orientation property can be set to Vertical or Horizontal(default)
WrapPanel은 DesiredSize Width로 element를 dispaly하고, 모든 elements in a row는 가장 큰 element Height에 맞춰 stretch된다.
모두 동일한 크기를 가지게 하려면 WrapPanel의 ItemWidth, ItemHeight를 설정해주면 된다.
(WrapPanel in UWP, see link)
The DockPanel (WPF only)
Docks elements at the left, top, right, and bottom
Set the attached property DockPanel.Dock on the children (Dock-enum Left(default), Top, Rigth, Bottom)
By default the LastChildFill property is true, so the last child fills the leftover space.
DockPanel can be used as a RootPanel for the layout of application, but there is no Splitter control
다른 XAML-base application과의 연동을 고려한다면 Grid를 사용하는 편이 낫다.
The VariableSizedWrapGrid (WinRT only)
Arranges children in rows and columns
Orientation property, Vertical(default, stacks children in columns) and Horizontal(stacks children in rows)
Define equal size for all children with ItemWidth and ItemHeight
MaximumRowsOrColumns property to define when break occurs(default -1)
창의 크기가 줄어들면 그에 맞추어 element가 재배열되지만 공간이 남더라도 아래의 사진처럼 아래로 2칸까지만 내려가고 오른쪽으로 배열되도록 설정할 수 있다(MaximumRowOrColumns = 2, Orientation = Vertical)
Attached Properties for children
VariableSizedWrapGrid.RowSpan
VariableSizedWrapGrid.ColumnSpan
Summary (생략)
출처
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 세번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents). 제가 정리한 것보다 더 많은 내용과 Demo를 포함하고 있으며 최종 Summary는 생략하겠습니다. Microsoft 지원을 통해 한달간 무료로 Pluralsight의 강의를 들으실 수도 있습니다.
(XAML Layout in Depth) Advanced Topics (0) | 2017.12.22 |
---|---|
(XAML Layout in Depth) Transformations and Projections (0) | 2017.12.20 |
(XAML Layout in Depth) Layout-properties of Element (0) | 2017.12.16 |
(XAML Layout in Depth) Layout Basics 2 (0) | 2017.12.15 |
(XAML Layout in Depth) Layout Basics 1 (0) | 2017.12.13 |
Copyright
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 두번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents).
Content
3. Layout properties of Element
4. Panels
5. Transformations and Projections
Outline
Alignments, Width Height, Margin
The Visibility
Using the Designer
Alignments
To align your element in its parent container, the FrameworkElement class has two properties.
HorizontalAlignment (Left, Right, Center, Stretch)
VerticalAlignment (Top, Bottom, Center, Stretch)
They only have an effect when the FinalSize is greater than the DesiredSize. (FinalSize > DesiredSize)
Left Right, Center, Top, Bottom, Center 를 하더라도 element는 DesiredSize Width/Height를 가지고, Stretch하면 element가 DesiredSize의 WIdth/Height 보다 커진다.
eg. //DiagonalPanel을 수정해 VerticalStackPanel을 만들었다. Width와 관련된 내용만 수정됨
public class VerticalStackPanel : Windows.UI.Xaml.Controls.Panel
{
protected override Size MeasureOverride(Size availableSize)
{
var mySize = new Size();
foreach(UIElement child in this.Children)
{
child.Measure(availableSize);
//가장 큰 너비를 가진 child의 너비를 mySize.Width로 넣어준다.
mySize.Width = Math.Max(mySize.Width, child.DesiredSize.Width);
mySize.Height += child.DesiredSize.Height;
}
return mySize;
}
protected override Size ArrangeOverride(Size finalSize)
{
var location = new Point();
foreach(UIElement child in this.Children)
{
//child의 너비에 VerticalStackPanel의 너비를 넘겨준다.
child.Arrange(new Rect(location, new Size(finalSize.Width, child.DesiredSize.Height)));
location.Y += child.DesiredSize.Height;
}
return finalSize;
}
}
//Child element의 DesiredSize.Width보다 fianlSize.Width가 더 크므로, HorizontalAlignment의 효과를 볼 수 있다.
//VerticalAlignment의 경우는 child.DesiredSize.Height가 finalSize가 되었으므로 그 효과를 볼 수 없다.
Width and Height
With the properties Width and Height you give your element an explicit size, and Default-Value is Double.NaN
To access the final size (the size that is calculated during the layout process) use ActualWidth/ActualHeight (readonly)
ActualWidth/ActualHeight는 Arrange()가 호출될때 값이 채워지고, they contain the FinalSize(FinalSize is used in the render step).
ActualWidth/ActualHeight의 값이 변하면 LayoutUpdated-event가 발생한다. This event is fired whenever a layout process has occurred(and that means the ActualWidth/ActualHeight properties of your element have changed). 따라서 LayoutUpdated-event내에서 ActualWidth/ActualHeight에 접근하는 것은 좋은 방법이다.
Explicit하게 Width/Height에 값을 넣으면 element의 final size will be based on those values(DesiredSize에 들어간다).
- Element does not resize anymore (fixed element)
Size-range : Use the properties MinWidth/MaxWidth, MinHeight/MaxHeight
They allow you to specify Arrange and your element will still resize inside of that range.
The Margin
of type Thickness, For XAML a three TypeConverter exist(값 한개, 두개, 네개를 XAML에서 Margin값으로 설정할 수 있게 한다.)
The Visibility
Takes a value of the Visibility-enum : Visible, Hidden(WPF only), Collapsed
Hidden의 경우 시야에서 사라지지만 여전히 공간을 차지한다, 즉 DesiredSize는 그대로이다.
다른 XAML-based framework들에서는 Opacity = 0, IsHitTestVisible = False 로 하는 방법을 사용할 수 있다.
Collapsed의 경우는 DesiredSize의 Width, Height값이 0이 된다.
Using the Designer (생략)
Summary (생략)
출처
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 두번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents). 제가 정리한 것보다 더 많은 내용과 Demo를 포함하고 있으며 최종 Summary는 생략하겠습니다. Microsoft 지원을 통해 한달간 무료로 Pluralsight의 강의를 들으실 수도 있습니다.
(XAML Layout in Depth) Transformations and Projections (0) | 2017.12.20 |
---|---|
(XAML Layout in Depth) Panels (0) | 2017.12.19 |
(XAML Layout in Depth) Layout Basics 2 (0) | 2017.12.15 |
(XAML Layout in Depth) Layout Basics 1 (0) | 2017.12.13 |
(XAML)Choosing the Right Tool for the job, (Visual studio vs Blend) (0) | 2017.12.12 |
Copyright
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 첫번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents).
Content
3. Layout properties of Element
4. Panels
5. Transformations and Projections
Dependency Property
In XAML-based applications, there are multiple sources for a property. 예를 들어 Style을 통해 Button의 Width를 설정하고, Style안의 Style Trigger로 마우스가 올라갔을때의 Width를 설정할 수 있다. 이것만으로도 이미 two sources for that property를 가진 것이고, 더 많아질 수 있다(Template Trigger, Local Value(버튼에 직접 설정하는 것). Animation etc). For example, each Dependency property has also a default value. So a Dependency Property has different sources, but it can only have one current value. And this current value is dependent of the different sources. That's why the properties are called Dependency Properties.
DependencyObject
이렇게 다양한 source들을 통해 값이 지정될 수 있으므로 어떤 값을 선택할 것인지에 대한 logic이 DependencyObject class에 implement되어 있다(DependencyObject class는 UIElement의 base class이다). DependencyObject에는 SetValue와 GetValue라는 메소드가 있다. link
eg. var btn = new Button();
btn.SetValue(Button.WidthProperty, 25.0);
//Button.WidthProperty는 DependencyProperty-Instance로써 it's of type DependencyProperty이다.
//이 DependencyProperty는 static field에 Property라는 suffix를 달고 저장되어있다.
Now behind the scenes, the SetValue call stores a Local Value for the Width property
그러나 GetValue를 호출하면 DependencyObject는 다양한 sources들을 뒤져서 적합한 값을 되돌려준다.
Attached Property
The special kind of a Dependency Property called Attached property is used extensively in layout.
An Attached Property is a property that you attach to any DependencyObject.
There are two kinds of Dependency properties
1. There are those Dependency properties that wrap SetValue and GetValue with a CLR-Property
var btn = new Button();
btn.SetValue(Button.WidthProperty, 25.0);
btn.Width = 25; //calling SetValue method with Button.WidthProperty and value(25)
Those Dependency properties that use a CLR-Property as a wrapper we normally just call DependencyProperty.
2. The second kind of a Dependency property is wrapping the SetValue and GetValue method with static Set and Get methods
var btn = new Button();
btn.SetValue(Grid.RowProperty, 1); //pass in a Dependency property that is defined in another class(Grid.RowProperty)
Grid.SetRow(btn, 1);
Grid.RowProperty는 Grid class에 정의되있지만 can set a value for this property on a Button object(attach a value for this property to the Button object). For those properties, there exists static methods, in this case, the SetRow method defined in the Grid class. where you pass in your DependencyObject and the value. As with the CLR-Property, this static method just calls SetValue behind the scenes. So in this case, we call SetValue on the Button instance that is passed as a first parameter to the SetRow method.
Those properties that are attached to specific objects are called AttachedProperty.
For Attached properties, Attached-Property-Syntax for XAML exist
eg. <Button Grid.Row="1" .../>
Behind the scenes, the framework is calling the SetValue method on the Button object and passing in as a first parameter Grid.RowProperty and as a second parameter, the value 1.
#Two snippets for Dependency properties
propdp : for normal Dependency Property wrapping the SetValue, GetValue method with CLR-Property
propa : for Attached Property wrapping SetValue, GetValue with two static methods
eg. public static double GetTop(DependencyObject obj)
{
return (double)obj.GetValue(TopProperty);
}
public static void SetTop(DependencyObject obj, double value)
{
obj.SetValue(TopProperty, value);
}
//To create DependencyProperty, static RegisterAttached method is called
public static readonly DependencyProperty TopProperty =
DependencyProperty.RegisterAttached("Top", typeof(double), typeof(SimpleCanvas), new PropertyMetadata(0.0));
//(Name of the property, Type, the ownerclass of the DependencyProperty, Metadata(defines the default value for the Dependency Property)
protected override Size MeasureOverride(Size availableSize)
{
foreach (UIElement child in this.Children)
{
//이 샘플에선 SimpleCanvas의 크기계산엔 관심이 없으나
//child들이 Arrange되기 위해선 Measure를 호출해줄 필요가 있다.
child.Measure(availableSize);
}
return base.MeasureOverride(availableSize);
}
protected override Size ArrangeOverride(Size finalSize)
{
var location = new Point();
foreach (UIElement child in this.Children)
{
//var top = (double)child.GetValue(TopProperty);
//Child로부터 위와 같이 GetValue를 호출해 TopProperty 값을 얻어낼 수도 있지만
//이와 동일한 과정이 GetTop static method에 구현되있다. 따라서 GetTop method를 사용한다.
location.Y = GetTop(child);
child.Arrange(new Rect(location, child.DesiredSize));
}
return base.ArrangeOverride(finalSize);
}
<panel:SimpleCanvas>
<Button Content="Hello" />
<Button panel:SimpleCanvas.Top="10" Content="world!" /> //Button의 위치가 위에서 10만큼 아래로 조정된다.
</panel:SimpleCanvas>
Layout Process Execution
The layout process is executed...
1. when an element is rendered its first time
2. when a child element was added or removed from the visual tree
3. when a specific dependency property has changed
4. when InvalidateMeasure, InvalidateArrange methods is called on a UIElement
These methods will set the UIElement into an invalidate state and the framework will do a deferred layout process execution. (If you don't want that deferred execution, you can call the UpdateLayout method on your UIElement after you have called InvalidateMeasure or InvalidateArrange. The UpdateLayout method will force a layout process execution.)
Trigger the Layout Process from code
1. Using dependency property metadata (WPF only)
2. Using InvalidateArrage from PropertyChangedCallback (WPF, WinRT, Silverlight, Windows Phone... all XAML-based frameworks)
e.g. 1번의 경우는 DependencyProperty TomProperty = DependencyProperty.RegisterAttached의 4번째 parameter인 PropertyMetadata를 FrameworkPropertyMetadata로 바꾸고 FrameworkPropertyMetadataOption(위 경우엔 AffectParrentArrange)을 선택해 넘겨주면 된다.
(c.f. FrameworkPropertyMetadata는 OnPropertyChanged뿐 아니라 OnPropertyValueChanged 콜백함수도 추가할 수 있다.)
e.g. //GetTop과 SetTop은 생략
public static readonly DependencyProperty TopProperty = DependencyProperty.RegisterAttached("Top", typeof(double), typeof(SimpleCanvas), new PropertyMetadata(0.0, OnTopPropertyChanged));
private static void OnTopPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
//First parameter is not SimpleCanvas, its child of SimpleCanvas. (Setting of TopProperty on children)
//From second parameter, we can access to new and old value and the DependencyProperty itself
{
if (d is FrameworkElement child) //Get child from d as FrameworkElement
{
if (child.Parent is SimpleCanvas simpleCanvas) //Get parent from child = SimpleCanvas
{
//We don't need to InvalidateMeasure in this case, we just want to arrange it again.
simpleCanvas.InvalidateArrange();
//Make layout Update right away
simpleCanvas.UpdateLayout();
}
}
}
Summary (생략)
출처
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 첫번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents). 제가 정리한 것보다 더 많은 내용과 Demo를 포함하고 있으며 최종 Summary는 생략하겠습니다. Microsoft 지원을 통해 한달간 무료로 Pluralsight의 강의를 들으실 수도 있습니다.
(XAML Layout in Depth) Panels (0) | 2017.12.19 |
---|---|
(XAML Layout in Depth) Layout-properties of Element (0) | 2017.12.16 |
(XAML Layout in Depth) Layout Basics 1 (0) | 2017.12.13 |
(XAML)Choosing the Right Tool for the job, (Visual studio vs Blend) (0) | 2017.12.12 |
(XAML)Debugging and Analyzing XAML (0) | 2017.12.12 |
Copyright
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 첫번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents).
Content
3. Layout properties of Element
4. Panels
5. Transformations and Projections
Outline
Layout Base Classes
Layout Process
Dependency and Attached Propertiees
Layout Base Classes
Class hierarchy in XAML
See Basic Element article
http://empisterian.tistory.com/9?category=980707
and Panels and Layout System article
http://empisterian.tistory.com/11?category=980707
UIElement has Measure and Arrage method which are used in the layout process. In addition, the UIElement class has the render transform property that is used to do transformations.
FrameworkElement과 Panel의 내용은 위 두 article에 수록됨
The Layout Process
The Layout Process is executed when an element is rendered its very first time and can be executed again during runtime
The Layout Process is a two step process
1. Measure - Elements calculate their desired size (each element calculates how big it wants to be)
- Calling Measure method on each direct child (after this Measure call, the DesiredSize property of the child is set)
- Access each child's DesiredSize (The parent element can access this DesiredSize and calculate its own desired size)
이 모든 과정은 element tree를 따라 내려간다, 즉 parent element가 자신의 각 direct child들에게 Measure method를 호출하고, 각각의 child들도 자신의 direct child들에게 Measure method를 호출한다. 이 과정이 모두 끝나면 각자 원하는 크기가 DesiredSize property에 저장된 상태가 된다.
2. Arrange - Elements arrange their children
- Calling Arrange on each direct child
(The Arrange method takes a position and a final size, that means the child knows the position where it has to render itself and it also knows the size in which it has to render. So all the information is now clear for the rendering)
3. Rendering - element are rendered on the screen.
Layout Process Participation
Override two methods in a FrameworkElement-subclass
1. MeasureOverride method : Participate in the first step of the layout process, the Measure step.
- Inside of the MeasureOverride method, Call Measure method on each direct child
The MeasureOverride method taskes an availableSize as a parameter which is passed in from the parent in the layout process
- Access each child's DesiredSize
- Return your own desired size
2. ArrangeOverride method : Participate in the second step of the layout process, the Arrange step.
- Inside of the ArrangeOverride method, Call Arrange method on each direct child (pass a location and a final size)
- Return your final size for your element
eg //Create new Class to make custom Panel
public class DiagonalPanel : Windows.UI.Xaml.Controls.Panel
{
//Layout Process의 첫 단계인 Measure단계에 관여하기 위해 MeasureOverride를 정의한다.
//availableSize is passed in from the parent in the layout process. 이경우엔 DiagonalPanel의 부모로부터 받는다.
protected override Size MeasureOverride(Size availableSize)
{
var mySize = new Size();
//DiagonalPanel의 DesiredSize를 반환하기 위해 element들을 돌면서 계산.
//InternalChildren property in WPF, Children property in UWP from Panel, It contains UIElementCollection
foreach(UIElement child in this.Children)
{
//Call Measure method on each child
//여기선 child가 DiagonalPanel보다 작을것이라 기대하며 availableSize를 그대로 넘겼다.
child.Measure(availableSize);
//Measure를 호출하면 child의 DesiredSize property가 채워지고 이걸로 DiagonalPanel의 크기를 계산한다.
mySize.Width += child.DesiredSize.Width;
mySize.Height += child.DesiredSize.Height;
}
return mySize;
}
//Layout Process의 두번째 단계인 Arrange단계에 관여하기 위해 ArrangeOverride를 정의한다.
//부모로부터 finalSize를 통보받고 그 안에서 arrange itself 해야한다. 그리고 layout에 arrange하고 난 실제 크기를 반환.
protected override Size ArrangeOverride(Size finalSize)
{
var location = new Point();
//Children들을 Arrange시킨다.
foreach(UIElement child in this.Children)
{
//각 child에게 Arrange method를 호출시켜 Arrange될 장소와 크기를 지정해준다.
//이 경우 대각선으로 child들이 연결되게 장소를 지정하고, 크기는 child의 DesiredSize를 그대로 받아주었다.
child.Arrange(new Rect(location, child.DesiredSize));
//여기선 다음 child가 지정될 위치를 현재 child의 오른쪽 하단 포인트로 잡아주었다.
location.X += child.DesiredSize.Width;
location.Y += child.DesiredSize.Height;
}
return finalSize;
}
//이것으로 child들을 대각선으로 배치하는 새로운 custom panel이 만들어졌고, .xaml에서 컨트롤들을 배치하며 사용가능하다.
Layout Process Internal
WPF WinRT
위에서 우리가 사용한 Measure/Arrange method와 MeasureOverride/ArrangeOverride method 사이에는 Internal logic이 있으며 그것이 몇가지 property들을 관리한다. UIElement의 경우에는 Clip과 Visibility property가 그렇고, FrameworkElement의 경우에는 Width, Height, Margin property가 그러하다.
eg. Visibility property를 Collapsed 로 설정하면, 해당 UIElement에서 Measure를 호출한 뒤의 시점에서 DesiredSize property의 Width, Height가 0으로 설정된다. 이렇게 Framework가 몇가지 속성을 관리하므로 개발자가 직접 Layout Process에 관여할 때에도 Child element의 Visibility property를 신경쓸 필요가 없다. Width, Height, Margin의 경우도 그렇다. Width를 설정한 경우, that element will have that DesiredSize Width after a Measure call. So in your layout process participation, you never have to care about if the Width or Height on an element is set. 마찬가지로 Margin의 경우도, Measure method가 호출된 뒤 얻어지는 DesiredSize property는 이미 그 Margin을 포함하고 있는 상태이다. 또한 Panel에 Margin을 설정하는 경우에도, MeasureOverride method의 반환값인 Size(Panel의 DesiredSize)에도 Framework가 개입해 Margin을 자동으로 포함시켜준다.
출처
이 모든 내용은 Pluralsight에 Thomas Claudius Huber가 올린 'XAML Layout in Depth'라는 강의의 첫번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-layout-in-depth/table-of-contents). 제가 정리한 것보다 더 많은 내용과 Demo를 포함하고 있으며 최종 Summary는 생략하겠습니다. Microsoft 지원을 통해 한달간 무료로 Pluralsight의 강의를 들으실 수도 있습니다.
(XAML Layout in Depth) Layout-properties of Element (0) | 2017.12.16 |
---|---|
(XAML Layout in Depth) Layout Basics 2 (0) | 2017.12.15 |
(XAML)Choosing the Right Tool for the job, (Visual studio vs Blend) (0) | 2017.12.12 |
(XAML)Debugging and Analyzing XAML (0) | 2017.12.12 |
(XAML)Using Resources and Styling Controls (0) | 2017.12.12 |
Copyright
이 모든 내용은 Pluralsight에 Kevin Dockx가 올린 'XAML Jumpstart : Getting Started With XAML' 이라는 강의의 마지막 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-jumpstart/table-of-contents). 강의 원작자분께 게시허가도 받았습니다.
Content
1. Basic Elements, Shapes, Brushes, and Masks
2. Control Basics and Interacting with them
3. Panels and the Layout System
6. Using Resources and Styling Controls
7. Debugging and Analyzing XAML
8. Choosing the Right Tool for the Job
The Right Tool for the Job
이 내용은 의견이 많이 엇갈릴 수 있는 부분이라고 강의하시는 분께선 설명하면서, Visual studio는 개발자용이고, Blend는 디자이너용이라는 구분은 옛날 일이며, 개발자라면 두 도구를 사용할 줄 알아야 한다고 주장한다. 아래의 pro & con 구분은 강의하시는 분의 개인적인 workflow를 바탕으로 한 것이다.
Visual Studio
- Code behind작업에 있어서는 VS가 우위를 가진다. (IntelliSense, Debugger, Tool of choice to write C# code)
- Defining simple elements and layouts (이정도는 VS에서도 충분히 쉽게 제작할 수 있다.)
- Events & event handlers
- Data Binding (simple)
- Debugging and analysing
Blend
- Defining more complicated layouts, especially templates
(It allows you to do this in a graphical manner rather than writing all the code yourself)
- Data Binding
huge DataContext, because in Blend you don't even have to write your data bindings yourself, you can just state, well this class is my DataContet, that DataContext contains those properties, and you combine them to the IDE, even adding converters if needed.
- Working with resources and styling (Definitely)
- Working with sample data (through blend you can generate sample data)
- Animation & the Visual State Manager
Course Summary (생략)
What's Next?
MVVM Light Toolkit Fundamentals
출처
이 모든 내용은 Pluralsight에 Kevin Dockx가 올린 'XAML Jumpstart : Getting Started With XAML' 이라는 강의의 마지막 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-jumpstart/table-of-contents). 제가 정리한 것보다 더 많은 내용과 Demo를 포함하고 있으며 최종 Summary는 생략하겠습니다. Microsoft 지원을 통해 한달간 무료로 Pluralsight의 강의를 들으실 수도 있습니다.
끝맺음 : XAML에 대해 간단하게 배워보았습니다. 다른 강의를 통해 XAML의 컨트롤과 레이아웃 등을 대강대강 배치하는 것을 배웠었는데, 이 강의를 통해서 좀더 확장된 내용을 배울 수 있었던 것 같습니다. 물론 XAML을 능숙하게 다루기 위해선 많은 실전이 필요하고 더 깊이있는 내용을 파봐야 custom control이나 layout을 제작할 수 있게 되는 것 같습니다.
(XAML Layout in Depth) Layout Basics 2 (0) | 2017.12.15 |
---|---|
(XAML Layout in Depth) Layout Basics 1 (0) | 2017.12.13 |
(XAML)Debugging and Analyzing XAML (0) | 2017.12.12 |
(XAML)Using Resources and Styling Controls (0) | 2017.12.12 |
(XAML)Working with ItemsControls (0) | 2017.12.11 |
Copyright
이 모든 내용은 Pluralsight에 Kevin Dockx가 올린 'XAML Jumpstart : Getting Started With XAML' 이라는 강의의 8번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-jumpstart/table-of-contents). 강의 원작자분께 게시허가도 받았습니다.
Content
1. Basic Elements, Shapes, Brushes, and Masks
2. Control Basics and Interacting with them
3. Panels and the Layout System
6. Using Resources and Styling Controls
7. Debugging and Analyzing XAML
8. Choosing the Right Tool for the Job
Debugging XAML
Setting breaakpoints in XAML isn't possible (except in Silverlight), but can use the output window to check for data binding errors
Output window를 통해서 대부분의 Data Binding error를 찾아낼 수 있다.
Analyzing XAML
After a while, it can become quite hard to find the source of a UI-related mistake in a XAML document, especially when you start creating custom templates
XAML Spy, develped by Koen Zwikstra, provides a real time view of your app state.From here, we can examine and modify properties of any element on the fly and see the changes reflected immediately in the running app.(www.xamlspy.com)
XAML Tips
How you write your XAML can have a tremendous impact on application performance
1. Minimize element count
eg. Use Background on Grid instead of filling the Grid with a Rectangle to achieve the same, Rectangle uses memory more!
2. Reuse resources
Resources you reuse are only created once
eg. specific brush that you're using throughout your application, create that brush, put it in a resource dictionary, and refer to it through StaticResource rather than setting the brush manually each time you use it. Not only is it a lot easier to change the brush afterwards, but the resource only has to be created once so you get one instance of that brush in memory instead of having multiple instances of that brush in memory.
3. Avoid unnecessary overdraw (specially important!)
Every pixels needs to be drawn, even if it's fully overlaid by another pixel. UWP app that has to run on a low-end ARM device, the maximum amount of overdraw that can take is about 4 pixels, and that's not a lot.
Memory profiing
Performance-related problems often arise in XAML applications, and contrary to what might look obvious, memory profiling is one of the best ways to locate the source of those problems. (You typically do not use a performance profiler to fix these issues because memory profiler is actually one of the best ways to locate the source of those problems. That's because a lot of those performance-related problems actually stem from bad memory management)
Telerik(http://telerik.com/)
JetBrains(http://jetbrains.com/)
Visual studio build in memory profiler
Summary (생략)
출처
이 모든 내용은 Pluralsight에 Kevin Dockx가 올린 'XAML Jumpstart : Getting Started With XAML' 이라는 강의의 8번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-jumpstart/table-of-contents). 제가 정리한 것보다 더 많은 내용과 Demo를 포함하고 있으며 최종 Summary는 생략하겠습니다. Microsoft 지원을 통해 한달간 무료로 Pluralsight의 강의를 들으실 수도 있습니다.
(XAML Layout in Depth) Layout Basics 1 (0) | 2017.12.13 |
---|---|
(XAML)Choosing the Right Tool for the job, (Visual studio vs Blend) (0) | 2017.12.12 |
(XAML)Using Resources and Styling Controls (0) | 2017.12.12 |
(XAML)Working with ItemsControls (0) | 2017.12.11 |
(XAML)Data Binding Essentials (0) | 2017.12.10 |
Copyright
이 모든 내용은 Pluralsight에 Kevin Dockx가 올린 'XAML Jumpstart : Getting Started With XAML' 이라는 강의의 7번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-jumpstart/table-of-contents). 강의 원작자분께 게시허가도 받았습니다.
Content
1. Basic Elements, Shapes, Brushes, and Masks
2. Control Basics and Interacting with them
3. Panels and the Layout System
6. Using Resources and Styling Controls
7. Debugging and Analyzing XAML
8. Choosing the Right Tool for the Job
Resources
Any object can be defined as a resource, effectively making it a reusable asset
Often-used are (Data)Templates, Brushes and Styles
Resources에 x:Key를 정의해야 접근할 수 있다. x:Key is a key to gets in the dictionary. The corresponding value is the resource itself.
They are defined in a Resource Dictionary. Framework(Content)Element에는 Resources property(of type ResourceDictionary)가 정의되있다.
The StaticResource markup extension is used to access the resource. (The StaticResource markup extension processes key by looking up the value for that key in all available resource dictionaries. This happens during loads, which is the point in time when the loading process needs to assign the property value that takes the StaticResource reference. So that's how it ties together.)
=>Define Resource in a ResourceDictionary, give it a key through x:Key, access it with the StaticResource markup extension providing set resource key.
ex <Page.Resources>
<SolidColorBrush x:Key="redColorBrush" Color="#FFFF1111" />
</Page.Resources>
<TextBlock Foreground="{StaticResource redColorBrush}" Text="first" />
Resource Scope
The scope of a resource depends on the resource dictionary it's defined in. ex Define it on a Grid(or Page), then it's accessible by all child elements on that Grid(or Page).
Full app as scope(the app root) => an app can be found in App.xaml's resource dictionary. It's the root of application. so if resources defined here are scoped to the complete application.
Resource Dictionaries in a separate file
Using separate files offers advantages - Better separation, More control, More reuse
=> make .xaml document starting with ResourceDictionary tag
=> Add the dictionary to the Resources tag where you want it scoped (App.xaml on page level or at element level...)
or merge multiple resource dictionaries through ResourceDictionary.MergedDictionaries syntax
#ResourceDictionary.MergedDictionaries syntax를 사용할 땐 hierarchy에 주의할 것, It the DataTemplates use a brush defined in the brush ResourceDictionary, we need to add that one first before the DataTemplates dictionary is added
ex //Create Brushes.xaml as ResourceDictionary and add...
<SolidColorBrush x:Key="redColorBrush" Color="#FFFF1111" />
//add above ResourceDictionary to App.xaml
<Application.Resources>
<ResourceDictionary Source="Style/Brushes.xaml"/>
</Application.Resources>
//then redColorBrush was added to application-wide scope and can be used everywhere
ex //Create DictionaryWithDataTemplate.xaml as ResourceDictionary and add...
<DataTemplate x:Key="dtPerson">
<StackPanel Orientation="Vertical">
<TextBlock Text="{Binding Name}"/>
<TextBlock Text="{Binding Age}" Foreground="{StaticResource redColorBrush}"/>
</StackPanel>
</DataTemplate>
//add App.xaml...
<Application.Resources>
<ResourceDictionary>
<ResourceDictionary.MergedDictionaries>
<ResourceDictionary Source="Style/Brushes.xaml" /> //ResourceDictionary의 배치 순서에 주의할 것
<ResourceDictionary Source="Resource/Templates/DictionaryWithDataTemplate.xaml" />
</ResourceDictionary.MergedDictionaries>
</ResourceDictionary>
</Application.Resources>
Styles
A Style is a convenient way to apply a set of property values to more than one element
Styles are defined in a Resource Dictionary
Through TargetType, we specify the type of target for the Style (eg : TextBlock, Button etc)
Every Style must have a key (x:Key), and Syles are assigned through StaticResource
implicit styling
Implicit styling allows us to specify a style that will be applied to all elements of a specific type
Through TargetType, we specify the type of target for the Style
We don't set a key : this is implicit
We no longer have to assign the Style explicitly
eg <Style x:Key="textBlockStyle" TargetType="TextBlock">
<Setter Property="FontSize" Value="40"/>
<Setter Property="HorizontalAlignment" Value="Left"/>
<Setter Property="Foreground" Value="Blue"/>
</Style>
<TextBlock
Style="{StaticResource textBlockStyle}"
Text="Hello world!" />
위의 Style에서 x:Key를 지우면 implicit style이 되고 이 Page내의 모든 TextBlock(target type)의 Style로 지정된다.
Style Inheritance
When most property values in different style sets are the same, we can inherit a base style to base the more specific styles on
1. Define a base Style, and give it a key
2. Define an inheriting Style, and set the BaseOn property to that Style through StaticResource
3. Inheritance depth is not limited, but multiple inheritance isn't allowed
eg <Style x:Key="textBlockStyle" TargetType="TextBlock">
<Setter Property="FontSize" Value="40"/>
<Setter Property="HorizontalAlignment" Value="Left"/>
<Setter Property="Foreground" Value="Blue"/>
</Style>
<Style x:Key="newTextBlockStyle" TargetType="TextBlock"
BasedOn="{StaticResource textBlockStyle}">
<Setter Property="Margin" Value="20"/>
</Style>
//Implicit Style을 상속받기 위해선 BaseOn="{StaticResource {x:Type TextBlock}}"과 같이 x:Type을 사용하면 되는 것 같으나 UWP에서는 지원되지 않는 기능으로 보인다. (https://stackoverflow.com/questions/33562864/xtype-missing-in-uwp-how-override-base-control-style)
Summery (생략)
출처
이 모든 내용은 Pluralsight에 Kevin Dockx가 올린 'XAML Jumpstart : Getting Started With XAML' 이라는 강의의 7번째 챕터를 듣고 정리한 것입니다(https://app.pluralsight.com/library/courses/xaml-jumpstart/table-of-contents). 제가 정리한 것보다 더 많은 내용과 Demo를 포함하고 있으며 최종 Summary는 생략하겠습니다. Microsoft 지원을 통해 한달간 무료로 Pluralsight의 강의를 들으실 수도 있습니다.
(XAML)Choosing the Right Tool for the job, (Visual studio vs Blend) (0) | 2017.12.12 |
---|---|
(XAML)Debugging and Analyzing XAML (0) | 2017.12.12 |
(XAML)Working with ItemsControls (0) | 2017.12.11 |
(XAML)Data Binding Essentials (0) | 2017.12.10 |
(XAML)Panels and the Layout System (0) | 2017.12.08 |