/* * File : workbench.c * This file is part of RTGUI in RT-Thread RTOS * COPYRIGHT (C) 2006 - 2009, RT-Thread Development Team * * The license and distribution terms for this file may be * found in the file LICENSE in this distribution or at * http://www.rt-thread.org/license/LICENSE * * Change Logs: * Date Author Notes * 2009-10-04 Bernard first version * 2010-09-24 Bernard fix workbench destroy issue */ #include #include #include static void _rtgui_workbench_constructor(rtgui_workbench_t *workbench) { /* set event handler */ rtgui_widget_set_event_handler(RTGUI_WIDGET(workbench), rtgui_workbench_event_handler); /* set attributes */ workbench->flag = RTGUI_WORKBENCH_FLAG_DEFAULT; workbench->panel = RT_NULL; workbench->title = RT_NULL; workbench->current_view = RT_NULL; workbench->modal_code = RTGUI_MODAL_OK; workbench->modal_widget = RT_NULL; } static void _rtgui_workbench_destructor(rtgui_workbench_t *workbench) { RT_ASSERT(workbench != RT_NULL); if (RTGUI_TOPLEVEL(workbench)->server != RT_NULL) { struct rtgui_event_panel_detach edetach; RTGUI_EVENT_PANEL_DETACH_INIT(&edetach); /* detach from panel */ edetach.panel = workbench->panel; /* send PANEL DETACH to server */ if (rtgui_thread_send_sync(RTGUI_TOPLEVEL(workbench)->server, RTGUI_EVENT(&edetach), sizeof(struct rtgui_event_panel_detach)) != RT_EOK) return; RTGUI_TOPLEVEL(workbench)->server = RT_NULL; } /* release title */ rt_free(workbench->title); workbench->title = RT_NULL; } DEFINE_CLASS_TYPE(workbench, "workbench", RTGUI_TOPLEVEL_TYPE, _rtgui_workbench_constructor, _rtgui_workbench_destructor, sizeof(struct rtgui_workbench)); rtgui_workbench_t *rtgui_workbench_create(const char* panel_name, const unsigned char* title) { struct rtgui_workbench* workbench; /* the server thread id */ rt_thread_t server = rtgui_thread_get_server(); if (server == RT_NULL) { rt_kprintf("can't find rtgui server\n"); return RT_NULL; } /* create workbench */ workbench = (rtgui_workbench_t*) rtgui_widget_create (RTGUI_WORKBENCH_TYPE); if (workbench != RT_NULL) { /* the buffer uses to receive event */ union { struct rtgui_event_panel_attach ecreate; struct rtgui_event_panel_info epanel; char buffer[256]; /* use to recv other information */ } event; /* set workbench title */ workbench->title = (unsigned char*)rt_strdup((char*)title); /* create application in server */ RTGUI_EVENT_PANEL_ATTACH_INIT(&(event.ecreate)); /* set the panel name and workbench */ rt_strncpy(event.ecreate.panel_name, panel_name, RTGUI_NAME_MAX); event.ecreate.workbench = workbench; /* send PANEL ATTACH to server */ if (rtgui_thread_send_sync(server, &(event.ecreate.parent), sizeof(struct rtgui_event_panel_attach)) != RTGUI_STATUS_OK) { return RT_NULL; } /* get PANEL INFO */ rtgui_thread_recv_filter(RTGUI_EVENT_PANEL_INFO, &(event.epanel.parent), sizeof(event)); /* set panel */ workbench->panel = (struct rtgui_panel*)event.epanel.panel; /* connected */ RTGUI_TOPLEVEL(workbench)->server = server; /* set extent of workbench */ rtgui_widget_set_rect(RTGUI_WIDGET(workbench), &(event.epanel.extent)); /* set workbench in thread */ rtgui_thread_set_widget(RTGUI_WIDGET(workbench)); } return workbench; } void rtgui_workbench_destroy(rtgui_workbench_t* workbench) { RT_ASSERT(workbench != RT_NULL); if (workbench->flag & RTGUI_WORKBENCH_FLAG_CLOSED) { rtgui_widget_destroy(RTGUI_WIDGET(workbench)); } else { /* just close workbench */ rtgui_workbench_close(workbench); } } void rtgui_workbench_close(rtgui_workbench_t* workbench) { /* detach workbench in server */ if (RTGUI_TOPLEVEL(workbench)->server != RT_NULL) { struct rtgui_event_panel_detach edetach; RTGUI_EVENT_PANEL_DETACH_INIT(&edetach); /* detach from panel */ edetach.panel = workbench->panel; edetach.workbench = workbench; /* send PANEL DETACH to server */ if (rtgui_thread_send_sync(RTGUI_TOPLEVEL(workbench)->server, RTGUI_EVENT(&edetach), sizeof(struct rtgui_event_panel_detach)) != RT_EOK) return; RTGUI_TOPLEVEL(workbench)->server = RT_NULL; } /* set status to close */ workbench->flag |= RTGUI_WORKBENCH_FLAG_CLOSED; } void rtgui_workbench_set_flag(rtgui_workbench_t* workbench, rt_uint8_t flag) { RT_ASSERT(workbench != RT_NULL); workbench->flag = flag; } rt_bool_t rtgui_workbench_event_loop(rtgui_workbench_t* workbench) { rt_err_t result; rtgui_thread_t* tid; struct rtgui_event* event; tid = rtgui_thread_self(); RT_ASSERT(tid != RT_NULL); /* point to event buffer */ event = (struct rtgui_event*)tid->event_buffer; if (workbench->flag & RTGUI_WORKBENCH_FLAG_MODAL_MODE) { /* event loop for modal mode shown view */ while (workbench->flag & RTGUI_WORKBENCH_FLAG_MODAL_MODE) { if (tid->on_idle != RT_NULL) { result = rtgui_thread_recv_nosuspend(event, RTGUI_EVENT_BUFFER_SIZE); if (result == RT_EOK) RTGUI_WIDGET(workbench)->event_handler(RTGUI_WIDGET(workbench), event); else if (result == -RT_ETIMEOUT) tid->on_idle(RTGUI_WIDGET(workbench), RT_NULL); } else { result = rtgui_thread_recv(event, RTGUI_EVENT_BUFFER_SIZE); if (result == RT_EOK) RTGUI_WIDGET(workbench)->event_handler(RTGUI_WIDGET(workbench), event); } } } else { /* show workbench firstly */ rtgui_workbench_show(workbench); while (!(workbench->flag & RTGUI_WORKBENCH_FLAG_CLOSED)) { if (tid->on_idle != RT_NULL) { result = rtgui_thread_recv_nosuspend(event, RTGUI_EVENT_BUFFER_SIZE); if (result == RT_EOK) RTGUI_WIDGET(workbench)->event_handler(RTGUI_WIDGET(workbench), event); else if (result == -RT_ETIMEOUT) tid->on_idle(RTGUI_WIDGET(workbench), RT_NULL); } else { result = rtgui_thread_recv(event, RTGUI_EVENT_BUFFER_SIZE); if (result == RT_EOK) RTGUI_WIDGET(workbench)->event_handler(RTGUI_WIDGET(workbench), event); } } } return RT_TRUE; } rt_err_t rtgui_workbench_show(rtgui_workbench_t* workbench) { RT_ASSERT(workbench != RT_NULL); if (RTGUI_TOPLEVEL(workbench)->server != RT_NULL) { struct rtgui_event_panel_show eraise; RTGUI_EVENT_PANEL_SHOW_INIT(&eraise); eraise.workbench = workbench; eraise.panel = workbench->panel; if (rtgui_thread_send_sync(workbench->parent.server, RTGUI_EVENT(&eraise), sizeof(struct rtgui_event_panel_show)) != RT_EOK) return -RT_ERROR; RTGUI_WIDGET_UNHIDE(RTGUI_WIDGET(workbench)); rtgui_toplevel_update_clip(RTGUI_TOPLEVEL(workbench)); } else return -RT_ERROR; return RT_EOK; } rt_err_t rtgui_workbench_hide(rtgui_workbench_t* workbench) { if (RTGUI_TOPLEVEL(workbench)->server != RT_NULL) { struct rtgui_event_panel_hide ehide; RTGUI_EVENT_PANEL_HIDE_INIT(&ehide); RT_ASSERT(workbench != RT_NULL); if (workbench->parent.server == RT_NULL) return -RT_ERROR; ehide.panel = workbench->panel; if (rtgui_thread_send_sync(RTGUI_TOPLEVEL(workbench)->server, RTGUI_EVENT(&ehide), sizeof(struct rtgui_event_panel_hide)) != RT_EOK) return -RT_ERROR; RTGUI_WIDGET_HIDE(RTGUI_WIDGET(workbench)); rtgui_toplevel_update_clip(RTGUI_TOPLEVEL(workbench)); } return RT_EOK; } rt_bool_t rtgui_workbench_event_handler(rtgui_widget_t* widget, rtgui_event_t* event) { struct rtgui_workbench* workbench = (struct rtgui_workbench*)widget; switch (event->type) { case RTGUI_EVENT_PANEL_DETACH: RTGUI_WIDGET_HIDE(RTGUI_WIDGET(workbench)); RTGUI_TOPLEVEL(workbench)->server = RT_NULL; return RT_TRUE; case RTGUI_EVENT_PANEL_SHOW: /* show workbench in server */ rtgui_workbench_show(workbench); break; case RTGUI_EVENT_PANEL_HIDE: /* hide widget */ RTGUI_WIDGET_HIDE(widget); break; case RTGUI_EVENT_MOUSE_MOTION: { struct rtgui_event_mouse* emouse = (struct rtgui_event_mouse*)event; struct rtgui_toplevel* top = RTGUI_TOPLEVEL(emouse->wid); /* check the destination window */ if (top != RT_NULL && RTGUI_WIDGET(top)->event_handler != RT_NULL) { RTGUI_WIDGET(top)->event_handler(RTGUI_WIDGET(top), event); } else { /* let viewer to handle it */ rtgui_view_t* view = workbench->current_view; if (view != RT_NULL && RTGUI_WIDGET(view)->event_handler != RT_NULL) { RTGUI_WIDGET(view)->event_handler(RTGUI_WIDGET(view), event); } } } break; case RTGUI_EVENT_MOUSE_BUTTON: { struct rtgui_event_mouse* emouse = (struct rtgui_event_mouse*)event; struct rtgui_toplevel* top = RTGUI_TOPLEVEL(emouse->wid); /* check whether has widget which handled mouse event before */ if (RTGUI_TOPLEVEL_LAST_MEVENT_WIDGET(workbench) != RT_NULL) { struct rtgui_event_mouse* emouse; emouse = (struct rtgui_event_mouse*)event; RTGUI_TOPLEVEL_LAST_MEVENT_WIDGET(workbench)->event_handler(RTGUI_TOPLEVEL_LAST_MEVENT_WIDGET(workbench), event); if (rtgui_rect_contains_point(&(RTGUI_TOPLEVEL_LAST_MEVENT_WIDGET(workbench)->extent), emouse->x, emouse->y) == RT_EOK) { RTGUI_TOPLEVEL_LAST_MEVENT_WIDGET(workbench) = RT_NULL; break; /* mouse event is inside of widget, do not handle it anymore */ } /* clean last mouse event handled widget */ RTGUI_TOPLEVEL_LAST_MEVENT_WIDGET(workbench) = RT_NULL; } /* check the destination window */ if (top != RT_NULL && RTGUI_WIDGET(top)->event_handler != RT_NULL) { RTGUI_WIDGET(top)->event_handler(RTGUI_WIDGET(top), event); } else { if (RTGUI_WORKBENCH_IS_MODAL_MODE(workbench)) { /* let modal widget to handle it */ if (workbench->modal_widget != RT_NULL && workbench->modal_widget->event_handler != RT_NULL) { workbench->modal_widget->event_handler(workbench->modal_widget, event); } } else { /* let viewer to handle it */ rtgui_view_t* view = workbench->current_view; if (view != RT_NULL && RTGUI_WIDGET(view)->event_handler != RT_NULL) { RTGUI_WIDGET(view)->event_handler(RTGUI_WIDGET(view), event); } } } } break; case RTGUI_EVENT_KBD: { struct rtgui_event_kbd* kbd = (struct rtgui_event_kbd*)event; struct rtgui_toplevel* top = RTGUI_TOPLEVEL(kbd->wid); /* check the destination window */ if (top != RT_NULL && RTGUI_WIDGET(top)->event_handler != RT_NULL) { RTGUI_WIDGET(top)->event_handler(RTGUI_WIDGET(top), event); } else { if (RTGUI_WORKBENCH_IS_MODAL_MODE(workbench)) { /* let modal widget to handle it */ if (workbench->modal_widget != RT_NULL && workbench->modal_widget->event_handler != RT_NULL) { workbench->modal_widget->event_handler(workbench->modal_widget, event); } } else { if (RTGUI_CONTAINER(widget)->focused == widget) { /* set focused widget to the current view */ if (workbench->current_view != RT_NULL) rtgui_widget_focus(RTGUI_WIDGET(RTGUI_CONTAINER(workbench->current_view)->focused)); } return rtgui_toplevel_event_handler(widget, event); } } } break; case RTGUI_EVENT_PAINT: { struct rtgui_event_paint* epaint = (struct rtgui_event_paint*)event; struct rtgui_toplevel* top = RTGUI_TOPLEVEL(epaint->wid); /* check the destination window */ if (top != RT_NULL && RTGUI_WIDGET(top)->event_handler != RT_NULL) { RTGUI_WIDGET(top)->event_handler(RTGUI_WIDGET(top), event); } else { rtgui_view_t* view; /* un-hide workbench */ RTGUI_WIDGET_UNHIDE(widget); /* paint a view */ view = workbench->current_view; if (view != RT_NULL) { /* remake a paint event */ RTGUI_EVENT_PAINT_INIT(epaint); epaint->wid = RT_NULL; /* send this event to the view */ if (RTGUI_WIDGET(view)->event_handler != RT_NULL) { RTGUI_WIDGET(view)->event_handler(RTGUI_WIDGET(view), event); } } else { struct rtgui_dc* dc; struct rtgui_rect rect; dc = rtgui_dc_begin_drawing(widget); rtgui_widget_get_rect(widget, &rect); rtgui_dc_fill_rect(dc, &rect); rtgui_dc_end_drawing(dc); } } } break; case RTGUI_EVENT_CLIP_INFO: { struct rtgui_event_clip_info* eclip = (struct rtgui_event_clip_info*)event; struct rtgui_widget* dest_widget = RTGUI_WIDGET(eclip->wid); if (dest_widget != RT_NULL && dest_widget->event_handler != RT_NULL) { dest_widget->event_handler(dest_widget, event); } else { return rtgui_toplevel_event_handler(widget, event); } } break; case RTGUI_EVENT_WIN_CLOSE: case RTGUI_EVENT_WIN_ACTIVATE: case RTGUI_EVENT_WIN_DEACTIVATE: { struct rtgui_event_win* wevent = (struct rtgui_event_win*)event; struct rtgui_widget* dest_widget = RTGUI_WIDGET(wevent->wid); if (dest_widget != RT_NULL && dest_widget->event_handler != RT_NULL) { dest_widget->event_handler(dest_widget, event); } } break; case RTGUI_EVENT_WIN_MOVE: { struct rtgui_event_win_move* wevent = (struct rtgui_event_win_move*)event; struct rtgui_toplevel* top = RTGUI_TOPLEVEL(wevent->wid); if (top != RT_NULL && RTGUI_WIDGET(top)->event_handler != RT_NULL) { RTGUI_WIDGET(top)->event_handler(RTGUI_WIDGET(top), event); } } break; default: return rtgui_toplevel_event_handler(widget, event); } return RT_TRUE; } /* * * view on workbench * */ void rtgui_workbench_add_view(rtgui_workbench_t* workbench, rtgui_view_t* view) { rtgui_container_add_child(RTGUI_CONTAINER(workbench), RTGUI_WIDGET(view)); /* hide view in default */ RTGUI_WIDGET_HIDE(RTGUI_WIDGET(view)); /* reset view extent */ rtgui_widget_set_rect(RTGUI_WIDGET(view), &(RTGUI_WIDGET(workbench)->extent)); } void rtgui_workbench_remove_view(rtgui_workbench_t* workbench, rtgui_view_t* view) { if (view == workbench->current_view) rtgui_workbench_hide_view(workbench, view); rtgui_container_remove_child(RTGUI_CONTAINER(workbench), RTGUI_WIDGET(view)); } void rtgui_workbench_show_view(rtgui_workbench_t* workbench, rtgui_view_t* view) { RT_ASSERT(workbench != RT_NULL); RT_ASSERT(view != RT_NULL); /* already shown */ if (workbench->current_view == view) return; if (workbench->current_view != RT_NULL) { /* hide old view */ RTGUI_WIDGET_HIDE(RTGUI_WIDGET(workbench->current_view)); } /* show view */ RTGUI_WIDGET_UNHIDE(RTGUI_WIDGET(view)); workbench->current_view = view; /* update workbench clip */ rtgui_toplevel_update_clip(RTGUI_TOPLEVEL(workbench)); if (!RTGUI_WIDGET_IS_HIDE(RTGUI_WIDGET(workbench))) { rtgui_widget_update(RTGUI_WIDGET(view)); } } void rtgui_workbench_hide_view(rtgui_workbench_t* workbench, rtgui_view_t* view) { RT_ASSERT(workbench != RT_NULL); RT_ASSERT(view != RT_NULL); /* hide view */ RTGUI_WIDGET_HIDE(RTGUI_WIDGET(view)); if (view == workbench->current_view) { rtgui_view_t *next_view; workbench->current_view = RT_NULL; next_view = RTGUI_VIEW(rtgui_widget_get_next_sibling(RTGUI_WIDGET(view))); if (next_view == RT_NULL) next_view = RTGUI_VIEW(rtgui_widget_get_prev_sibling(RTGUI_WIDGET(view))); if (next_view != RT_NULL) { rtgui_view_show(next_view, RT_FALSE); } else { /* update workbench clip */ rtgui_toplevel_update_clip(RTGUI_TOPLEVEL(workbench)); } } }