Blame keyboard.c

452870
452870
#include "app.h"
452870
452870
452870
843e7a
int keyInit(Key *k, Layout *l) {
8864eb
  CLEARFROM(k, l);
843e7a
  k->l = l;
63daec
  
63daec
  inputPrepareKeysyms(&k->keySym, &k->keySym2, &k->isLetter, &k->isKeypad);
63daec
  if (!k->label[0]) keysymString(k->keySym, k->label);
63daec
  if (!k->label2[0]) keysymString(k->keySym2, k->label2);
63daec
  
843e7a
  textLayoutInit(&k->tl, &k->l->kbd->app->graph, k->label);
061fcf
  textLayoutInit(&k->tl2, &k->l->kbd->app->graph, k->label2);
452870
  
452870
  return 1;
452870
}
452870
452870
452870
void keyDeinit(Key *k) {
843e7a
  keyUp(k, 1);
843e7a
  textLayoutDeinit(&k->tl);
843e7a
  textLayoutDeinit(&k->tl2);
452870
}
452870
452870
  
d942fc
void keyInvalidateRect(Key* k) {
d942fc
  if (k->orig) k = k->orig;
d942fc
  while(k) {
d942fc
    appInvalidateRect(k->l->kbd->app, k->x, k->y, k->w, k->h);
d942fc
    k = k->clone;
d942fc
  }
d942fc
}
d942fc
d942fc
452870
void keyDraw(Key *k, int cx, int cy, int cw, int ch) {
452870
  if ( k->x + k->w <= cx || k->x >= cx + cw
452870
    || k->y + k->h <= cy || k->y >= cy + ch ) return;
843e7a
  
63daec
  App *app = k->l->kbd->app;
63daec
  
63daec
  int active = k->down;
5b6cfc
  int longActive = k->longDown;
5b6cfc
  if (k->orig) {
d942fc
    active = k->orig->down;
5b6cfc
    longActive = k->orig->longDown;
5b6cfc
  }
5b6cfc
  
5b6cfc
  active = !!active;
5b6cfc
  if (active && longActive) active = 2; 
5b6cfc
  
569832
  if (!active && (k->flags & KF_MOD))
5b6cfc
    active = !!(app->input.modifiers & (unsigned int)k->optValue);
843e7a
  
db2d04
  int w = k->w - 2*KEY_BORDER;
db2d04
  int h = k->h - 2*KEY_BORDER;
37122c
  int highlight = !!(k->flags & KF_HIGHLIGHT);
37122c
  graphDrawButton(&k->l->kbd->app->graph, k->x + KEY_BORDER, k->y + KEY_BORDER, w, h, active, highlight);
843e7a
  
4b2399
  TextLayout *tl = &k->tl;
843e7a
  TextLayout *tl2 = &k->tl2;
63daec
  if (inputChooseKeysym(app->input.modifiers, k->keySym, k->keySym2, k->isLetter, k->isKeypad))
63daec
    tl = tl2, tl2 = &k->tl; // swap labels
843e7a
  
dbe1a1
  int lx, lx2, ly, ly2, lw, lw2, lh, lh2;
db2d04
  lx = lx2 = k->x + w/2;
db2d04
  ly = ly2 = k->y + h/2;
db2d04
  lw = lw2 = w - w/8;
db2d04
  lh = lh2 = h - h/8;
061fcf
  if (!k->isLetter && k->label2[0]) {
db2d04
    lx -= w/8;
db2d04
    ly += h/8;
db2d04
    lw -= w/4;
db2d04
    lh -= h/4;
dbe1a1
    
db2d04
    lx2 += w/4;
db2d04
    ly2 -= h/4;
db2d04
    lw2 -= w/2;
db2d04
    lh2 -= h/2;
843e7a
  }
061fcf
  
37122c
  textLayoutDraw(tl, lx, ly, lw, lh, active, highlight);
061fcf
  if (!k->isLetter)
37122c
    textLayoutDraw(tl2, lx2, ly2, lw2, lh2, active, highlight);
452870
}
452870
452870
843e7a
void keyDown(Key *k, int x, int y) {
d942fc
  if (k->orig)
d942fc
    return keyDown(k->orig, x, y);
843e7a
  if (k->down == 2) k->down = 1;
843e7a
  if (k->down) return;
843e7a
  
843e7a
  Keyboard *kbd = k->l->kbd;
452870
  k->down = 1;
843e7a
  
63daec
  if (k->flags & KF_MOVE) {
63daec
    k->mx = x;
63daec
    k->my = y;
63daec
  } else
63daec
  if (k->flags & KF_SIZE) {
63daec
    k->mx = kbd->app->w - x;
63daec
    k->my = kbd->app->h - y;
843e7a
  }
63daec
  
843e7a
  if (k->flags & KF_HOLD) {
843e7a
    k->down = 2;
843e7a
    // insert to held list
843e7a
    if (!k->prevHeld && !k->nextHeld && kbd->firstHeld != k) {
843e7a
      k->nextHeld = kbd->firstHeld;
843e7a
      if (k->nextHeld) k->nextHeld->prevHeld = k;
843e7a
      kbd->firstHeld = k;
843e7a
    }
843e7a
  }
843e7a
  
843e7a
  // choose keySym
63daec
  unsigned int keySym = inputChooseKeysym(kbd->app->input.modifiers, k->keySym, k->keySym2, k->isLetter, k->isKeypad)
63daec
                      ? k->keySym2 : k->keySym;
843e7a
  
63daec
  // press
843e7a
  if (keySym) {
843e7a
    k->downKeycode = inputKeycode(&kbd->app->input, keySym);
843e7a
    if (k->downKeycode) inputEvent(&kbd->app->input, k->downKeycode, 1);
843e7a
  }
843e7a
  
843e7a
  // redraw
d942fc
  keyInvalidateRect(k);
843e7a
}
843e7a
843e7a
843e7a
void keyMotion(Key *k, int x, int y) {
d942fc
  if (k->orig)
d942fc
    return keyMotion(k->orig, x, y);
843e7a
  App *app = k->l->kbd->app;
843e7a
  if (k->flags & KF_MOVE) {
63daec
    appMove(app, app->x + x - k->mx, app->y + y - k->my, app->w, app->h);
843e7a
  } else
843e7a
  if (k->flags & KF_SIZE) {
63daec
    appMove(app, app->x, app->y, x + k->mx, y + k->my);
843e7a
  }
452870
}
452870
452870
5b6cfc
void keyLongDown(Key *k) {
5b6cfc
  if (k->orig)
5b6cfc
    return keyLongDown(k->orig);
5b6cfc
  if (k->down != 2 || k->longDown)
5b6cfc
    return;
5b6cfc
5b6cfc
  // remove from held list, this key will not unhold automatically
5b6cfc
  if (k->prevHeld || k->nextHeld || k->l->kbd->firstHeld == k) {
5b6cfc
    if (k->nextHeld) k->nextHeld->prevHeld = k->prevHeld;
5b6cfc
    if (k->prevHeld) k->prevHeld->nextHeld = k; else k->l->kbd->firstHeld = k->nextHeld;
5b6cfc
    k->nextHeld = k->prevHeld = NULL;
5b6cfc
  }
5b6cfc
  
5b6cfc
  LOGDBG("key: long down: activated");
5b6cfc
  k->longDown = 1;
5b6cfc
  keyInvalidateRect(k);
5b6cfc
}
5b6cfc
5b6cfc
843e7a
void keyUp(Key *k, int force) {
d942fc
  if (k->orig) {
d942fc
    // clones must not be in held list
d942fc
    assert( !(k->prevHeld || k->nextHeld || k->l->kbd->firstHeld == k) );
d942fc
    return keyUp(k->orig, force);
d942fc
  }
d942fc
  if (!force && k->down != 1)
d942fc
    return;
843e7a
  
843e7a
  Keyboard *kbd = k->l->kbd;
843e7a
  int wasDown = k->down;
452870
  k->down = 0;
5b6cfc
  k->longDown = 0;
843e7a
  
843e7a
  // release key
843e7a
  if (k->downKeycode) {
843e7a
    inputEvent(&kbd->app->input, k->downKeycode, 0);
843e7a
    k->downKeycode = 0;
843e7a
  }
843e7a
  
d942fc
  
843e7a
  // remove from held list
843e7a
  if (k->prevHeld || k->nextHeld || kbd->firstHeld == k) {
843e7a
    if (k->nextHeld) k->nextHeld->prevHeld = k->prevHeld;
843e7a
    if (k->prevHeld) k->prevHeld->nextHeld = k; else kbd->firstHeld = k->nextHeld;
843e7a
    k->nextHeld = k->prevHeld = NULL;
843e7a
  }
63daec
63daec
  // following action should be done only for real button up
63daec
  if (!wasDown)
63daec
    return;
63daec
  
63daec
  if (k->flags & KF_CLOSE)
63daec
    appStop(kbd->app, 0);
63daec
  if (k->flags & KF_LAYOUT)
63daec
    keyboardSwitchLayout(kbd, k->optValue);
843e7a
  
843e7a
  // on releasing a regular key release all held special keys
37122c
  if (!force && !(k->flags & KF_SPECIAL))
843e7a
    keyboardRelaseHeld(kbd);
843e7a
  
843e7a
  // redraw
d942fc
  keyInvalidateRect(k);
843e7a
}
843e7a
843e7a
843e7a
843e7a
int layoutInit(Layout *l, Keyboard *kbd) {
843e7a
  LOGDBG("layout: init");
8864eb
  
8864eb
  CLEARFROM(l, kbd);
843e7a
  l->kbd = kbd;
63daec
  
63daec
  if (!l->keysCount)
63daec
    return 1;
63daec
  
37122c
  l->hidden = 0;
37122c
  
63daec
  // fix keys coords
63daec
  Key *k = &l->keys[0];
061fcf
  int x0 = k->ox;
061fcf
  int y0 = k->oy;
061fcf
  int x1 = x0 + k->ow;
061fcf
  int y1 = y0 + k->oh;
63daec
  for(int i = 1; i < l->keysCount; ++i) {
63daec
    k = &l->keys[i];
63daec
    Key *p = k - 1;
63daec
    
63daec
    // coord < 0 means relative offset from previos key -1
63daec
    // zero coord of size means copy value from previous key
061fcf
    if (!k->ox) k->ox = p->ox; else
061fcf
      if (k->ox < 0) k->ox = p->ox + p->ow - k->ox - 1;
061fcf
    if (!k->oy) k->oy = p->oy; else
061fcf
      if (k->oy < 0) k->oy = p->oy + p->oh - k->oy - 1;
061fcf
    if (!k->ow) k->ow = p->ow;
061fcf
    if (!k->oh) k->oh = p->oh;
63daec
    
63daec
    // calc bounds
061fcf
    if (x0 > k->ox) x0 = k->ox;
061fcf
    if (y0 > k->oy) y0 = k->oy;
061fcf
    if (x1 < k->ox + k->ow) x1 = k->ox + k->ow;
061fcf
    if (y1 < k->oy + k->oh) y1 = k->oy + k->oh;
37122c
    
37122c
    if ((k->flags & KF_LAYOUT) && k->optValue == LI_REVERT)
37122c
      l->hidden = 1;
63daec
  }
63daec
    
63daec
  // fix layout size
061fcf
  if (!l->ow) l->ow = x1 + x0;
061fcf
  if (!l->oh) l->oh = y1 + y0;
37122c
  LOGDBG("layout: init: size x0=%d y0=%d x1=%d y1=%d w=%d h=%d, hedden: %d", x0, y0, x1, y1, l->ow, l->oh, l->hidden);
63daec
63daec
  // init keys
061fcf
  for(int i = 0; i < l->keysCount; ++i) {
843e7a
    if (!keyInit(&l->keys[i], l)) {
843e7a
      for(--i; i >= 0; --i) keyDeinit(&l->keys[i]);
843e7a
      return 0;
843e7a
    }
061fcf
  }
061fcf
  
843e7a
  return 1;
843e7a
}
843e7a
843e7a
843e7a
void layoutDeinit(Layout *l) {
843e7a
  LOGDBG("layout: deinit");
843e7a
  layoutMouseUp(l);
843e7a
  for(int i = 0; i < l->keysCount; ++i)
843e7a
    keyDeinit(&l->keys[i]);
843e7a
}
843e7a
843e7a
061fcf
void layoutResize(Layout *l) {
061fcf
  if (l->w == l->kbd->app->w && l->h == l->kbd->app->h) return;
061fcf
  LOGDBG("layout: resize w=%d h=%d", l->kbd->app->w, l->kbd->app->h);
061fcf
  l->w = l->kbd->app->w;
061fcf
  l->h = l->kbd->app->h;
061fcf
  float kx = l->w/(float)l->ow;
061fcf
  float ky = l->h/(float)l->oh;
061fcf
  for(int i = 0; i < l->keysCount; ++i) {
061fcf
    Key *k = &l->keys[i];
061fcf
    k->x = (int)(k->ox*kx + 0.5f);
061fcf
    k->y = (int)(k->oy*ky + 0.5f);
4b2399
    k->w = (int)((k->ox + k->ow)*kx + 0.5f) - k->x;
4b2399
    k->h = (int)((k->oy + k->oh)*ky + 0.5f) - k->y;
061fcf
  }
061fcf
}
061fcf
061fcf
843e7a
void layoutDraw(Layout *l, int cx, int cy, int cw, int ch) {
843e7a
  for(int i = 0; i < l->keysCount; ++i)
843e7a
    keyDraw(&l->keys[i], cx, cy, cw, ch);
843e7a
}
843e7a
843e7a
843e7a
void layoutMouseDown(Layout *l, int x, int y) {
843e7a
  LOGDBG("layout: mouse down: x=%d y=%d", x, y);
843e7a
  for(int i = l->keysCount - 1; i >= 0; --i) {
843e7a
    Key *k = &l->keys[i];
843e7a
    if ( k->x <= x && x < k->x + k->w
843e7a
      && k->y <= y && y < k->y + k->h )
843e7a
    {
843e7a
      if (l->downKey != k) {
843e7a
        layoutMouseUp(l);
843e7a
        l->downKey = k;
843e7a
        keyDown(k, x, y);
843e7a
      }
843e7a
      break;
843e7a
    }
843e7a
  }
843e7a
}
843e7a
843e7a
843e7a
void layoutMouseMotion(Layout *l, int x, int y) {
843e7a
  LOGDBG("layout: mouse motion: x=%d y=%d", x, y);
843e7a
  if (!l->downKey) return;
843e7a
  keyMotion(l->downKey, x, y);
843e7a
}
843e7a
843e7a
5b6cfc
void layoutMouseLongDown(Layout *l) {
5b6cfc
  LOGDBG("layout: mouse long down");
5b6cfc
  if (!l->downKey) return;
5b6cfc
  keyLongDown(l->downKey);
5b6cfc
}
5b6cfc
5b6cfc
843e7a
void layoutMouseUp(Layout *l) {
843e7a
  LOGDBG("layout: mouse up");
843e7a
  if (!l->downKey) return;
843e7a
  keyUp(l->downKey, 0);
843e7a
  l->downKey = NULL;
452870
}
452870
452870
452870
452870
int keyboardInit(Keyboard *kbd, App *app) {
452870
  LOGDBG("keyboard: init");
8864eb
  
8864eb
  CLEARFROM(kbd, app);
452870
  kbd->app = app;
8864eb
  
061fcf
  kbd->lastModifiers = kbd->app->input.modifiers;
843e7a
  kbd->current = kbd->layoutsCount > 0 ? kbd->layouts : NULL;
63daec
  
843e7a
  for(int i = 0; i < kbd->layoutsCount; ++i) {
843e7a
    if (!layoutInit(&kbd->layouts[i], kbd)) {
843e7a
      for(--i; i >= 0; --i) layoutDeinit(&kbd->layouts[i]);
452870
      return 0;
452870
    }
843e7a
  }
63daec
63daec
  // fix size
63daec
  for(int i = 0; i < kbd->layoutsCount; ++i) {
061fcf
    if (kbd->ow < kbd->layouts[i].ow) kbd->ow = kbd->layouts[i].ow;
061fcf
    if (kbd->oh < kbd->layouts[i].oh) kbd->oh = kbd->layouts[i].oh;
63daec
  }
061fcf
  LOGDBG("keyboard: init: size ow=%d oh=%d", kbd->ow, kbd->oh);
63daec
  
d942fc
  // search clones
d942fc
  for(int ila = 0; ila < kbd->layoutsCount; ++ila)
d942fc
  for(int ika = 0; ika < kbd->layouts[ila].keysCount; ++ika) {
d942fc
    Key *a = &kbd->layouts[ila].keys[ika];
d942fc
    
d942fc
    if (!a->orig)
d942fc
    for(int ilb = ila; ilb < kbd->layoutsCount; ++ilb)
d942fc
    for(int ikb = ilb==ila ? ika+1: 0; ikb < kbd->layouts[ilb].keysCount; ++ikb) {
d942fc
      Key *b = &kbd->layouts[ilb].keys[ikb];
d942fc
      if ( a->flags    == b->flags
d942fc
        && a->optValue == b->optValue
d942fc
        && a->keySym   == b->keySym
d942fc
        && a->keySym2  == b->keySym2 )
d942fc
      {
d942fc
        b->orig = a;
d942fc
        b->clone = a->clone;
d942fc
        a->clone = b;
d942fc
      }
d942fc
    }
d942fc
  }
d942fc
  
061fcf
  //appMove(app, app->x, app->y, kbd->ow, kbd->oh);
452870
  return 1;
452870
}
452870
452870
452870
void keyboardDeinit(Keyboard *kbd) {
452870
  LOGDBG("keyboard: deinit");
452870
  keyboardMouseUp(kbd);
843e7a
  for(int i = 0; i < kbd->layoutsCount; ++i)
843e7a
    layoutDeinit(&kbd->layouts[i]);
843e7a
}
843e7a
843e7a
843e7a
void keyboardResize(Keyboard *kbd) {
061fcf
  if (kbd->current)
061fcf
    layoutResize(kbd->current);
061fcf
}
061fcf
061fcf
061fcf
void keyboardUpdateModifiers(Keyboard* kbd) {
061fcf
  if (kbd->lastModifiers == kbd->app->input.modifiers) return;
061fcf
  LOGDBG("keyboard: update modifiers");
061fcf
  kbd->lastModifiers = kbd->app->input.modifiers;
061fcf
  appInvalidateRect(kbd->app, 0, 0, kbd->app->w, kbd->app->h);
452870
}
452870
452870
452870
void keyboardDraw(Keyboard *kbd, int cx, int cy, int cw, int ch) {
843e7a
  if (!kbd->current) return;
843e7a
  layoutDraw(kbd->current, cx, cy, cw, ch);
452870
}
452870
452870
452870
void keyboardMouseDown(Keyboard *kbd, int x, int y) {
843e7a
  LOGDBG("keyboard: mouse down: x=%d y=%d", x, y);
843e7a
  if (!kbd->current) return;
843e7a
  layoutMouseDown(kbd->current, x, y);
843e7a
}
843e7a
843e7a
843e7a
void keyboardMouseMotion(Keyboard *kbd, int x, int y) {
843e7a
  LOGDBG("keyboard: mouse motion: x=%d y=%d", x, y);
843e7a
  if (!kbd->current) return;
843e7a
  layoutMouseMotion(kbd->current, x, y);
452870
}
452870
452870
5b6cfc
void keyboardMouseLongDown(Keyboard *kbd) {
5b6cfc
  LOGDBG("keyboard: mouse long down");
5b6cfc
  if (!kbd->current) return;
5b6cfc
  layoutMouseLongDown(kbd->current);
5b6cfc
}
5b6cfc
5b6cfc
452870
void keyboardMouseUp(Keyboard *kbd) {
843e7a
  LOGDBG("keyboard: mouse up");
843e7a
  if (!kbd->current) return;
843e7a
  layoutMouseUp(kbd->current);
843e7a
}
843e7a
843e7a
843e7a
void keyboardRelaseHeld(Keyboard *kbd) {
843e7a
  LOGDBG("keyboard: release held");
843e7a
  while(kbd->firstHeld) keyUp(kbd->firstHeld, 1);
843e7a
}
843e7a
843e7a
843e7a
void keyboardSwitchLayout(Keyboard *kbd, int index) {
843e7a
843e7a
  if (!kbd->layoutsCount) return;
d942fc
d942fc
  int prevIndex = kbd->current - kbd->layouts;
d942fc
  int remember = 1;
d942fc
d942fc
  LOGDBG("keyboard: switch layout: prevIndex=%d nextIndex=%d", prevIndex, index);
d942fc
  //keyboardRelaseHeld(kbd);
843e7a
843e7a
  if (index == LI_PREV) {
37122c
    for(int i = 1; i <= kbd->layoutsCount; ++i) {
37122c
      index = (prevIndex + kbd->layoutsCount - i) % kbd->layoutsCount;
37122c
      if (!kbd->layouts[index].hidden) break;
37122c
    }
843e7a
  } else
843e7a
  if (index == LI_NEXT) {
37122c
    for(int i = 1; i <= kbd->layoutsCount; ++i) {
37122c
      index = (prevIndex + i) % kbd->layoutsCount;
37122c
      if (!kbd->layouts[index].hidden) break;
37122c
    }
843e7a
  } else
843e7a
  if (index == LI_REVERT) {
37122c
    if (!kbd->current || !kbd->current->prev) {
37122c
      LOGWRN("keyboard: switch layout: cannot revert");
d942fc
      return;
37122c
    }
37122c
    Layout *l = kbd->current->prev;
37122c
    while(l->hidden && l->prev) l = l->prev;
37122c
    index = l - kbd->layouts;
d942fc
    remember = 0;
843e7a
  }
843e7a
  
843e7a
  if (index < 0 || index >= kbd->layoutsCount) {
843e7a
    LOGWRN("keyboard: switch layout: bad index %d", index);
843e7a
    return;
452870
  }
843e7a
  
d942fc
  LOGDBG("keyboard: switch layout: actual index=%d", index);
d942fc
  if (remember)
d942fc
    kbd->layouts[index].prev = kbd->current;
843e7a
  kbd->current = &kbd->layouts[index];
061fcf
  
061fcf
  keyboardResize(kbd);
061fcf
  appInvalidateRect(kbd->app, 0, 0, kbd->app->w, kbd->app->h);
452870
}
843e7a