leocad/linux/toolbar.cpp

1233 lines
42 KiB
C++

// 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);
}
// =========================================================
// 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 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_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_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);
}