-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathact-touch.lisp
512 lines (428 loc) · 20.5 KB
/
act-touch.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; Author : Frank Tamborello
;;; Copyright : (c) 2012-8 Cogscent, LLC
;;; Availability: GNU LGPL, see LGPL.txt
;;; Address : Cogscent, LLC
;;; : PMB 7431
;;; : 2711 Centerville Rd, Ste 120
;;; : Wilmington DE, USA 19808-1676
;;; : frank.tamborello@cogscent.com
;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; This library is free software; you can redistribute it and/or
;;; modify it under the terms of the Lisp Lesser General Public
;;; License: the GNU Lesser General Public License as published by the
;;; Free Software Foundation (either version 2.1 of the License,
;;; or, at your option, any later version),
;;; and the Franz, Inc Lisp-specific preamble.
;;;
;;; This library is distributed in the hope that it will be useful,
;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
;;; Lesser General Public License for more details.
;;;
;;; You should have received a copy of the Lisp Lesser General Public
;;; License along with this library; if not, write to the Free Software
;;; Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
;;; and see Franz, Inc.'s preamble to the GNU Lesser General Public License,
;;; http://opensource.franz.com/preamble.html.
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; Acknowledgements
;;; : This research is sponsored by Measurement Science and
;;; Engineering grants 60NANB12D134 and 70NANB15H252 from the
;;; National Institute of Standards and Technology (NIST).
;;; Special acknowledgements are due to Drs. Ross Micheals, Mary
;;; Theofanos, and Kristen K. Greene of NIST's Information
;;; Technology Laboratory.
;;; Thanks also to Dr. Michael D. Byrne, upon whose experiment
;;; library code I based the device code.
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; Filename : act-touch.lisp
;;; Revision : 16
;;;
;;; Description : This code extends the ACT-R 7 motor module to implement
;;; several movement styles commonly used with multi-touch handheld
;;; computers as well as defines a device with which to perform those
;;; movement styles.
;;;
;;; Usage : Place in ACT-R folder "User Loads." This file will load
;;; automatically after ACT-R loads.
;;;
;;; Bugs : None known
;;;
;;; To do : 1. Index-z should be a property of the hand or finger (probably both), not of the device! XY coordinate units should all be pixels, not keyboard keys.
;;;
;;;
;;; Issues :
;;; 2014.10.05 fpt
;;; : 1. Long-term, it'd probably make more sense to add a z-dimension to hand locations.
;;; For now movement time methods use "z-index", a slot of the device object.
;;;
;;; 2012.06.14 fpt
;;; : 1. What's the target of a swipe? Is it the entire side of the
;;; device?
;;; 2012.06.14 fpt
;;; : 2. There's something funny about using Fitts' Law to compute
;;; pinch execution time. How wide should the target be? That's sort
;;; of a nonsensical question to ask in the context of a pinch, but
;;; Fitts' Law requires a width and on the surface the pinch movement
;;; seems fairly corrected-ballistic like other pointing movements.
;;; For now let's assume that the thumb is the target-yes, even for
;;; reverse-pinches-and that the typical human thumb is 1 inch (72px
;;; @ 72ppi) wide. So the pinch class has a slot for thumb-width
;;; initializing to 72, which gets passed to the fitts function for
;;; calculating pinch execution time.
;;; 2012.08.18 fpt
;;; : 3. Does it make more sense to construct two-, three-, and
;;; four-fingered swipes as different movement styles instead of
;;; one movement style with a feature specifying the number of
;;; fingers?
;;; 2012.08.18 fpt
;;; : 4. What about qualitatively different swipes used for
;;; scrolling precisely starting and ending within one page
;;; vs page turning?
;;;
;;; ----- History -----
;;; 2012.06.01 fpt 1
;;; : Inception: Extend the motor module with a movement style that
;;; allows ACT-R to respond to that style's request and produce some
;;; model output: swipe.
;;; 2012.06.02 fpt 2
;;; : Implement other movement styles: tap, pinch (also works for
;;; reverse-pinch), and rotate.
;;; 2012.06.07 fpt 3
;;; : Gave tap its real execution time computation, removed the r &
;;; theta features since it taps in place, and gave it a stub device
;;; method.
;;; 2012.06.14 fpt 4
;;; : Defined additional multitouch movement styles with assumed-to-
;;; be-sensible time computations: peck-tap, peck-tap-recoil, tap-
;;; hold, tap-release, tap-drag-release, swipe, pinch, rotate.
;;; 2012.06.19 fpt 5
;;; : The Device is now instrumented to record things like time and
;;; location of taps. It uses the virtual-experiment-window library,
;;; adapted by me from Mike Byrne's experiment-window4 library.
;;;
;;; 2012.06.29 fpt 6
;;; Demo task is now a procedure-window rather than a timed-exp-window so that the
;;; scope of one "trial" is one performance of one multi-action task rather than
;;; one performance of one action.
;;;
;;; 2012.07.13 fpt 7
;;; Move-hand-touch allows the model to move its hand to a visual-location or
;;; a visual-object, just like move-cursor. ...except that noisy movement
;;; is not yet implemented.
;;;
;;; 2012.08.18 fpt 8
;;; Wrote a feat-differences method for tap since it didn't inherit it from a parent
;;; movement-style.
;;;
;;; 2012.08.26 fpt 9
;;; Increased the length of the procedure from 1 to 8 steps & made some of the steps
;;; require other gestures besides tap, namely swipe, rotate, & pinch.
;;;
;;; 2012.09.29 fpt 10
;;; 1. Changed license from public domain to LGPL.
;;; 2. Forked the virtual multitouch display device and
;;;demonstration model into their own files.
;;;
;;; 2013.04.03 fpt 11
;;; Quickly hacked move-hand-touch so it could generate some noisy movements
;;; probability of a miss considering index finger tip area and target area
;;;
;;; 2013.11.21 fpt 12
;;; 1. Incorporate Melissa Gallagher's dual-distribution noisy movement mechanism,
;;; based on May & Byrne.
;;; 2. Push ':virtual-multitouch-device onto *features* to ease checking for &
;;; loading
;;;
;;; 2014.10.04 fpt 13
;;; Update to make compatible with ACT-R 6.1:
;;; 1. Move-hand-touch conforms to the new argument list of verify-single-explicit-
;;; value introduced with ACT-R 6.1.
;;;
;;; 2. A new abstraction became available in ACT-R 6.1, handle-style-request,
;;; to schedule the event for a defined movement style. Now all ACT-Touch's
;;; defined movement styles, such as tap, use that instead of their own
;;; request methods.
;;;
;;; 2016.06.02 fpt 14
;;; 1. Revised move-hand-touch to comply with xy-loc's new declamation (r1802)
;;; and apparently an undocumented change in approach-width that now requires
;;; the vision module to be passed as a third argument.
;;;
;;; 2. Replaced call to rand-time with randomize-time because r1878.
;;;
;;; NB: This version complies with ACT-R 7 (r2018), at least for move-hand-touch
;;; and tap.
;;;
;;; 2017.11.04 fpt 15
;;; I deleted index-z so that act-touch now works with devices other than objects
;;; with an index-z slot, such as ACT-Droid.
;;;
;;; 2018.02.06 fpt 16
;;; Implement a new feature to allow the model to "consciously" control swiping
;;; swipe-speed: an integer from the set [1,5], where 1 is slowest and 5 is fastest.
;;;
;;; Note that this update breaks old code calling the swipe movement command
;;; without specifying a swipe-speed feature: ACT-R simply won't execute the swipe
;;; command, instead proceeding otherwise.
;;;
;;; Note: I lack a principled way to map user intention of swiping speed to
;;; computing its execution time, so I'll just do what's expedient and otherwise
;;; seems at least not stupidly unreasonable: I give the Fitts function a
;;; coefficient of difficulty that decreases with swipe-speed:
;;; (/ 1 (expt (swipe-speed self) 2))
;;;
;;; Note: I think swipe should allow for a nice, wide target parameter for the
;;; Fitts function, like the entire side of the device to which the model is
;;; swiping. For when you run with a NIL device, EG for testing, I gave it a
;;; value of 500. Replace that in compute-exec-time with a call to the width
;;; of your target.
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;; ---------------------------------------------------------------------- ;;;;
;;;; Macros from "On Lisp"
;;;; ---------------------------------------------------------------------- ;;;;
(defmacro aif (test-form then-form &optional else-form)
`(let ((it ,test-form))
(if it ,then-form ,else-form)))
;;;; ---------------------------------------------------------------------- ;;;;
;;;; ACT-R Motor Module Extension
;;;; ---------------------------------------------------------------------- ;;;;
(defparameter *index-finger-tip-size* #(45 27) ; ~ 5/8" x 1/4"
"Width and height of the user's index finger tip, in pixels at 72 ppi.")
;;; tap movement style
;;; analogous to punch movement style, the indicated finger strikes the surface
;;; of the display at the finger's current x, y location and returns the finger
;;; to a z-position where it is ready to act again
(defStyle tap () hand finger)
(defmethod compute-exec-time ((mtr-mod motor-module) (self tap))
(+ (init-time mtr-mod)
(max (burst-time mtr-mod)
(randomize-time (fitts mtr-mod
;; borrow peck's "b" coefficient, although really should just be 0
(peck-fitts-coeff mtr-mod)
;; get the current device's index-z
;; no, don't use that kludge anymore
;; (index-z (current-device))
72)))))
(defmethod feat-differences ((t1 tap) (t2 tap))
(cond ((not (eq (hand t1) (hand t2))) 2)
((not (eq (finger t1) (finger t2))) 1)
(t 0)))
(defmethod queue-output-events ((mtr-mod motor-module) (self tap))
(schedule-event-relative
(exec-time self)
'device-handle-tap
:module :motor
:output 'high
:params (list (current-device)
(ecase (hand self)
(right (finger-loc (right-hand mtr-mod) (finger self)))
(left (finger-loc (left-hand mtr-mod) (finger self))))
(hand self)
(finger self))))
;;; tap-hold movement style
;;; Performs a tap movement, but holds the finger to the multitouch display surface.
(defstyle tap-hold tap hand finger)
(defmethod queue-output-events ((mtr-mod motor-module) (self tap-hold))
(schedule-event-relative
(exec-time self)
'device-handle-tap-hold
:module :motor
:output 'high
:params (list (device (current-device-interface))
(hand self)
(finger self))))
;;; tap-release movement style
;;; If the finger is held against the surface of the multitouch display,
;;; as in a tap-hold movement, then tap-release moves the finger away from
;;; the multitouch display surface.
(defstyle tap-release tap hand finger)
(defmethod queue-output-events ((mtr-mod motor-module) (self tap-release))
(schedule-event-relative
(exec-time self)
'device-handle-tap-release
:module :motor
:output 'high
:params (list (device (current-device-interface))
(hand self)
(finger self))))
(defgeneric noisy-loc-em? (mtr-mod xy-loc w theta)
(:documentation "If the Motor Module is set up for it, make the output location noisy."))
;;; NOISY-LOC-em? [Method]
;;; Description : Adds noise to the output location if noise is on.
;;; : Rather than adding the same amount of error on both axis, more is on axis than
;;; : off axis. uses weighted-error to determine how much it is
;;;
(defmethod noisy-loc-em? ((mm motor-module) (xy-loc vector) (w number) (theta number))
(if (not (cursor-noise mm))
xy-loc
(if (zerop w)
xy-loc
(weighted-error xy-loc w theta))))
;;; WEIGHTED-ERROR [Method]
;;; Description : samples from the normal distribution with different error on axis and off axis
;;; : based on work by may and byrne (see eq 6)
;;; : Treats the error area as a circle with a diameter of pixw
(defmethod weighted-error ((xy-loc vector) (pixw number) (theta number))
(model-output "pixw: ~a theta: ~a~%" pixw theta)
(let ((on-noise (act-r-noise (* (/ pixw 4.133) (/ (sqrt 3) pi))))
(off-noise (act-r-noise (* 0.75 (/ pixw 4.133) (/ (sqrt 3) pi)))))
(polar-move-xy
xy-loc
(vector
(sqrt
(+ (* on-noise on-noise) (* off-noise off-noise)))
(+ theta (atan (/ off-noise on-noise)))))))
;;; move-hand-touch
;;; Allows the model to move its hand to what it sees.
;;; Adapted from motor.lisp's move-cursor.
(defmethod move-hand-touch ((mtr-mod motor-module) &key loc object)
(unless (or
(check-jam mtr-mod)
(check-specs (or loc object)))
(let (r-theta feat w vision)
(setf
vision
(get-module :vision)
feat ;; always refer back to the visicon chunks if possible
(cond ((and object (chunk-visicon-entry object) (chunk-p-fct (gethash (chunk-visicon-entry object) (visicon vision))))
(gethash (chunk-visicon-entry object) (visicon vision)))
((and object (chunk-slot-value-fct object 'screen-pos)
(chunk-p-fct (chunk-slot-value-fct object 'screen-pos))
(numberp (chunk-slot-value-fct (chunk-slot-value-fct object 'screen-pos) 'screen-x))
(numberp (chunk-slot-value-fct (chunk-slot-value-fct object 'screen-pos) 'screen-y)))
(if (chunk-p-fct (gethash (chunk-visicon-entry (chunk-slot-value-fct object 'screen-pos)) (visicon vision)))
(gethash (chunk-visicon-entry (chunk-slot-value-fct object 'screen-pos)) (visicon vision))
(chunk-slot-value-fct object 'screen-pos)))
((and loc (chunk-visicon-entry loc) (chunk-p-fct (gethash (chunk-visicon-entry loc) (visicon vision))))
(gethash (chunk-visicon-entry loc) (visicon vision)))
((and loc (chunk-p-fct loc) (numberp (chunk-slot-value-fct loc 'screen-x)) (numberp (chunk-slot-value-fct loc 'screen-y)))
loc)
(t
(print-warning "No valid location could be generated from ~s or ~s when trying to move the mouse." object loc)
(return-from move-hand-touch nil))))
(setf r-theta (xy-to-polar (loc (right-hand mtr-mod)) (xy-loc feat vision)))
(if (= 0 (vr r-theta)) ; r=0 is a no-op
(model-warning "Move-hand-touch action aborted because hand is at
requested target ~S" (if object object loc))
(progn
(setf w (pm-angle-to-pixels (approach-width feat (vtheta r-theta) vision)))
(let ((r-theta-new (xy-to-polar
(loc (right-hand mtr-mod))
(noisy-loc-em? mtr-mod (xy-loc feat vision) w (vtheta r-theta)))))
(prepare-movement
mtr-mod
(make-instance
'hand-ply
:hand 'right
:r (vr r-theta-new)
:theta (vtheta r-theta-new)
:target-width w))))))))
(defmethod move-hand-touch-request ((mtr-mod motor-module) chunk-spec)
(let ((object (if (slot-in-chunk-spec-p chunk-spec 'object)
(verify-single-explicit-value
chunk-spec
'object
:motor
'move-hand-touch)
nil))
(location (if (slot-in-chunk-spec-p chunk-spec 'loc)
(verify-single-explicit-value
chunk-spec
'loc
:motor
'move-hand-touch)
nil)))
(when (or object location)
(schedule-event-relative
0
'move-hand-touch
:destination :motor
:params (list :object object
:loc location)
:module :motor
:output 'high))))
;;; Index-Thumb
;;; Subclass for pinch, reverse-pinch, and rotate
(defclass index-thumb (hfrt-movement)
((move-time :accessor move-time))
(:default-initargs
:style-name :index-thumb))
(defmethod compute-exec-time ((mtr-mod motor-module) (self index-thumb))
(setf (move-time self)
(max (burst-time mtr-mod)
(fitts mtr-mod (peck-fitts-coeff mtr-mod)
(ecase (style-name self)
(:index-thumb (r self))
(:pinch (abs (- (start-width self) (end-width self))))
(:rotate (dist
(finger-loc-m mtr-mod 'right 'thumb)
(finger-loc-m mtr-mod 'right 'index)))))))
(+ (init-time mtr-mod)
(max (burst-time mtr-mod)
(randomize-time (move-time self)))))
(defmethod compute-finish-time ((mtr-mod motor-module) (self index-thumb))
(+ (exec-time self)
(burst-time mtr-mod)
(max (burst-time mtr-mod)
(randomize-time (move-time self)))))
(defmethod index-thumb ((mtr-mod motor-module) &key hand finger r theta)
(unless (or (check-jam mtr-mod) (check-specs 'index-thumb hand finger r theta))
(when (symbolp theta)
(setf theta (symbol-value theta)))
(prepare-movement mtr-mod (make-instance 'index-thumb :hand hand :finger finger :r r :theta theta))))
(defStyle pinch index-thumb hand finger start-width end-width)
(defmethod queue-output-events ((mtr-mod motor-module) (self pinch))
(schedule-event-relative (exec-time self) 'device-handle-pinch :module :motor :output 'medium
:params (list (current-device) (start-width self) (end-width self))))
(defStyle rotate index-thumb hand finger rotation)
(defmethod queue-output-events ((mtr-mod motor-module) (self rotate))
(schedule-event-relative (exec-time self) 'device-handle-rotate :module :motor :output 'medium
:params (list
(device (current-device-interface))
(rotation self))))
(defStyle swipe index-thumb hand finger r theta num-fngrs swipe-speed)
(defmethod compute-exec-time ((mtr-mod motor-module) (self swipe))
(setf (move-time self)
(max (burst-time mtr-mod)
(fitts mtr-mod
(/ 1 (expt (swipe-speed self) 2))
(r self)
;; width because the width of a target of a swipe should be wide, ie the edge of the device
500))) ; or whatever the width of your device may be
(+ (init-time mtr-mod)
(max (burst-time mtr-mod)
(randomize-time (move-time self)))))
(defmethod queue-output-events ((mtr-mod motor-module) (self swipe))
(schedule-event-relative (exec-time self) 'device-handle-swipe :module :motor :output 'medium
:params (list
; (device (current-device-interface))
(current-device)
(polar-move-xy (finger-loc-m mtr-mod (hand self) (finger self))
(vector (r self) (theta self)))
(num-fngrs self))))
(extend-manual-requests-fct '((tap (:include motor-command)) (hand right) (finger index)) 'handle-style-request)
; If you need to redefine a movement style, first remove it:
; (remove-manual-request tap)
(extend-manual-requests-fct '((tap-hold (:include motor-command)) (hand right) (finger index)) 'handle-style-request)
; (remove-manual-request tap-hold)
(extend-manual-requests-fct '((tap-release (:include motor-command)) (hand right) (finger index)) 'handle-style-request)
; (remove-manual-request tap-release)
(extend-manual-requests-fct '((move-hand-touch (:include motor-command)) loc object) 'move-hand-touch-request)
; (remove-manual-request move-hand-touch)
(extend-manual-requests-fct '((index-thumb (:include motor-command)) (hand right) (finger index) r theta) 'index-thumb)
; (remove-manual-request swipe)
(extend-manual-requests-fct '((swipe (:include motor-command)) (hand right) (finger index) r theta (num-fngrs 1)) 'handle-style-request)
; (remove-manual-request swipe)
(extend-manual-requests-fct '((pinch (:include motor-command)) (hand right) (finger index) start-width end-width) 'handle-style-request)
; (remove-manual-request pinch)
(extend-manual-requests-fct '((rotate (:include motor-command)) (hand right) (finger index) rotation) 'handle-style-request)
; (remove-manual-request rotate)
(unless
(member ':act-touch *features*)
(push :act-touch *features*))