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