// Toolbar creation and related functions // #include "lc_global.h" #include "lc_colors.h" #include "lc_math.h" #include <gdk/gdkkeysyms.h> #include <gtk/gtk.h> #include <stdio.h> #include "opengl.h" #include "gtktools.h" #include "main.h" #include "globals.h" #include "project.h" #include "pieceinf.h" #include "toolbar.h" #include "message.h" #include "preview.h" #include "lc_library.h" #include "lc_application.h" #include "gtkmisc.h" // ============================================================================= // Variables GtkWidget *piecetree; GtkWidget *pieceentry; GtkWidget *piecemenu; GtkWidget *colorlist; GtkWidget *grouptoolbar; PiecePreview *preview; TOOL_TOOLBAR tool_toolbar; MAIN_TOOLBAR main_toolbar; ANIM_TOOLBAR anim_toolbar; extern GtkWidget* create_snap_menu(); extern GtkWidget* create_lock_menu(); // ========================================================= void create_toolbars(GtkWidget *window, GtkWidget *vbox) { #include "pixmaps/ac-brick.xpm" #include "pixmaps/ac-light.xpm" #include "pixmaps/ac-spot.xpm" #include "pixmaps/ac-cam.xpm" #include "pixmaps/ac-sel.xpm" #include "pixmaps/ac-move.xpm" #include "pixmaps/ac-rot.xpm" #include "pixmaps/ac-erase.xpm" #include "pixmaps/ac-paint.xpm" #include "pixmaps/ac-zoom.xpm" #include "pixmaps/ac-pan.xpm" #include "pixmaps/ac-rotv.xpm" #include "pixmaps/ac-roll.xpm" #include "pixmaps/ac-zoomr.xpm" #include "pixmaps/ac-zoome.xpm" #include "pixmaps/ac-prev.xpm" #include "pixmaps/ac-next.xpm" #include "pixmaps/an-anim.xpm" #include "pixmaps/an-key.xpm" #include "pixmaps/an-next.xpm" #include "pixmaps/an-prev.xpm" #include "pixmaps/an-first.xpm" #include "pixmaps/an-last.xpm" #include "pixmaps/an-play.xpm" #include "pixmaps/an-stop.xpm" //#include "pixmaps/st-about.xpm" #include "pixmaps/st-fast.xpm" //#include "pixmaps/st-paste.xpm" //#include "pixmaps/st-save.xpm" //#include "pixmaps/st-help.xpm" //#include "pixmaps/st-prev.xpm" #include "pixmaps/st-snap.xpm" #include "pixmaps/st-lock.xpm" //#include "pixmaps/st-copy.xpm" //#include "pixmaps/st-new.xpm" //#include "pixmaps/st-print.xpm" #include "pixmaps/st-snapa.xpm" //#include "pixmaps/st-cut.xpm" //#include "pixmaps/st-open.xpm" //#include "pixmaps/st-redo.xpm" //#include "pixmaps/st-undo.xpm" GtkToolItem* item; // Main Toolbar main_toolbar.handle_box = gtk_handle_box_new(); gtk_box_pack_start(GTK_BOX(vbox), main_toolbar.handle_box, FALSE, FALSE, 0); gtk_widget_show(main_toolbar.handle_box); main_toolbar.toolbar = gtk_toolbar_new(); gtk_toolbar_set_orientation(GTK_TOOLBAR(main_toolbar.toolbar), GTK_ORIENTATION_HORIZONTAL); gtk_toolbar_set_style(GTK_TOOLBAR(main_toolbar.toolbar), GTK_TOOLBAR_ICONS); gtk_container_add(GTK_CONTAINER(main_toolbar.handle_box), main_toolbar.toolbar); item = gtk_tool_button_new_from_stock(GTK_STOCK_NEW); gtk_tool_item_set_tooltip_text(item, "Create a new project"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommandDirect), GINT_TO_POINTER(LC_FILE_NEW)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); item = gtk_tool_button_new_from_stock(GTK_STOCK_OPEN); gtk_tool_item_set_tooltip_text(item, "Open an existing project"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommandDirect), GINT_TO_POINTER(LC_FILE_OPEN)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); item = gtk_tool_button_new_from_stock(GTK_STOCK_SAVE); gtk_tool_item_set_tooltip_text(item, "Save the current project"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommandDirect), GINT_TO_POINTER(LC_FILE_SAVE)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); item = gtk_separator_tool_item_new(); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); item = gtk_tool_button_new_from_stock(GTK_STOCK_UNDO); gtk_tool_item_set_tooltip_text(item, "Undo the last action"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommandDirect), GINT_TO_POINTER(LC_EDIT_UNDO)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); main_toolbar.undo = (GtkWidget*)item; item = gtk_tool_button_new_from_stock(GTK_STOCK_REDO); gtk_tool_item_set_tooltip_text(item, "Redo the last undone action"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommandDirect), GINT_TO_POINTER(LC_EDIT_REDO)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); main_toolbar.redo = (GtkWidget*)item; item = gtk_separator_tool_item_new(); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); item = gtk_tool_button_new_from_stock(GTK_STOCK_CUT); gtk_tool_item_set_tooltip_text(item, "Cut the selection"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommandDirect), GINT_TO_POINTER(LC_EDIT_CUT)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); main_toolbar.cut = (GtkWidget*)item; item = gtk_tool_button_new_from_stock(GTK_STOCK_COPY); gtk_tool_item_set_tooltip_text(item, "Copy the selection"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommandDirect), GINT_TO_POINTER(LC_EDIT_COPY)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); main_toolbar.copy = (GtkWidget*)item; item = gtk_tool_button_new_from_stock(GTK_STOCK_PASTE); gtk_tool_item_set_tooltip_text(item, "Insert the clipboard contents"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommandDirect), GINT_TO_POINTER(LC_EDIT_PASTE)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); main_toolbar.paste = (GtkWidget*)item; item = gtk_separator_tool_item_new(); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); item = gtk_menu_tool_button_new(new_pixmap(window, st_lock), "3D lock"); gtk_tool_item_set_tooltip_text(item, "Toggle 3D lock"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommand), GINT_TO_POINTER(ID_LOCK_ON)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); main_toolbar.lock = (GtkWidget*)item; main_toolbar.lock_menu = create_lock_menu(); gtk_menu_tool_button_set_menu(GTK_MENU_TOOL_BUTTON(item), main_toolbar.lock_menu); item = gtk_menu_tool_button_new(new_pixmap(window, st_snap), "3D snap"); gtk_tool_item_set_tooltip_text(item, "Toggle 3D snap"); g_signal_connect(item, "clicked", G_CALLBACK(OnCommand), GINT_TO_POINTER(ID_SNAP_ON)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); main_toolbar.snap = (GtkWidget*)item; main_toolbar.snap_menu = create_snap_menu(); gtk_menu_tool_button_set_menu(GTK_MENU_TOOL_BUTTON(item), main_toolbar.snap_menu); item = gtk_toggle_tool_button_new(); gtk_tool_item_set_tooltip_text(item, "Toggle angle snap"); gtk_tool_button_set_label(GTK_TOOL_BUTTON(item), "Angle snap"); gtk_tool_button_set_icon_widget(GTK_TOOL_BUTTON(item), new_pixmap(window, st_snapa)); g_signal_connect(item, "clicked", G_CALLBACK(OnCommand), GINT_TO_POINTER(ID_SNAP_A)); gtk_toolbar_insert(GTK_TOOLBAR(main_toolbar.toolbar), item, -1); main_toolbar.angle = (GtkWidget*)item; item = gtk_toggle_tool_button_new(); gtk_tool_item_set_tooltip_text(item, "Toggle fast rendering"); gtk_tool_button_set_label(GTK_TOOL_BUTTON(item), "Fast render"); gtk_tool_button_set_icon_widget(GTK_TOOL_BUTTON(item), new_pixmap(window, st_fast)); g_signal_connect(item, "clicked", G_CALLBACK(OnCommandDirect), GINT_TO_POINTER(LC_TOOLBAR_FASTRENDER)); gtk_toolbar_insert (GTK_TOOLBAR (main_toolbar.toolbar), item, -1); main_toolbar.fast = (GtkWidget*)item; gtk_widget_show_all(main_toolbar.handle_box); GtkWidget *button; // Tools Toolbar tool_toolbar.handle_box = gtk_handle_box_new (); gtk_box_pack_start (GTK_BOX (vbox),tool_toolbar. handle_box, FALSE, FALSE, 0); gtk_widget_show (tool_toolbar.handle_box); tool_toolbar.toolbar = gtk_toolbar_new(); gtk_toolbar_set_orientation(GTK_TOOLBAR(tool_toolbar.toolbar), GTK_ORIENTATION_HORIZONTAL); gtk_toolbar_set_style(GTK_TOOLBAR(tool_toolbar.toolbar), GTK_TOOLBAR_ICONS); gtk_container_add (GTK_CONTAINER (tool_toolbar.handle_box), tool_toolbar.toolbar); gtk_widget_show (tool_toolbar.toolbar); tool_toolbar.brick = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, NULL, "Piece", "Insert Piece", "", new_pixmap (window, ac_brick), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_INSERT); tool_toolbar.light = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Light", "Insert Light", "", new_pixmap (window, ac_light), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_LIGHT); tool_toolbar.spot = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Spot", "Insert Spotlight", "", new_pixmap (window, ac_spot), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_SPOTLIGHT); tool_toolbar.camera = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Camera", "Insert Camera", "", new_pixmap (window, ac_cam), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_CAMERA); tool_toolbar.select = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Select", "Select Objects", "", new_pixmap (window, ac_sel), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_SELECT); tool_toolbar.move = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Move", "Move Objects", "", new_pixmap (window, ac_move), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_MOVE); tool_toolbar.rotate = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Rotate", "Rotate Pieces", "", new_pixmap (window, ac_rot), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_ROTATE); tool_toolbar.erase = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Delete", "Remove Objects", "", new_pixmap (window, ac_erase), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_ERASER); tool_toolbar.paint = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Paint", "Paint Bricks", "", new_pixmap (window, ac_paint), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_PAINT); tool_toolbar.zoom = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Zoom", "Zoom", "", new_pixmap (window, ac_zoom), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_ZOOM); tool_toolbar.pan = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Pan", "Pan", "", new_pixmap (window, ac_pan), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_PAN); tool_toolbar.rotview = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Rot. View", "Rotate View", "", new_pixmap (window, ac_rotv), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_ROTATE_VIEW); tool_toolbar.roll = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Roll", "Roll", "", new_pixmap (window, ac_roll), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_ROLL); tool_toolbar.zoomreg = button = gtk_toolbar_append_element (GTK_TOOLBAR (tool_toolbar.toolbar), GTK_TOOLBAR_CHILD_RADIOBUTTON, button, "Zoom Box", "Zoom Region", "", new_pixmap (window, ac_zoomr), GTK_SIGNAL_FUNC (OnCommand), (void*)ID_ACTION_ZOOM_REGION); gtk_toolbar_append_item (GTK_TOOLBAR (tool_toolbar.toolbar), "Zoom Ext.", "Zoom Extents", "", new_pixmap (window, ac_zoome), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_VIEW_ZOOMEXTENTS); tool_toolbar.prev = gtk_toolbar_append_item (GTK_TOOLBAR (tool_toolbar.toolbar), "Show Previous", "Show piece on previous step", "", new_pixmap (window, ac_prev), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_PIECE_PREVIOUS); tool_toolbar.next = gtk_toolbar_append_item (GTK_TOOLBAR (tool_toolbar.toolbar), "Show Next", "Show Piece on next step", "", new_pixmap (window, ac_next), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_PIECE_NEXT); // Animation Toolbar anim_toolbar.handle_box = gtk_handle_box_new (); gtk_box_pack_start (GTK_BOX (vbox), anim_toolbar.handle_box, FALSE, FALSE, 0); gtk_widget_show (anim_toolbar.handle_box); anim_toolbar.toolbar = gtk_toolbar_new(); gtk_toolbar_set_orientation(GTK_TOOLBAR(anim_toolbar.toolbar), GTK_ORIENTATION_HORIZONTAL); gtk_toolbar_set_style(GTK_TOOLBAR(anim_toolbar.toolbar), GTK_TOOLBAR_ICONS); gtk_container_add (GTK_CONTAINER (anim_toolbar.handle_box), anim_toolbar.toolbar); gtk_widget_show (anim_toolbar.toolbar); anim_toolbar.first = gtk_toolbar_append_item (GTK_TOOLBAR (anim_toolbar.toolbar), "First", "Go to the Start", "", new_pixmap (window, an_first), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_VIEW_STEP_FIRST); anim_toolbar.prev = gtk_toolbar_append_item (GTK_TOOLBAR (anim_toolbar.toolbar), "Previous", "Go Back", "", new_pixmap (window, an_prev), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_VIEW_STEP_PREVIOUS); anim_toolbar.play = gtk_toolbar_append_item (GTK_TOOLBAR (anim_toolbar.toolbar), "Play", "Play Animation", "", new_pixmap (window, an_play), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_VIEW_PLAY); anim_toolbar.stop = gtk_toolbar_append_item (GTK_TOOLBAR (anim_toolbar.toolbar), "Stop", "Stop Animation", "", new_pixmap (window, an_stop), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_VIEW_STOP); anim_toolbar.next = gtk_toolbar_append_item (GTK_TOOLBAR (anim_toolbar.toolbar), "Next", "Go Forward", "", new_pixmap (window, an_next), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_VIEW_STEP_NEXT); anim_toolbar.last = gtk_toolbar_append_item (GTK_TOOLBAR (anim_toolbar.toolbar), "Last", "Go to the End", "", new_pixmap (window, an_last), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_VIEW_STEP_LAST); anim_toolbar.anim = gtk_toolbar_append_element (GTK_TOOLBAR (anim_toolbar.toolbar), GTK_TOOLBAR_CHILD_TOGGLEBUTTON, NULL, "Mode", "Toggle Animation or Instructions", "", new_pixmap (window, an_anim), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_TOOLBAR_ANIMATION); anim_toolbar.keys = gtk_toolbar_append_element (GTK_TOOLBAR (anim_toolbar.toolbar), GTK_TOOLBAR_CHILD_TOGGLEBUTTON, NULL, "Keys", "Add Keys", "", new_pixmap (window, an_key), GTK_SIGNAL_FUNC (OnCommandDirect), (void*)LC_TOOLBAR_ADDKEYS); } static void drag_end(GtkWidget *widget, GdkDragContext *context, gpointer data) { dragged_piece = NULL; } // ========================================================= // Pieces toolbar //static bool list_subparts = false; static int cur_color = 0; int PiecesSortFunc(const PieceInfo* a, const PieceInfo* b, void* SortData) { if (a->IsSubPiece()) { if (b->IsSubPiece()) { return strcmp(a->m_strDescription, b->m_strDescription); } else { return 1; } } else { if (b->IsSubPiece()) { return -1; } else { return strcmp(a->m_strDescription, b->m_strDescription); } } return 0; } void fill_piecetree() { lcPiecesLibrary* Lib = lcGetPiecesLibrary(); GtkTreeStore* model = GTK_TREE_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(piecetree))); for (int i = 0; i < Lib->mCategories.GetSize(); i++) { GtkTreeIter iter; gtk_tree_store_append(model, &iter, NULL); gtk_tree_store_set(model, &iter, 0, (const char*)Lib->mCategories[i].Name, 1, NULL, -1); PtrArray<PieceInfo> SinglePieces, GroupedPieces; Lib->GetCategoryEntries(i, true, SinglePieces, GroupedPieces); SinglePieces += GroupedPieces; SinglePieces.Sort(PiecesSortFunc, NULL); for (int j = 0; j < SinglePieces.GetSize(); j++) { PieceInfo* Info = SinglePieces[j]; GtkTreeIter entry; gtk_tree_store_append(model, &entry, &iter); gtk_tree_store_set(model, &entry, 0, Info->m_strDescription, 1, Info, -1); if (GroupedPieces.FindIndex(Info) != -1) { PtrArray<PieceInfo> Patterns; Lib->GetPatternedPieces(Info, Patterns); for (int k = 0; k < Patterns.GetSize(); k++) { GtkTreeIter pat; PieceInfo* child = Patterns[k]; if (!Lib->PieceInCategory(child, Lib->mCategories[i].Keywords)) continue; const char* desc = child->m_strDescription; int len = strlen(Info->m_strDescription); if (!strncmp(child->m_strDescription, Info->m_strDescription, len)) desc += len; gtk_tree_store_append(model, &pat, &entry); gtk_tree_store_set(model, &pat, 0, desc, 1, child, -1); } } } } GtkTreeIter iter; gtk_tree_store_append(model, &iter, NULL); gtk_tree_store_set(model, &iter, 0, "Search Results", 1, NULL, -1); GtkTreeIter entry; gtk_tree_store_append(GTK_TREE_STORE(model), &entry, &iter); gtk_tree_store_set(GTK_TREE_STORE(model), &entry, 0, "No pieces found", 1, NULL, -1); } // Callback for the pieces list. static void piecetree_changed(GtkTreeSelection* selection, gpointer data) { GtkTreeIter iter; GtkTreeModel* model; if (gtk_tree_selection_get_selected(selection, &model, &iter)) { gpointer sel; gtk_tree_model_get(model, &iter, 1, &sel, -1); if (sel) preview->SetCurrentPiece((PieceInfo*)sel); } } static void piecetree_drag_begin(GtkWidget *widget, GdkDragContext *context, gpointer data) { GtkTreeIter iter; GtkTreeModel *model; dragged_piece = NULL; if (gtk_tree_selection_get_selected(gtk_tree_view_get_selection(GTK_TREE_VIEW(widget)), &model, &iter)) { gtk_tree_model_get(model, &iter, 1, &dragged_piece, -1); } } static void piececombo_popup_position (GtkMenu *menu, gint *x, gint *y, gboolean* push, gpointer data) { gdk_window_get_origin (pieceentry->window, x, y); *y += pieceentry->allocation.height; *push = true; } static void piececombo_popup (GtkWidget *widget, gpointer data) { if (piecemenu != NULL) gtk_menu_popup (GTK_MENU (piecemenu), NULL, NULL, piececombo_popup_position, NULL, 1, GDK_CURRENT_TIME); } static void piececombo_selected (GtkWidget *widget, gpointer data) { gchar *str; gtk_label_get (GTK_LABEL (GTK_BIN (widget)->child), &str); gtk_entry_set_text (GTK_ENTRY (pieceentry), str); } // Add a new piece to the combobox void piececombo_add (const char* str) { if (str == NULL) { // Clear the list if (piecemenu != NULL) { gtk_widget_destroy (piecemenu); piecemenu = NULL; } } else { GtkWidget *item; GList *children; int pos = 0; if (piecemenu == NULL) piecemenu = gtk_menu_new (); children = gtk_container_children (GTK_CONTAINER (piecemenu)); while (children) { gchar *label; int i; gtk_label_get (GTK_LABEL (GTK_BIN (children->data)->child), &label); i = strcmp (str, label); if (i == 0) return; else if (i < 0) break; children = children->next; pos++; } item = gtk_menu_item_new_with_label (str); gtk_signal_connect (GTK_OBJECT (item), "activate", GTK_SIGNAL_FUNC (piececombo_selected), NULL); gtk_widget_show (item); gtk_menu_insert (GTK_MENU (piecemenu), item, pos); } } static gint piececombo_key(GtkWidget* widget, GdkEventKey* event) { if (event->keyval == GDK_Return) { const gchar* str = gtk_entry_get_text(GTK_ENTRY(pieceentry)); lcPiecesLibrary* Lib = lcGetPiecesLibrary(); // Save search. int Index = Lib->FindCategoryIndex("Search Results"); if (Index == -1) { Lib->AddCategory("Search Results", (const char*)str); Index = Lib->mCategories.GetSize() - 1; } else Lib->SetCategory(Index, "Search Results", (const char*)str); // Find search category row. GtkTreeModel* model = gtk_tree_view_get_model(GTK_TREE_VIEW(piecetree)); GtkTreeIter iter; if (!gtk_tree_model_get_iter_first(model, &iter)) return FALSE; do { gchar* name; gtk_tree_model_get(model, &iter, 0, &name, -1); if (strcmp(name, "Search Results")) continue; GtkTreeIter child; // Remove all children. while (gtk_tree_model_iter_children(model, &child, &iter)) gtk_tree_store_remove(GTK_TREE_STORE(model), &child); // Perform search. PtrArray<PieceInfo> SinglePieces, GroupedPieces; Lib->GetCategoryEntries(Index, true, SinglePieces, GroupedPieces); // Merge and sort the arrays. SinglePieces += GroupedPieces; SinglePieces.Sort(PiecesSortFunc, NULL); // Add results. for (int i = 0; i < SinglePieces.GetSize(); i++) { PieceInfo* Info = SinglePieces[i]; GtkTreeIter entry; gtk_tree_store_append(GTK_TREE_STORE(model), &entry, &iter); gtk_tree_store_set(GTK_TREE_STORE(model), &entry, 0, Info->m_strDescription, 1, Info, -1); } if (SinglePieces.GetSize() == 0) { GtkTreeIter entry; gtk_tree_store_append(GTK_TREE_STORE(model), &entry, &iter); gtk_tree_store_set(GTK_TREE_STORE(model), &entry, 0, "No pieces found", 1, NULL, -1); } // Expand results. GtkTreePath* path = gtk_tree_model_get_path(model, &iter); gtk_tree_view_expand_row(GTK_TREE_VIEW(piecetree), path, FALSE); gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(piecetree), path, NULL, TRUE, 0.5f, 0.0f); gtk_tree_path_free(path); } while (gtk_tree_model_iter_next(model, &iter)); } return FALSE; } static void piececombo_changed(GtkWidget *widget, gpointer data) { lcPiecesLibrary *pLib = lcGetPiecesLibrary(); const gchar* str; int i; str = gtk_entry_get_text(GTK_ENTRY(pieceentry)); for (i = 0; i < pLib->mPieces.GetSize(); i++) { PieceInfo* pInfo = pLib->mPieces[i]; if (strcmp(str, pInfo->m_strDescription) == 0) { // Select the piece // i = gtk_clist_find_row_from_data(GTK_CLIST(piecelist), pInfo); // gtk_clist_select_row(GTK_CLIST(piecelist), i, 0); // if(gtk_clist_row_is_visible(GTK_CLIST(piecelist), i) != GTK_VISIBILITY_FULL) // gtk_clist_moveto(GTK_CLIST(piecelist), i, 0, 0.5f, 0); return; } } } struct ColorListCell { GdkRectangle Rect; GdkColor Color; int ColorIndex; }; class ColorListData { public: ColorListData() { mWidth = 0; mHeight = 0; } ~ColorListData() { } void UpdateColors() { mCells.RemoveAll(); mGroups.RemoveAll(); mGroups.SetSize(LC_NUM_COLORGROUPS); for (int GroupIdx = 0; GroupIdx < LC_NUM_COLORGROUPS; GroupIdx++) { lcColorGroup* Group = &gColorGroups[GroupIdx]; for (int ColorIdx = 0; ColorIdx < Group->Colors.GetSize(); ColorIdx++) { lcColor* Color = &gColorList[Group->Colors[ColorIdx]]; ColorListCell Cell; Cell.Color.red = (gushort)(Color->Value[0] * 0xFFFF); Cell.Color.green = (gushort)(Color->Value[1] * 0xFFFF); Cell.Color.blue = (gushort)(Color->Value[2] * 0xFFFF); Cell.ColorIndex = Group->Colors[ColorIdx]; mCells.Add(Cell); } } mColumns = 14; mRows = 0; for (int GroupIdx = 0; GroupIdx < LC_NUM_COLORGROUPS; GroupIdx++) mRows += (gColorGroups[GroupIdx].Colors.GetSize() + mColumns - 1) / mColumns; } void UpdateLayout(GtkWidget* widget) { PangoRenderer *renderer; PangoContext *context; PangoLayout *layout; int width, height; if (mWidth == widget->allocation.width && mHeight == widget->allocation.height) return; renderer = gdk_pango_renderer_get_default(gtk_widget_get_screen(widget)); gdk_pango_renderer_set_drawable(GDK_PANGO_RENDERER(renderer), widget->window); gdk_pango_renderer_set_gc(GDK_PANGO_RENDERER(renderer), widget->style->black_gc); context = gtk_widget_create_pango_context(widget); layout = pango_layout_new(context); int TextHeight = 0; for (int GroupIdx = 0; GroupIdx < LC_NUM_COLORGROUPS; GroupIdx++) { lcColorGroup* Group = &gColorGroups[GroupIdx]; pango_layout_set_text(layout, Group->Name, -1); pango_layout_get_size(layout, &width, &height); TextHeight += height / PANGO_SCALE; } float CellWidth = (float)widget->allocation.width / (float)mColumns; float CellHeight = (float)(widget->allocation.height - TextHeight) / (float)mRows; int CurCell = 0; float CurY = 0.0f; for (int GroupIdx = 0; GroupIdx < LC_NUM_COLORGROUPS; GroupIdx++) { lcColorGroup* Group = &gColorGroups[GroupIdx]; int CurColumn = 0; pango_layout_set_text(layout, Group->Name, -1); pango_layout_get_size(layout, &width, &height); mGroups[GroupIdx].x = 0; mGroups[GroupIdx].y = (int)CurY * PANGO_SCALE; mGroups[GroupIdx].width = width; mGroups[GroupIdx].height = height; CurY += height / PANGO_SCALE; for (int ColorIdx = 0; ColorIdx < Group->Colors.GetSize(); ColorIdx++) { float Left = CurColumn * CellWidth; float Right = (CurColumn + 1) * CellWidth; float Top = CurY; float Bottom = CurY + CellHeight; mCells[CurCell].Rect.x = Left; mCells[CurCell].Rect.y = Top; mCells[CurCell].Rect.width = (int)Right - (int)Left; mCells[CurCell].Rect.height = Bottom - Top; // lcColor* Color = &gColorList[mCells[CurCell].ColorIndex]; // Text.Format("%s (%d)", Color->Name, Color->Code); // mToolTip.AddTool(this, Text, CellRect, CurCell + 1); CurColumn++; if (CurColumn == mColumns) { CurColumn = 0; CurY += CellHeight; } CurCell++; } if (CurColumn != 0) CurY += CellHeight; } g_object_unref(layout); g_object_unref(context); mWidth = widget->allocation.width; mHeight = widget->allocation.height; } ObjArray<GdkRectangle> mGroups; ObjArray<ColorListCell> mCells; int mColumns; int mRows; int mWidth; int mHeight; }; static gboolean colorlist_expose(GtkWidget *widget, GdkEventExpose *event, gpointer data) { PangoRenderer *renderer; PangoContext *context; PangoLayout *layout; ColorListData* Data; Data = (ColorListData*)gtk_object_get_data(GTK_OBJECT(widget), "colorlist"); Data->UpdateLayout(widget); renderer = gdk_pango_renderer_get_default(gtk_widget_get_screen(widget)); gdk_pango_renderer_set_drawable(GDK_PANGO_RENDERER(renderer), widget->window); gdk_pango_renderer_set_gc(GDK_PANGO_RENDERER(renderer), widget->style->black_gc); context = gtk_widget_create_pango_context(widget); layout = pango_layout_new(context); pango_layout_set_width(layout, widget->allocation.width * PANGO_SCALE); pango_layout_set_alignment(layout, PANGO_ALIGN_CENTER); for (int GroupIdx = 0; GroupIdx < LC_NUM_COLORGROUPS; GroupIdx++) { lcColorGroup* Group = &gColorGroups[GroupIdx]; pango_layout_set_text(layout, Group->Name, -1); pango_renderer_draw_layout(renderer, layout, Data->mGroups[GroupIdx].x, Data->mGroups[GroupIdx].y); } gdk_pango_renderer_set_drawable(GDK_PANGO_RENDERER(renderer), NULL); gdk_pango_renderer_set_gc(GDK_PANGO_RENDERER(renderer), NULL); g_object_unref(layout); g_object_unref(context); GdkGC* gc = widget->style->fg_gc[GTK_WIDGET_STATE(widget)]; GdkGCValues values; gdk_gc_get_values(gc, &values); for (int CellIdx = 0; CellIdx < Data->mCells.GetSize(); CellIdx++) { GdkRectangle& rect = Data->mCells[CellIdx].Rect; gdk_color_alloc(gtk_widget_get_colormap(widget), &Data->mCells[CellIdx].Color); gdk_gc_set_foreground(gc, &Data->mCells[CellIdx].Color); gdk_draw_rectangle(widget->window, gc, TRUE, rect.x, rect.y, rect.width, rect.height); } GdkColor Color; Color.red = (gushort)(0.4f * 0xFFFF); Color.green = (gushort)(0.8f * 0xFFFF); Color.blue = (gushort)(0.4f * 0xFFFF); gdk_color_alloc(gtk_widget_get_colormap(widget), &Color); gdk_gc_set_foreground(gc, &Color); GdkRectangle& rect = Data->mCells[cur_color].Rect; gdk_draw_rectangle(widget->window, gc, FALSE, rect.x, rect.y, rect.width, rect.height); gdk_gc_set_values(gc, &values, GDK_GC_FOREGROUND); return FALSE; } static gboolean colorlist_realize(GtkWidget* widget, gpointer user) { ColorListData* Data; Data = (ColorListData*)gtk_object_get_data(GTK_OBJECT(widget), "colorlist"); delete Data; Data = new ColorListData; gtk_object_set_data(GTK_OBJECT(widget), "colorlist", Data); Data->UpdateColors(); Data->UpdateLayout(widget); return FALSE; } static gboolean colorlist_unrealize(GtkWidget* widget, gpointer user) { ColorListData* Data; Data = (ColorListData*)gtk_object_get_data(GTK_OBJECT(widget), "colorlist"); delete Data; return FALSE; } static gint colorlist_key_press(GtkWidget* widget, GdkEventKey* event, gpointer data) { ColorListData* Data = (ColorListData*)gtk_object_get_data(GTK_OBJECT(widget), "colorlist"); int color = cur_color; if (event->keyval == GDK_Left) { if (cur_color > 0) color = cur_color - 1; } else if (event->keyval == GDK_Right) { if (cur_color < Data->mCells.GetSize() - 1) color = cur_color + 1; } else if (event->keyval == GDK_Up || event->keyval == GDK_Down) { if (cur_color < 0 || cur_color >= Data->mCells.GetSize()) cur_color = 0; int CurGroup = 0; int NumCells = 0; for (CurGroup = 0; CurGroup < LC_NUM_COLORGROUPS; CurGroup++) { int NumColors = gColorGroups[CurGroup].Colors.GetSize(); if (cur_color < NumCells + NumColors) break; NumCells += NumColors; } int Row = (cur_color - NumCells) / Data->mColumns; int Column = (cur_color - NumCells) % Data->mColumns; if (event->keyval == GDK_Up) { if (Row > 0) color = cur_color - Data->mColumns; else if (CurGroup > 0) { int NumColors = gColorGroups[CurGroup - 1].Colors.GetSize(); int NumColumns = NumColors % Data->mColumns; if (NumColumns <= Column + 1) color = cur_color - NumColumns - Data->mColumns; else color = cur_color - NumColumns; } } else if (event->keyval == GDK_Down) { int NumColors = gColorGroups[CurGroup].Colors.GetSize(); if (cur_color + Data->mColumns < NumCells + NumColors) color = cur_color + Data->mColumns; else { int NumColumns = NumColors % Data->mColumns; if (NumColumns > Column) { if (cur_color + NumColumns < Data->mCells.GetSize()) color = cur_color + NumColumns; } else color = cur_color + Data->mColumns + NumColumns; } } } if (color != cur_color) { cur_color = color; lcGetActiveProject()->HandleNotify(LC_COLOR_CHANGED, Data->mCells[cur_color].ColorIndex); gtk_widget_draw(widget, NULL); preview->Redraw(); } return TRUE; } static gboolean colorlist_button_press(GtkWidget *widget, GdkEventButton *event, gpointer data) { if (event->button != 1) return FALSE; ColorListData* Data = (ColorListData*)gtk_object_get_data(GTK_OBJECT(widget), "colorlist"); int NewColor = cur_color; for (int CellIdx = 0; CellIdx < Data->mCells.GetSize(); CellIdx++) { GdkRectangle& rect = Data->mCells[CellIdx].Rect; if (event->x >= rect.x && event->y >= rect.y && event->x <= rect.x + rect.width && event->y <= rect.y + rect.height) { NewColor = CellIdx; break; } } if (NewColor != cur_color) { cur_color = NewColor; lcGetActiveProject()->HandleNotify(LC_COLOR_CHANGED, Data->mCells[cur_color].ColorIndex); gtk_widget_draw(widget, NULL); preview->Redraw(); } gtk_window_set_focus(GTK_WINDOW(((GtkWidget*)(*main_window))), widget); return TRUE; } static gboolean colorlist_tooltip(GtkWidget *widget, gint x, gint y, gboolean keyboard_mode, GtkTooltip *tooltip, gpointer user_data) { if (keyboard_mode) return FALSE; ColorListData* Data = (ColorListData*)gtk_object_get_data(GTK_OBJECT(widget), "colorlist"); for (int CellIdx = 0; CellIdx < Data->mCells.GetSize(); CellIdx++) { GdkRectangle& rect = Data->mCells[CellIdx].Rect; if (x >= rect.x && y >= rect.y && x <= rect.x + rect.width && y <= rect.y + rect.height) { int ColorIndex = Data->mCells[CellIdx].ColorIndex; gtk_tooltip_set_text(tooltip, gColorList[ColorIndex].Name); GdkPixbuf* pixbuf = gdk_pixbuf_new(GDK_COLORSPACE_RGB, FALSE, 8, 24, 24); float* Value = gColorList[ColorIndex].Value; guint32 Color = ((int)(Value[0] * 255) << 24) | ((int)(Value[1] * 255) << 16) | ((int)(Value[2] * 255) << 8); gdk_pixbuf_fill(pixbuf, Color); gtk_tooltip_set_icon(tooltip, pixbuf); g_object_unref(pixbuf); return TRUE; } } return FALSE; } void colorlist_set(int new_color) { if (new_color != cur_color) { cur_color = new_color; if (colorlist) gtk_widget_draw(colorlist, NULL); if (preview) preview->Redraw(); } } // Create the pieces toolbar GtkWidget* create_piecebar(GtkWidget *window, GLWindow *share) { GtkWidget *vbox1, *hbox, *vpan, *scroll_win, *frame, *button, *arrow; frame = gtk_frame_new(NULL); gtk_widget_show(frame); gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_IN); vbox1 = gtk_vbox_new(FALSE, 0); gtk_widget_show(vbox1); gtk_container_add(GTK_CONTAINER(frame), vbox1); gtk_container_border_width(GTK_CONTAINER(vbox1), 2); vpan = gtk_vpaned_new(); gtk_widget_show(vpan); gtk_box_pack_start(GTK_BOX(vbox1), vpan, TRUE, TRUE, 0); GtkWidget *w; preview = new PiecePreview(share); preview->CreateFromWindow(&w); gtk_widget_set_usize(w, 100, 100); gtk_widget_show(w); gtk_container_add(GTK_CONTAINER(vpan), w); scroll_win = gtk_scrolled_window_new(NULL, NULL); gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll_win), GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS); gtk_widget_show(scroll_win); gtk_container_add(GTK_CONTAINER(vpan), scroll_win); GtkTreeStore* store = gtk_tree_store_new(2, G_TYPE_STRING, G_TYPE_POINTER); piecetree = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store)); gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(piecetree), false); GtkCellRenderer* renderer = gtk_cell_renderer_text_new(); GtkTreeViewColumn* column; column = gtk_tree_view_column_new_with_attributes("Piece", renderer, "text", 0, NULL); gtk_tree_view_append_column(GTK_TREE_VIEW(piecetree), column); GtkTreeSelection* select; select = gtk_tree_view_get_selection(GTK_TREE_VIEW(piecetree)); gtk_tree_selection_set_mode(select, GTK_SELECTION_SINGLE); g_signal_connect(G_OBJECT(select), "changed", G_CALLBACK(piecetree_changed), NULL); gtk_drag_source_set(GTK_WIDGET(piecetree), GDK_BUTTON1_MASK, drag_target_list, 1, GDK_ACTION_COPY); g_signal_connect(G_OBJECT(piecetree), "drag_begin", G_CALLBACK(piecetree_drag_begin), NULL); g_signal_connect(G_OBJECT(piecetree), "drag_end", G_CALLBACK(drag_end), NULL); gtk_container_add(GTK_CONTAINER(scroll_win), piecetree); gtk_widget_show(piecetree); // Piece combo hbox = gtk_hbox_new(FALSE, 1); gtk_widget_show(hbox); gtk_box_pack_start(GTK_BOX(vbox1), hbox, FALSE, TRUE, 1); pieceentry = gtk_entry_new(); gtk_widget_show(pieceentry); gtk_box_pack_start(GTK_BOX(hbox), pieceentry, TRUE, TRUE, 0); gtk_signal_connect(GTK_OBJECT(pieceentry), "changed", GTK_SIGNAL_FUNC(piececombo_changed), NULL); gtk_signal_connect(GTK_OBJECT(pieceentry), "key_press_event", GTK_SIGNAL_FUNC(piececombo_key), NULL); button = gtk_button_new(); gtk_widget_show(button); gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, TRUE, 0); gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(piececombo_popup), NULL); arrow = gtk_arrow_new(GTK_ARROW_DOWN, GTK_SHADOW_OUT); gtk_widget_show(arrow); gtk_container_add(GTK_CONTAINER(button), arrow); // Color list colorlist = gtk_drawing_area_new(); gtk_widget_set_events(colorlist, GDK_EXPOSURE_MASK | GDK_KEY_PRESS_MASK | GDK_BUTTON_PRESS_MASK); GTK_WIDGET_SET_FLAGS(colorlist, GTK_CAN_FOCUS); gtk_drawing_area_size(GTK_DRAWING_AREA(colorlist), 200, 160); gtk_box_pack_start(GTK_BOX(vbox1), colorlist, FALSE, TRUE, 0); gtk_widget_set_has_tooltip(colorlist, TRUE); gtk_signal_connect(GTK_OBJECT(colorlist), "expose_event", GTK_SIGNAL_FUNC(colorlist_expose), NULL); gtk_signal_connect(GTK_OBJECT(colorlist), "button_press_event", GTK_SIGNAL_FUNC(colorlist_button_press), NULL); gtk_signal_connect(GTK_OBJECT(colorlist), "key_press_event", GTK_SIGNAL_FUNC(colorlist_key_press), NULL); gtk_signal_connect(GTK_OBJECT(colorlist), "realize", GTK_SIGNAL_FUNC(colorlist_realize), NULL); gtk_signal_connect(GTK_OBJECT(colorlist), "unrealize", GTK_SIGNAL_FUNC(colorlist_unrealize), NULL); gtk_signal_connect(GTK_OBJECT(colorlist), "query-tooltip", GTK_SIGNAL_FUNC(colorlist_tooltip), NULL); gtk_signal_connect(GTK_OBJECT(colorlist), "drag_end", G_CALLBACK(drag_end), NULL); gtk_widget_show(colorlist); fill_piecetree(); PieceInfo* Info = lcGetPiecesLibrary()->FindPiece("3005", false); if (!Info) Info = lcGetPiecesLibrary()->mPieces[0]; if (Info) { lcGetActiveProject()->SetCurrentPiece(Info); // preview->SetCurrentPiece(Info); } return frame; } // ============================================================================= // Status bar GtkWidget *label_message, *label_position, *label_snap, *label_step; static void statusbar_listener (int message, void *data, void *user) { if (message == LC_MSG_FOCUS_CHANGED) { char text[64]; lcVector3 pos; lcGetActiveProject()->GetFocusPosition(pos); lcGetActiveProject()->ConvertToUserUnits(pos); sprintf (text, "X: %.2f Y: %.2f Z: %.2f", pos[0], pos[1], pos[2]); gtk_label_set (GTK_LABEL (label_position), text); } } static gint statusbar_popup(GtkWidget *widget, GdkEvent *event, GtkWidget *menu_widget) { GtkMenu *menu; GdkEventButton *event_button; g_return_val_if_fail(widget != NULL, FALSE); g_return_val_if_fail(event != NULL, FALSE); g_return_val_if_fail(menu_widget != NULL, FALSE); g_return_val_if_fail(GTK_IS_MENU(menu_widget), FALSE); menu = GTK_MENU(menu_widget); if (event->type == GDK_BUTTON_PRESS) { event_button = (GdkEventButton *) event; if (event_button->button == 3) { gtk_menu_popup(menu, NULL, NULL, NULL, NULL, event_button->button, event_button->time); return TRUE; } } return FALSE; } void create_statusbar(GtkWidget *window, GtkWidget *vbox) { GtkWidget *hbox, *hbox1, *frame, *ebox_snap; hbox = gtk_hbox_new (FALSE, 0); gtk_widget_show (hbox); gtk_container_border_width (GTK_CONTAINER (hbox), 1); gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 2); frame = gtk_frame_new (NULL); gtk_widget_show (frame); gtk_box_pack_start (GTK_BOX (hbox), frame, TRUE, TRUE, 0); gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN); hbox1 = gtk_hbox_new (FALSE, 0); gtk_container_add (GTK_CONTAINER (frame), hbox1); gtk_container_border_width (GTK_CONTAINER (hbox1), 0); gtk_widget_show (hbox1); label_message = gtk_label_new (" "); gtk_widget_show (label_message); gtk_box_pack_start (GTK_BOX (hbox1), label_message, FALSE, TRUE, 0); gtk_label_set_justify (GTK_LABEL (label_message), GTK_JUSTIFY_LEFT); gtk_misc_set_padding (GTK_MISC (label_message), 3, 0); frame = gtk_frame_new (NULL); gtk_widget_show (frame); gtk_box_pack_start (GTK_BOX (hbox), frame, FALSE, TRUE, 0); gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN); hbox1 = gtk_hbox_new (FALSE, 0); gtk_container_add (GTK_CONTAINER (frame), hbox1); gtk_container_border_width (GTK_CONTAINER (hbox1), 0); gtk_widget_show (hbox1); label_position = gtk_label_new (" "); gtk_widget_show (label_position); gtk_box_pack_start (GTK_BOX (hbox1), label_position, TRUE, TRUE, 0); frame = gtk_frame_new (NULL); gtk_widget_show (frame); gtk_box_pack_start (GTK_BOX (hbox), frame, FALSE, TRUE, 0); gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN); hbox1 = gtk_hbox_new (FALSE, 0); gtk_container_add (GTK_CONTAINER (frame), hbox1); gtk_container_border_width (GTK_CONTAINER (hbox1), 0); gtk_widget_show (hbox1); ebox_snap = gtk_event_box_new(); gtk_widget_show(ebox_snap); gtk_box_pack_start(GTK_BOX(hbox1), ebox_snap, TRUE, TRUE, 0); label_snap = gtk_label_new (" "); gtk_widget_show (label_snap); gtk_container_add(GTK_CONTAINER(ebox_snap), label_snap); frame = gtk_frame_new (NULL); gtk_widget_show (frame); gtk_box_pack_start (GTK_BOX (hbox), frame, FALSE, TRUE, 0); gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN); hbox1 = gtk_hbox_new (FALSE, 0); gtk_container_add (GTK_CONTAINER (frame), hbox1); gtk_container_border_width (GTK_CONTAINER (hbox1), 0); gtk_widget_show (hbox1); label_step = gtk_label_new (" "); gtk_widget_show (label_step); gtk_box_pack_start (GTK_BOX (hbox1), label_step, TRUE, TRUE, 0); messenger->Listen (&statusbar_listener, NULL); // Add snap popup menu GtkWidget *menu, *item; menu = gtk_menu_new(); item = gtk_menu_item_new_with_label("XY Snap"); gtk_widget_set_sensitive(item, FALSE); gtk_menu_append(menu, item); create_menu_item(menu, "None", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_0, NULL); create_menu_item(menu, "1/20 Stud", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_1, NULL); create_menu_item(menu, "1/4 Stud", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_2, NULL); create_menu_item(menu, "1 Flat", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_3, NULL); create_menu_item(menu, "1/2 Stud", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_4, NULL); create_menu_item(menu, "1 Stud", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_5, NULL); create_menu_item(menu, "2 Studs", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_6, NULL); create_menu_item(menu, "3 Studs", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_7, NULL); create_menu_item(menu, "4 Studs", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_8, NULL); create_menu_item(menu, "8 Studs", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEXY_SNAP_9, NULL); menu_separator(menu); item = gtk_menu_item_new_with_label("Z Snap"); gtk_widget_set_sensitive(item, FALSE); gtk_menu_append(menu, item); create_menu_item(menu, "None", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_0, NULL); create_menu_item(menu, "1/20 Stud", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_1, NULL); create_menu_item(menu, "1/4 Stud", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_2, NULL); create_menu_item(menu, "1 Flat", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_3, NULL); create_menu_item(menu, "1/2 Stud", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_4, NULL); create_menu_item(menu, "1 Stud", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_5, NULL); create_menu_item(menu, "1 Brick", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_6, NULL); create_menu_item(menu, "2 Bricks", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_7, NULL); create_menu_item(menu, "4 Bricks", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_8, NULL); create_menu_item(menu, "8 Bricks", NULL, GTK_SIGNAL_FUNC(OnCommandDirect), NULL, LC_EDIT_MOVEZ_SNAP_9, NULL); gtk_widget_show_all(menu); gtk_signal_connect(GTK_OBJECT(ebox_snap), "button_press_event", GTK_SIGNAL_FUNC(statusbar_popup), menu); }