Skip to content
This repository was archived by the owner on Sep 25, 2024. It is now read-only.

Commit 1f5e720

Browse files
Dominique LouisCartBlanche
Dominique Louis
authored andcommitted
[Mac] Initial Variations Implementation
1 parent 50cbb49 commit 1f5e720

21 files changed

+516
-75
lines changed

Xamarin.PropertyEditing.Mac/Controls/Custom/BasePopOverViewModelControl.cs

+3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
using System;
22
using System.Collections.Generic;
33
using System.Linq;
4+
using AppKit;
45
using Xamarin.PropertyEditing.ViewModels;
56

67
namespace Xamarin.PropertyEditing.Mac
@@ -9,6 +10,8 @@ internal class BasePopOverViewModelControl : BasePopOverControl
910
{
1011
internal PropertyViewModel ViewModel { get; }
1112

13+
public AutoClosePopOver PopOver { get; internal set; }
14+
1215
public BasePopOverViewModelControl (IHostResourceProvider hostResources, PropertyViewModel viewModel, string title, string imageNamed)
1316
: base (hostResources, title, imageNamed)
1417
{

Xamarin.PropertyEditing.Mac/Controls/Custom/CommandButton.cs

+28-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
using System.Windows.Input;
33
using AppKit;
44

5-
namespace Xamarin.PropertyEditing.Mac.Controls.Custom
5+
namespace Xamarin.PropertyEditing.Mac
66
{
77
internal class CommandButton : NSButton
88
{
@@ -34,7 +34,33 @@ public CommandButton ()
3434
private void CanExecuteChanged (object sender, EventArgs e)
3535
{
3636
if (sender is ICommand cmd)
37-
this.Enabled = cmd.CanExecute (null);
37+
Enabled = cmd.CanExecute (null);
38+
}
39+
}
40+
41+
internal class FocusableCommandButton : CommandButton
42+
{
43+
public override bool CanBecomeKeyView { get { return Enabled; } }
44+
45+
public FocusableCommandButton ()
46+
{
47+
AllowsExpansionToolTips = true;
48+
AllowsMixedState = true;
49+
Cell.LineBreakMode = NSLineBreakMode.TruncatingTail;
50+
Cell.UsesSingleLineMode = true;
51+
ControlSize = NSControlSize.Small;
52+
Font = NSFont.SystemFontOfSize (NSFont.SystemFontSizeForControlSize (NSControlSize.Small));
53+
Title = string.Empty;
54+
TranslatesAutoresizingMaskIntoConstraints = false;
55+
}
56+
57+
public override bool BecomeFirstResponder ()
58+
{
59+
var willBecomeFirstResponder = base.BecomeFirstResponder ();
60+
if (willBecomeFirstResponder) {
61+
ScrollRectToVisible (Bounds);
62+
}
63+
return willBecomeFirstResponder;
3864
}
3965
}
4066
}

Xamarin.PropertyEditing.Mac/Controls/Custom/DrawingExtensions.cs

+28
Original file line numberDiff line numberDiff line change
@@ -116,5 +116,33 @@ public static CommonColor UpdateCMYK (
116116
k: k ?? color.K,
117117
alpha: alpha ?? color.A);
118118
}
119+
120+
public static CGPath ToCGPath (this NSBezierPath nsPath)
121+
{
122+
var cgPath = new CGPath ();
123+
for (var i = 0; i < nsPath.ElementCount; i++) {
124+
NSBezierPathElement type = nsPath.ElementAt (i, out CGPoint[] points);
125+
126+
switch (type) {
127+
case NSBezierPathElement.ClosePath:
128+
cgPath.CloseSubpath ();
129+
break;
130+
131+
case NSBezierPathElement.CurveTo:
132+
cgPath.AddCurveToPoint (points[0], points[1], points[2]);
133+
break;
134+
135+
case NSBezierPathElement.LineTo:
136+
cgPath.AddLineToPoint (points[0]);
137+
break;
138+
139+
case NSBezierPathElement.MoveTo:
140+
cgPath.MoveToPoint (points[0]);
141+
break;
142+
}
143+
}
144+
145+
return cgPath;
146+
}
119147
}
120148
}

Xamarin.PropertyEditing.Mac/Controls/Custom/PropertyButton.cs

+51-33
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
using System;
2+
using System.Threading.Tasks;
23
using AppKit;
34
using CoreGraphics;
45
using Xamarin.PropertyEditing.ViewModels;
@@ -19,11 +20,16 @@ internal PropertyViewModel ViewModel
1920
set {
2021
if (this.viewModel != null) {
2122
this.viewModel.PropertyChanged -= OnPropertyChanged;
23+
if (this.viewModel.HasVariations)
24+
this.viewModel.CreateVariantRequested -= OnCreateVariantRequested;
2225
}
2326

2427
this.viewModel = value;
2528
if (this.viewModel != null) {
2629
this.viewModel.PropertyChanged += OnPropertyChanged;
30+
if (this.viewModel.HasVariations)
31+
this.viewModel.CreateVariantRequested += OnCreateVariantRequested;
32+
2733
ValueSourceChanged (this.viewModel.ValueSource);
2834
}
2935
}
@@ -82,7 +88,7 @@ private void PopUpContextMenu ()
8288
AttributedTitle = new Foundation.NSAttributedString (
8389
Properties.Resources.CustomExpressionEllipsis,
8490
new CoreText.CTStringAttributes {
85-
Font = new CoreText.CTFont (PropertyEditorControl.DefaultFontName, PropertyEditorControl.DefaultFontSize + 1),
91+
Font = new CoreText.CTFont (PropertyEditorControl.DefaultFontName, PropertyEditorControl.DefaultMenuFontSize),
8692
})
8793
};
8894

@@ -99,7 +105,7 @@ private void PopUpContextMenu ()
99105
AttributedTitle = new Foundation.NSAttributedString (
100106
Properties.Resources.ResourceEllipsis,
101107
new CoreText.CTStringAttributes {
102-
Font = new CoreText.CTFont (PropertyEditorControl.DefaultFontName, PropertyEditorControl.DefaultFontSize + 1),
108+
Font = new CoreText.CTFont (PropertyEditorControl.DefaultFontName, PropertyEditorControl.DefaultMenuFontSize),
103109
})
104110
};
105111

@@ -110,11 +116,11 @@ private void PopUpContextMenu ()
110116
this.popUpContextMenu.AddItem (NSMenuItem.SeparatorItem);
111117

112118
// TODO If we add more menu items consider making the Label/Command a dictionary that we can iterate over to populate everything.
113-
this.popUpContextMenu.AddItem (new CommandMenuItem (Properties.Resources.Reset, viewModel.ClearValueCommand) {
119+
this.popUpContextMenu.AddItem (new CommandMenuItem (Properties.Resources.Reset, this.viewModel.ClearValueCommand) {
114120
AttributedTitle = new Foundation.NSAttributedString (
115121
Properties.Resources.Reset,
116122
new CoreText.CTStringAttributes {
117-
Font = new CoreText.CTFont (PropertyEditorControl.DefaultFontName, PropertyEditorControl.DefaultFontSize + 1),
123+
Font = new CoreText.CTFont (PropertyEditorControl.DefaultFontName, PropertyEditorControl.DefaultMenuFontSize),
118124
})
119125
});
120126
}
@@ -131,34 +137,34 @@ private void ToggleFocusImage (bool focused = false)
131137
if (this.viewModel != null) {
132138

133139
switch (this.viewModel.ValueSource) {
134-
case ValueSource.Binding:
135-
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-bound-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-bound-mac-10");
136-
break;
137-
138-
case ValueSource.Default:
139-
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-default-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-default-mac-10");
140-
break;
141-
142-
case ValueSource.Local:
143-
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-local-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-local-mac-10");
144-
break;
145-
146-
case ValueSource.Inherited:
147-
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-inherited-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-inherited-mac-10");
148-
break;
149-
150-
case ValueSource.Resource:
151-
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-inherited-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-inherited-mac-10");
152-
break;
153-
154-
case ValueSource.Unset:
155-
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-default-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-default-mac-10");
156-
break;
157-
158-
default:
159-
// To Handle ValueSource.DefaultStyle, ValueSource.Style etc.
160-
Image = null;
161-
break;
140+
case ValueSource.Binding:
141+
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-bound-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-bound-mac-10");
142+
break;
143+
144+
case ValueSource.Default:
145+
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-default-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-default-mac-10");
146+
break;
147+
148+
case ValueSource.Local:
149+
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-local-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-local-mac-10");
150+
break;
151+
152+
case ValueSource.Inherited:
153+
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-inherited-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-inherited-mac-10");
154+
break;
155+
156+
case ValueSource.Resource:
157+
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-inherited-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-inherited-mac-10");
158+
break;
159+
160+
case ValueSource.Unset:
161+
Image = focused ? this.hostResources.GetNamedImage ("pe-property-button-default-mac-active-10") : this.hostResources.GetNamedImage ("pe-property-button-default-mac-10");
162+
break;
163+
164+
default:
165+
// To Handle ValueSource.DefaultStyle, ValueSource.Style etc.
166+
Image = null;
167+
break;
162168
}
163169
}
164170
}
@@ -228,13 +234,25 @@ private void OnResourceRequested (object sender, EventArgs e)
228234
Appearance = EffectiveAppearance
229235
};
230236

231-
var resourceSelectorPopOver = new AutoClosePopOver(this.hostResources) {
237+
var resourceSelectorPopOver = new AutoClosePopOver (this.hostResources) {
232238
ContentViewController = new NSViewController (null, null) { View = requestResourceView },
233239
};
234240

235241
requestResourceView.PopOver = resourceSelectorPopOver;
236242

237243
resourceSelectorPopOver.Show (requestResourceView.Frame, (NSView)this, NSRectEdge.MinYEdge);
238244
}
245+
246+
private void OnCreateVariantRequested (object sender, CreateVariantEventArgs e)
247+
{
248+
var createVariantWindow = new CreateVariantWindow (this.hostResources, this.viewModel) {
249+
Appearance = EffectiveAppearance,
250+
};
251+
252+
var result = (NSModalResponse)(int)NSApplication.SharedApplication.RunModalForWindow (createVariantWindow);
253+
if (result == NSModalResponse.OK) {
254+
e.Variation = Task.FromResult (createVariantWindow.ViewModel.Variation);
255+
}
256+
}
239257
}
240258
}

Xamarin.PropertyEditing.Mac/Controls/Custom/UnfocusableTextField.cs

+5
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,11 @@ public NSColor TextColor {
3939
internal set { this.label.TextColor = value; }
4040
}
4141

42+
public bool Bordered {
43+
get { return this.label.Bordered; }
44+
internal set { this.label.Bordered = value; }
45+
}
46+
4247
public virtual NSBackgroundStyle BackgroundStyle
4348
{
4449
[Export ("backgroundStyle")] get => this.label.Cell.BackgroundStyle;

Xamarin.PropertyEditing.Mac/Controls/CustomExpressionView.cs

-2
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,6 @@ internal class CustomExpressionView : BasePopOverViewModelControl
1515
private const string PreviewCustomExpressionString = "PreviewCustomExpression";
1616
private const string AutocompleteItemsString = "AutocompleteItems";
1717

18-
public AutoClosePopOver PopOver { get; internal set; }
19-
2018
public CustomExpressionView (IHostResourceProvider hostResources, PropertyViewModel viewModel)
2119
: base (hostResources, viewModel, Properties.Resources.CustomExpression, "pe-custom-expression-32")
2220
{

0 commit comments

Comments
 (0)