Blob Blame Raw

#include "app.h"




int appInit(App *app) {
  LOGDBG("app: init");
  app->run = 0;
  app->x = app->y = app->w = app->h = 0;
  app->irx = app->iry = app->irw = app->irh = 0;
  
  LOGDBG("app: init: connect to xcb");
  app->xcb = xcb_connect(NULL, NULL);
  if (!app->xcb)
    return LOGERR("app: init: cannot connect to xcb");
  
  LOGDBG("app: init: get screen");
  app->screen = xcb_setup_roots_iterator(xcb_get_setup(app->xcb)).data;
  if (app->screen) {
    LOGDBG("app: init: create window");
    
    app->x = 0;
    app->w = app->screen->width_in_pixels;
    app->h = app->screen->height_in_pixels/4;
    app->y = app->screen->height_in_pixels - app->h;
    
    #ifdef NOBORDER
    int noborder = 1;
    #else
    int noborder = 0;
    #endif
    
    app->win = xcb_generate_id(app->xcb);
    uint32_t mask = XCB_CW_OVERRIDE_REDIRECT | XCB_CW_EVENT_MASK;
    uint32_t values[] = {
      noborder,                         // override redirect
        XCB_EVENT_MASK_EXPOSURE         // event mask
      | XCB_EVENT_MASK_STRUCTURE_NOTIFY
      | XCB_EVENT_MASK_BUTTON_PRESS
      | XCB_EVENT_MASK_BUTTON_1_MOTION
      | XCB_EVENT_MASK_BUTTON_RELEASE };
    xcb_create_window(
      app->xcb,                      // connection
      XCB_COPY_FROM_PARENT,          // depth (same as root)
      app->win,                      // window id
      app->screen->root,             // parent window
      app->x, app->y,                // origin
      app->w, app->h,                // size
      10,                            // border_width
      XCB_WINDOW_CLASS_INPUT_OUTPUT, // class
      app->screen->root_visual,      // visual
      mask,                          // additional options mask
      values );                      // additional options
    
    uint32_t hints[9] = {
      1, // mask for input hints
      0, // value of input hint
    };
    xcb_change_property_checked(
      app->xcb,
      XCB_PROP_MODE_REPLACE,
      app->win,
      XCB_ATOM_WM_HINTS,
      XCB_ATOM_WM_HINTS,
      32,
      sizeof(hints)/sizeof(hints[0]), hints );
    
    // init submodules
    if (graphInit(&app->graph, app)) {
      graphResize(&app->graph);
      if (inputInit(&app->input, app)) {
        if (keyboardInit(&app->keyboard, app)) {
          return 1;
        }
        inputDeinit(&app->input);
      }
      graphDeinit(&app->graph);
    }
  } else {
    LOGERR("app: init: no screen found");
  }
  
  xcb_disconnect(app->xcb);
  return 0;
}


void appDeinit(App *app) {
  LOGDBG("app: deinit");
  keyboardDeinit(&app->keyboard);
  inputDeinit(&app->input);
  graphDeinit(&app->graph);
  xcb_disconnect(app->xcb);
}


int appRun(App *app) {
  LOGDBG("app: run");

  if (app->run)
    return LOGERR("app: run: seems already run");
  app->run = 1;
  
  LOGDBG("app: run: show window");
  xcb_map_window(app->xcb, app->win);
  xcb_flush(app->xcb);

  LOGDBG("app: run: event loop");
  xcb_generic_event_t *event;
  while ((event = xcb_wait_for_event(app->xcb))) {
    switch (event->response_type & ~0x80) {
    case XCB_EXPOSE: {
      xcb_expose_event_t *e = (xcb_expose_event_t*)event;
      LOGDBG("app: expose: x=%d, y=%d, w=%d, h=%d", e->x, e->y, e->width, e->height);
      appInvalidateRect(app, e->x, e->y, e->width, e->height);
      break;
    }
    case XCB_CONFIGURE_NOTIFY: {
      xcb_configure_notify_event_t *e = (xcb_configure_notify_event_t*)event;
      if ( e->width
        && e->height
        && ( app->x != e->x
          || app->y != e->y
          || app->w != e->width
          || app->h != e->height ))
      {
        LOGDBG("app: moved: x=%d y=%d w=%d, h=%d", e->x, e->y, e->width, e->height);
        int resized = app->w != e->width || app->h != e->height;
        app->x = e->x;
        app->y = e->y;
        app->w = e->width;
        app->h = e->height;
        if (resized) graphResize(&app->graph);
      }
      break;
    }
    case XCB_BUTTON_PRESS: {
      xcb_button_press_event_t *e = (xcb_button_press_event_t*)event;
      if (e->detail == 1) {
        LOGDBG("app: mouse down: x=%d, y=%d", e->event_x, e->event_y);
        keyboardMouseDown(&app->keyboard, e->event_x, e->event_y);
      }
      break;
    }
    case XCB_BUTTON_RELEASE: {
      xcb_button_press_event_t *e = (xcb_button_press_event_t*)event;
      if (e->detail == 1) {
        LOGDBG("app: mouse up");
        keyboardMouseUp(&app->keyboard);
      }
      break;
    }
    case XCB_MOTION_NOTIFY: {
      xcb_motion_notify_event_t *e = (xcb_motion_notify_event_t*)event;
      if (e->state & Button1Mask) {
        LOGDBG("app: mouse motion: x=%d, y=%d", e->event_x, e->event_y);
        keyboardMouseMotion(&app->keyboard, e->event_x, e->event_y);
      }
      break;
    }
    default: 
        break;
    }
    free(event);
    
    if (app->run != 1) break;
    
    inputUpdateModifiers(&app->input);
    keyboardResize(&app->keyboard);
    keyboardUpdateModifiers(&app->keyboard);
    if (app->irw > 0 && app->irh > 0) {
      LOGDBG("app: draw: x=%d, y=%d, w=%d, h=%d", app->irx, app->iry, app->irw, app->irh);
      graphDrawBackgound(&app->graph, app->irx, app->iry, app->irw, app->irh);
      keyboardDraw(&app->keyboard, app->irx, app->iry, app->irw, app->irh);
      xcb_flush(app->xcb);
      app->irw = 0;
    }
  }
  
  int err = app->run == -2;
  app->run = 0;
  LOGDBG("app: run: done err=%d", err);
  return !err;
}


void appStop(App *app, int err) {
  LOGDBG("app: stop: err=%d", err);
  if (!app->run) {
    LOGERR("app: stop: seems not started");
    return;
  }
  if (app->run == 1) app->run = -1;
  if (err) app->run = -2;
}


void appMove(App *app, int x, int y, int w, int h) {
  LOGDBG("app: move: x=%d, y=%d, w=%d, h=%d", x, y, w, h);
  if (w <= 0 || h <= 0) return;
  if (app->x == x && app->y == y && app->w == w && app->h == h)
    return;
  uint32_t mask = XCB_CONFIG_WINDOW_X | XCB_CONFIG_WINDOW_Y | XCB_CONFIG_WINDOW_WIDTH | XCB_CONFIG_WINDOW_HEIGHT;
  uint32_t values[] = { x, y, w, h };
  xcb_configure_window(app->xcb, app->win, mask, values);
  xcb_flush(app->xcb);
}


void appInvalidateRect(App *app, int x, int y, int w, int h) {
  LOGDBG("app: invalidate rect: x=%d, y=%d, w=%d, h=%d", x, y, w, h);
  rectIntersect(&x, &y, &w, &h, 0, 0, app->w, app->h);
  rectMerge(&app->irx, &app->iry, &app->irw, &app->irh, x, y, w, h);
  LOGDBG("app: invalidate rect: summary x=%d, y=%d, w=%d, h=%d", app->irx, app->iry, app->irw, app->irh);
}