UI: Add search-create callback
[blender-staging.git] / source / blender / editors / space_view3d / view3d_toolbar.c
1 /*
2  * ***** BEGIN GPL LICENSE BLOCK *****
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License
6  * as published by the Free Software Foundation; either version 2
7  * of the License, or (at your option) any later version. 
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software Foundation,
16  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17  *
18  * The Original Code is Copyright (C) 2009 Blender Foundation.
19  * All rights reserved.
20  *
21  * 
22  * Contributor(s): Blender Foundation
23  *
24  * ***** END GPL LICENSE BLOCK *****
25  */
26
27 /** \file blender/editors/space_view3d/view3d_toolbar.c
28  *  \ingroup spview3d
29  */
30
31
32 #include <string.h>
33 #include <stdio.h>
34 #include <math.h>
35 #include <float.h>
36
37 #include "DNA_object_types.h"
38 #include "DNA_scene_types.h"
39
40 #include "MEM_guardedalloc.h"
41
42 #include "BLI_blenlib.h"
43 #include "BLI_utildefines.h"
44 #include "BLI_ghash.h"
45
46 #include "BLT_translation.h"
47
48 #include "BKE_context.h"
49 #include "BKE_screen.h"
50
51
52 #include "WM_api.h"
53 #include "WM_types.h"
54
55 #include "RNA_access.h"
56
57 #include "ED_screen.h"
58 #include "ED_util.h"
59
60 #include "UI_interface.h"
61 #include "UI_resources.h"
62
63 #include "view3d_intern.h"  /* own include */
64
65
66 /* ******************* view3d space & buttons ************** */
67
68 static void view3d_panel_operator_redo_buts(const bContext *C, Panel *pa, wmOperator *op)
69 {
70         uiLayoutOperatorButs(C, pa->layout, op, NULL, 'V', 0);
71 }
72
73 static void view3d_panel_operator_redo_header(const bContext *C, Panel *pa)
74 {
75         wmOperator *op = WM_operator_last_redo(C);
76
77         if (op)
78                 BLI_strncpy(pa->drawname, RNA_struct_ui_name(op->type->srna), sizeof(pa->drawname));
79         else
80                 BLI_strncpy(pa->drawname, IFACE_("Operator"), sizeof(pa->drawname));
81 }
82
83 static void view3d_panel_operator_redo_operator(const bContext *C, Panel *pa, wmOperator *op)
84 {
85         if (op->type->flag & OPTYPE_MACRO) {
86                 for (op = op->macro.first; op; op = op->next) {
87                         uiItemL(pa->layout, RNA_struct_ui_name(op->type->srna), ICON_NONE);
88                         view3d_panel_operator_redo_operator(C, pa, op);
89                 }
90         }
91         else {
92                 view3d_panel_operator_redo_buts(C, pa, op);
93         }
94 }
95
96 /* TODO de-duplicate redo panel functions - campbell */
97 static void view3d_panel_operator_redo(const bContext *C, Panel *pa)
98 {
99         wmOperator *op = WM_operator_last_redo(C);
100         ARegion *ar;
101         ARegion *ar1;
102
103         if (op == NULL) {
104                 return;
105         }
106
107         /* keep in sync with logic in ED_undo_operator_repeat() */
108         ar = CTX_wm_region(C);
109         ar1 = BKE_area_find_region_active_win(CTX_wm_area(C));
110         if (ar1)
111                 CTX_wm_region_set((bContext *)C, ar1);
112
113         if (WM_operator_poll((bContext *)C, op->type)) {
114                 uiBlock *block = uiLayoutGetBlock(pa->layout);
115
116                 if (!WM_operator_check_ui_enabled(C, op->type->name))
117                         uiLayoutSetEnabled(pa->layout, false);
118
119                 /* note, blockfunc is a default but->func, use Handle func to allow button callbacks too */
120                 UI_block_func_handle_set(block, ED_undo_operator_repeat_cb_evt, op);
121
122                 view3d_panel_operator_redo_operator(C, pa, op);
123         }
124
125         /* set region back */
126         CTX_wm_region_set((bContext *)C, ar);
127 }
128
129 /* ******************* */
130
131 typedef struct CustomTool {
132         struct CustomTool *next, *prev;
133         char opname[OP_MAX_TYPENAME];
134         char context[OP_MAX_TYPENAME];
135 } CustomTool;
136
137 static void operator_call_cb(struct bContext *C, void *arg_listbase, void *arg2)
138 {
139         wmOperatorType *ot = arg2;
140         
141         if (ot) {
142                 CustomTool *ct = MEM_callocN(sizeof(CustomTool), "CustomTool");
143                 
144                 BLI_addtail(arg_listbase, ct);
145                 BLI_strncpy(ct->opname, ot->idname, OP_MAX_TYPENAME);
146                 BLI_strncpy(ct->context, CTX_data_mode_string(C), OP_MAX_TYPENAME);
147         }
148                 
149 }
150
151 static void operator_search_cb(const struct bContext *C, void *UNUSED(arg), const char *str, uiSearchItems *items)
152 {
153         GHashIterator iter;
154
155         for (WM_operatortype_iter(&iter); !BLI_ghashIterator_done(&iter); BLI_ghashIterator_step(&iter)) {
156                 wmOperatorType *ot = BLI_ghashIterator_getValue(&iter);
157
158                 if (BLI_strcasestr(ot->name, str)) {
159                         if (WM_operator_poll((bContext *)C, ot)) {
160                                 
161                                 if (false == UI_search_item_add(items, ot->name, ot, 0))
162                                         break;
163                         }
164                 }
165         }
166 }
167
168
169 /* ID Search browse menu, open */
170 static uiBlock *tool_search_menu(bContext *C, ARegion *ar, void *arg_listbase)
171 {
172         static char search[OP_MAX_TYPENAME];
173         wmEvent event;
174         wmWindow *win = CTX_wm_window(C);
175         uiBlock *block;
176         uiBut *but;
177         
178         /* clear initial search string, then all items show */
179         search[0] = 0;
180         
181         block = UI_block_begin(C, ar, "_popup", UI_EMBOSS);
182         UI_block_flag_enable(block, UI_BLOCK_LOOP | UI_BLOCK_SEARCH_MENU);
183         
184         /* fake button, it holds space for search items */
185         uiDefBut(block, UI_BTYPE_LABEL, 0, "", 10, 15, UI_searchbox_size_x(), UI_searchbox_size_y(), NULL, 0, 0, 0, 0, NULL);
186         
187         but = uiDefSearchBut(block, search, 0, ICON_VIEWZOOM, sizeof(search), 10, 0, 150, 19, 0, 0, "");
188         UI_but_func_search_set(but, NULL, operator_search_cb, arg_listbase, operator_call_cb, NULL);
189         
190         UI_block_bounds_set_normal(block, 6);
191         UI_block_direction_set(block, UI_DIR_DOWN);
192         UI_block_end(C, block);
193         
194         wm_event_init_from_window(win, &event);
195         event.type = EVT_BUT_OPEN;
196         event.val = KM_PRESS;
197         event.customdata = but;
198         event.customdatafree = false;
199         wm_event_add(win, &event);
200         
201         return block;
202 }
203
204
205 static void view3d_panel_tool_shelf(const bContext *C, Panel *pa)
206 {
207         SpaceLink *sl = CTX_wm_space_data(C);
208         SpaceType *st = NULL;
209         uiLayout *col;
210         const char *context = CTX_data_mode_string(C);
211         
212         if (sl)
213                 st = BKE_spacetype_from_id(sl->spacetype);
214         
215         if (st && st->toolshelf.first) {
216                 CustomTool *ct;
217                 
218                 for (ct = st->toolshelf.first; ct; ct = ct->next) {
219                         if (STREQLEN(context, ct->context, OP_MAX_TYPENAME)) {
220                                 col = uiLayoutColumn(pa->layout, true);
221                                 uiItemFullO(col, ct->opname, NULL, ICON_NONE, NULL, WM_OP_INVOKE_REGION_WIN, 0);
222                         }
223                 }
224         }
225         col = uiLayoutColumn(pa->layout, true);
226         uiDefBlockBut(uiLayoutGetBlock(pa->layout), tool_search_menu, &st->toolshelf, "Add Tool", 0, 0, UI_UNIT_X, UI_UNIT_Y, "Add Tool in shelf, gets saved in files");
227 }
228
229
230 void view3d_toolshelf_register(ARegionType *art)
231 {
232         PanelType *pt;
233
234         pt = MEM_callocN(sizeof(PanelType), "spacetype view3d panel tools");
235         strcpy(pt->idname, "VIEW3D_PT_tool_shelf");
236         strcpy(pt->label, N_("Tool Shelf"));
237         strcpy(pt->translation_context, BLT_I18NCONTEXT_DEFAULT_BPYRNA);
238         pt->draw = view3d_panel_tool_shelf;
239         BLI_addtail(&art->paneltypes, pt);
240 }
241
242 void view3d_tool_props_register(ARegionType *art)
243 {
244         PanelType *pt;
245         
246         pt = MEM_callocN(sizeof(PanelType), "spacetype view3d panel last operator");
247         strcpy(pt->idname, "VIEW3D_PT_last_operator");
248         strcpy(pt->label, N_("Operator"));
249         strcpy(pt->translation_context, BLT_I18NCONTEXT_DEFAULT_BPYRNA);
250         pt->draw_header = view3d_panel_operator_redo_header;
251         pt->draw = view3d_panel_operator_redo;
252         BLI_addtail(&art->paneltypes, pt);
253 }
254
255 /* ********** operator to open/close toolshelf region */
256
257 static int view3d_toolshelf_toggle_exec(bContext *C, wmOperator *UNUSED(op))
258 {
259         ScrArea *sa = CTX_wm_area(C);
260         ARegion *ar = view3d_has_tools_region(sa);
261         
262         if (ar)
263                 ED_region_toggle_hidden(C, ar);
264
265         return OPERATOR_FINISHED;
266 }
267
268 void VIEW3D_OT_toolshelf(wmOperatorType *ot)
269 {
270         ot->name = "Tool Shelf";
271         ot->description = "Toggles tool shelf display";
272         ot->idname = "VIEW3D_OT_toolshelf";
273         
274         ot->exec = view3d_toolshelf_toggle_exec;
275         ot->poll = ED_operator_view3d_active;
276         
277         /* flags */
278         ot->flag = 0;
279 }
280