Add back 2 patches for histogram and overlayframe that are working correctly and...
[goodguy/cinelerra.git] / cinelerra-5.1 / cinelerra / shuttle.C
1 #ifdef HAVE_SHUTTLE
2 // Copyright 2013 Eric Messick (FixedImagePhoto.com/Contact)
3 // reworked 2019 for cinelerra-gg by William Morrow
4
5 #include "arraylist.h"
6 #include "cstrdup.h"
7 #include "file.h"
8 #include "guicast.h"
9 #include "keys.h"
10 #include "linklist.h"
11 #include "loadfile.h"
12 #include "mainmenu.h"
13 #include "shuttle.h"
14 #include "thread.h"
15
16 #include "mwindow.h"
17 #include "mwindowgui.h"
18 #include "awindow.h"
19 #include "awindowgui.h"
20 #include "cwindow.h"
21 #include "cwindowgui.h"
22 #include "vwindow.h"
23 #include "vwindowgui.h"
24
25 #include <stdio.h>
26 #include <stdint.h>
27 #include <stdlib.h>
28 #include <unistd.h>
29 #include <fcntl.h>
30 #include <string.h>
31 #include <sys/time.h>
32 #include <sys/stat.h>
33
34 #include <X11/Xlib.h>
35 #include <X11/keysym.h>
36
37 static Time milliTimeClock()
38 {
39         struct timeval tv;
40         gettimeofday(&tv, 0);
41         return (tv.tv_sec * 1000) + (tv.tv_usec / 1000);
42 }
43
44 KeySymMapping KeySymMapping::key_sym_mapping[] = {
45 // button keycodes
46         { "XK_Button_1", XK_Button_1 },
47         { "XK_Button_2", XK_Button_2 },
48         { "XK_Button_3", XK_Button_3 },
49         { "XK_Scroll_Up", XK_Scroll_Up },
50         { "XK_Scroll_Down", XK_Scroll_Down },
51 #include "shuttle_keys.h"
52         { NULL, 0 }
53 };
54
55 int KeySymMapping::get_mask(const char *&str)
56 {
57         int mask = 0;
58         while( *str ) {
59                 if( !strncmp("Shift-",str,6) ) {
60                         mask |= ShiftMask;
61                         str += 6;  continue;
62                 }
63                 if( !strncmp("Ctrl-",str,5) ) {
64                         mask |= ControlMask;
65                         str += 5;  continue;
66                 }
67                 else if( !strncmp("Alt-",str,4) ) {
68                         mask |= Mod1Mask;
69                         str += 4;  continue;
70                 }
71                 break;
72         }
73         return mask;
74 }
75
76 SKeySym KeySymMapping::to_keysym(const char *str)
77 {
78         if( !strncmp("FWD_",str, 4) ) {
79                 float speed = atof(str+4) / SHUTTLE_MAX_SPEED;
80                 if( speed > SHUTTLE_MAX_SPEED ) return 0;
81                 int key_code = (SKEY_MAX+SKEY_MIN)/2. +
82                         (SKEY_MAX-SKEY_MIN)/2. * speed;
83                 if( key_code > SKEY_MAX ) key_code = SKEY_MAX;
84                 return key_code;
85         }
86         if( !strncmp("REV_",str, 4) ) {
87                 float speed = atof(str+4) / SHUTTLE_MAX_SPEED;
88                 if( speed > SHUTTLE_MAX_SPEED ) return 0;
89                 int key_code = (SKEY_MAX+SKEY_MIN)/2. -
90                         (SKEY_MAX-SKEY_MIN)/2. * speed;
91                 if( key_code < SKEY_MIN ) key_code = SKEY_MIN;
92                 return key_code;
93         }
94         int mask = get_mask(str);
95         for( KeySymMapping *ksp = &key_sym_mapping[0]; ksp->str; ++ksp )
96                 if( !strcmp(str, ksp->str) )
97                         return SKeySym(ksp->sym, mask);
98         return 0;
99 }
100
101 const char *KeySymMapping::to_string(SKeySym ks)
102 {
103         for( KeySymMapping *ksp = &key_sym_mapping[0]; ksp->sym.key; ++ksp ) {
104                 if( ksp->sym.key == ks.key ) {
105                         static char string[BCSTRLEN];
106                         char *sp = string, *ep = sp+sizeof(string);
107                         if( ks.msk & Mod1Mask ) sp += snprintf(sp, ep-sp, "Alt-");
108                         if( ks.msk & ControlMask ) sp += snprintf(sp, ep-sp, "Ctrl-");
109                         if( ks.msk & ShiftMask ) sp += snprintf(sp, ep-sp, "Shift-");
110                         snprintf(sp, ep-sp, "%s", ksp->str);
111                         return string;
112                 }
113         }
114         if( ks >= SKEY_MIN && ks <= SKEY_MAX ) {
115                 double speed = SHUTTLE_MAX_SPEED *
116                         (ks-(SKEY_MAX+SKEY_MIN)/2.) / ((SKEY_MAX-SKEY_MIN)/2.);
117                 static char text[BCSTRLEN];
118                 sprintf(text, "%s_%0.3f", speed>=0 ? "FWD" : "REV", fabs(speed));
119                 char *bp = strchr(text,'.');
120                 if( bp ) {
121                         char *cp = bp+strlen(bp);
122                         while( --cp>bp && *cp=='0' ) *cp=0;
123                         if( cp == bp ) *cp = 0;
124                 }
125                 return text;
126         }
127         return 0;
128 }
129
130 TransName::TransName(int cin, const char *nm, const char *re)
131 {
132         this->cin = cin;
133         this->name = cstrdup(nm);
134 }
135 TransName::~TransName()
136 {
137         delete [] name;
138 }
139
140 void Translation::init(int def)
141 {
142         is_default = def;
143         is_key = 0;
144         first_release_stroke = 0;
145         pressed = 0;
146         released = 0;
147         keysym_down = 0;
148 }
149
150 Translation::Translation(Shuttle *shuttle)
151  : modifiers(this)
152 { // initial default translation
153         init(1);
154         this->shuttle = shuttle;
155         this->name = cstrdup("Default");
156         names.append(new TransName(FOCUS_DEFAULT, name, ""));
157         key_down[K6].add_stroke(XK_Button_1, 1);
158         key_up[K6].add_stroke(XK_Button_1, 0);
159         key_down[K7].add_stroke(XK_Button_2, 1);
160         key_up[K7].add_stroke(XK_Button_2, 0);
161         key_down[K8].add_stroke(XK_Button_3, 1);
162         key_up[K8].add_stroke(XK_Button_3, 0);
163         jog[JL].add_stroke(XK_Scroll_Up, 1);
164         jog[JL].add_stroke(XK_Scroll_Up, 0);
165         jog[JR].add_stroke(XK_Scroll_Down, 0);
166         jog[JR].add_stroke(XK_Scroll_Down, 1);
167 }
168
169 Translation::Translation(Shuttle *shuttle, const char *name)
170  : modifiers(this)
171 {
172         init(0);
173         this->shuttle = shuttle;
174         this->name = cstrdup(name);
175 }
176
177 Translation::~Translation()
178 {
179         delete [] name;
180 }
181
182 void Translation::clear()
183 {
184         names.remove_all_objects();
185         init(0);
186         for( int i=0; i<NUM_KEYS; ++i ) key_down[i].clear();
187         for( int i=0; i<NUM_KEYS; ++i ) key_up[i].clear();
188         for( int i=0; i<NUM_SHUTTLES; ++i ) shuttles[i].clear();
189         for( int i=0; i<NUM_JOGS; ++i ) jog[i].clear();
190 }
191
192 void Translation::append_stroke(SKeySym sym, int press)
193 {
194         Stroke *s = pressed_strokes->append();
195         s->keysym = sym;
196         s->press = press;
197 }
198
199 void Translation::add_keysym(SKeySym sym, int press_release)
200 {
201 //printf("add_keysym(0x%x, %d)\n", (int)sym, press_release);
202         switch( press_release ) {
203         case PRESS:
204                 append_stroke(sym, 1);
205                 modifiers.mark_as_down(sym, 0);
206                 break;
207         case RELEASE:
208                 append_stroke(sym, 0);
209                 modifiers.mark_as_up(sym);
210                 break;
211         case HOLD:
212                 append_stroke(sym, 1);
213                 modifiers.mark_as_down(sym, 1);
214                 break;
215         case PRESS_RELEASE:
216         default:
217                 if( first_release_stroke ) {
218                         modifiers.re_press();
219                         first_release_stroke = 0;
220                 }
221                 if( keysym_down ) {
222                         append_stroke(keysym_down, 0);
223                 }
224                 append_stroke(sym, 1);
225                 keysym_down = sym;
226                 break;
227         }
228 }
229
230 void Translation::add_release(int all_keys)
231 {
232 //printf("add_release(%d)\n", all_keys);
233         modifiers.release(all_keys);
234         if( !all_keys )
235                 pressed_strokes = released_strokes;
236         if( keysym_down ) {
237                 append_stroke(keysym_down, 0);
238                 keysym_down = 0;
239         }
240         first_release_stroke = 1;
241 }
242
243 void Translation::add_keystroke(const char *keySymName, int press_release)
244 {
245         SKeySym sym;
246
247         if( is_key && !strncmp(keySymName, "RELEASE", 8) ) {
248                 add_release(0);
249                 return;
250         }
251         sym = KeySymMapping::to_keysym(keySymName);
252         if( sym != 0 ) {
253                 add_keysym(sym, press_release);
254         }
255         else
256                 fprintf(stderr, "unrecognized KeySym: %s\n", keySymName);
257 }
258
259 void Translation::add_string(char *&str)
260 {
261         int delim = *str++;
262         if( !delim ) return;
263         while( str && *str && *str!=delim ) {
264                 if( *str < ' ' || *str > '~' ) continue;
265                 int mask = KeySymMapping::get_mask((const char *&)str);
266                 if( str[0] == '\\' && str[1] ) ++str;
267                 add_keysym(SKeySym(*str++, mask), PRESS_RELEASE);
268                 mask = 0;
269         }
270         if( *str == delim ) ++str;
271 }
272
273 int Translation::start_line(const char *key)
274 {
275         pressed_strokes = 0;
276         released_strokes = 0;
277         pressed = released = 0;
278         is_key = 0;
279         if( !strcasecmp("JL", key) ) {
280                 pressed = &jog[0];
281         }
282         else if( !strcasecmp("JR", key) ) {
283                 pressed = &jog[1];
284         }
285         else {
286                 char c = 0;  int k = -1, n = 0;
287                 if( sscanf(key, "%c%d%n", &c, &k, &n) != 2 ) return 1;
288                 switch( c ) {
289                 case 'K': case 'k':
290                         --k;
291                         if( k >= K1 && k <= K15 ) {
292                                 pressed = &key_down[k];
293                                 released = &key_up[k];
294                                 is_key = 1;
295                         }
296                         break;
297                 case 'S': case 's':
298                         if( k >= S_7 && k <= S7 ) {
299                                 pressed = &shuttles[k-S_7];
300                         }
301                         break;
302                 }
303                 if( !pressed ) {
304                         fprintf(stderr, "bad key name: [%s]%s\n", name, key);
305                         return 1;
306                 }
307                 if( pressed->first ) {
308                         fprintf(stderr, "dupl key name: [%s]%s\n", name, key);
309                         return 1;
310                 }
311         }
312         pressed_strokes = pressed;
313         released_strokes = released;
314         return 0;
315 }
316
317 void Translation::print_stroke(Stroke *s)
318 {
319         if( !s ) return;
320         const char *cp = KeySymMapping::to_string(s->keysym);
321         if( !cp ) { printf("0x%x", (int)s->keysym); cp = "???"; }
322         printf("%s/%c ", cp, s->press ? 'D' : 'U');
323 }
324
325 void Translation::print_strokes(const char *name, const char *up_dn, Strokes *strokes)
326 {
327         printf("%s[%s]: ", name, up_dn);
328         for( Stroke *s=strokes->first; s; s=s->next )
329                 print_stroke(s);
330         printf("\n");
331 }
332
333 void Translation::finish_line()
334 {
335 //printf("finish_line()\n");
336         if( is_key ) {
337                 add_release(0);
338         }
339         add_release(1);
340 }
341
342 void Translation::print_line(const char *key)
343 {
344         if( is_key ) {
345                 print_strokes(key, "D", pressed);
346                 print_strokes(key, "U", released);
347         }
348         else {
349                 print_strokes(key, "", pressed);
350         }
351         printf("\n");
352 }
353
354 // press values in Modifiers:
355 // PRESS -> down
356 // HOLD -> held
357 // PRESS_RELEASE -> released, but to be re-pressed if necessary
358 // RELEASE -> up
359
360 void Modifiers::mark_as_down(SKeySym sym, int hold)
361 {
362         Modifiers &modifiers = *this;
363         for( int i=0; i<size(); ++i ) {
364                 Stroke &s = modifiers[i];
365                 if( s.keysym == sym ) {
366                         s.press = hold ? HOLD : PRESS;
367                         return;
368                 }
369         }
370         Stroke &s = append();
371         s.keysym = sym;
372         s.press = hold ? HOLD : PRESS;
373 }
374
375 void Modifiers::mark_as_up(SKeySym sym)
376 {
377         Modifiers &modifiers = *this;
378         for( int i=0; i<size(); ++i ) {
379                 Stroke &s = modifiers[i];
380                 if( s.keysym == sym ) {
381                         s.press = RELEASE;
382                         return;
383                 }
384         }
385 }
386
387 void Modifiers::release(int allkeys)
388 {
389         Modifiers &modifiers = *this;
390         for( int i=0; i<size(); ++i ) {
391                 Stroke &s = modifiers[i];
392                 if( s.press == PRESS ) {
393                         trans->append_stroke(s.keysym, 0);
394                         s.press = PRESS_RELEASE;
395                 }
396                 else if( allkeys && s.press == HOLD ) {
397                         trans->append_stroke(s.keysym, 0);
398                         s.press = RELEASE;
399                 }
400         }
401 }
402
403 void Modifiers::re_press()
404 {
405         Modifiers &modifiers = *this;
406         for( int i=0; i<size(); ++i ) {
407                 Stroke &s = modifiers[i];
408                 if( s.press == PRESS_RELEASE ) {
409                         trans->append_stroke(s.keysym, 1);
410                         s.press = PRESS;
411                 }
412         }
413 }
414
415
416 Shuttle::Shuttle(MWindow *mwindow)
417  : Thread(1, 0, 0)
418 {
419         this->mwindow = mwindow;
420
421         fd = -1;
422         wdw = 0;
423         win = 0;
424         msk = 0;
425         rx = ry = 0;
426         wx = wy = 0;
427         jogvalue = 0xffff;
428         shuttlevalue = 0xffff;
429         dev_index = -1;
430
431         done = -1;
432         failed = 0;
433         first_time = 1;
434         tr = 0;
435         debug = 0;
436         usb_direct = 0;
437
438         last_translation = 0;
439         last_focused = 0;
440
441 #ifdef HAVE_SHUTTLE_USB
442         devsh = 0;
443         claimed = -1;
444         last_jog = 0;
445         last_shuttle = 0;
446         last_btns = 0;
447 #endif
448
449         default_translation = new Translation(this);
450         config_path = 0;
451         config_mtime = 0;
452         ev.type = ~0;
453 }
454
455 Shuttle::~Shuttle()
456 {
457         stop();
458         delete default_translation;
459         delete [] config_path;
460 }
461
462 int Shuttle::send_button(unsigned int button, int press)
463 {
464         if( debug )
465                 printf("btn: %u %d\n", button, press);
466         XButtonEvent *b = new XButtonEvent();
467         memset(b, 0, sizeof(*b));
468         b->type = press ? ButtonPress : ButtonRelease;
469         b->time = milliTimeClock();
470         b->send_event = 1;
471         b->display = wdw->top_level->display;
472         b->root = wdw->top_level->rootwin;
473         b->window = win;
474         b->x_root = rx;
475         b->y_root = ry;
476         b->x = wx;
477         b->y = wy;
478         b->state = msk;
479         b->button = button;
480         b->same_screen = 1;
481         wdw->top_level->put_event((XEvent *) b);
482         return 0;
483 }
484 int Shuttle::send_keycode(unsigned key, unsigned msk, int press, int send)
485 {
486         if( debug ) {
487                 const char *cp = !send ? 0 :
488                         KeySymMapping::to_string(SKeySym(key, msk));
489                 if( cp )
490                         printf("key: %s %d\n", cp, press);
491                 else
492                         printf("key: %04x/%04x %d\n", key, msk, press);
493         }
494         XKeyEvent *k = new XKeyEvent();
495         memset(k, 0, sizeof(*k));
496         k->type = press ? KeyPress : KeyRelease;
497         k->time = milliTimeClock();
498         k->send_event = send;
499         k->display = wdw->top_level->display;
500         k->root = wdw->top_level->rootwin;
501         k->window = win;
502         k->x_root = rx;
503         k->y_root = ry;
504         k->x = wx;
505         k->y = wy;
506         k->state = msk;
507         k->keycode = key;
508         k->same_screen = 1;
509         wdw->top_level->put_event((XEvent *) k);
510         return 0;
511 }
512
513 int Shuttle::send_keysym(SKeySym keysym, int press)
514 {
515         return keysym >= XK_Button_1 && keysym <= XK_Scroll_Down ?
516                 send_button((unsigned int)keysym - XK_Button_0, press) :
517                 send_keycode(keysym.key, keysym.msk, press, 1);
518 //      unsigned int keycode = XKeysymToKeycode(wdw->top_level->display, keysym);
519 //      return send_keycode(keycode, press, 0);
520 }
521
522
523 static Stroke *fetch_stroke(Translation *translation, int kjs, int index)
524 {
525         Stroke *ret = 0;
526         if( translation ) {
527                 switch( kjs ) {
528                 default:
529                 case KJS_KEY_DOWN:  ret = translation->key_down[index].first;     break;
530                 case KJS_KEY_UP:    ret = translation->key_up[index].first;       break;
531                 case KJS_JOG:       ret = translation->jog[index].first;          break;
532                 case KJS_SHUTTLE:   ret = translation->shuttles[index-S_7].first; break;
533                 }
534         }
535         return ret;
536 }
537
538 void Shuttle::send_stroke_sequence(int kjs, int index)
539 {
540         if( !wdw ) return;
541         Stroke *s = fetch_stroke(tr, kjs, index);
542         if( !s ) s = fetch_stroke(default_translation, kjs, index);
543         while( s ) {
544                 send_keysym(s->keysym, s->press);
545                 s = s->next;
546         }
547 }
548
549 void Shuttle::key(unsigned short code, unsigned int value)
550 {
551         code -= EVENT_CODE_KEY1;
552         if( code >= NUM_KEYS ) {
553                 fprintf(stderr, "key(%d, %d) out of range\n", code + EVENT_CODE_KEY1, value);
554                 return;
555         }
556         send_stroke_sequence(value ? KJS_KEY_DOWN : KJS_KEY_UP, code);
557 }
558
559
560 void Shuttle::shuttle(int value)
561 {
562         if( value < S_7 || value > S7 ) {
563                 fprintf(stderr, "shuttle(%d) out of range\n", value);
564                 return;
565         }
566         if( value != (int)shuttlevalue ) {
567                 shuttlevalue = value;
568                 send_stroke_sequence(KJS_SHUTTLE, value);
569         }
570 }
571
572 // Due to a bug (?) in the way Linux HID handles the ShuttlePro, the
573 // center position is not reported for the shuttle wheel.       Instead,
574 // a jog event is generated immediately when it returns.        We check to
575 // see if the time since the last shuttle was more than a few ms ago
576 // and generate a shuttle of 0 if so.
577 //
578 // Note, this fails if jogvalue happens to be 0, as we don't see that
579 // event either!
580 void Shuttle::jog(unsigned int value)
581 {
582         if( jogvalue != 0xffff ) {
583                 value = value & 0xff;
584                 int direction = ((value - jogvalue) & 0x80) ? -1 : 1;
585                 int index = direction > 0 ? 1 : 0;
586                 while( jogvalue != value ) {
587                         // driver fails to send an event when jogvalue == 0
588                         if( jogvalue != 0 ) {
589                                 send_stroke_sequence(KJS_JOG, index);
590                         }
591                         jogvalue = (jogvalue + direction) & 0xff;
592                 }
593         }
594         jogvalue = value;
595 }
596
597 void Shuttle::jogshuttle(unsigned short code, unsigned int value)
598 {
599         switch( code ) {
600         case EVENT_CODE_JOG:
601                 jog(value);
602                 break;
603         case EVENT_CODE_SHUTTLE:
604                 shuttle(value);
605                 break;
606         case EVENT_CODE_HI_JOG:  // redundant report of JOG value*120
607                 break;
608         default:
609                 fprintf(stderr, "jogshuttle(%d, %d) invalid code\n", code, value);
610                 break;
611         }
612 }
613
614 static const struct shuttle_dev {
615         const char *path;
616         unsigned vendor, product;
617 } shuttle_devs[] = {
618         { "/dev/input/by-id/usb-Contour_Design_ShuttleXpress-event-if00",
619                 0x0b33, 0x0020 },
620         { "/dev/input/by-id/usb-Contour_Design_ShuttlePRO_v2-event-if00",
621                 0x0b33, 0x0030 },
622         { "/dev/input/by-id/usb-Contour_Design_ShuttlePro-event-if00",
623                 0x0b33, 0x0030 },
624         { "/dev/input/by-id/usb-Contour_Design_ShuttlePRO_v2-event-joystick",
625                 0x0b33, 0x0030 },
626 };
627
628 #ifdef HAVE_SHUTTLE_USB
629 void Shuttle::usb_probe(int idx)
630 {
631         int ret = libusb_init(0);
632         if( ret < 0 ) return;
633         claimed = 0;
634         const struct shuttle_dev *s = &shuttle_devs[idx];
635         devsh = libusb_open_device_with_vid_pid(0, s->vendor, s->product);
636         if( devsh ) {
637                 int sh_iface = SHUTTLE_INTERFACE;
638                 libusb_detach_kernel_driver(devsh, sh_iface);
639                 ret = libusb_claim_interface(devsh, sh_iface);
640                 if( ret >= 0 ) claimed = 1;
641         }
642         if( !claimed )
643                 usb_done();
644 }
645
646 void Shuttle::usb_done()
647 {
648         if( devsh ) {
649                 if( claimed > 0 ) {
650                         int sh_iface = SHUTTLE_INTERFACE;
651                         libusb_release_interface(devsh, sh_iface);
652                         libusb_attach_kernel_driver(devsh, sh_iface);
653                         claimed = 0;
654                 }
655                 libusb_close(devsh);
656                 devsh = 0;
657         }
658         if( claimed >= 0 ) {
659                 libusb_exit(0);
660                 claimed = -1;
661         }
662 }
663 #endif
664
665 int Shuttle::probe()
666 {
667         struct stat st;
668         int ret = sizeof(shuttle_devs) / sizeof(shuttle_devs[0]);
669         while( --ret >= 0 && stat(shuttle_devs[ret].path , &st) );
670         return ret;
671 }
672
673 void Shuttle::start(int idx)
674 {
675         this->dev_index = idx;
676         first_time = 1;
677         done = 0;
678         Thread::start();
679 }
680
681 void Shuttle::stop()
682 {
683         if( running() && !done ) {
684                 done = 1;
685                 cancel();
686                 join();
687         }
688 }
689
690 BC_WindowBase *Shuttle::owns(BC_WindowBase *wdw, Window win)
691 {
692         if( wdw->win == win ) return wdw;
693         if( (wdw=wdw->top_level)->win == win ) return wdw;
694         for( int i=wdw->popups.size(); --i>=0; )
695                 if( wdw->popups[i]->win == win ) return wdw;
696         return 0;
697 }
698
699 int Shuttle::get_focused_window_translation()
700 {
701         MWindowGUI *gui = mwindow->gui;
702         Display *dpy = gui->display;
703         Window focus = 0;
704         int ret = 0, revert = 0;
705         char win_title[BCTEXTLEN];  win_title[0] = 0;
706         gui->lock_window("Shuttle::get_focused_window_translation");
707         XGetInputFocus(dpy, &focus, &revert);
708         if( last_focused != focus ) {
709                 last_focused = focus;
710                 Atom prop = XInternAtom(dpy, "WM_NAME", False);
711                 Atom type;  int form;
712                 unsigned long remain, len;
713                 unsigned char *list;
714                 if( XGetWindowProperty(dpy, focus, prop, 0, sizeof(win_title)-1, False,
715                         AnyPropertyType, &type, &form, &len, &remain, &list) == Success ) {
716                         len = len*(form/8) - remain;
717                         memcpy(win_title, list, len);
718                         win_title[len] = 0;
719                         XFree(list);
720                         if( debug )
721                                 printf("new focus: %08x %s\n", (unsigned)focus, win_title);
722                 }
723                 else {
724                         last_focused = 0;
725                         fprintf(stderr, "XGetWindowProperty failed for window 0x%x\n",
726                                         (int)focus);
727                         ret = 1;
728                 }
729         }
730         gui->unlock_window();
731         if( ret ) return -1;
732
733         this->wdw = 0;  this->win = 0;
734         this->wx = 0;   this->wy = 0;
735         this->rx = 0;   this->ry = 0;
736         this->msk = 0;
737         BC_WindowBase *wdw = 0;
738         int cin = -1;
739         if( (wdw=owns(mwindow->gui, focus)) != 0 )
740                 cin = FOCUS_MWINDOW;
741         else if( (wdw=owns(mwindow->awindow->gui, focus)) != 0 )
742                 cin = FOCUS_AWINDOW;
743         else if( (wdw=owns(mwindow->cwindow->gui, focus)) != 0 ) {
744                 if( mwindow->cwindow->gui->canvas->get_fullscreen() )
745                         wdw = mwindow->cwindow->gui->canvas->get_canvas();
746                 cin = FOCUS_CWINDOW;
747         }
748         else if( mwindow->gui->mainmenu->load_file->thread->running() &&
749                  (wdw=mwindow->gui->mainmenu->load_file->thread->window) != 0 &&
750                  wdw->win == focus )
751                 cin = FOCUS_LOAD;
752         else {
753                 int i = mwindow->vwindows.size();
754                 while( --i >= 0 ) {
755                         VWindow *vwdw =  mwindow->vwindows[i];
756                         if( !vwdw->is_running() ) continue;
757                         if( (wdw=owns(vwdw->gui, focus)) != 0 ) {
758                                 if( vwdw->gui->canvas->get_fullscreen() )
759                                         wdw = vwdw->gui->canvas->get_canvas();
760                                 cin = FOCUS_VIEWER;  break;
761                         }
762                 }
763         }
764         if( cin < 0 ) return -1;
765         Window root = 0, child = 0;
766         int root_x = 0, root_y = 0, win_x = 0, win_y = 0, x = 0, y = 0;
767         unsigned int mask = 0, width = 0, height = 0, border_width = 0, depth = 0;
768         wdw->lock_window("Shuttle::get_focused_window_translation 1");
769         if( XQueryPointer(wdw->top_level->display, focus, &root, &child,
770                         &root_x, &root_y, &win_x, &win_y, &mask) ) {
771                 if( !child ) {
772                         if( wdw->active_menubar )
773                                 child = wdw->active_menubar->win;
774                         else if( wdw->active_popup_menu )
775                                 child = wdw->active_popup_menu->win;
776                         else if( wdw->active_subwindow )
777                                 child = wdw->active_subwindow->win;
778                         else
779                                 child = wdw->win;
780                 }
781                 else
782                         XGetGeometry(wdw->top_level->display, child, &root, &x, &y,
783                                 &width, &height, &border_width, &depth);
784         }
785         wdw->unlock_window();
786         if( !child || !wdw->match_window(child) ) return -1;
787 // success
788         this->wdw = wdw;
789         this->win = child;
790         this->msk = mask;
791         this->rx = root_x;
792         this->ry = root_y;
793         this->wx = win_x - x;
794         this->wy = win_y - y;
795         for( tr=translations.first; tr; tr=tr->next ) {
796                 if( tr->is_default ) return 1;
797                 for( int i=0; i<tr->names.size(); ++i ) {
798                         TransName *name = tr->names[i];
799                         if( name->cin == cin ) return 1;
800                 }
801         }
802         tr = default_translation;
803         return 0;
804 }
805
806 int Shuttle::load_translation()
807 {
808         if( read_config_file() > 0 )
809                 return done = 1;
810         if( get_focused_window_translation() < 0 )
811                 return 0;
812         if( last_translation != tr ) {
813                 last_translation = tr;
814                 if( debug )
815                         printf("new translation: %s\n", tr->name);
816         }
817         return 0;
818 }
819
820 void Shuttle::handle_event()
821 {
822         if( load_translation() ) return;
823 //      if( debug )
824 //              printf("event: (%d, %d, 0x%x)\n", ev.type, ev.code, ev.value);
825         switch( ev.type ) {
826         case EVENT_TYPE_DONE:
827         case EVENT_TYPE_ACTIVE_KEY:
828                 break;
829         case EVENT_TYPE_KEY:
830                 key(ev.code, ev.value);
831                 break;
832         case EVENT_TYPE_JOGSHUTTLE:
833                 jogshuttle(ev.code, ev.value);
834                 break;
835         default:
836                 fprintf(stderr, "handle_event() invalid type code\n");
837                 break;
838         }
839 }
840
841 void Shuttle::run()
842 {
843         if( dev_index < 0 ) return;
844         const char *dev_name = shuttle_devs[dev_index].path;
845
846 #ifdef HAVE_SHUTTLE_USB
847         if( usb_direct )
848                 usb_probe(dev_index);
849
850         disable_cancel();
851         while( devsh && !done ) {
852                 int len = 0;
853                 static const int IN_ENDPOINT = 0x81;
854                 unsigned char dat[5];
855                 int ret = libusb_interrupt_transfer(devsh,
856                                 IN_ENDPOINT, dat, sizeof(dat), &len, 100);
857                 if( ret != 0 ) {
858                         if( ret == LIBUSB_ERROR_TIMEOUT ) continue;
859                         printf("shuttle: %s\n  %s\n",
860                                 dev_name, libusb_strerror((libusb_error)ret));
861                         sleep(1);  continue;
862                 }
863                 if( load_translation() ) break;
864                 if( debug ) {
865                         printf("shuttle: ");
866                         for( int i=0; i<len; ++i ) printf(" %02x", dat[i]);
867                         printf("\n");
868                 }
869                 if( last_shuttle != dat[0] )
870                         shuttle((char)(last_shuttle = dat[0]));
871
872                 if( last_jog != dat[1] )
873                         jog(last_jog = dat[1]);
874
875                 unsigned btns = (dat[4]<<8) | dat[3];
876                 unsigned dif = last_btns ^ btns;
877                 if( dif ) {
878                         last_btns = btns;
879                         for( int i=0; i<15; ++i ) {
880                                 unsigned msk = 1 << i;
881                                 if( !(dif & msk) ) continue;
882                                 key(i+EVENT_CODE_KEY1, btns & msk ? 1 : 0);
883                         }
884                 }
885         }
886         usb_done();
887 #endif
888         usb_direct = 0;
889         enable_cancel();
890
891         for( ; !done; sleep(1) ) {
892                 fd = open(dev_name, O_RDONLY);
893                 if( fd < 0 ) {
894                         perror(dev_name);
895                         if( first_time ) break;
896                         continue;
897                 }
898                 if( 1 || !ioctl(fd, EVIOCGRAB, 1) ) { // exclusive access
899                         first_time = 0;
900                         while( !done ) {
901                                 int ret = read(fd, &ev, sizeof(ev));
902                                 if( done ) break;
903                                 if( ret != sizeof(ev) ) {
904                                         if( ret < 0 ) { perror("read event"); break; }
905                                         fprintf(stderr, "bad read: %d\n", ret);
906                                         break;
907                                 }
908                                 handle_event();
909                         }
910                 }
911                 else
912                         perror( "evgrab ioctl" );
913                 close(fd);
914         }
915         done = 2;
916 }
917
918 int Shuttle::read_config_file()
919 {
920         if( !config_path ) {
921                 const char *env;
922                 config_path = (env=getenv("SHUTTLE_CONFIG_FILE")) != 0 ? cstrdup(env) :
923                         (env=getenv("HOME")) != 0 ? cstrcat(2, env, "/.shuttlerc") : 0;
924                 if( !config_path ) { fprintf(stderr, "no config file\n");  return 1; }
925         }
926         struct stat st;
927         if( stat(config_path, &st) ) {
928                 perror(config_path);
929                 char shuttlerc[BCTEXTLEN];
930                 snprintf(shuttlerc, sizeof(shuttlerc), "%s/shuttlerc",
931                                 File::get_cindat_path());
932                 if( stat(shuttlerc, &st) ) {
933                         perror(shuttlerc);
934                         return 1;
935                 }
936                 delete [] config_path;
937                 config_path = cstrdup(shuttlerc);
938         }
939         if( config_mtime > 0 &&
940             config_mtime == st.st_mtime ) return 0;
941         FILE *fp = fopen(config_path, "r");
942         if( !fp ) {
943                 perror(config_path);
944                 return 1;
945         }
946
947         config_mtime = st.st_mtime;
948         translations.clear();
949         debug = 0;
950         usb_direct = 0;
951 #define ws(ch) (ch==' ' || ch=='\t')
952         char line[BCTEXTLEN], *cp;
953         Translation *trans = 0;
954         int no = 0;
955         int ret = fgets(cp=line,sizeof(line),fp) ? 0 : -1;
956         if( !ret ) ++no;
957 // lines
958         while( !ret ) {
959 // translation names
960                 while( !ret && *cp == '[' ) {
961                         char *name = ++cp;
962                         while( *cp && *cp != ']' ) ++cp;
963                         *cp++ = 0;
964                         if( !name || !*name ) { ret = 1;  break; }
965                         int cin = -1;
966                         if( !strcasecmp("default", name) )
967                                 cin = FOCUS_DEFAULT;
968                         else if( !strcasecmp("cinelerra", name) )
969                                 cin = FOCUS_MWINDOW;
970                         else if( !strcasecmp("resources", name) )
971                                 cin = FOCUS_AWINDOW;
972                         else if( !strcasecmp("composer", name) )
973                                 cin = FOCUS_CWINDOW;
974                         else if( !strcasecmp("viewer", name) )
975                                 cin = FOCUS_VIEWER;
976                         else if( !strcasecmp("load", name) )
977                                 cin = FOCUS_LOAD;
978                         else {
979                                 fprintf(stderr, "unknown focus target window: %s\n",
980                                          name);
981                                 ret = 1;  break;
982                         }
983                         if( !trans ) {
984                                 if( cin == FOCUS_DEFAULT ) {
985                                         trans = default_translation;
986                                         trans->clear();
987                                 }
988                                 else {
989                                         trans = new Translation(this, name);
990                                 }
991                         }
992                         while( ws(*cp) ) ++cp;
993                         trans->names.append(new TransName(cin, name, cp));
994                         ret = fgets(cp=line,sizeof(line),fp) ? 0 : 1;
995                         if( ret ) {
996                                 fprintf(stderr, "hit eof, no translation def for: %s\n",
997                                         trans->names.last()->name);
998                                 ret = 1;  break;
999                         }
1000                         ++no;
1001                 }
1002                 if( ret ) break;
1003                 if( debug && trans ) {
1004                         printf("------------------------\n");
1005                         TransNames &names = trans->names;
1006                         for( int i=0; i<names.size(); ++i ) {
1007                                 TransName *tp = names[i];
1008                                 printf("[%s] # %d\n\n", tp->name, tp->cin);
1009                         }
1010                 }
1011 // rules lines: "tok <stroke list>\n"
1012                 while( !ret && *cp != '[' ) {
1013                         const char *key = 0, *tok = 0;
1014                         while( ws(*cp) ) ++cp;
1015                         if( !*cp || *cp == '\n' || *cp == '#' ) goto skip;
1016                         tok = cp;
1017                         while( *cp && !ws(*cp) && *cp != '\n' ) ++cp;
1018                         *cp++ = 0;
1019                         if( !strcmp(tok, "DEBUG") ) {
1020                                 debug = 1;  goto skip;
1021                         }
1022                         if( !strcmp(tok, "USB_DIRECT") ) {
1023                                 usb_direct = 1;  goto skip;
1024                         }
1025                         key = tok;
1026                         if( !trans ) {
1027                                 fprintf(stderr, "no translation section defining key: %s\n", key);
1028                                 ret = 1;  break;
1029                         }
1030         
1031                         ret = trans->start_line(key);
1032                         while( !ret && *cp && *cp != '\n' ) {
1033                                 while( ws(*cp) ) ++cp;
1034                                 if( !*cp || *cp == '#' || *cp == '\n' ) break;
1035                                 if( *cp == '"' ) {
1036                                         while( *cp ) {
1037                                                 if( *cp == '"' )
1038                                                         trans->add_string(cp);
1039                                                 while( ws(*cp) ) ++cp;
1040                                                 if( !*cp || *cp == '#' || *cp == '\n' ) break;
1041                                                 tok = cp;
1042                                                 while( *cp && !ws(*cp) && *cp != '\n' ) ++cp;
1043                                                 *cp = 0;
1044                                                 SKeySym sym = KeySymMapping::to_keysym(tok);
1045                                                 if( !sym ) {
1046                                                         fprintf(stderr, "unknown keysym: %s\n", tok);
1047                                                         ret = 1;  break;
1048                                                 }
1049                                                 trans->add_keysym(sym, PRESS_RELEASE);
1050                                         }
1051                                         continue;
1052                                 }
1053                                 tok = cp;
1054                                 while( *cp && !ws(*cp) && *cp!='/' && *cp != '\n' ) ++cp;
1055                                 int dhu = PRESS_RELEASE;
1056                                 if( *cp == '/' ) {
1057                                         *cp++ = 0;
1058                                         switch( *cp ) {
1059                                         case 'D':  dhu = PRESS;    break;
1060                                         case 'H':  dhu = HOLD;     break;
1061                                         case 'U':  dhu = RELEASE;  break;
1062                                         default:
1063                                                 fprintf(stderr, "invalid up/down modifier [%s]%s: '%c'\n",
1064                                                         trans->name, tok, *cp);
1065                                                 ret = 1;  break;
1066                                         }
1067                                         ++cp;
1068                                 }
1069                                 else
1070                                         *cp++ = 0;
1071                                 trans->add_keystroke(tok, dhu);
1072                         }
1073                         if( ret ) break;
1074                         trans->finish_line();
1075                         if( debug )
1076                                 trans->print_line(key);
1077                 skip:   ret = fgets(cp=line,sizeof(line),fp) ? 0 : -1;
1078                         if( !ret ) ++no;
1079                 }
1080                 if( trans ) {
1081                         if( trans != default_translation )
1082                                 translations.append(trans);
1083                         trans = 0;
1084                 }
1085         }
1086         if( ret > 0 )
1087                 fprintf(stderr, "shuttle config err file: %s, line:%d\n",
1088                         config_path, no);
1089
1090         fclose(fp);
1091         return ret;
1092 }
1093 #endif