Develop and Download Open Source Software

Browse CVS Repository

Contents of /shiki/shiki/shiki.c

Parent Directory Parent Directory | Revision Log Revision Log | View Revision Graph Revision Graph


Revision 1.14 - (show annotations) (download) (as text)
Sat Nov 11 05:03:37 2006 UTC (17 years, 5 months ago) by aloha
Branch: MAIN
Changes since 1.13: +53 -7 lines
File MIME type: text/x-csrc
add load_scheme_file, load_current_buffer

1 /* vim: set encoding=utf8:
2 *
3 * shiki.c
4 *
5 * Copyright(C)2006 WAKATSUKI toshihiro
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a
8 * copy of this software and associated documentation files (the "Software"),
9 * to deal in the Software without restriction, including without limitation
10 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
11 * and/or sell copies of the Software, and to permit persons to whom the
12 * Software is furnished to do so, subject to the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be included in
15 * all copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
23 * SOFTWARE.
24 *
25 * $Id: shiki.c,v 1.13 2006/11/11 04:05:06 aloha Exp $
26 */
27
28 #include<gauche.h>
29 #include<gtk/gtk.h>
30 #include<gdk/gdkkeysyms.h>
31
32 static gint editor_indent_width = 2;
33
34 static GtkWidget *editor_window;
35 static GtkScrolledWindow *current_tabpage;
36 static gint current_tabpage_num;
37 static const gchar *current_tabpage_label;
38 static GtkTextView *current_text_view;
39 static GtkTextBuffer *current_text_buffer;
40
41 static GtkWidget *statusbar;
42 static GtkWidget *modeline_label;
43
44 static gchar *get_all_buffer_contents(GtkTextBuffer *buffer);
45 static gchar *load_cstring_by_gauche(gchar *s);
46
47 /* ������������������������������������ */
48 static void load_buffer_by_gauche() {
49 GtkTextIter p;
50 gtk_text_buffer_get_end_iter(current_text_buffer, &p);
51 gtk_text_buffer_insert(current_text_buffer, &p, "\n\n", -1);
52 gtk_text_buffer_insert(current_text_buffer, &p, load_cstring_by_gauche(get_all_buffer_contents(current_text_buffer)), -1);
53 }
54
55 static gchar *get_filename_from_dialog(const gchar *msg);
56
57 /* ������������������������ */
58 static void load_scheme_file_by_gauche(GtkNotebook *notebook) {
59 gchar *contents, *text;
60 gsize br, bw, len;
61 GError *err = NULL;
62 gchar *filename = get_filename_from_dialog("File Selection");
63 GtkTextIter p;
64
65 if(!filename) return;
66
67 gtk_text_buffer_get_end_iter(current_text_buffer, &p);
68 gtk_text_buffer_insert(current_text_buffer, &p, "\n\n", -1);
69
70 if(g_file_get_contents(filename, &contents, &len, NULL)) {
71 if(!(text = g_locale_to_utf8(contents, -1, &br, &bw, &err)))
72 gtk_text_buffer_insert(current_text_buffer, &p, load_cstring_by_gauche(text), -1);
73 else
74 gtk_text_buffer_insert(current_text_buffer, &p, load_cstring_by_gauche(contents), -1);
75 }
76 g_free(text); g_free(contents); g_free(filename);
77 }
78
79 /* gauche ������������������������������������ */
80 static gchar *load_cstring_by_gauche(gchar *s) {
81 gchar *msg;
82
83 ScmObj result, error;
84 /* ��������������������������������� */
85 ScmObj is = Scm_MakeInputStringPort(SCM_STRING(SCM_MAKE_STR(s)), TRUE);
86 /* ������������������������������ */
87 ScmObj os = Scm_MakeOutputStringPort(TRUE);
88
89 Scm_Define(Scm_UserModule(), SCM_SYMBOL(SCM_INTERN("*input*")), is);
90 Scm_Define(Scm_UserModule(), SCM_SYMBOL(SCM_INTERN("*error*")), SCM_FALSE);
91 /* Scheme ��������������������������������������������������������������������������������� S ��������������������������������������������������������������������������� *error* ������������������ */
92 result = Scm_EvalCString("(guard (e (else (set! *error* e) #f)) (eval (load-from-port *input*) (current-module)))", SCM_OBJ(Scm_UserModule()));
93
94 error = Scm_GlobalVariableRef(Scm_UserModule(), SCM_SYMBOL(SCM_INTERN("*error*")), 0);
95
96 /* ��������������������������������������������������������� */
97 if (!SCM_FALSEP(error))
98 Scm_Write(error, os, SCM_WRITE_DISPLAY);
99 else
100 Scm_Write(result, os, SCM_WRITE_DISPLAY);
101
102 msg = Scm_GetString(SCM_STRING(Scm_GetOutputString(SCM_PORT(os))));
103 /* ������������������ */
104 Scm_ClosePort(SCM_PORT(is));
105 Scm_ClosePort(SCM_PORT(os));
106
107 return msg;
108 }
109
110 static void font_selection_ok(GtkWidget *button, GtkWidget *font_dialog) {
111 gchar *font_name = gtk_font_selection_dialog_get_font_name (GTK_FONT_SELECTION_DIALOG (font_dialog));
112 if(font_name) {
113 GtkRcStyle *style = gtk_rc_style_new ();
114 pango_font_description_free(style->font_desc);
115 style->font_desc = pango_font_description_from_string(font_name);
116 gtk_widget_modify_style (GTK_WIDGET(current_text_view), style);
117 g_free (font_name);
118 }
119 }
120
121 /* ������������������������������������������������������ */
122 static void select_font(){
123 GtkWidget *font_dialog = gtk_font_selection_dialog_new("Font Selection Dialog");
124 g_signal_connect (GTK_FONT_SELECTION_DIALOG (font_dialog)->ok_button, "clicked", G_CALLBACK(font_selection_ok), font_dialog);
125 gtk_dialog_run(GTK_DIALOG(font_dialog));
126 gtk_widget_destroy(font_dialog);
127 }
128
129 /* ������������������������������������������ */
130 static void about_this_application() {
131 GtkAboutDialog *about = GTK_ABOUT_DIALOG(gtk_about_dialog_new());
132 const gchar *authors[] = {
133 "������������ (���������) <alohakun@gmail.com>"
134 };
135 gtk_about_dialog_set_authors(about, authors);
136 gtk_about_dialog_set_copyright(about, "Copyright(C)2006 WAKATSUKI Toshihiro");
137 gtk_about_dialog_set_name(about, "��� (SHIKI)");
138 gtk_about_dialog_set_website_label(about, "���������30������������������������������������������������������������Blog");
139 gtk_about_dialog_set_website(about, "http://alohakun.blog7.fc2.com/blog-category-29.html");
140 gtk_dialog_run(GTK_DIALOG(about));
141 gtk_widget_destroy(GTK_WIDGET(about));
142 }
143
144 /* ��������������������������������������������������������������� */
145 static gint get_current_line_number(GtkTextBuffer *b) {
146 GtkTextIter p;
147 gtk_text_buffer_get_iter_at_mark(b, &p, gtk_text_buffer_get_insert(b));
148 return gtk_text_iter_get_line(&p) + 1;
149 }
150
151 /* ��������������������������������������������������������������� */
152 static void update_modeline_label() {
153 gchar* basename = g_path_get_basename(current_tabpage_label);
154 gchar* l = g_strdup_printf("-E:%s %-10s (Gauche Interaction)--L%d--------------------------------------",
155 gtk_text_buffer_get_modified(current_text_buffer) ? "**" : "--",
156 basename, get_current_line_number(current_text_buffer));
157 gtk_label_set_text(GTK_LABEL(modeline_label), l);
158 g_free(l); g_free(basename);
159 }
160
161 static void text_buffer_cursor_moved_handler(){
162 update_modeline_label();
163 }
164
165 /* ��������������������������������������������������������������� */
166 static gchar* get_all_buffer_contents(GtkTextBuffer *buffer) {
167 GtkTextIter start, end;
168 gtk_text_buffer_get_start_iter(buffer, &start);
169 gtk_text_buffer_get_end_iter(buffer, &end);
170 return gtk_text_buffer_get_text(buffer, &start, &end, FALSE);
171 }
172
173 /* buffer ������������������������ filename ��������� */
174 static gboolean save_text_buffer(const gchar *filename, GtkTextBuffer *buffer) {
175 gchar *contents, *text;
176 gsize br, bw;
177 GError *err = NULL;
178
179 if(!filename) return FALSE;
180 contents = get_all_buffer_contents(buffer);
181 text = g_locale_from_utf8(contents, -1, &br, &bw, &err);
182 /* ��������������������������������� */
183 g_file_set_contents(filename, text, -1, NULL);
184 gtk_text_buffer_set_modified(buffer, FALSE);
185 update_modeline_label();
186 g_free(contents); g_free(text);
187 return TRUE;
188 }
189
190 /* ������������������������������������������������������msg ������������������������������������ */
191 static gchar *get_filename_from_dialog(const gchar *msg) {
192
193 GtkWidget *dialog = gtk_file_selection_new(msg);
194 int resp = gtk_dialog_run(GTK_DIALOG(dialog));
195 gchar *filename = NULL;
196
197 /* gtk_file_selection_get_filename ������������������������������������������������������������������������������������������������������������������������������ */
198 if(resp == GTK_RESPONSE_OK)
199 filename = g_strdup(gtk_file_selection_get_filename(GTK_FILE_SELECTION(dialog)));
200
201 gtk_widget_destroy(dialog);
202 return filename;
203 }
204
205 /* ��������������������������������������������������������������������� */
206 static void save_file_from_notebook(GtkNotebook *notebook) {
207
208 /* ������������������������������������ */
209 if(!gtk_text_buffer_get_modified(current_text_buffer)) return;
210
211 /* ������������������������������������������������������������������������������������������������������ */
212 if(g_ascii_strcasecmp("*scratch*", current_tabpage_label) == 0) {
213 gchar *filename = get_filename_from_dialog("Save File As ...");
214 if(!filename) return;
215 if(!save_text_buffer(filename, current_text_buffer)) return;
216 gtk_notebook_set_tab_label_text(notebook, GTK_WIDGET(current_tabpage), filename);
217 gtk_window_set_title (GTK_WINDOW(editor_window), filename);
218 g_free(filename);
219 } else
220 save_text_buffer(current_tabpage_label, current_text_buffer);
221 }
222
223 /* ��������������������������������������������������� */
224 static void save_file_handler(GtkWidget *widget, GtkWidget *notebook) {
225 save_file_from_notebook(GTK_NOTEBOOK(notebook));
226 }
227
228 /* ��������������������������������������������������������������������������� */
229 static void save_file_as_from_notebook(GtkNotebook *notebook) {
230 gchar *filename = get_filename_from_dialog("Save File As ...");
231
232 if(!filename) return;
233 if(!save_text_buffer(filename, current_text_buffer)) return;
234
235 gtk_notebook_set_tab_label_text(notebook, GTK_WIDGET(current_tabpage), filename);
236 gtk_window_set_title (GTK_WINDOW (editor_window), filename);
237
238 g_free(filename);
239 }
240
241 /* ��������������������������������������������������������� */
242 static void save_file_as_handler(GtkWidget *widget, GtkWidget *notebook) {
243 save_file_as_from_notebook(GTK_NOTEBOOK(notebook));
244 }
245
246 /* YES ������������NO ������������������������������������ callback */
247 void really_quit_dialog_yes(GtkWidget *widget, gboolean *flag){*flag = FALSE;}
248 void really_quit_dialog_no(GtkWidget *widget, gint *flag){*flag = TRUE;}
249
250 /* ��������������������������������������������� ? */
251 gboolean not_yet_save_changes_really_quit(GtkTextBuffer *buffer) {
252 GtkWidget *yes_button, *no_button;
253 static GtkWidget *dialog_window = NULL;
254
255 /* ��������������������������������������� */
256 if(!gtk_text_buffer_get_modified(buffer)) return FALSE;
257
258 if(dialog_window == NULL) {
259 gboolean flag = TRUE;
260 dialog_window = gtk_dialog_new ();
261
262 /* ��������������������������������������������� ? ��������������������������� */
263 g_signal_connect(G_OBJECT(dialog_window), "delete_event", G_CALLBACK(gtk_false), NULL);
264 g_signal_connect(G_OBJECT(dialog_window), "destroy", G_CALLBACK(gtk_main_quit), NULL);
265 gtk_box_pack_start (GTK_BOX (GTK_DIALOG (dialog_window)->vbox),
266 gtk_label_new("������������������������������������������\n��������������������������������������� ?"), TRUE, TRUE, 0);
267 gtk_window_set_title(GTK_WINDOW (dialog_window), "Really Quit ?");
268 /* YES ������������ */
269 yes_button = gtk_button_new_with_mnemonic("������ (_Y)");
270 g_signal_connect(GTK_OBJECT(yes_button), "clicked", G_CALLBACK(really_quit_dialog_yes), &flag);
271 g_signal_connect_swapped(GTK_OBJECT(yes_button), "clicked", G_CALLBACK(gtk_widget_destroy), G_OBJECT(dialog_window));
272 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog_window)->action_area), yes_button, TRUE, TRUE, 0);
273
274 /* NO ������������ */
275 no_button = gtk_button_new_with_mnemonic("��������� (_N)");
276 g_signal_connect(GTK_OBJECT(no_button), "clicked", G_CALLBACK(really_quit_dialog_no), &flag);
277 g_signal_connect_swapped(GTK_OBJECT(no_button), "clicked", G_CALLBACK(gtk_widget_destroy), G_OBJECT(dialog_window));
278 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog_window)->action_area), no_button, TRUE, TRUE, 0);
279
280 gtk_window_set_modal(GTK_WINDOW(dialog_window), TRUE);
281 gtk_window_set_transient_for(GTK_WINDOW(dialog_window), GTK_WINDOW (editor_window));
282
283 gtk_widget_show_all(dialog_window);
284 gtk_main ();
285 dialog_window = NULL;
286
287 /* "delete_event" ��������������� FALSE ���������"destory" ������������������window ������������������ */
288 return flag;
289 }
290 return TRUE;
291 }
292
293 /* ������������������������������������������������������������������������������������������ */
294 gboolean delete_event_handler(GtkWidget *widget, GdkEvent *event, GtkWidget *buffer){
295 return not_yet_save_changes_really_quit(GTK_TEXT_BUFFER(buffer));
296 }
297
298 /* ������������������������������ */
299 static GtkWidget *new_scrolled_text_buffer() {
300
301 GtkWidget *scrolledwindow, *view;
302 GtkTextBuffer *buffer;
303
304 /* ������������������������������������ */
305 scrolledwindow = gtk_scrolled_window_new(NULL, NULL);
306 gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW(scrolledwindow), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
307
308 /* ������������������������������������������������ */
309 view = gtk_text_view_new();
310 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(view));
311 gtk_container_add(GTK_CONTAINER(scrolledwindow), view);
312 g_signal_connect(G_OBJECT(editor_window), "delete_event", G_CALLBACK(delete_event_handler), buffer);
313 gtk_widget_set_size_request(GTK_WIDGET(view), 500, 600);
314 g_signal_connect(buffer, "mark_set", G_CALLBACK(text_buffer_cursor_moved_handler), view);
315 /* ������������������������ */
316 /* ������������������������������������������������ */
317 gtk_text_buffer_create_tag (buffer, "parent_emphasis_background", "background", "green", NULL);
318
319 return scrolledwindow;
320 }
321
322 /* ��������������������� */
323 static void open_file_from_notebook(GtkNotebook *notebook) {
324 gchar *contents, *text;
325 gsize br, bw, len;
326 GError *err = NULL;
327 gchar *filename = get_filename_from_dialog("File Selection");
328
329 if(!filename) return;
330
331 if(g_file_get_contents(filename, &contents, &len, NULL)) {
332 GtkTextBuffer *buffer;
333 GtkWidget *scrolledwindow;
334 GtkTextIter p;
335
336 /* ������������������������������ */
337 gtk_notebook_append_page(GTK_NOTEBOOK(notebook),
338 scrolledwindow = new_scrolled_text_buffer(),
339 gtk_label_new(filename));
340 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtk_bin_get_child(GTK_BIN(scrolledwindow))));
341
342 if(!(text = g_locale_to_utf8(contents, -1, &br, &bw, &err)))
343 gtk_text_buffer_set_text(buffer, contents, len);
344 else
345 gtk_text_buffer_set_text(buffer, text, len);
346
347 /* ������������������������ */
348 gtk_text_buffer_set_modified(buffer, FALSE);
349 /* ������������������������������ */
350 gtk_text_buffer_get_start_iter(buffer, &p);
351 gtk_text_buffer_place_cursor(buffer, &p);
352 update_modeline_label();
353 gtk_window_set_title (GTK_WINDOW (editor_window), filename);
354 gtk_widget_show_all(GTK_WIDGET(notebook));
355 g_free(contents); g_free(text); g_free(filename);
356 } else
357 g_printerr("Get file contents error !\n");
358 }
359
360 /* ��������������������������������������������� */
361 static void open_file_handler(GtkWidget *widget, GtkWidget *notebook) {
362 open_file_from_notebook(GTK_NOTEBOOK(notebook));
363 }
364
365 /* gauche ��������������������������������� */
366 static gchar *eval_cstring_by_gauche(gchar *s) {
367 gchar *msg;
368
369 ScmObj result, error;
370 /* ������������������������������ */
371 ScmObj os = Scm_MakeOutputStringPort(TRUE);
372
373 /* Scheme ��������������������������������������� */
374 /* http://alohakun.blog7.fc2.com/blog-entry-517.html */
375 Scm_Define(Scm_UserModule(), SCM_SYMBOL(SCM_INTERN("*input*")), SCM_MAKE_STR(s));
376 Scm_Define(Scm_UserModule(), SCM_SYMBOL(SCM_INTERN("*error*")), SCM_FALSE);
377
378 result = Scm_EvalCString("(guard (e (else (set! *error* e) #f)) (eval (read-from-string *input*) (current-module)))", SCM_OBJ(Scm_UserModule()));
379
380 error = Scm_GlobalVariableRef(Scm_UserModule(), SCM_SYMBOL(SCM_INTERN("*error*")), 0);
381
382 /* ��������������������������������������������������������� */
383 if (!SCM_FALSEP(error))
384 Scm_Write(error, os, SCM_WRITE_DISPLAY);
385 else
386 Scm_Write(result, os, SCM_WRITE_DISPLAY);
387
388 msg = Scm_GetString(SCM_STRING(Scm_GetOutputString(SCM_PORT(os))));
389 /* ������������������ */
390 Scm_ClosePort(SCM_PORT(os));
391
392 return msg;
393 }
394
395 /* ������������������������������������������������������������������ S ��������������� (������������) */
396 static void load_region_handler(GtkWidget *widget, GtkWidget *notebook) {
397
398 GtkTextIter start, end, p;
399 gchar *code;
400 gtk_text_buffer_get_end_iter(current_text_buffer, &p);
401 gtk_text_buffer_insert(current_text_buffer, &p, "\n\n", -1);
402
403 /* ������������������������������������������������������������ */
404 if(gtk_text_buffer_get_selection_bounds(current_text_buffer, &start, &end)) {
405 code = gtk_text_buffer_get_text(current_text_buffer, &start, &end, FALSE);
406 gtk_text_buffer_insert(current_text_buffer, &p, load_cstring_by_gauche(code), -1);
407 g_free(code);
408 }
409 }
410
411 // GtkTextCharPredicate
412 static gboolean is_kakko_or_kokka(gunichar ch, gpointer p) {
413 return ch == '(' || ch == ')';
414 }
415 static gboolean is_kokka(gunichar ch, gpointer p) {return ch == ')';}
416
417
418 /* ')' ��������������� '(' ������������������ (S ���) ��������������� */
419 static gboolean search_sexp_string(GtkTextIter *start) {
420 gint nest_level = 0;
421 /* ��������������������������������� S ������������������ */
422 while(1) {
423 if(!gtk_text_iter_backward_find_char(start, is_kakko_or_kokka, NULL, NULL))
424 return FALSE;
425
426 if(gtk_text_iter_get_char(start) == ')')
427 nest_level++;
428 else {
429 if(!nest_level)
430 break;
431 else
432 nest_level--;
433 }
434 }
435 return TRUE;
436 }
437
438 /* ������������������������������������������������ */
439 static gint get_parent_nest_level_at_cursor(GtkTextBuffer *buffer) {
440 gint nest_level = 0;
441 GtkTextIter start, end;
442 gtk_text_buffer_get_start_iter(buffer, &start);
443 if(gtk_text_iter_get_char(&start) == '(') nest_level++;
444
445 /* ��������������������� (= end) ��������� */
446 gtk_text_buffer_get_iter_at_mark(buffer,&end, gtk_text_buffer_get_insert(buffer));
447
448 while(1) {
449 /* end ������ '(' ��� ')' ��������������������������������������������� */
450 if(!gtk_text_iter_forward_find_char(&start, is_kakko_or_kokka, NULL, &end))
451 return nest_level;
452
453 if(gtk_text_iter_get_char(&start) == '(')
454 nest_level++;
455 else
456 nest_level--;
457 }
458 }
459
460 /* ������������������������������������������������������������ */
461 static void switch_page(GtkNotebook *notebook, GtkNotebookPage *page, guint pagenum, gpointer p) {
462
463 /* ������������������������������������������������������������ */
464 current_tabpage = GTK_SCROLLED_WINDOW(gtk_notebook_get_nth_page(notebook, pagenum));
465 current_tabpage_num = pagenum;
466 /* GtkBin ������������������������������������������������������������ */
467 current_text_view = GTK_TEXT_VIEW(gtk_bin_get_child(GTK_BIN(current_tabpage)));
468 current_text_buffer = gtk_text_view_get_buffer(current_text_view);
469
470 /* ������������������������������������������������������ */
471 gtk_window_set_title (GTK_WINDOW(editor_window), current_tabpage_label = gtk_notebook_get_tab_label_text(notebook, GTK_WIDGET(current_tabpage)));
472
473 update_modeline_label();
474 }
475
476 /* ��������������������������������� on/off */
477 static void tabsborder_on_off(GtkButton *button, GtkNotebook *notebook) {
478 gint tval = FALSE;
479 gint bval = FALSE;
480 if(notebook->show_tabs == FALSE)
481 tval = TRUE;
482 if(notebook->show_border == FALSE)
483 bval = TRUE;
484
485 gtk_notebook_set_show_tabs(notebook, tval);
486 gtk_notebook_set_show_border(notebook, bval);
487 }
488
489 /* ������������������������������������������ */
490 static void remove_tabpage(GtkNotebook *notebook) {
491 if(!not_yet_save_changes_really_quit(current_text_buffer)) {
492 gtk_notebook_remove_page(notebook, current_tabpage_num);
493 /* ��������������������������������������� */
494 gtk_widget_queue_draw(GTK_WIDGET(notebook));
495 }
496 }
497
498 static void remove_tabpage_handler(GtkButton *button, GtkWidget *notebook) {
499 remove_tabpage(GTK_NOTEBOOK(notebook));
500 }
501
502 /* ������������������������������������������������ */
503 static void append_tabpage(GtkButton *button, GtkNotebook *notebook) {
504 gtk_notebook_append_page(notebook, new_scrolled_text_buffer(), gtk_label_new("*scratch*"));
505 gtk_widget_show_all(GTK_WIDGET(notebook));
506 }
507
508 /* ������������������������ */
509 static void rotate_tab_position(GtkButton *button, GtkNotebook *notebook ) {
510 gtk_notebook_set_tab_pos(notebook, (notebook->tab_pos + 1) % 4);
511 }
512
513 /* ������������������������������������������ */
514
515 /* ��������������������� ^npfb */
516 static void forward_current_buffer() {
517 GtkTextIter p;
518 gtk_text_buffer_get_iter_at_mark(current_text_buffer,&p, gtk_text_buffer_get_insert(current_text_buffer));
519 gtk_text_iter_forward_char(&p);
520 gtk_text_buffer_place_cursor(current_text_buffer, &p);
521 }
522 static void backward_current_buffer() {
523 GtkTextIter p;
524 gtk_text_buffer_get_iter_at_mark(current_text_buffer,&p, gtk_text_buffer_get_insert(current_text_buffer));
525 gtk_text_iter_backward_char(&p);
526 gtk_text_buffer_place_cursor(current_text_buffer, &p);
527 }
528 static void line_forward_current_buffer() {
529 GtkTextIter p;
530 gtk_text_buffer_get_iter_at_mark(current_text_buffer, &p, gtk_text_buffer_get_insert(current_text_buffer));
531 gtk_text_view_forward_display_line(current_text_view, &p);
532 gtk_text_buffer_place_cursor(current_text_buffer, &p);
533 }
534 static void line_backward_current_buffer() {
535 GtkTextIter p;
536 gtk_text_buffer_get_iter_at_mark(current_text_buffer,&p, gtk_text_buffer_get_insert(current_text_buffer));
537 gtk_text_view_backward_display_line(current_text_view, &p);
538 gtk_text_buffer_place_cursor(current_text_buffer, &p);
539 }
540
541 /* ��������������������� */
542 static gboolean signal_key_press_handler (GtkWidget *notebook, GdkEventKey *event, gpointer contextid) {
543 GtkTextIter start, end;
544
545 /* ������������������������������������ */
546 gtk_text_buffer_get_start_iter(current_text_buffer, &start);
547 gtk_text_buffer_get_end_iter(current_text_buffer, &end);
548 gtk_text_buffer_remove_tag_by_name(current_text_buffer, "parent_emphasis_background", &start, &end);
549
550 if(event->state & GDK_CONTROL_MASK) {
551 switch(event->keyval) {
552 case GDK_f : /* Ctrl + f : forward */
553 forward_current_buffer();
554 break;
555 case GDK_b : /* Ctrl + b : backward */
556 backward_current_buffer();
557 break;
558 case GDK_n : /* Ctrl + n : next line */
559 line_forward_current_buffer();
560 break;
561 case GDK_p : /* Ctrl + p : previous line */
562 line_backward_current_buffer();
563 break;
564
565 case GDK_j : /* Ctrl + j : ��������������������� S ������������ */
566 {
567 gchar *code;
568 GtkTextIter start, end;
569
570 /* ������������������������������ */
571 gtk_text_buffer_get_iter_at_mark(current_text_buffer, &end, gtk_text_buffer_get_insert(current_text_buffer));
572
573 gtk_text_iter_backward_find_char(&end, is_kokka, NULL, NULL);
574 start = end;
575 gtk_text_iter_forward_char(&end);
576
577 /* ��������������������������������� S ������������������ */
578 if(!search_sexp_string(&start)) return FALSE;
579
580 code = gtk_text_buffer_get_text(current_text_buffer, &start, &end, FALSE);
581 gtk_text_buffer_insert(current_text_buffer, &end, "\n\n", -1);
582 gtk_text_buffer_insert(current_text_buffer, &end, eval_cstring_by_gauche(code), -1);
583 g_free(code);
584 }
585 break;
586
587 case GDK_t : /* Ctrl + t : ��������������� */
588 gtk_notebook_append_page(GTK_NOTEBOOK(notebook), new_scrolled_text_buffer(), gtk_label_new("*scratch*"));
589 gtk_widget_show_all(GTK_WIDGET(notebook));
590 break;
591
592 case GDK_k : /* Ctrl + k : ������������������ */
593 remove_tabpage(GTK_NOTEBOOK(notebook));
594 break;
595 }
596 }
597 return FALSE;
598 }
599
600 /* ��������������������� */
601 static gboolean signal_key_release_handler (GtkWidget *notebook, GdkEventKey *event, gpointer contextid) {
602 static gint metakey_pressed = 0;
603 static gint controlx_pressed = 0;
604
605 if(event->keyval == GDK_parenright && event->state & GDK_SHIFT_MASK) {
606 GtkTextIter start, end;
607
608 /* ������������������������������ */
609 gtk_text_buffer_get_iter_at_mark(current_text_buffer, &end, gtk_text_buffer_get_insert(current_text_buffer));
610
611 start = end;
612 gtk_text_iter_backward_char(&start);
613
614 /* ��������������������������������� S ������������������ */
615 if(!search_sexp_string(&start)) return FALSE;
616
617 gtk_text_buffer_apply_tag_by_name(current_text_buffer, "parent_emphasis_background", &start, &end);
618 }
619
620 /* ������������������������������������������������������������������������������������������������ (���������������) ������������������ */
621 if(event->keyval == GDK_Return) {
622 gint indentWidth = get_parent_nest_level_at_cursor(current_text_buffer) * editor_indent_width;
623 gchar *indent = g_strnfill(indentWidth, ' ');
624 gtk_text_buffer_insert_at_cursor(current_text_buffer, indent, -1);
625 g_free(indent);
626 }
627
628 /* C-x */
629 if(event->keyval == GDK_x && event->state & GDK_CONTROL_MASK) {
630 controlx_pressed++;
631 gtk_statusbar_push(GTK_STATUSBAR(statusbar), GPOINTER_TO_INT(contextid), "C-x -");
632 } else if(event->state & GDK_CONTROL_MASK) {
633
634 if(controlx_pressed > 0) {
635 switch(event->keyval) {
636 case GDK_c :/* C-x C-c : ������ */
637 gtk_statusbar_push(GTK_STATUSBAR(statusbar), GPOINTER_TO_INT(contextid), "C-c");
638 {/* "delete-event" ��������������������������������������� �� ������������������������������������ */
639 GdkEvent ev;
640
641 ev.any.type = GDK_DELETE;
642 ev.any.window = editor_window->window;
643 ev.any.send_event = FALSE;
644 gdk_event_put (&ev);
645 }
646 break;
647
648 case GDK_f : /* C-x C-f : ������������������ */
649 gtk_statusbar_push(GTK_STATUSBAR(statusbar), GPOINTER_TO_INT(contextid), "C-f");
650 open_file_from_notebook(GTK_NOTEBOOK(notebook));
651 break;
652
653 case GDK_s : /* C-x C-s : ������������������ */
654 gtk_statusbar_push(GTK_STATUSBAR(statusbar), GPOINTER_TO_INT(contextid), "C-s");
655 save_file_from_notebook(GTK_NOTEBOOK(notebook));
656 break;
657
658 case GDK_w : /* C-x C-w : ������������������������ */
659 gtk_statusbar_push(GTK_STATUSBAR(statusbar), GPOINTER_TO_INT(contextid), "C-w");
660 save_file_as_from_notebook(GTK_NOTEBOOK(notebook));
661 break;
662 }
663 controlx_pressed = 0;
664 }
665
666 switch(event->keyval) {
667 case GDK_g :/* C-g : ��������������� */
668 metakey_pressed = 0;
669 controlx_pressed = 0;
670
671 gtk_statusbar_push(GTK_STATUSBAR(statusbar), GPOINTER_TO_INT(contextid), "Quit");
672 break;
673 }
674
675 }
676 return FALSE;
677 }
678
679 /* ��������������������������������������� */
680 static void editor_window_init() {
681 GtkWidget *vbox, *toolbar, *notebook;
682 GtkToolItem *icon;
683 GtkIconSize iconsize;
684 GtkTooltips *toolbar_tips = gtk_tooltips_new();
685 /* ��������������������������������������������������������������������������������� */
686 GtkToolItem *oicon, *sicon, *saicon, *eicon;
687
688 gint contextid;
689
690 /* ������������ */
691 editor_window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
692 g_signal_connect(G_OBJECT(editor_window), "destroy", G_CALLBACK(gtk_main_quit), NULL);
693
694 /* ������������������������������������ */
695 vbox = gtk_vbox_new(FALSE, 0);
696 /* ��������������������� */
697 toolbar = gtk_toolbar_new();
698 gtk_box_pack_start(GTK_BOX(vbox), toolbar, FALSE, FALSE, 0);
699
700 notebook = gtk_notebook_new();
701 g_signal_connect(G_OBJECT(notebook), "switch-page", GTK_SIGNAL_FUNC(switch_page), NULL);
702
703 /* ������������������������������������������������ */
704 gtk_toolbar_set_style(GTK_TOOLBAR (toolbar), GTK_TOOLBAR_ICONS);
705 iconsize = gtk_toolbar_get_icon_size (GTK_TOOLBAR (toolbar));
706
707 /* ������������������ */
708
709 /* ������������������ */
710 oicon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-open", iconsize), "");
711 /* ������������������������������������������������������������������������������������ */
712 g_signal_connect(G_OBJECT(oicon), "clicked", G_CALLBACK(open_file_handler), G_OBJECT(notebook));
713 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(oicon));
714 gtk_tool_item_set_tooltip(oicon, toolbar_tips, "���������������������������",
715 "���������������������������������������������������������������������������������������");
716
717 /* ������������������ */
718 sicon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-save", iconsize), "");
719 /* ������������������������������������������������������������������������������������ */
720 g_signal_connect(G_OBJECT(sicon), "clicked", G_CALLBACK(save_file_handler), G_OBJECT(notebook));
721 gtk_container_add (GTK_CONTAINER (toolbar), GTK_WIDGET(sicon));
722 gtk_tool_item_set_tooltip(sicon, toolbar_tips, "������������������������������",
723 "������������������������������������������������������������������������������������������������������������������������������������");
724
725 /* ��������������������������� */
726 saicon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-save-as", iconsize), "");
727 /* ������������������������������������������������������������������������������������������������������������������ */
728 g_signal_connect(G_OBJECT(saicon), "clicked", G_CALLBACK(save_file_as_handler), G_OBJECT(notebook));
729 gtk_container_add (GTK_CONTAINER (toolbar), GTK_WIDGET(saicon));
730 gtk_tool_item_set_tooltip(saicon, toolbar_tips, "������������������������������������",
731 "");
732
733 /* ������������������ */
734 eicon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-execute", iconsize), "");
735 /* ������������������������������������������ libgauche ������������������ */
736 g_signal_connect(G_OBJECT(eicon), "clicked", G_CALLBACK(load_region_handler), G_OBJECT(notebook));
737 gtk_container_add (GTK_CONTAINER (toolbar), GTK_WIDGET(eicon));
738 gtk_tool_item_set_tooltip(eicon, toolbar_tips, "��������������� S ������������������������ (load-region-lisp)",
739 "Scheme (gauche) ������������������ S ������������������������");
740
741 gtk_container_add(GTK_CONTAINER(editor_window), vbox);
742 gtk_container_add(GTK_CONTAINER(vbox), notebook);
743
744 icon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-apply", iconsize), "append");
745 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(tabsborder_on_off), G_OBJECT(notebook));
746 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
747 gtk_tool_item_set_tooltip(icon, toolbar_tips, "��������� on/off", "");
748
749 icon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-preferences", iconsize), "append");
750 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(rotate_tab_position), G_OBJECT( notebook));
751 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
752 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������", "");
753
754 icon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-add", iconsize), "append");
755 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(append_tabpage), G_OBJECT( notebook));
756 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
757 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������������������", "");
758
759 icon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-close", iconsize), "remove");
760 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(remove_tabpage_handler), G_OBJECT( notebook));
761 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
762 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������������",
763 "���������������������������������������������������������������");
764 icon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-bold", iconsize), "append");
765 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(select_font), NULL);
766 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
767 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������", "");
768
769 icon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-connect", iconsize), "append");
770 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(load_scheme_file_by_gauche), G_OBJECT( notebook));
771 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
772 gtk_tool_item_set_tooltip(icon, toolbar_tips, "Scheme ������������������������", "");
773
774 icon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-convert", iconsize), "append");
775 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(load_buffer_by_gauche), G_OBJECT( notebook));
776 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
777 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������������������", "");
778
779 icon = gtk_tool_button_new(gtk_image_new_from_stock ("gtk-dialog-info", iconsize), "append");
780 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(about_this_application), NULL);
781 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
782 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������������������������������������", "");
783
784 gtk_box_pack_start(GTK_BOX(vbox), modeline_label = gtk_label_new("-E:** *scratch* (Gauche Interaction)--L1--All---------------------------------"), TRUE, TRUE, 0);
785
786 /* C-x C-s ��������������������������������������������������������������������������������������� */
787 statusbar = gtk_statusbar_new();
788 gtk_box_pack_start(GTK_BOX(vbox), statusbar, TRUE, TRUE, 0);
789 contextid = gtk_statusbar_get_context_id(GTK_STATUSBAR(statusbar), "");
790
791 /* ������������������������������������������������ */
792 g_signal_connect(G_OBJECT(notebook), "key-press-event", G_CALLBACK (signal_key_press_handler), GINT_TO_POINTER(contextid));
793 g_signal_connect(G_OBJECT(notebook), "key-release-event", G_CALLBACK (signal_key_release_handler), GINT_TO_POINTER(contextid));
794
795 /* ������������������������������������ */
796 gtk_notebook_prepend_page(GTK_NOTEBOOK(notebook), new_scrolled_text_buffer(), gtk_label_new("*scratch*"));
797
798 gtk_widget_grab_focus(notebook);
799 gtk_widget_show_all(editor_window);
800 }
801
802 int main(int argc, char *argv[]) {
803 /* ������������������������������������ */
804 gtk_set_locale();
805 gtk_init(&argc, &argv);
806 GC_INIT(); Scm_Init(GAUCHE_SIGNATURE);
807 editor_window_init();
808 gtk_main();
809 Scm_Exit(0);
810 return 0;
811 }

Back to OSDN">Back to OSDN
ViewVC Help
Powered by ViewVC 1.1.26