svn merge https://svn.blender.org/svnroot/bf-blender/trunk/blender -r22625:22668
[blender.git] / source / blender / editors / space_text / space_text.c
1 /**
2  * $Id:
3  *
4  * ***** BEGIN GPL LICENSE BLOCK *****
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License
8  * as published by the Free Software Foundation; either version 2
9  * of the License, or (at your option) any later version. 
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software Foundation,
18  * Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
19  *
20  * The Original Code is Copyright (C) 2008 Blender Foundation.
21  * All rights reserved.
22  *
23  * 
24  * Contributor(s): Blender Foundation
25  *
26  * ***** END GPL LICENSE BLOCK *****
27  */
28
29 #include <string.h>
30 #include <stdio.h>
31
32 #include "DNA_text_types.h"
33 #include "DNA_object_types.h"
34 #include "DNA_space_types.h"
35 #include "DNA_scene_types.h"
36 #include "DNA_screen_types.h"
37 #include "DNA_userdef_types.h"
38
39 #include "MEM_guardedalloc.h"
40
41 #include "BLI_blenlib.h"
42 #include "BLI_arithb.h"
43 #include "BLI_rand.h"
44
45 #include "BKE_colortools.h"
46 #include "BKE_context.h"
47 #include "BKE_screen.h"
48
49 #include "ED_space_api.h"
50 #include "ED_screen.h"
51
52 #include "BIF_gl.h"
53
54 #include "WM_api.h"
55 #include "WM_types.h"
56
57 #include "UI_interface.h"
58 #include "UI_resources.h"
59 #include "UI_view2d.h"
60
61 #include "RNA_access.h"
62
63 #include "ED_markers.h"
64
65 #include "text_intern.h"        // own include
66
67 /* ******************** default callbacks for text space ***************** */
68
69 static SpaceLink *text_new(const bContext *C)
70 {
71         ARegion *ar;
72         SpaceText *stext;
73         
74         stext= MEM_callocN(sizeof(SpaceText), "inittext");
75         stext->spacetype= SPACE_TEXT;
76
77         stext->lheight= 12;
78         stext->tabnumber= 4;
79         
80         /* header */
81         ar= MEM_callocN(sizeof(ARegion), "header for text");
82         
83         BLI_addtail(&stext->regionbase, ar);
84         ar->regiontype= RGN_TYPE_HEADER;
85         ar->alignment= RGN_ALIGN_BOTTOM;
86         
87         /* main area */
88         ar= MEM_callocN(sizeof(ARegion), "main area for text");
89         
90         BLI_addtail(&stext->regionbase, ar);
91         ar->regiontype= RGN_TYPE_WINDOW;
92         
93         return (SpaceLink *)stext;
94 }
95
96 /* not spacelink itself */
97 static void text_free(SpaceLink *sl)
98 {       
99         SpaceText *stext= (SpaceText*) sl;
100         
101         stext->text= NULL;
102 }
103
104
105 /* spacetype; init callback */
106 static void text_init(struct wmWindowManager *wm, ScrArea *sa)
107 {
108
109 }
110
111 static SpaceLink *text_duplicate(SpaceLink *sl)
112 {
113         SpaceText *stextn= MEM_dupallocN(sl);
114         
115         /* clear or remove stuff from old */
116         
117         return (SpaceLink *)stextn;
118 }
119
120 static void text_listener(ScrArea *sa, wmNotifier *wmn)
121 {
122         SpaceText *st= sa->spacedata.first;
123
124         /* context changes */
125         switch(wmn->category) {
126                 case NC_TEXT:
127                         if(!wmn->reference || wmn->reference == st->text) {
128                                 ED_area_tag_redraw(sa);
129
130                                 if(wmn->data == ND_CURSOR || wmn->action == NA_EDITED) {
131                                         ARegion *ar;
132
133                                         for(ar=sa->regionbase.first; ar; ar= ar->next)
134                                                 if(ar->regiontype==RGN_TYPE_WINDOW)
135                                                         text_update_cursor_moved(st, ar);
136                                 }
137
138                                 if(wmn->action == NA_EDITED)
139                                         if(st->text)
140                                                 text_update_edited(st->text);
141                         }
142                         else if(wmn->data == ND_DISPLAY)
143                                 ED_area_tag_redraw(sa);
144
145                         break;
146         }
147 }
148
149 static void text_operatortypes(void)
150 {
151         WM_operatortype_append(TEXT_OT_new);
152         WM_operatortype_append(TEXT_OT_open);
153         WM_operatortype_append(TEXT_OT_reload);
154         WM_operatortype_append(TEXT_OT_unlink);
155         WM_operatortype_append(TEXT_OT_save);
156         WM_operatortype_append(TEXT_OT_save_as);
157         WM_operatortype_append(TEXT_OT_make_internal);
158         WM_operatortype_append(TEXT_OT_run_script);
159         WM_operatortype_append(TEXT_OT_refresh_pyconstraints);
160
161         WM_operatortype_append(TEXT_OT_paste);
162         WM_operatortype_append(TEXT_OT_copy);
163         WM_operatortype_append(TEXT_OT_cut);
164
165         WM_operatortype_append(TEXT_OT_convert_whitespace);
166         WM_operatortype_append(TEXT_OT_uncomment);
167         WM_operatortype_append(TEXT_OT_comment);
168         WM_operatortype_append(TEXT_OT_unindent);
169         WM_operatortype_append(TEXT_OT_indent);
170
171         WM_operatortype_append(TEXT_OT_markers_clear);
172         WM_operatortype_append(TEXT_OT_next_marker);
173         WM_operatortype_append(TEXT_OT_previous_marker);
174
175         WM_operatortype_append(TEXT_OT_select_line);
176         WM_operatortype_append(TEXT_OT_select_all);
177
178         WM_operatortype_append(TEXT_OT_jump);
179         WM_operatortype_append(TEXT_OT_move);
180         WM_operatortype_append(TEXT_OT_move_select);
181         WM_operatortype_append(TEXT_OT_delete);
182         WM_operatortype_append(TEXT_OT_overwrite_toggle);
183
184         WM_operatortype_append(TEXT_OT_cursor_set);
185         WM_operatortype_append(TEXT_OT_scroll);
186         WM_operatortype_append(TEXT_OT_scroll_bar);
187         WM_operatortype_append(TEXT_OT_line_number);
188
189         WM_operatortype_append(TEXT_OT_line_break);
190         WM_operatortype_append(TEXT_OT_insert);
191
192         WM_operatortype_append(TEXT_OT_properties);
193
194         WM_operatortype_append(TEXT_OT_find);
195         WM_operatortype_append(TEXT_OT_find_set_selected);
196         WM_operatortype_append(TEXT_OT_replace);
197         WM_operatortype_append(TEXT_OT_replace_set_selected);
198         WM_operatortype_append(TEXT_OT_mark_all);
199
200         WM_operatortype_append(TEXT_OT_to_3d_object);
201
202         WM_operatortype_append(TEXT_OT_resolve_conflict);
203 }
204
205 static void text_keymap(struct wmWindowManager *wm)
206 {
207         ListBase *keymap= WM_keymap_listbase(wm, "Text", SPACE_TEXT, 0);
208         
209         WM_keymap_add_item(keymap, "TEXT_OT_new", NKEY, KM_PRESS, KM_ALT, 0);
210         WM_keymap_add_item(keymap, "TEXT_OT_open", OKEY, KM_PRESS, KM_ALT, 0);
211         WM_keymap_add_item(keymap, "TEXT_OT_reload", RKEY, KM_PRESS, KM_ALT, 0);
212         WM_keymap_add_item(keymap, "TEXT_OT_save", SKEY, KM_PRESS, KM_ALT, 0);
213
214         WM_keymap_add_item(keymap, "TEXT_OT_run_script", PKEY, KM_PRESS, KM_ALT, 0);
215
216         WM_keymap_add_item(keymap, "TEXT_OT_cut", XKEY, KM_PRESS, KM_ALT, 0);
217         WM_keymap_add_item(keymap, "TEXT_OT_cut", XKEY, KM_PRESS, KM_CTRL, 0);
218         WM_keymap_add_item(keymap, "TEXT_OT_copy", CKEY, KM_PRESS, KM_ALT, 0);
219         WM_keymap_add_item(keymap, "TEXT_OT_copy", CKEY, KM_PRESS, KM_CTRL, 0);
220         WM_keymap_add_item(keymap, "TEXT_OT_paste", VKEY, KM_PRESS, KM_ALT, 0);
221         WM_keymap_add_item(keymap, "TEXT_OT_paste", VKEY, KM_PRESS, KM_CTRL, 0);
222
223         if(U.uiflag & USER_MMB_PASTE) // XXX not dynamic
224                 RNA_boolean_set(WM_keymap_add_item(keymap, "TEXT_OT_paste", MIDDLEMOUSE, KM_PRESS, 0, 0)->ptr, "selection", 1);
225
226         WM_keymap_add_item(keymap, "TEXT_OT_jump", JKEY, KM_PRESS, KM_ALT, 0);
227         WM_keymap_add_item(keymap, "TEXT_OT_find", FKEY, KM_PRESS, KM_ALT, 0);
228         WM_keymap_add_item(keymap, "TEXT_OT_properties", FKEY, KM_PRESS, KM_ALT, 0);
229         WM_keymap_add_item(keymap, "TEXT_OT_properties", FKEY, KM_PRESS, KM_CTRL, 0);
230         WM_keymap_add_item(keymap, "TEXT_OT_replace", HKEY, KM_PRESS, KM_ALT, 0);
231         WM_keymap_add_item(keymap, "TEXT_OT_replace", HKEY, KM_PRESS, KM_CTRL, 0);
232
233         WM_keymap_add_item(keymap, "TEXT_OT_to_3d_object", MKEY, KM_PRESS, KM_ALT, 0);
234
235         WM_keymap_add_item(keymap, "TEXT_OT_select_all", AKEY, KM_PRESS, KM_CTRL, 0);
236
237         WM_keymap_add_item(keymap, "TEXT_OT_indent", TABKEY, KM_PRESS, 0, 0);
238         WM_keymap_add_item(keymap, "TEXT_OT_unindent", TABKEY, KM_PRESS, KM_SHIFT, 0);
239         WM_keymap_add_item(keymap, "TEXT_OT_uncomment", DKEY, KM_PRESS, KM_CTRL|KM_SHIFT, 0);
240
241         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", HOMEKEY, KM_PRESS, 0, 0)->ptr, "type", LINE_BEGIN);
242         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", LEFTARROWKEY, KM_PRESS, KM_OSKEY, 0)->ptr, "type", LINE_BEGIN);
243         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", ENDKEY, KM_PRESS, 0, 0)->ptr, "type", LINE_END);
244         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", RIGHTARROWKEY, KM_PRESS, KM_OSKEY, 0)->ptr, "type", LINE_BEGIN);
245         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", EKEY, KM_PRESS, KM_CTRL, 0)->ptr, "type", LINE_END);
246         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", EKEY, KM_PRESS, KM_CTRL|KM_SHIFT, 0)->ptr, "type", LINE_END);
247         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", ENDKEY, KM_PRESS, 0, 0)->ptr, "type", LINE_END);
248         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", LEFTARROWKEY, KM_PRESS, 0, 0)->ptr, "type", PREV_CHAR);
249         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", RIGHTARROWKEY, KM_PRESS, 0, 0)->ptr, "type", NEXT_CHAR);
250         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", LEFTARROWKEY, KM_PRESS, KM_CTRL, 0)->ptr, "type", PREV_WORD);
251         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", RIGHTARROWKEY, KM_PRESS, KM_CTRL, 0)->ptr, "type", NEXT_WORD);
252         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", UPARROWKEY, KM_PRESS, 0, 0)->ptr, "type", PREV_LINE);
253         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", DOWNARROWKEY, KM_PRESS, 0, 0)->ptr, "type", NEXT_LINE);
254         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", PAGEUPKEY, KM_PRESS, 0, 0)->ptr, "type", PREV_PAGE);
255         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move", PAGEDOWNKEY, KM_PRESS, 0, 0)->ptr, "type", NEXT_PAGE);
256
257         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", HOMEKEY, KM_PRESS, KM_SHIFT, 0)->ptr, "type", LINE_BEGIN);
258         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", ENDKEY, KM_PRESS, KM_SHIFT, 0)->ptr, "type", LINE_END);
259         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", LEFTARROWKEY, KM_PRESS, KM_SHIFT|KM_OSKEY, 0)->ptr, "type", LINE_BEGIN);
260         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", RIGHTARROWKEY, KM_PRESS, KM_SHIFT|KM_OSKEY, 0)->ptr, "type", LINE_BEGIN);
261         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", LEFTARROWKEY, KM_PRESS, KM_SHIFT, 0)->ptr, "type", PREV_CHAR);
262         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", RIGHTARROWKEY, KM_PRESS, KM_SHIFT, 0)->ptr, "type", NEXT_CHAR);
263         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", LEFTARROWKEY, KM_PRESS, KM_SHIFT|KM_CTRL, 0)->ptr, "type", PREV_WORD);
264         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", RIGHTARROWKEY, KM_PRESS, KM_SHIFT|KM_CTRL, 0)->ptr, "type", NEXT_WORD);
265         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", UPARROWKEY, KM_PRESS, KM_SHIFT, 0)->ptr, "type", PREV_LINE);
266         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", DOWNARROWKEY, KM_PRESS, KM_SHIFT, 0)->ptr, "type", NEXT_LINE);
267         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", PAGEUPKEY, KM_PRESS, KM_SHIFT, 0)->ptr, "type", PREV_PAGE);
268         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_move_select", PAGEDOWNKEY, KM_PRESS, KM_SHIFT, 0)->ptr, "type", NEXT_PAGE);
269
270         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_delete", DELKEY, KM_PRESS, 0, 0)->ptr, "type", DEL_NEXT_CHAR);
271         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_delete", DKEY, KM_PRESS, KM_CTRL, 0)->ptr, "type", DEL_NEXT_CHAR);
272         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_delete", BACKSPACEKEY, KM_PRESS, 0, 0)->ptr, "type", DEL_PREV_CHAR);
273         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_delete", DELKEY, KM_PRESS, KM_CTRL, 0)->ptr, "type", DEL_NEXT_WORD);
274         RNA_enum_set(WM_keymap_add_item(keymap, "TEXT_OT_delete", BACKSPACEKEY, KM_PRESS, KM_CTRL, 0)->ptr, "type", DEL_PREV_WORD);
275
276         WM_keymap_add_item(keymap, "TEXT_OT_overwrite_toggle", INSERTKEY, KM_PRESS, 0, 0);
277
278         WM_keymap_add_item(keymap, "TEXT_OT_scroll", MIDDLEMOUSE, KM_PRESS, 0, 0);
279         WM_keymap_add_item(keymap, "TEXT_OT_scroll_bar", LEFTMOUSE, KM_PRESS, 0, 0);
280         WM_keymap_add_item(keymap, "TEXT_OT_cursor_set", LEFTMOUSE, KM_PRESS, 0, 0);
281         RNA_boolean_set(WM_keymap_add_item(keymap, "TEXT_OT_cursor_set", LEFTMOUSE, KM_PRESS, KM_SHIFT, 0)->ptr, "select", 1);
282         RNA_int_set(WM_keymap_add_item(keymap, "TEXT_OT_scroll", WHEELUPMOUSE, KM_PRESS, 0, 0)->ptr, "lines", -1);
283         RNA_int_set(WM_keymap_add_item(keymap, "TEXT_OT_scroll", WHEELDOWNMOUSE, KM_PRESS, 0, 0)->ptr, "lines", 1);
284
285         WM_keymap_add_item(keymap, "TEXT_OT_to_3d_object", MKEY, KM_PRESS, KM_ALT, 0);
286
287         WM_keymap_add_item(keymap, "TEXT_OT_line_break", RETKEY, KM_PRESS, 0, 0);
288
289         WM_keymap_add_item(keymap, "TEXT_OT_line_number", KM_TEXTINPUT, KM_ANY, KM_ANY, 0);
290         WM_keymap_add_item(keymap, "TEXT_OT_insert", KM_TEXTINPUT, KM_ANY, KM_ANY, 0); // last!
291 }
292
293 static int text_context(const bContext *C, const char *member, bContextDataResult *result)
294 {
295         SpaceText *st= CTX_wm_space_text(C);
296
297         if(CTX_data_dir(member)) {
298                 static const char *dir[] = {"edit_text", NULL};
299                 CTX_data_dir_set(result, dir);
300                 return 1;
301         }
302         else if(CTX_data_equals(member, "edit_text")) {
303                 CTX_data_id_pointer_set(result, &st->text->id);
304                 return 1;
305         }
306
307         return 0;
308 }
309
310 /********************* main region ********************/
311
312 /* add handlers, stuff you only do once or on area/region changes */
313 static void text_main_area_init(wmWindowManager *wm, ARegion *ar)
314 {
315         ListBase *keymap;
316         
317         UI_view2d_region_reinit(&ar->v2d, V2D_COMMONVIEW_STANDARD, ar->winx, ar->winy);
318         
319         /* own keymap */
320         keymap= WM_keymap_listbase(wm, "Text", SPACE_TEXT, 0);  /* XXX weak? */
321         WM_event_add_keymap_handler_bb(&ar->handlers, keymap, &ar->v2d.mask, &ar->winrct);
322 }
323
324 static void text_main_area_draw(const bContext *C, ARegion *ar)
325 {
326         /* draw entirely, view changes should be handled here */
327         SpaceText *st= CTX_wm_space_text(C);
328         //View2D *v2d= &ar->v2d;
329         float col[3];
330         
331         /* clear and setup matrix */
332         UI_GetThemeColor3fv(TH_BACK, col);
333         glClearColor(col[0], col[1], col[2], 0.0);
334         glClear(GL_COLOR_BUFFER_BIT);
335         
336         // UI_view2d_view_ortho(C, v2d);
337                 
338         /* data... */
339         draw_text_main(st, ar);
340         
341         /* reset view matrix */
342         // UI_view2d_view_restore(C);
343         
344         /* scrollers? */
345 }
346
347 static void text_cursor(wmWindow *win, ScrArea *sa, ARegion *ar)
348 {
349         WM_cursor_set(win, BC_TEXTEDITCURSOR);
350 }
351
352 /****************** header region ******************/
353
354 /* add handlers, stuff you only do once or on area/region changes */
355 static void text_header_area_init(wmWindowManager *wm, ARegion *ar)
356 {
357         ED_region_header_init(ar);
358 }
359
360 static void text_header_area_draw(const bContext *C, ARegion *ar)
361 {
362         ED_region_header(C, ar);
363 }
364
365 /****************** properties region ******************/
366
367 /* add handlers, stuff you only do once or on area/region changes */
368 static void text_properties_area_init(wmWindowManager *wm, ARegion *ar)
369 {
370         ED_region_panels_init(wm, ar);
371 }
372
373 static void text_properties_area_draw(const bContext *C, ARegion *ar)
374 {
375         ED_region_panels(C, ar, 1, NULL, -1);
376 }
377
378 /********************* registration ********************/
379
380 /* only called once, from space/spacetypes.c */
381 void ED_spacetype_text(void)
382 {
383         SpaceType *st= MEM_callocN(sizeof(SpaceType), "spacetype text");
384         ARegionType *art;
385         
386         st->spaceid= SPACE_TEXT;
387         
388         st->new= text_new;
389         st->free= text_free;
390         st->init= text_init;
391         st->duplicate= text_duplicate;
392         st->operatortypes= text_operatortypes;
393         st->keymap= text_keymap;
394         st->listener= text_listener;
395         st->context= text_context;
396         
397         /* regions: main window */
398         art= MEM_callocN(sizeof(ARegionType), "spacetype text region");
399         art->regionid = RGN_TYPE_WINDOW;
400         art->init= text_main_area_init;
401         art->draw= text_main_area_draw;
402         art->cursor= text_cursor;
403
404         BLI_addhead(&st->regiontypes, art);
405         
406         /* regions: properties */
407         art= MEM_callocN(sizeof(ARegionType), "spacetype text region");
408         art->regionid = RGN_TYPE_UI;
409         art->minsizex= UI_COMPACT_PANEL_WIDTH;
410         art->keymapflag= ED_KEYMAP_UI|ED_KEYMAP_VIEW2D;
411         
412         art->init= text_properties_area_init;
413         art->draw= text_properties_area_draw;
414         BLI_addhead(&st->regiontypes, art);
415
416         /* regions: header */
417         art= MEM_callocN(sizeof(ARegionType), "spacetype text region");
418         art->regionid = RGN_TYPE_HEADER;
419         art->minsizey= HEADERY;
420         art->keymapflag= ED_KEYMAP_UI|ED_KEYMAP_VIEW2D;
421         
422         art->init= text_header_area_init;
423         art->draw= text_header_area_draw;
424
425         BLI_addhead(&st->regiontypes, art);
426
427         BKE_spacetype_register(st);
428 }
429