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.59 - (show annotations) (download) (as text)
Wed Nov 29 04:37:23 2006 UTC (17 years, 4 months ago) by aloha
Branch: MAIN
Changes since 1.58: +17 -151 lines
File MIME type: text/x-csrc
bug fix (C-x C-s) and add no-or-yes-p API

1 /* vim: set encoding=utf8:
2 *
3 * shiki.c
4 *
5 * This file is main file of Shiki.
6 *
7 * Copyright(C)2006 WAKATSUKI toshihiro
8 *
9 * Permission is hereby granted, free of charge, to any person obtaining a
10 * copy of this software and associated documentation files (the "Software"),
11 * to deal in the Software without restriction, including without limitation
12 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
13 * and/or sell copies of the Software, and to permit persons to whom the
14 * Software is furnished to do so, subject to the following conditions:
15 *
16 * The above copyright notice and this permission notice shall be included in
17 * all copies or substantial portions of the Software.
18 *
19 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
20 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
22 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
23 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
24 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
25 * SOFTWARE.
26 *
27 * $Id: shiki.c,v 1.58 2006/11/28 16:29:22 aloha Exp $
28 */
29
30 #include"shiki.h"
31
32 static gint editor_indent_width = 2;
33
34 /* ������������������������������������������������������������������������������������������������������������������������������ */
35 ShikiEditorType Shiki_editor;
36
37 static gchar *R5RS_keywords[] = {"and", "begin", "case", "cond-expand", "cond", "define-accessor", "define-class", "defined?", "define-generic", "define", "define-macro", "define-method", "define-module", "define-private", "define-public", "define-reader-ctor", "define-syntax", "define-syntax-macro", "defmacro", "defmacro*-public", "delay", "do", "else", "fluid-let", "if", "lambda", "let", "let*", "letrec", "letrec-syntax", "let-syntax", "or", "quasiquote", "quote", "set!", "syntax-rules", "unquote", NULL};
38
39 static gchar *R5RS_functions[] = {"*", "+", "-", "/", "<", ">", "<=", ">=", "?", "`", "=", "abs", "acos", "angle", "append", "apply", "asin", "assoc", "assq", "assv", "atan", "boolean?", "caaar", "caadr", "caar", "cadar", "caddr", "cadr", "call/cc", "call-with-current-continuation", "call-with-input-file", "call-with-output-file", "call-with-values", "car", "catch", "cdaar", "cdadr", "cdar", "cddar", "cdddr", "cddr", "cdr", "ceiling", "char-alphabetic?", "char-ci>=?", "char-ci>?", "char-ci=?", "char-ci<=?", "char-ci<?", "char-downcase", "char->integer", "char>=?", "char>?", "char=?", "char?", "char-lower-case?", "char<=?", "char<?", "char-numeric?", "char-ready?", "char-upcase", "char-upper-case?", "char-whitespace?", "close-input-port", "close-output-port", "complex?", "cons", "cos", "current-input-port", "current-output-port", "delete-file", "display", "dynamic-wind", "eof-object?", "eq?", "equal?", "eqv?", "eval", "even?", "exact->inexact", "exact?", "exit", "exp", "expt", "file-exists?", "file-or-directory-modify-seconds", "floor", "force", "for-each", "gcd", "gensym", "getenv", "get-output-string", "imag-part", "inexact?", "input-port?", "integer->char", "integer?", "lcm", "length", "list->string", "list->vector", "list", "list?", "list-ref", "list-tail", "load", "log", "magnitude", "make-polar", "make-rectangular", "make-string", "make-vector", "map", "max", "member", "memq", "memv", "min", "modulo", "negative?", "newline", "nil", "not", "null?", "number->string", "number?", "odd?", "open-input-file", "open-input-string", "open-output-file", "open-output-string", "output-port?", "pair?", "peek-char", "port?", "positive?", "procedure?", "quotient", "rational?", "read-char", "read", "read-line", "real?", "real-part", "remainder", "reverse", "reverse!", "round", "set-car!", "set-cdr!", "sin", "sqrt", "string-append", "string-ci>=?", "string-ci>?", "string-ci=?", "string-ci<=?", "string-ci<?", "string-copy", "string-fill!", "string>=?", "string>?", "string->list", "string->number", "string->symbol", "string", "string=?", "string?", "string-length", "string<=?", "string<?", "string-ref", "string-set!", "substring", "symbol->string", "symbol?", "system", "tan", "truncate", "values", "vector-fill!", "vector->list", "vector", "vector?", "vector-length", "vector-ref", "vector-set!", "with-input-from-file", "with-output-to-file", "write-char", "write", "zero", NULL};
40
41 static GHashTable *keywords_hash = NULL;
42
43 typedef enum {
44 R5RS_KEYWORD_COLOR = 1,
45 R5RS_FUNCTION_COLOR,
46 GAUCHE_KEYWORD_COLOR,
47 GAUCHE_FUNCTION_COLOR
48 } HIGHILIGHT_COLOR;
49
50 GdkColor COLOR_BLACK;
51 GdkColor COLOR_GREEN;
52
53 /* ������������������ */
54
55 /* foo_bar_handler() ������������������������������������������������������ */
56 static void kill_buffer_handler();
57 static void append_default_tabpage_handler();
58
59 /* ������������������ */
60 static void open_file(const gchar *filename);
61 static void open_file_handler();
62 static void save_file();
63 static void save_file_as();
64
65 /* ������������������������������ */
66 static gchar* get_all_buffer_contents(GtkTextBuffer *buffer);
67 static gboolean save_text_buffer(const gchar *filename, GtkTextBuffer *buffer);
68 static void clear_current_buffer_handler();
69 static void undo();
70 static void search_current_buffer();
71
72 /* Gauche ��� S ��������������������������������������������������������������������������� */
73 static gchar *eval_cstring_by_gauche(gchar *s);
74 static gchar *load_cstring_by_gauche(gchar *s);
75 static void load_buffer_by_gauche();
76 static void load_region_by_gauche();
77 static void load_scheme_file_by_gauche();
78 static gboolean is_kakko_or_kokka(gunichar ch, gpointer);
79 static gboolean is_kakko(gunichar ch, gpointer);
80 static gboolean is_kokka(gunichar ch, gpointer);
81 static gboolean search_sexp(GtkTextIter *start, GtkTextIter *end);
82 static gboolean search_sexp_kokka(GtkTextIter *end);
83 static gboolean search_last_sexp(GtkTextIter *start, GtkTextIter *end);
84 static gboolean search_last_sexp_kakko(GtkTextIter *start);
85 static gint get_parent_nest_level_at_cursor(GtkTextBuffer *buffer);
86 static gboolean is_not_scheme_delimita_p(gunichar ch, gpointer user_data);
87 static gboolean is_double_quote(gunichar ch, gpointer user_data);
88 static gboolean is_scheme_delimita_p(gunichar ch, gpointer user_data);
89 static void scheme_keyword_highlighting_current_buffer();
90
91 /* ������ */
92 static void select_font();
93 static void font_selection_ok(GtkWidget *button, GtkWidget *font_dialog);
94 static void switch_tabpage_handler(GtkNotebook *notebook, GtkNotebookPage *page, guint pagenum) ;
95 static void tabsborder_on_off(GtkButton *button, GtkNotebook *notebook);
96 static void rotate_tab_position(GtkButton *button, GtkNotebook *notebook);
97
98 /* ������������������������ */
99 static gboolean signal_key_press_handler(GtkWidget *notebook, GdkEventKey *event, gpointer contextid);
100 static gboolean signal_key_release_handler(GtkWidget *notebook, GdkEventKey *event, gpointer contextid);
101
102 /* ������������������ */
103 static void open_online_help();
104 static void about_this_application();
105
106 /* ������������������������ */
107 static void Shiki_editor_window_init(int argc, char **argv);
108
109 /* ������������������������������ */
110 static void clear_current_buffer_handler() {
111 Shiki_erase_buffer(Shiki_CURRENT_TEXT_BUFFER);
112 }
113
114 static void destroy_handler(GtkWidget *button, GtkWidget *widget) {gtk_widget_destroy(widget);}
115
116 static void toggled_handler(GtkToggleButton *togglebutton, gboolean *flag) {
117 *flag = !*flag;
118 }
119
120 /* ��������������������������������������������������������������� */
121 static void update_modeline_label() {
122 static gchar label[1024];
123 GtkTextIter p;
124 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, &p, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
125
126 g_snprintf(label, 1024, "--%s- %-10s (Gauche Interaction) [%s] L%d:%d ",
127 gtk_text_buffer_get_modified(Shiki_CURRENT_TEXT_BUFFER) ? "**" : "--",
128 Shiki_CURRENT_BASENAME,
129 Shiki_CURRENT_CES,
130 gtk_text_iter_get_line(&p) + 1,
131 gtk_text_iter_get_line_offset (&p) + 1);
132 gtk_label_set_text(GTK_LABEL(Shiki_EDITOR_MODELINE_LABEL), label);
133 }
134
135 static void replace_current_buffer() {
136 GtkWidget *dialog = gtk_dialog_new_with_buttons ("������������������", GTK_WINDOW(Shiki_EDITOR_WINDOW), GTK_DIALOG_DESTROY_WITH_PARENT, NULL);
137 GtkWidget *table = gtk_table_new(7, 3, FALSE);
138 GtkWidget *find_label = gtk_label_new("������ : ");
139 GtkWidget *find = gtk_entry_new();
140 GtkWidget *rep_label = gtk_label_new("������ : ");
141 GtkWidget *replace = gtk_entry_new();
142 GtkWidget *check1 = gtk_check_button_new_with_label("���������������������������������");
143 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check1), TRUE);
144 GtkWidget *check2 = gtk_check_button_new_with_label("���������������������������");
145 GtkWidget *check3 = gtk_check_button_new_with_label("������������");
146 GtkWidget *check4 = gtk_check_button_new_with_label("���������������������������������������������");
147 GtkWidget *check5 = gtk_check_button_new_with_label("���������������������������");
148 GtkWidget *interactive = gtk_button_new_with_label ("������������");
149 GtkWidget *all = gtk_button_new_with_label ("������������");
150 GtkWidget *cancel = gtk_button_new_with_label ("���������������");
151 gboolean f1 = TRUE, f2, f3, f4, f5;
152 f2 = f3 = f4 = f5 = FALSE;
153 g_signal_connect (check1, "toggled", G_CALLBACK (toggled_handler), &f1);
154 g_signal_connect (check2, "toggled", G_CALLBACK (toggled_handler), &f2);
155 g_signal_connect (check3, "toggled", G_CALLBACK (toggled_handler), &f3);
156 g_signal_connect (check4, "toggled", G_CALLBACK (toggled_handler), &f4);
157 g_signal_connect (check5, "toggled", G_CALLBACK (toggled_handler), &f5);
158
159 g_signal_connect (G_OBJECT(dialog), "delete_event", G_CALLBACK(gtk_widget_destroy), NULL);
160 g_signal_connect (G_OBJECT(cancel), "clicked", G_CALLBACK(destroy_handler), dialog);
161 gtk_table_set_row_spacings(GTK_TABLE(table), 10);
162 gtk_table_set_col_spacings(GTK_TABLE(table), 10);
163 gtk_container_border_width (GTK_CONTAINER (dialog), 10);
164
165 gtk_table_attach_defaults (GTK_TABLE(table), find_label, 0, 1, 0, 1);
166 gtk_table_attach_defaults (GTK_TABLE(table), find, 1, 2, 0, 1);
167 gtk_table_attach_defaults (GTK_TABLE(table), interactive, 2, 3, 0, 1);
168
169 gtk_table_attach_defaults (GTK_TABLE(table), rep_label, 0, 1, 1, 2);
170 gtk_table_attach_defaults (GTK_TABLE(table), replace, 1, 2, 1, 2);
171 gtk_table_attach_defaults (GTK_TABLE(table), all, 2, 3, 1, 2);
172
173 gtk_table_attach_defaults (GTK_TABLE(table), check1, 1, 2, 2, 3);
174 gtk_table_attach_defaults (GTK_TABLE(table), cancel, 2, 3, 2, 3);
175
176 gtk_table_attach_defaults (GTK_TABLE(table), check2, 1, 2, 3, 4);
177 gtk_table_attach_defaults (GTK_TABLE(table), check3, 1, 2, 4, 5);
178 gtk_table_attach_defaults (GTK_TABLE(table), check4, 1, 2, 5, 6);
179 gtk_table_attach_defaults (GTK_TABLE(table), check5, 1, 2, 6, 7);
180
181 gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), table);
182 gtk_widget_show_all(table);
183 gtk_dialog_run(GTK_DIALOG(dialog));
184 }
185
186 static void search_current_buffer() {
187 GtkWidget *dialog = gtk_dialog_new_with_buttons ("������������������", GTK_WINDOW(Shiki_EDITOR_WINDOW), GTK_DIALOG_DESTROY_WITH_PARENT, NULL);
188 GtkWidget *table = gtk_table_new(6, 3, FALSE);
189 GtkWidget *label = gtk_label_new("������ : ");
190 GtkWidget *input = gtk_entry_new();
191 GtkWidget *check1 = gtk_check_button_new_with_label("���������������������������������");
192 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check1), TRUE);
193 GtkWidget *check2 = gtk_check_button_new_with_label("���������������������������");
194 GtkWidget *check3 = gtk_check_button_new_with_label("������������");
195 GtkWidget *check4 = gtk_check_button_new_with_label("���������������������������������������������");
196 GtkWidget *check5 = gtk_check_button_new_with_label("���������������������������������");
197 GtkWidget *prev = gtk_button_new_with_label ("���������");
198 GtkWidget *next = gtk_button_new_with_label ("���������");
199 GtkWidget *cancel = gtk_button_new_with_label ("���������������");
200 gboolean f1 = TRUE, f2, f3, f4, f5;
201 f2 = f3 = f4 = f5 = FALSE;
202 g_signal_connect (check1, "toggled", G_CALLBACK (toggled_handler), &f1);
203 g_signal_connect (check2, "toggled", G_CALLBACK (toggled_handler), &f2);
204 g_signal_connect (check3, "toggled", G_CALLBACK (toggled_handler), &f3);
205 g_signal_connect (check4, "toggled", G_CALLBACK (toggled_handler), &f4);
206 g_signal_connect (check5, "toggled", G_CALLBACK (toggled_handler), &f5);
207
208 g_signal_connect (G_OBJECT(dialog), "delete_event", G_CALLBACK(gtk_widget_destroy), NULL);
209 g_signal_connect (G_OBJECT(cancel), "clicked", G_CALLBACK(destroy_handler), dialog);
210 gtk_table_set_row_spacings(GTK_TABLE(table), 10);
211 gtk_table_set_col_spacings(GTK_TABLE(table), 10);
212 gtk_container_border_width (GTK_CONTAINER (dialog), 10);
213 gtk_table_attach_defaults (GTK_TABLE(table), label, 0, 1, 0, 1);
214 gtk_table_attach_defaults (GTK_TABLE(table), input, 1, 2, 0, 1);
215 gtk_table_attach_defaults (GTK_TABLE(table), prev, 2, 3, 0, 1);
216 gtk_table_attach_defaults (GTK_TABLE(table), check1, 1, 2, 1, 2);
217 gtk_table_attach_defaults (GTK_TABLE(table), check2, 1, 2, 2, 3);
218 gtk_table_attach_defaults (GTK_TABLE(table), check3, 1, 2, 3, 4);
219 gtk_table_attach_defaults (GTK_TABLE(table), check4, 1, 2, 4, 5);
220 gtk_table_attach_defaults (GTK_TABLE(table), check5, 1, 2, 5, 6);
221 gtk_table_attach_defaults (GTK_TABLE(table), next, 2, 3, 1, 2);
222 gtk_table_attach_defaults (GTK_TABLE(table), cancel, 2, 3, 2, 3);
223 gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), table);
224 gtk_widget_show_all(table);
225 gtk_dialog_run(GTK_DIALOG(dialog));
226 }
227
228 static gboolean is_not_scheme_delimita_p(gunichar ch, gpointer user_data) {
229 return ch != '(' && ch != ')' && !g_unichar_isspace(ch);
230 }
231
232 static gboolean is_double_quote(gunichar ch, gpointer user_data) {
233 return ch == '\"';
234 }
235
236 static gboolean is_scheme_delimita_p(gunichar ch, gpointer user_data) {
237 return ch == ' ' || ch == '(' || ch == ')' || ch == '\"' || g_unichar_isspace(ch);
238 }
239
240 /* ������������������������������ */
241 static void scheme_keyword_highlighting_current_buffer() {
242 GtkTextIter s, e;
243 HIGHILIGHT_COLOR c;
244 gchar *word;
245 gboolean is_comment, is_string;
246 gunichar ch;
247
248 gtk_text_buffer_get_start_iter(Shiki_CURRENT_TEXT_BUFFER, &s);
249
250 /* ��������� Scheme ������������������������������ */
251 while(TRUE) {
252 is_comment = FALSE;
253 is_string = FALSE;
254 if((ch = gtk_text_iter_get_char(&s)) != ';' && ch != '\"')
255 gtk_text_iter_forward_find_char(&s, is_not_scheme_delimita_p, NULL, NULL);
256 e = s;
257 if(gtk_text_iter_get_char(&s) == ';') {
258 gtk_text_iter_forward_line(&e);
259 gtk_text_iter_backward_char(&e);
260 is_comment = TRUE;
261 } else if(gtk_text_iter_get_char(&s) == '\"') {
262 while(TRUE) {
263 gtk_text_iter_forward_find_char(&e, is_double_quote, NULL, NULL);
264 gtk_text_iter_backward_char(&e);
265 if(gtk_text_iter_get_char(&e) != '\\') {
266 is_string = TRUE;
267 gtk_text_iter_forward_char(&e);
268 gtk_text_iter_forward_char(&e);
269 break;
270 }
271 gtk_text_iter_forward_char(&e);
272 gtk_text_iter_forward_char(&e);
273 }
274
275 } else
276 gtk_text_iter_forward_find_char(&e, is_scheme_delimita_p, NULL, NULL);
277
278 word = gtk_text_buffer_get_text(Shiki_CURRENT_TEXT_BUFFER, &s, &e, FALSE);
279
280 /* ������������������������������������������������������������ */
281 if(is_comment) /* ������������ */
282 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "comment_highlighting", &s, &e);
283 else if(is_string) /* ��������� */
284 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "string_highlighting", &s, &e);
285 else if(R5RS_KEYWORD_COLOR == (c = GPOINTER_TO_INT(g_hash_table_lookup(keywords_hash, word)))) /* R5RS ��������������� */
286 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "keyword_highlighting", &s, &e);
287 else if(R5RS_FUNCTION_COLOR == c) /* R5RS ������ */
288 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "function_highlighting", &s, &e);
289
290 /* XXX : get_text() ������������������������������������������������������������������������������������������������������ GtkTextBuffer ������ const gchar * ������������������������������ */
291 g_free(word);
292
293 if(gtk_text_iter_is_end(&e)) break;
294 s = e;
295 }
296 }
297
298 /* ������������������������������������������������������������ */
299 static void switch_tabpage_handler(GtkNotebook *notebook, GtkNotebookPage *page, guint pagenum) {
300 /* ��������������������������������������������������������������������� */
301 Shiki_CURRENT_TAB_INFO = (ShikiBuffer *)g_list_nth_data(Shiki_EDITOR_BUFFER_LIST, pagenum);
302
303 /* ������������������������������������ */
304 Shiki_CURRENT_TAB_NUM = pagenum;
305
306 /* ������������������������������������������������������ */
307 if(!Shiki_CURRENT_TAB_INFO) return;
308 gtk_window_set_title (GTK_WINDOW(Shiki_EDITOR_WINDOW), Shiki_CURRENT_FILENAME);
309
310 update_modeline_label();
311 }
312
313 static void undo() {
314 g_print("Undo\n");
315 GtkTextIter start, end;
316 ShikiUndoInfo *undoInfo = g_list_nth_data(Shiki_CURRENT_UNDO_INFO_LIST, 0);
317 if(!undoInfo) {
318 g_print("������������ Undo ���������������\n");
319 return;
320 }
321 gtk_text_buffer_get_iter_at_offset(Shiki_CURRENT_TEXT_BUFFER, &start, undoInfo->start);
322 gtk_text_buffer_get_iter_at_offset(Shiki_CURRENT_TEXT_BUFFER, &end, undoInfo->end);
323
324 if(undoInfo->action == SHIKI_UNDO_INSERT) {
325 Shiki_CURRENT_UNDO_INFO_LIST = g_list_delete_link(Shiki_CURRENT_UNDO_INFO_LIST, g_list_first(Shiki_CURRENT_UNDO_INFO_LIST));
326 gtk_text_buffer_delete(Shiki_CURRENT_TEXT_BUFFER, &start, &end);
327 g_free(undoInfo->str);
328 g_free(undoInfo);
329 } else if(undoInfo->action == SHIKI_UNDO_DELETE) {
330 Shiki_CURRENT_UNDO_INFO_LIST = g_list_delete_link(Shiki_CURRENT_UNDO_INFO_LIST, g_list_first(Shiki_CURRENT_UNDO_INFO_LIST));
331 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &start, undoInfo->str, -1);
332 g_free(undoInfo->str);
333 g_free(undoInfo);
334 }
335
336 }
337
338 /* ��������������������� */
339 static gboolean signal_key_press_handler (GtkWidget *notebook, GdkEventKey *event, gpointer contextid) {
340 GtkTextIter start, end;
341
342 /* ������������������������������������ */
343 gtk_text_buffer_get_start_iter(Shiki_CURRENT_TEXT_BUFFER, &start);
344 gtk_text_buffer_get_end_iter(Shiki_CURRENT_TEXT_BUFFER, &end);
345 gtk_text_buffer_remove_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "parent_emphasis_background", &start, &end);
346
347 if(event->state & GDK_CONTROL_MASK && event->state & GDK_MOD1_MASK) {
348 switch(event->keyval) {
349 case GDK_at : /* C-M-SPC */
350 { GtkTextIter start, end;
351 if(!search_sexp(&start, &end)) return FALSE;
352 gtk_text_buffer_select_range(Shiki_CURRENT_TEXT_BUFFER, &start, &end);
353 }
354 break;
355 case GDK_space : /* C-M-SPC */
356 { GtkTextIter start, end;
357 if(!search_last_sexp(&start, &end)) return FALSE;
358 gtk_text_buffer_select_range(Shiki_CURRENT_TEXT_BUFFER, &start, &end);
359 }
360 break;
361 }
362 } else if(event->state & GDK_CONTROL_MASK) {
363 switch(event->keyval) {
364 case GDK_f : /* Ctrl + f : forward */
365 Shiki_forward_char();
366 break;
367 case GDK_b : /* Ctrl + b : backward */
368 Shiki_backward_char();
369 break;
370 case GDK_n : /* Ctrl + n : next line */
371 Shiki_forward_line(1);
372 break;
373 case GDK_p : /* Ctrl + p : previous line */
374 Shiki_forward_line(-1);
375 break;
376 case GDK_h :
377 { GtkTextIter p;
378 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER,&p, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
379 gtk_text_buffer_backspace(Shiki_CURRENT_TEXT_BUFFER, &p, FALSE, TRUE);
380 }
381 break;
382
383 case GDK_e : /* Ctrl + e : eval-expression */
384 {
385 gchar *code, *result;
386 GtkTextIter start, end;
387
388 if(!search_sexp(&start, &end)) return FALSE;
389
390 code = gtk_text_buffer_get_text(Shiki_CURRENT_TEXT_BUFFER, &start, &end, FALSE);
391 result = eval_cstring_by_gauche(code);
392 g_free(code);
393
394 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, &end, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
395
396 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, "\n", -1);
397 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, result, -1);
398 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, "\n", -1);
399 }
400 break;
401
402 case GDK_j : /* Ctrl + j : eval-last-sexp */
403 {
404 gchar *code, *result;
405 GtkTextIter start, end;
406
407 if(!search_last_sexp(&start, &end)) return FALSE;
408
409 code = gtk_text_buffer_get_text(Shiki_CURRENT_TEXT_BUFFER, &start, &end, FALSE);
410 result = eval_cstring_by_gauche(code);
411 g_free(code);
412
413 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, &end, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
414
415 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, "\n", -1);
416 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, result, -1);
417 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, "\n", -1);
418 }
419 break;
420
421 case GDK_underscore : /* Ctrl + _ : Undo */
422 undo();
423 break;
424
425 case GDK_t : /* Ctrl + t : ��������������� */
426 append_default_tabpage_handler();
427 break;
428
429 case GDK_k : /* Ctrl + k : ������������������ */
430 kill_buffer_handler();
431 break;
432
433 case GDK_w : /* Ctrl + w : ��������� */
434 gtk_text_buffer_cut_clipboard(Shiki_CURRENT_TEXT_BUFFER, Shiki_EDITOR_CLIPBOARD, TRUE);
435 break;
436
437 case GDK_y : /* Ctrl + y : ��������� */
438 {GtkTextIter p;
439 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER,&p, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
440 gtk_text_buffer_paste_clipboard(Shiki_CURRENT_TEXT_BUFFER, Shiki_EDITOR_CLIPBOARD, &p, TRUE);
441 }
442 break;
443 }
444 }
445 return FALSE;
446 }
447
448 static void append_default_tabpage_handler() {
449 Shiki_new_buffer_create(g_strdup("*scratch*"));
450 }
451
452 /* ������������������������������������������ (������������) ��������� */
453 static void kill_buffer_handler() {
454 Shiki_kill_buffer(Shiki_CURRENT_TEXT_BUFFER);
455 }
456
457 /* ������������������������������������ */
458 static void load_buffer_by_gauche() {
459 GtkTextIter p;
460 gtk_text_buffer_get_end_iter(Shiki_CURRENT_TEXT_BUFFER, &p);
461 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, "\n\n", -1);
462 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, load_cstring_by_gauche(get_all_buffer_contents(Shiki_CURRENT_TEXT_BUFFER)), -1);
463 }
464
465 /* ������������������������ */
466 static void load_scheme_file_by_gauche() {
467 gchar *contents, *text = NULL;
468 gsize br, bw, len;
469 GError *err = NULL;
470 const gchar *filename = Shiki_file_name_dialog("File Selection");
471 GtkTextIter p;
472
473 if(!filename) return;
474
475 gtk_text_buffer_get_end_iter(Shiki_CURRENT_TEXT_BUFFER, &p);
476 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, "\n\n", -1);
477
478 if(g_file_get_contents(filename, &contents, &len, NULL)) {
479 if(!(text = g_locale_to_utf8(contents, -1, &br, &bw, &err)))
480 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, load_cstring_by_gauche(text), -1);
481 else
482 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, load_cstring_by_gauche(contents), -1);
483 }
484 g_free(text); g_free(contents);
485 }
486
487 /* gauche ������������������������������������ */
488 static gchar *load_cstring_by_gauche(gchar *s) {
489 gchar *msg;
490
491 ScmObj result, error;
492 /* ��������������������������������� */
493 ScmObj is = Scm_MakeInputStringPort(SCM_STRING(SCM_MAKE_STR_COPYING(s)), TRUE);
494 /* ������������������������������ */
495 ScmObj os = Scm_MakeOutputStringPort(TRUE);
496
497 Scm_Define(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*input*")), is);
498 Scm_Define(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*error*")), SCM_FALSE);
499 /* Scheme ��������������������������������������������������������������������������������� S ��������������������������������������������������������������������������� *error* ������������������ */
500 result = Scm_EvalCString("(guard (e (else (set! *error* e) #f)) (eval (load-from-port *input*) (current-module)))", SCM_OBJ(Shiki_CURRENT_BUFFER_ENV));
501
502 error = Scm_GlobalVariableRef(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*error*")), 0);
503
504 /* ��������������������������������������������������������� */
505 if (!SCM_FALSEP(error))
506 Scm_Write(error, os, SCM_WRITE_DISPLAY);
507 else
508 Scm_Write(result, os, SCM_WRITE_DISPLAY);
509
510 msg = Scm_GetString(SCM_STRING(Scm_GetOutputString(SCM_PORT(os))));
511 /* ������������������ */
512 Scm_ClosePort(SCM_PORT(is));
513 Scm_ClosePort(SCM_PORT(os));
514
515 return msg;
516 }
517
518 static void font_selection_ok(GtkWidget *button, GtkWidget *font_dialog) {
519 gchar *font_name = gtk_font_selection_dialog_get_font_name (GTK_FONT_SELECTION_DIALOG (font_dialog));
520 if(font_name) {
521 GtkRcStyle *style = gtk_rc_style_new ();
522 pango_font_description_free(style->font_desc);
523 style->font_desc = pango_font_description_from_string(font_name);
524 gtk_widget_modify_style (GTK_WIDGET(Shiki_CURRENT_TEXT_VIEW), style);
525 gtk_rc_style_unref (style);
526 g_free (font_name);
527 }
528 }
529
530 /* ������������������������������������������������������ */
531 static void select_font(){
532 GtkWidget *font_dialog = gtk_font_selection_dialog_new("Font Selection Dialog");
533 g_signal_connect (GTK_FONT_SELECTION_DIALOG (font_dialog)->ok_button, "clicked", G_CALLBACK(font_selection_ok), font_dialog);
534 gtk_dialog_run(GTK_DIALOG(font_dialog));
535 gtk_widget_destroy(font_dialog);
536 }
537
538 /* ������������������������������������������ */
539 static void about_this_application() {
540 GtkAboutDialog *about = GTK_ABOUT_DIALOG(gtk_about_dialog_new());
541 const gchar *authors[] = {
542 "������������ (���������) <alohakun@gmail.com>\n",
543 "Contribute : tkng ������",
544 "(http://d.hatena.ne.jp/tkng/20061113)", NULL
545 };
546 gtk_about_dialog_set_authors(about, authors);
547 gtk_about_dialog_set_copyright(about, "Copyright(C)2006 WAKATSUKI Toshihiro");
548 gtk_about_dialog_set_name(about, "��� (SHIKI)");
549 gtk_about_dialog_set_website_label(about, "���������30������������������������������������������������������������Blog");
550 gtk_about_dialog_set_website(about, "http://alohakun.blog7.fc2.com/blog-category-29.html");
551 gtk_dialog_run(GTK_DIALOG(about));
552 gtk_widget_destroy(GTK_WIDGET(about));
553 }
554
555 /* ��������� */
556 static void dummy_handler() {
557 GtkWidget *dummy = gtk_message_dialog_new(GTK_WINDOW(Shiki_EDITOR_WINDOW),
558 GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_WARNING, GTK_BUTTONS_CLOSE,
559 "Sorry... This Button is Dummy.");
560 gtk_dialog_run(GTK_DIALOG(dummy));
561 gtk_widget_destroy(dummy);
562 }
563
564 /* ��������������������������������������������������������������� */
565 static gchar* get_all_buffer_contents(GtkTextBuffer *buffer) {
566 GtkTextIter start, end;
567 gtk_text_buffer_get_start_iter(buffer, &start);
568 gtk_text_buffer_get_end_iter(buffer, &end);
569 return gtk_text_buffer_get_text(buffer, &start, &end, FALSE);
570 }
571
572 /* buffer ������������������������ filename ��������� */
573 static gboolean save_text_buffer(const gchar *filename, GtkTextBuffer *buffer) {
574 gchar *contents, *text;
575 gsize br, bw;
576 GError *err = NULL;
577
578 if(!filename) return FALSE;
579 contents = get_all_buffer_contents(buffer);
580 text = g_locale_from_utf8(contents, -1, &br, &bw, &err);
581 /* ��������������������������������� */
582 g_file_set_contents(filename, text, -1, NULL);
583 gtk_text_buffer_set_modified(buffer, FALSE);
584 update_modeline_label();
585 g_free(contents); g_free(text);
586 return TRUE;
587 }
588
589 /* ��������������������������������������������������������������������� */
590 static void save_file() {
591
592 /* ��������������������������������������� */
593 if(g_ascii_strcasecmp("*help*", Shiki_CURRENT_TAB_TITLE) == 0) return;
594
595 /* ������������������������������������ */
596 if(!gtk_text_buffer_get_modified(Shiki_CURRENT_TEXT_BUFFER)) return;
597
598 /* ������������������������������������������������������������������������������������������������������ */
599 if(g_ascii_strcasecmp("*scratch*", Shiki_CURRENT_FILENAME) == 0) {
600 const gchar *filename = Shiki_file_name_dialog("Save File As ...");
601 if(!filename) return;
602 if(!save_text_buffer(filename, Shiki_CURRENT_TEXT_BUFFER)) return;
603 gtk_notebook_set_tab_label_text(Shiki_EDITOR_NOTEBOOK, GTK_WIDGET(Shiki_CURRENT_TAB), filename);
604 gtk_window_set_title (GTK_WINDOW(Shiki_EDITOR_WINDOW), filename);
605 } else
606 save_text_buffer(Shiki_CURRENT_TAB_TITLE, Shiki_CURRENT_TEXT_BUFFER);
607 }
608
609 /* ��������������������������������������������������������������������������� */
610 static void save_file_as() {
611 const gchar *filename = Shiki_file_name_dialog("Save File As ...");
612
613 if(!filename) return;
614 if(!save_text_buffer(filename, Shiki_CURRENT_TEXT_BUFFER)) return;
615
616 gtk_notebook_set_tab_label_text(Shiki_EDITOR_NOTEBOOK, GTK_WIDGET(Shiki_CURRENT_TAB), filename);
617 gtk_window_set_title (GTK_WINDOW (Shiki_EDITOR_WINDOW), filename);
618 }
619
620 /* ��������������������� */
621 static void open_file(const gchar *filename) {
622 gchar *text;
623 gchar *utf8filename = g_locale_to_utf8(filename, -1, NULL, NULL, NULL);
624 GtkTextIter p;
625 ScmObj s;
626
627 /* g_file_get_contents(filename, &contents, &len, NULL); */
628
629 /* ������������������������������ */
630 Shiki_new_buffer_create(g_strdup(filename));
631 gtk_window_set_title (GTK_WINDOW (Shiki_EDITOR_WINDOW), filename);
632
633 Scm_Define(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*filename*")), SCM_MAKE_STR_COPYING(utf8filename));
634 g_free(utf8filename);
635
636 Scm_EvalCString("(use gauche.charconv)", Shiki_CURRENT_BUFFER_ENV);
637
638 /* ������������������������������������������������������������������������������ */
639 s = Scm_EvalCString("(port->string (open-input-conversion-port (open-input-file *filename*) \"*jp\" :owner? #t))", Shiki_CURRENT_BUFFER_ENV);
640 text = Scm_GetString(SCM_STRING(s));
641 if(text)
642 gtk_text_buffer_set_text(Shiki_CURRENT_TEXT_BUFFER, text, -1);
643 else {
644 /* open-input-conversion-port ������������������������������������������������������
645 * ��������������������������������������������������������������������������� UTF8
646 */
647 gchar *contents;
648 gsize br, bw, len;
649 GError *err = NULL;
650
651 if(g_file_get_contents(filename, &contents, &len, NULL)) {
652 if(!(text = g_locale_to_utf8(contents, -1, &br, &bw, &err)))
653 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, text, -1);
654 else
655 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, contents, -1);
656 g_free(contents);
657 }
658 }
659
660 /* ������������������������ */
661 gtk_text_buffer_set_modified(Shiki_CURRENT_TEXT_BUFFER, FALSE);
662 /* ������������������������������ */
663 gtk_text_buffer_get_start_iter(Shiki_CURRENT_TEXT_BUFFER, &p);
664 gtk_text_buffer_place_cursor(Shiki_CURRENT_TEXT_BUFFER, &p);
665 update_modeline_label();
666 /* ��������������������������������� */
667 scheme_keyword_highlighting_current_buffer();
668 gtk_widget_show_all(GTK_WIDGET(Shiki_EDITOR_NOTEBOOK));
669 }
670
671 /* ��������������������������������������������������� */
672 static void open_file_handler() {
673 const gchar *filename = Shiki_file_name_dialog("File Selection");
674
675 if(!filename) return;
676 open_file(filename);
677 }
678
679 /* gauche ��������������������������������� */
680 static gchar *eval_cstring_by_gauche(gchar *s) {
681 gchar *msg;
682
683 ScmObj result, error;
684 /* ������������������������������ */
685 ScmObj os = Scm_MakeOutputStringPort(TRUE);
686
687 /* Scheme ��������������������������������������� */
688 /* http://alohakun.blog7.fc2.com/blog-entry-517.html */
689 Scm_Define(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*input*")), SCM_MAKE_STR_COPYING(s));
690 Scm_Define(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*error*")), SCM_FALSE);
691
692 result = Scm_EvalCString("(guard (e (else (set! *error* e) #f)) (eval (read-from-string *input*) (current-module)))", SCM_OBJ(Shiki_CURRENT_BUFFER_ENV));
693
694 error = Scm_GlobalVariableRef(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*error*")), 0);
695
696 /* ��������������������������������������������������������� */
697 if (!SCM_FALSEP(error))
698 Scm_Write(error, os, SCM_WRITE_DISPLAY);
699 else
700 Scm_Write(result, os, SCM_WRITE_DISPLAY);
701
702 msg = Scm_GetString(SCM_STRING(Scm_GetOutputString(SCM_PORT(os))));
703 /* ������������������ */
704 Scm_ClosePort(SCM_PORT(os));
705
706 return msg;
707 }
708
709 /* ������������������������������������������������������������������ S ��������������� (������������) */
710 static void load_region_by_gauche() {
711
712 GtkTextIter start, end, p;
713 gchar *code;
714 gtk_text_buffer_get_end_iter(Shiki_CURRENT_TEXT_BUFFER, &p);
715 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, "\n\n", -1);
716
717 /* ������������������������������������������������������������ */
718 if(gtk_text_buffer_get_selection_bounds(Shiki_CURRENT_TEXT_BUFFER, &start, &end)) {
719 code = gtk_text_buffer_get_text(Shiki_CURRENT_TEXT_BUFFER, &start, &end, FALSE);
720 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, load_cstring_by_gauche(code), -1);
721 g_free(code);
722 }
723 }
724
725 // GtkTextCharPredicate
726 static gboolean is_kakko_or_kokka(gunichar ch, gpointer p) {
727 return ch == '(' || ch == ')';
728 }
729 static gboolean is_kakko(gunichar ch, gpointer p) {return ch == '(';}
730 static gboolean is_kokka(gunichar ch, gpointer p) {return ch == ')';}
731
732 /* ��������������������� '(' ��������������� ')' ������������������ (S ���) ��������������� */
733 static gboolean search_sexp(GtkTextIter *start, GtkTextIter *end) {
734
735 /* ������������������������������ */
736 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, start, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
737
738 if(gtk_text_iter_get_char(start) != '(')
739 gtk_text_iter_forward_find_char(start, is_kakko, NULL, NULL);
740
741 *end = *start;
742
743 /* ��������������������������������� S ������������������ */
744 if(!search_sexp_kokka(end)) return FALSE;
745 gtk_text_iter_forward_char(end);
746 return TRUE;
747 }
748
749 static gboolean search_sexp_kokka(GtkTextIter *end) {
750 gint nest_level = 0;
751
752 /* ������������ ')' ��������� */
753 while(1) {
754 if(!gtk_text_iter_forward_find_char(end, is_kakko_or_kokka, NULL, NULL))
755 return FALSE;
756
757 if(gtk_text_iter_get_char(end) == '(')
758 nest_level++;
759 else {
760 if(!nest_level)
761 break;
762 else
763 nest_level--;
764 }
765 }
766 return TRUE;
767 }
768
769 /* ��������������������� ')' ��������������� '(' ������������������ (S ���) ��������������� */
770 static gboolean search_last_sexp(GtkTextIter *start, GtkTextIter *end) {
771
772 /* ������������������������������ */
773 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, end, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
774
775 gtk_text_iter_backward_char(end);
776
777 if(gtk_text_iter_get_char(end) != ')')
778 gtk_text_iter_backward_find_char(end, is_kokka, NULL, NULL);
779 *start = *end;
780 gtk_text_iter_forward_char(end);
781
782 /* ��������������������������������� S ������������������ */
783 if(!search_last_sexp_kakko(start)) return FALSE;
784
785 return TRUE;
786 }
787
788 /* ')' ��������������� '(' ������������������ (S ���) ��������������� */
789 static gboolean search_last_sexp_kakko(GtkTextIter *start) {
790 gint nest_level = 0;
791 /* ��������������������������������������� ')' ��������� */
792 while(1) {
793 if(!gtk_text_iter_backward_find_char(start, is_kakko_or_kokka, NULL, NULL))
794 return FALSE;
795
796 if(gtk_text_iter_get_char(start) == ')')
797 nest_level++;
798 else {
799 if(!nest_level)
800 break;
801 else
802 nest_level--;
803 }
804 }
805 return TRUE;
806 }
807
808 /* ������������������������������������������������ */
809 static gint get_parent_nest_level_at_cursor(GtkTextBuffer *buffer) {
810 gint nest_level = 0;
811 GtkTextIter start, end;
812 gtk_text_buffer_get_start_iter(buffer, &start);
813 if(gtk_text_iter_get_char(&start) == '(') nest_level++;
814
815 /* ��������������������� (= end) ��������� */
816 gtk_text_buffer_get_iter_at_mark(buffer,&end, gtk_text_buffer_get_insert(buffer));
817
818 while(1) {
819 /* end ������ '(' ��� ')' ��������������������������������������������� */
820 if(!gtk_text_iter_forward_find_char(&start, is_kakko_or_kokka, NULL, &end))
821 return nest_level;
822
823 if(gtk_text_iter_get_char(&start) == '(')
824 nest_level++;
825 else
826 nest_level--;
827 }
828 }
829
830 /* ��������������������������������� on/off */
831 static void tabsborder_on_off(GtkButton *button, GtkNotebook *notebook) {
832 gint tval = FALSE;
833 gint bval = FALSE;
834 if(notebook->show_tabs == FALSE)
835 tval = TRUE;
836 if(notebook->show_border == FALSE)
837 bval = TRUE;
838
839 gtk_notebook_set_show_tabs(notebook, tval);
840 gtk_notebook_set_show_border(notebook, bval);
841 }
842
843 /* ������������������������ */
844 static void rotate_tab_position(GtkButton *button, GtkNotebook *notebook ) {
845 gtk_notebook_set_tab_pos(notebook, (notebook->tab_pos + 1) % 4);
846 }
847
848 /* ��������������������� */
849 static gboolean signal_key_release_handler (GtkWidget *notebook, GdkEventKey *event, gpointer contextid) {
850 static gint metakey_pressed = 0;
851 static gint controlx_pressed = 0;
852
853 if(event->keyval == GDK_parenright && event->state & GDK_SHIFT_MASK) {
854 GtkTextIter start, end;
855
856 /* ������������������������������ */
857 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, &end, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
858
859 start = end;
860 gtk_text_iter_backward_char(&start);
861
862 /* ��������������������������������� S ������������������ */
863 if(!search_last_sexp_kakko(&start)) return FALSE;
864
865 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "parent_emphasis_background", &start, &end);
866 }
867
868 /* ������������������������������������������������������������������������������������������������ (���������������) ������������������ */
869 if(event->keyval == GDK_Return) {
870 gint indentWidth = get_parent_nest_level_at_cursor(Shiki_CURRENT_TEXT_BUFFER) * editor_indent_width;
871 gchar *indent = g_strnfill(indentWidth, ' ');
872 gtk_text_buffer_insert_at_cursor(Shiki_CURRENT_TEXT_BUFFER, indent, -1);
873 g_free(indent);
874 }
875
876 /* C-x */
877 if(event->keyval == GDK_x && event->state & GDK_CONTROL_MASK) {
878 controlx_pressed++;
879 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-x -");
880 } else if(event->state & GDK_CONTROL_MASK) {
881
882 if(controlx_pressed > 0) {
883 switch(event->keyval) {
884 case GDK_c :/* C-x C-c : ������ */
885 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-c");
886 {/* "delete-event" ��������������������������������������� �� ������������������������������������ */
887 GdkEvent ev;
888
889 ev.any.type = GDK_DELETE;
890 ev.any.window = Shiki_EDITOR_WINDOW->window;
891 ev.any.send_event = FALSE;
892 gdk_event_put (&ev);
893 }
894 break;
895
896 case GDK_f : /* C-x C-f : ������������������ */
897 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-f");
898 open_file_handler();
899 break;
900
901 case GDK_s : /* C-x C-s : ������������������ */
902 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-s");
903 save_file();
904 break;
905
906 case GDK_w : /* C-x C-w : ������������������������ */
907 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-w");
908 save_file_as();
909 break;
910 }
911 controlx_pressed = 0;
912 }
913
914 switch(event->keyval) {
915 case GDK_g :/* C-g : ��������������� */
916 metakey_pressed = 0;
917 controlx_pressed = 0;
918
919 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "Quit");
920 break;
921 }
922
923 }
924 return FALSE;
925 }
926 static void open_online_help() {
927 GtkTextIter p;
928 Shiki_new_buffer_create(g_strdup("*help*"));
929 gtk_text_buffer_set_text(Shiki_CURRENT_TEXT_BUFFER,
930 "������������������������������������������\n"
931 "$ ./shiki [file1 file2 ....]\n\n"
932 "[���������������������������] ��������������������� (C-x C-f)\n"
933 "[������������������������������������������] ��������������������� (C-x C-s)\n"
934 "[���������������������������������������������] ��������������������������� (C-x C-w)\n"
935 "[���������������������] ��������������� gauche ������������\n"
936 "[��������������� (���������) ���������������] ��������� on/off\n"
937 "[������ (���������) ���������������] ������������������������\n"
938 "[������������������] ��������������������������� (C-t)\n"
939 "[���������������������������] ������������ (C-_)\n"
940 "[���������������������] ������������������������\n"
941 "\n"
942 "(��� : ������������������������������������������������������������������������������������������������������������������������������������������������������������������������������������������������������ Undo ������������������)\n"
943 "\n"
944 "[�� ������������] ������������������������ (C-k)\n"
945 "[A ������������] ���������������������\n"
946 "[���������������������������] Scheme ������������������������\n"
947 "[���������������������������������] ���������������������������������������\n"
948 "[��������� (?) ������������] ���������������������������������������\n"
949 "[info ������������] ���������������������������������������������������\n"
950 "\n"
951 "C-f : ��� ��������� (forward)\n"
952 "C-b : ��� ��������� (backward)\n"
953 "C-n : ��� ��������� (next line)\n"
954 "C-p : ��� ��������� (previous line)\n"
955 "\n"
956 "C-h : ���������������������\n"
957 "C-w : ���������\n"
958 "C-y : ��������� (������������)\n"
959 "\n"
960 "C-e : ��������������������� S ������������ (eval-expression)\n"
961 "C-j : ��������������������� S ������������ (eval-last-sexp)\n"
962 "(emacs/xyzzy ��� *scratch* ���������������������)\n"
963 "\n"
964 "C-M-@ : ��������������������� S ������������ (mark-sexp)\n"
965 "C-M-SPC : ��������������������� S ������������ (mark-last-sexp)\n"
966 "C-x C-c : ��������������������������� �� ���������������������������������\n"
967 "\n"
968 "������������ API\n"
969 "(clear-current-buffer) : ��������������������������������� (������������)\n"
970 , -1);
971 gtk_text_buffer_set_modified(Shiki_CURRENT_TEXT_BUFFER, FALSE);
972 /* ������������������������������ */
973 gtk_text_buffer_get_start_iter(Shiki_CURRENT_TEXT_BUFFER, &p);
974 gtk_text_buffer_place_cursor(Shiki_CURRENT_TEXT_BUFFER, &p);
975 }
976
977 /* ��������������������������������������� */
978 static void Shiki_editor_window_init(int argc, char **argv) {
979 GtkWidget *vbox, *toolbar, *modeline_bg = gtk_event_box_new();
980 GtkToolItem *icon;
981 GtkIconSize iconsize;
982 GtkTooltips *toolbar_tips = gtk_tooltips_new();
983 /* ��������������������������������������������������������������������������������� */
984 GtkToolItem *oicon, *sicon, *saicon, *eicon;
985
986 gint contextid, i;
987
988 /* ������������������������������������������������������������������������������ */
989 keywords_hash = g_hash_table_new(g_str_hash, g_str_equal);
990 i = 0;
991 while(R5RS_keywords[i] != NULL)
992 g_hash_table_insert(keywords_hash, R5RS_keywords[i++], GINT_TO_POINTER(R5RS_KEYWORD_COLOR));
993 i = 0;
994 while(R5RS_functions[i] != NULL)
995 g_hash_table_insert(keywords_hash, R5RS_functions[i++], GINT_TO_POINTER(R5RS_FUNCTION_COLOR));
996
997 /* ������������ */
998 Shiki_EDITOR_WINDOW = gtk_window_new(GTK_WINDOW_TOPLEVEL);
999 g_signal_connect(G_OBJECT(Shiki_EDITOR_WINDOW), "destroy", G_CALLBACK(gtk_main_quit), NULL);
1000
1001 /* ������������������������������������������������������������ */
1002 Shiki_EDITOR_CLIPBOARD = gtk_clipboard_get(GDK_SELECTION_PRIMARY);
1003
1004 /* ������������������������������������ */
1005 vbox = gtk_vbox_new(FALSE, 0);
1006 /* ��������������������� */
1007 toolbar = gtk_toolbar_new();
1008 gtk_box_pack_start(GTK_BOX(vbox), toolbar, FALSE, FALSE, 0);
1009
1010 Shiki_EDITOR_NOTEBOOK = GTK_NOTEBOOK(gtk_notebook_new());
1011 g_signal_connect(G_OBJECT(Shiki_EDITOR_NOTEBOOK), "switch-page", GTK_SIGNAL_FUNC(switch_tabpage_handler), NULL);
1012
1013 /* ������������������������������������������������ */
1014 gtk_toolbar_set_style(GTK_TOOLBAR (toolbar), GTK_TOOLBAR_ICONS);
1015 iconsize = gtk_toolbar_get_icon_size (GTK_TOOLBAR (toolbar));
1016
1017 /* ������������������ */
1018
1019 /* ������������������ */
1020 oicon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_OPEN, iconsize), "");
1021 /* ������������������������������������������������������������������������������������ */
1022 g_signal_connect(G_OBJECT(oicon), "clicked", G_CALLBACK(open_file_handler), NULL);
1023 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(oicon));
1024 gtk_tool_item_set_tooltip(oicon, toolbar_tips, "���������������������������",
1025 "���������������������������������������������������������������������������������������");
1026
1027 /* ������������������ */
1028 sicon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_SAVE, iconsize), "");
1029 /* ������������������������������������������������������������������������������������ */
1030 g_signal_connect(G_OBJECT(sicon), "clicked", G_CALLBACK(save_file), NULL);
1031 gtk_container_add (GTK_CONTAINER (toolbar), GTK_WIDGET(sicon));
1032 gtk_tool_item_set_tooltip(sicon, toolbar_tips, "������������������������������",
1033 "������������������������������������������������������������������������������������������������������������������������������������");
1034
1035 /* ��������������������������� */
1036 saicon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_SAVE_AS, iconsize), "");
1037 /* ������������������������������������������������������������������������������������������������������������������ */
1038 g_signal_connect(G_OBJECT(saicon), "clicked", G_CALLBACK(save_file_as), NULL);
1039 gtk_container_add (GTK_CONTAINER (toolbar), GTK_WIDGET(saicon));
1040 gtk_tool_item_set_tooltip(saicon, toolbar_tips, "������������������������������������",
1041 "");
1042
1043 /* ������������������ */
1044 eicon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_EXECUTE, iconsize), "");
1045 /* ������������������������������������������ libgauche ������������������ */
1046 g_signal_connect(G_OBJECT(eicon), "clicked", G_CALLBACK(load_region_by_gauche), NULL);
1047 gtk_container_add (GTK_CONTAINER (toolbar), GTK_WIDGET(eicon));
1048 gtk_tool_item_set_tooltip(eicon, toolbar_tips, "��������������� S ������������������������ (load-region-lisp)",
1049 "Scheme (gauche) ������������������ S ������������������������");
1050
1051 gtk_container_add(GTK_CONTAINER(Shiki_EDITOR_WINDOW), vbox);
1052 gtk_container_add(GTK_CONTAINER(vbox), GTK_WIDGET(Shiki_EDITOR_NOTEBOOK));
1053
1054 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_NEW, iconsize), "");
1055 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(append_default_tabpage_handler), NULL);
1056 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1057 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������������������", "");
1058
1059 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_UNDO, iconsize), "");
1060 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(undo), G_OBJECT(Shiki_EDITOR_NOTEBOOK));
1061 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1062 gtk_tool_item_set_tooltip(icon, toolbar_tips, "Undo","");
1063
1064 /* XXX : TODO */
1065 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_REDO, iconsize), "");
1066 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(dummy_handler), NULL);
1067 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1068 gtk_tool_item_set_tooltip(icon, toolbar_tips, "Redo", "");
1069
1070 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_FIND, iconsize), "");
1071 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(search_current_buffer), NULL);
1072 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1073 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������", "");
1074
1075
1076 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_FIND_AND_REPLACE, iconsize), "");
1077 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(replace_current_buffer), NULL);
1078 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1079 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������", "");
1080
1081 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_REFRESH, iconsize), "");
1082 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(dummy_handler), NULL);
1083 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1084 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������", "");
1085
1086 /* TODO ������������ */
1087
1088 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_DELETE, iconsize), "");
1089 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(clear_current_buffer_handler), NULL);
1090 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1091 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������������������",
1092 "���������������������������������������������������������������������");
1093
1094 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_CLOSE, iconsize), "");
1095 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(kill_buffer_handler), NULL);
1096 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1097 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������������",
1098 "���������������������������������������������������������������");
1099
1100 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_CONNECT, iconsize), "");
1101 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(load_scheme_file_by_gauche), NULL);
1102 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1103 gtk_tool_item_set_tooltip(icon, toolbar_tips, "Scheme ������������������������", "");
1104
1105 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_CONVERT, iconsize), "");
1106 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(load_buffer_by_gauche), G_OBJECT(Shiki_EDITOR_NOTEBOOK));
1107 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1108 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������������������", "");
1109
1110 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_SPELL_CHECK, iconsize), "");
1111 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(scheme_keyword_highlighting_current_buffer), NULL);
1112 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1113 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������������������������������������", "");
1114
1115
1116 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_DIALOG_QUESTION, iconsize), "");
1117 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(open_online_help), NULL);
1118 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1119 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������", "");
1120
1121 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_BOLD, iconsize), "");
1122 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(select_font), NULL);
1123 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1124 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������", "");
1125
1126 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_APPLY, iconsize), "");
1127 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(tabsborder_on_off), G_OBJECT(Shiki_EDITOR_NOTEBOOK));
1128 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1129 gtk_tool_item_set_tooltip(icon, toolbar_tips, "��������� on/off", "");
1130
1131 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_PREFERENCES, iconsize), "");
1132 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(rotate_tab_position), G_OBJECT(Shiki_EDITOR_NOTEBOOK));
1133 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1134 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������", "");
1135
1136 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_DIALOG_INFO, iconsize), "");
1137 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(about_this_application), NULL);
1138 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1139 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������������������������������������", "");
1140
1141 /* ������������������������ */
1142 Shiki_EDITOR_MODELINE_LABEL = gtk_label_new(NULL);
1143 gtk_container_add(GTK_CONTAINER (modeline_bg), Shiki_EDITOR_MODELINE_LABEL);
1144
1145 gdk_color_parse("black", &COLOR_BLACK);
1146 gdk_color_parse("green", &COLOR_GREEN);
1147
1148 gtk_widget_modify_fg(Shiki_EDITOR_MODELINE_LABEL, GTK_STATE_NORMAL, &COLOR_GREEN);
1149 gtk_widget_modify_bg(modeline_bg, GTK_STATE_NORMAL, &COLOR_BLACK);
1150
1151 gtk_box_pack_start(GTK_BOX(vbox), modeline_bg, TRUE, TRUE, 0);
1152
1153 /* C-x C-s ��������������������������������������������������������������������������������������� */
1154 Shiki_EDITOR_STATUSBAR = gtk_statusbar_new();
1155 gtk_box_pack_start(GTK_BOX(vbox), Shiki_EDITOR_STATUSBAR, TRUE, TRUE, 0);
1156 contextid = gtk_statusbar_get_context_id(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), "");
1157
1158 /* ������������������������������������������������ */
1159 g_signal_connect(G_OBJECT(Shiki_EDITOR_NOTEBOOK), "key-press-event", G_CALLBACK (signal_key_press_handler), GINT_TO_POINTER(contextid));
1160 g_signal_connect(G_OBJECT(Shiki_EDITOR_NOTEBOOK), "key-release-event", G_CALLBACK (signal_key_release_handler), GINT_TO_POINTER(contextid));
1161
1162 /* ��������������������������������������������� */
1163 if(argc >= 2) {
1164 int i;
1165 for(i = 1; i < argc; i++)
1166 open_file(g_strdup(argv[i]));
1167 } else /* ������������������������������������������������������������������ */
1168 open_online_help(Shiki_EDITOR_NOTEBOOK);
1169
1170 gtk_widget_grab_focus(GTK_WIDGET(Shiki_EDITOR_NOTEBOOK));
1171 gtk_widget_show_all(Shiki_EDITOR_WINDOW);
1172 }
1173
1174 int main(int argc, char *argv[]) {
1175 /* ������������������������������������ */
1176 Shiki_EDITOR_DEFAULT_LOCALE = g_locale_to_utf8(gtk_set_locale(), -1, NULL, NULL, NULL);
1177 gtk_init(&argc, &argv);
1178 GC_INIT(); Scm_Init(GAUCHE_SIGNATURE);
1179 Scm_Load("gauche-init.scm", 0);
1180 Shiki_editor_window_init(argc, argv);
1181 gtk_main();
1182 Scm_Exit(0);
1183 return 0;
1184 }

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