author | PAMPHILE Jonathan <pamphile@efrei.fr> |
Wed, 28 Oct 2009 11:57:38 +0100 | |
changeset 178 | 56041bd3761e |
parent 176 | 0896f36b9d57 |
child 189 | b37888f59cf2 |
permissions | -rw-r--r-- |
174 | 1 |
using System; |
2 |
using System.Collections.Generic; |
|
3 |
using System.Linq; |
|
4 |
using System.Diagnostics; |
|
5 |
using System.Text; |
|
6 |
using System.Windows; |
|
7 |
using System.Windows.Controls; |
|
8 |
using System.Windows.Data; |
|
9 |
using System.Windows.Documents; |
|
10 |
using System.Windows.Input; |
|
11 |
using System.Windows.Media; |
|
12 |
using System.Windows.Media.Imaging; |
|
13 |
using System.Windows.Media.Animation; |
|
14 |
using System.Windows.Navigation; |
|
15 |
using System.Windows.Shapes; |
|
16 |
using Microsoft.Surface; |
|
17 |
using Microsoft.Surface.Presentation; |
|
18 |
using Microsoft.Surface.Presentation.Controls; |
|
19 |
using System.Windows.Ink; |
|
176 | 20 |
using System.Threading; |
21 |
using System.IO; |
|
22 |
using System.Xml; |
|
23 |
using System.Reflection; |
|
24 |
using System.Xml.Linq; |
|
174 | 25 |
|
26 |
namespace GestureControl |
|
27 |
{ |
|
28 |
public class GestureControl : ContentControl |
|
29 |
{ |
|
30 |
private double angle = 0; |
|
31 |
public static readonly DependencyProperty GestureProperty; |
|
176 | 32 |
Thread inProgess; |
33 |
List<SurfaceGesture> _Gestures = new List<SurfaceGesture>(); |
|
34 |
||
35 |
/// <summary> |
|
36 |
/// List of pattern readed in the patterns.xml |
|
37 |
/// </summary> |
|
38 |
public List<SurfaceGesturePattern> Pattern = new List<SurfaceGesturePattern>(); |
|
39 |
||
174 | 40 |
public String Gesture |
41 |
{ |
|
42 |
get { return (String)GetValue(GestureControl.GestureProperty); } |
|
43 |
set { SetValue(GestureControl.GestureProperty, value); } |
|
44 |
} |
|
45 |
||
46 |
#region GestureEvent |
|
176 | 47 |
|
174 | 48 |
public delegate void GestureRoutedEventHandler(object sender, GestureRoutedEventArgs e); |
176 | 49 |
|
50 |
public static readonly RoutedEvent gestureEvent = EventManager.RegisterRoutedEvent("GestureEvent", RoutingStrategy.Bubble, typeof(GestureRoutedEventHandler), typeof(GestureControl)); |
|
174 | 51 |
|
52 |
public event GestureRoutedEventHandler GestureEvent |
|
53 |
{ |
|
54 |
add { AddHandler(gestureEvent, value); } |
|
55 |
remove { RemoveHandler(gestureEvent, value); } |
|
56 |
} |
|
176 | 57 |
|
58 |
public delegate void RaiseGestureEventCallBack(string gesture); |
|
59 |
||
60 |
public void RaiseGestureEvent(string gesture) |
|
174 | 61 |
{ |
176 | 62 |
if (this.Dispatcher.CheckAccess()) |
174 | 63 |
{ |
176 | 64 |
try |
65 |
{ |
|
66 |
GestureRoutedEventArgs newEventArgs = new GestureRoutedEventArgs(GestureControl.gestureEvent, gesture); |
|
67 |
RaiseEvent(newEventArgs); |
|
68 |
} |
|
69 |
catch (Exception e) { } |
|
174 | 70 |
} |
176 | 71 |
else |
72 |
{ |
|
73 |
this.Dispatcher.BeginInvoke(new RaiseGestureEventCallBack(RaiseGestureEvent), gesture); |
|
74 |
} |
|
174 | 75 |
} |
176 | 76 |
|
174 | 77 |
#endregion |
78 |
||
79 |
static GestureControl() |
|
80 |
{ |
|
81 |
DefaultStyleKeyProperty.OverrideMetadata(typeof(GestureControl), new FrameworkPropertyMetadata(typeof(GestureControl))); |
|
82 |
||
83 |
GestureControl.GestureProperty = DependencyProperty.Register("Gesture", typeof(String), typeof(GestureControl), |
|
84 |
new FrameworkPropertyMetadata("None", new PropertyChangedCallback(OnGestureChanged))); |
|
85 |
} |
|
176 | 86 |
|
174 | 87 |
private static void OnGestureChanged(DependencyObject o, DependencyPropertyChangedEventArgs e) |
88 |
{ |
|
89 |
//Debug.WriteLine("changed", "Gesture"); |
|
90 |
} |
|
91 |
||
92 |
protected override void OnInitialized(EventArgs e) |
|
93 |
{ |
|
94 |
base.OnInitialized(e); |
|
95 |
this.InitializeControl(); |
|
96 |
} |
|
176 | 97 |
|
174 | 98 |
protected void InitializeControl() |
99 |
{ |
|
176 | 100 |
base.AddHandler(SurfaceControl.PreviewContactDownEvent, new RoutedEventHandler(AreaDown)); |
101 |
base.AddHandler(SurfaceInkCanvas.StrokeCollectedEvent, new RoutedEventHandler(AreaStrokeCollected)); |
|
174 | 102 |
} |
176 | 103 |
|
174 | 104 |
// get angle from the first contact of the stroke |
105 |
protected void AreaDown(object source, RoutedEventArgs e) |
|
106 |
{ |
|
107 |
ContactEventArgs c = (ContactEventArgs)e; |
|
108 |
this.angle = c.Contact.GetOrientation(this) - 270; |
|
109 |
} |
|
110 |
||
111 |
protected void AreaStrokeCollected(object source, RoutedEventArgs e) |
|
112 |
{ |
|
178
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
113 |
try |
176 | 114 |
{ |
178
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
115 |
if (inProgess == null) |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
116 |
{ |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
117 |
inProgess = new Thread(RunCapture); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
118 |
inProgess.Start(); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
119 |
} |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
120 |
else |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
121 |
{ |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
122 |
inProgess.Abort(); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
123 |
inProgess = null; |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
124 |
inProgess = new Thread(RunCapture); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
125 |
inProgess.Start(); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
126 |
} |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
127 |
Debug.WriteLine("collected", "Stroke"); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
128 |
System.Windows.Controls.InkCanvasStrokeCollectedEventArgs tmp = e as System.Windows.Controls.InkCanvasStrokeCollectedEventArgs; |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
129 |
// Apply a rotation with the angle of the first contact |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
130 |
Matrix rot = Matrix.Identity; |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
131 |
rot.Rotate(-this.angle); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
132 |
Stroke s = tmp.Stroke; |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
133 |
s.Transform(rot, true); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
134 |
// Get a list of point from a Bezier curve |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
135 |
StylusPointCollection tmp2 = s.GetBezierStylusPoints(); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
136 |
// Generate a list of SurfaceGesturePoint, just X and Y but can be improve |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
137 |
List<SurfaceGesturePoint> pointList = new List<SurfaceGesturePoint>(); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
138 |
foreach (StylusPoint p in tmp2) |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
139 |
pointList.Add(new SurfaceGesturePoint { X = p.X, Y = p.Y }); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
140 |
// create the gesture analyser and set the list |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
141 |
SurfaceGesture gesture = new SurfaceGesture(pointList, 1); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
142 |
// try to get a pattern from the list, if one, raise a event |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
143 |
_Gestures.Add(gesture); |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
144 |
SurfaceInkCanvas ink = e.OriginalSource as SurfaceInkCanvas; |
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
145 |
ink.Strokes.Clear(); |
176 | 146 |
} |
178
56041bd3761e
Mise en place de la reconnaissance de formes
PAMPHILE Jonathan <pamphile@efrei.fr>
parents:
176
diff
changeset
|
147 |
catch { } |
174 | 148 |
} |
176 | 149 |
|
150 |
void RunCapture() |
|
151 |
{ |
|
152 |
try |
|
153 |
{ |
|
154 |
Thread.Sleep(1000); |
|
155 |
CommonPattern(); |
|
156 |
string gesture = GetPattern(_Gestures); |
|
157 |
if (gesture != "None") |
|
158 |
{ |
|
159 |
this.RaiseGestureEvent(gesture); |
|
160 |
} |
|
161 |
_Gestures.Clear(); |
|
162 |
inProgess = null; |
|
163 |
} |
|
164 |
catch (Exception) |
|
165 |
{ |
|
166 |
inProgess = null; |
|
167 |
} |
|
168 |
} |
|
169 |
||
170 |
#region Pattern |
|
171 |
/// <ary> |
|
172 |
/// return a String with the recognized pattern, "None" if no pattern |
|
173 |
/// </summary> |
|
174 |
/// <returns></returns> |
|
175 |
public String GetPattern(List<SurfaceGesture> gestures) |
|
176 |
{ |
|
177 |
int found = 0; |
|
178 |
List<SurfaceGesture> tmp = new List<SurfaceGesture>(); |
|
179 |
foreach (SurfaceGesturePattern p in Pattern) |
|
180 |
{ |
|
181 |
if (p.Count == gestures.Count) |
|
182 |
{ |
|
183 |
int index = 0; |
|
184 |
for (index = 0; index < p.Count; index++) |
|
185 |
{ |
|
186 |
if (p[index].Count == gestures[index].Count) |
|
187 |
{ |
|
188 |
int i; |
|
189 |
for (i = 0; i < p[index].Count; i++) |
|
190 |
{ |
|
191 |
if (gestures[index][i].Direction != p[index][i].Direction) |
|
192 |
i = p[index].Count + 1; |
|
193 |
} |
|
194 |
if (i == p[index].Count) |
|
195 |
found++; |
|
196 |
||
197 |
} |
|
198 |
} |
|
199 |
if (found == p.Count) |
|
200 |
return p.Name; |
|
201 |
} |
|
202 |
} |
|
203 |
||
204 |
return "None"; |
|
205 |
} |
|
206 |
||
207 |
/// <summary> |
|
208 |
/// Load know patterns from the Resources/Patterns.xml file |
|
209 |
/// </summary> |
|
210 |
private void CommonPattern() |
|
211 |
{ |
|
212 |
try |
|
213 |
{ |
|
214 |
#region Load Patterns |
|
215 |
Pattern.Clear(); |
|
216 |
System.IO.Stream file = Assembly.GetExecutingAssembly().GetManifestResourceStream("GestureControl.Resources.Patterns.xml"); |
|
217 |
XmlDocument xml = new XmlDocument(); |
|
218 |
xml.Load(file); |
|
219 |
XmlElement root = xml.DocumentElement; |
|
220 |
XmlNodeList nodes = root.SelectNodes("//Pattern"); |
|
221 |
SurfaceGesturePattern item; |
|
222 |
foreach (XmlNode nod in nodes) |
|
223 |
{ |
|
224 |
string name = nod["Name"].InnerText; |
|
225 |
if (nod.ChildNodes.Count < 2) |
|
226 |
continue; |
|
227 |
item = new SurfaceGesturePattern() { Name = name }; |
|
228 |
List<XmlNode> childs = new List<XmlNode>(); |
|
229 |
foreach (XmlNode elt in nod.ChildNodes) |
|
230 |
if (elt.Name.Equals("Childs")) |
|
231 |
foreach (XmlNode c in elt.ChildNodes) |
|
232 |
childs.Add(c); |
|
233 |
||
234 |
SurfaceGesturePatternItem p; |
|
235 |
int j = 0; |
|
236 |
foreach (XmlNode node in childs) |
|
237 |
{ |
|
238 |
string childname = node["Name"].InnerText; |
|
239 |
List<XmlNode> subNodes = new List<XmlNode>(); |
|
240 |
foreach (XmlNode elt in node.ChildNodes) |
|
241 |
if (elt.Name.Equals("Directions")) |
|
242 |
foreach (XmlNode c in elt.ChildNodes) |
|
243 |
subNodes.Add(c); |
|
244 |
if (subNodes.Count == 0) |
|
245 |
continue; |
|
246 |
||
247 |
p = new SurfaceGesturePatternItem() { Name = childname }; |
|
248 |
||
249 |
foreach (XmlNode subNode in subNodes) |
|
250 |
{ |
|
251 |
switch (subNode.InnerText) |
|
252 |
{ |
|
253 |
case "Up": |
|
254 |
p.Add(new SurfaceGestureVector() { Direction = SurfaceGestureVectorDirection.UP }); |
|
255 |
break; |
|
256 |
case "Down": |
|
257 |
p.Add(new SurfaceGestureVector() { Direction = SurfaceGestureVectorDirection.DOWN }); |
|
258 |
break; |
|
259 |
case "Left": |
|
260 |
p.Add(new SurfaceGestureVector() { Direction = SurfaceGestureVectorDirection.LEFT }); |
|
261 |
break; |
|
262 |
case "Right": |
|
263 |
p.Add(new SurfaceGestureVector() { Direction = SurfaceGestureVectorDirection.RIGHT }); |
|
264 |
break; |
|
265 |
case "DownRight": |
|
266 |
p.Add(new SurfaceGestureVector() { Direction = SurfaceGestureVectorDirection.DOWNRIGHT }); |
|
267 |
break; |
|
268 |
case "DownLeft": |
|
269 |
p.Add(new SurfaceGestureVector() { Direction = SurfaceGestureVectorDirection.DOWNLEFT }); |
|
270 |
break; |
|
271 |
case "UpRight": |
|
272 |
p.Add(new SurfaceGestureVector() { Direction = SurfaceGestureVectorDirection.UPRIGHT }); |
|
273 |
break; |
|
274 |
case "UpLeft": |
|
275 |
p.Add(new SurfaceGestureVector() { Direction = SurfaceGestureVectorDirection.UPLEFT }); |
|
276 |
break; |
|
277 |
default: |
|
278 |
break; |
|
279 |
} |
|
280 |
} |
|
281 |
item.Add(p); |
|
282 |
} |
|
283 |
this.Pattern.Add(item); |
|
284 |
} |
|
285 |
#endregion |
|
286 |
} |
|
287 |
catch |
|
288 |
{ |
|
289 |
throw new Exception("Error loading Patterns.xml"); |
|
290 |
} |
|
291 |
} |
|
292 |
||
293 |
#endregion |
|
174 | 294 |
} |
295 |
} |