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.60 - (show annotations) (download) (as text)
Wed Nov 29 05:08:46 2006 UTC (17 years, 4 months ago) by aloha
Branch: MAIN
Changes since 1.59: +6 -67 lines
File MIME type: text/x-csrc
add some 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.59 2006/11/29 04:37:23 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 save_file();
61 static void save_file_as();
62
63 /* ������������������������������ */
64 static gchar* get_all_buffer_contents(GtkTextBuffer *buffer);
65 static gboolean save_text_buffer(const gchar *filename, GtkTextBuffer *buffer);
66 static void clear_current_buffer_handler();
67 static void undo();
68 static void search_current_buffer();
69
70 /* Gauche ��� S ��������������������������������������������������������������������������� */
71 static gchar *eval_cstring_by_gauche(gchar *s);
72 static gchar *load_cstring_by_gauche(gchar *s);
73 static void load_buffer_by_gauche();
74 static void load_region_by_gauche();
75 static void load_scheme_file_by_gauche();
76 static gboolean is_kakko_or_kokka(gunichar ch, gpointer);
77 static gboolean is_kakko(gunichar ch, gpointer);
78 static gboolean is_kokka(gunichar ch, gpointer);
79 static gboolean search_sexp(GtkTextIter *start, GtkTextIter *end);
80 static gboolean search_sexp_kokka(GtkTextIter *end);
81 static gboolean search_last_sexp(GtkTextIter *start, GtkTextIter *end);
82 static gboolean search_last_sexp_kakko(GtkTextIter *start);
83 static gint get_parent_nest_level_at_cursor(GtkTextBuffer *buffer);
84 static gboolean is_not_scheme_delimita_p(gunichar ch, gpointer user_data);
85 static gboolean is_double_quote(gunichar ch, gpointer user_data);
86 static gboolean is_scheme_delimita_p(gunichar ch, gpointer user_data);
87 static void scheme_keyword_highlighting_current_buffer();
88
89 /* ������ */
90 static void select_font();
91 static void font_selection_ok(GtkWidget *button, GtkWidget *font_dialog);
92 static void switch_tabpage_handler(GtkNotebook *notebook, GtkNotebookPage *page, guint pagenum) ;
93 static void tabsborder_on_off(GtkButton *button, GtkNotebook *notebook);
94 static void rotate_tab_position(GtkButton *button, GtkNotebook *notebook);
95
96 /* ������������������������ */
97 static gboolean signal_key_press_handler(GtkWidget *notebook, GdkEventKey *event, gpointer contextid);
98 static gboolean signal_key_release_handler(GtkWidget *notebook, GdkEventKey *event, gpointer contextid);
99
100 /* ������������������ */
101 static void open_online_help();
102 static void about_this_application();
103
104 /* ������������������������ */
105 static void Shiki_editor_window_init(int argc, char **argv);
106
107 /* ������������������������������ */
108 static void clear_current_buffer_handler() {
109 Shiki_erase_buffer(Shiki_CURRENT_TEXT_BUFFER);
110 }
111
112 static void destroy_handler(GtkWidget *button, GtkWidget *widget) {gtk_widget_destroy(widget);}
113
114 static void toggled_handler(GtkToggleButton *togglebutton, gboolean *flag) {
115 *flag = !*flag;
116 }
117
118 /* ��������������������������������������������������������������� */
119 static void update_modeline_label() {
120 static gchar label[1024];
121 GtkTextIter p;
122 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, &p, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
123
124 g_snprintf(label, 1024, "--%s- %-10s (Gauche Interaction) [%s] L%d:%d ",
125 gtk_text_buffer_get_modified(Shiki_CURRENT_TEXT_BUFFER) ? "**" : "--",
126 Shiki_CURRENT_BASENAME,
127 Shiki_CURRENT_CES,
128 gtk_text_iter_get_line(&p) + 1,
129 gtk_text_iter_get_line_offset (&p) + 1);
130 gtk_label_set_text(GTK_LABEL(Shiki_EDITOR_MODELINE_LABEL), label);
131 }
132
133 static void replace_current_buffer() {
134 GtkWidget *dialog = gtk_dialog_new_with_buttons ("������������������", GTK_WINDOW(Shiki_EDITOR_WINDOW), GTK_DIALOG_DESTROY_WITH_PARENT, NULL);
135 GtkWidget *table = gtk_table_new(7, 3, FALSE);
136 GtkWidget *find_label = gtk_label_new("������ : ");
137 GtkWidget *find = gtk_entry_new();
138 GtkWidget *rep_label = gtk_label_new("������ : ");
139 GtkWidget *replace = gtk_entry_new();
140 GtkWidget *check1 = gtk_check_button_new_with_label("���������������������������������");
141 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check1), TRUE);
142 GtkWidget *check2 = gtk_check_button_new_with_label("���������������������������");
143 GtkWidget *check3 = gtk_check_button_new_with_label("������������");
144 GtkWidget *check4 = gtk_check_button_new_with_label("���������������������������������������������");
145 GtkWidget *check5 = gtk_check_button_new_with_label("���������������������������");
146 GtkWidget *interactive = gtk_button_new_with_label ("������������");
147 GtkWidget *all = gtk_button_new_with_label ("������������");
148 GtkWidget *cancel = gtk_button_new_with_label ("���������������");
149 gboolean f1 = TRUE, f2, f3, f4, f5;
150 f2 = f3 = f4 = f5 = FALSE;
151 g_signal_connect (check1, "toggled", G_CALLBACK (toggled_handler), &f1);
152 g_signal_connect (check2, "toggled", G_CALLBACK (toggled_handler), &f2);
153 g_signal_connect (check3, "toggled", G_CALLBACK (toggled_handler), &f3);
154 g_signal_connect (check4, "toggled", G_CALLBACK (toggled_handler), &f4);
155 g_signal_connect (check5, "toggled", G_CALLBACK (toggled_handler), &f5);
156
157 g_signal_connect (G_OBJECT(dialog), "delete_event", G_CALLBACK(gtk_widget_destroy), NULL);
158 g_signal_connect (G_OBJECT(cancel), "clicked", G_CALLBACK(destroy_handler), dialog);
159 gtk_table_set_row_spacings(GTK_TABLE(table), 10);
160 gtk_table_set_col_spacings(GTK_TABLE(table), 10);
161 gtk_container_border_width (GTK_CONTAINER (dialog), 10);
162
163 gtk_table_attach_defaults (GTK_TABLE(table), find_label, 0, 1, 0, 1);
164 gtk_table_attach_defaults (GTK_TABLE(table), find, 1, 2, 0, 1);
165 gtk_table_attach_defaults (GTK_TABLE(table), interactive, 2, 3, 0, 1);
166
167 gtk_table_attach_defaults (GTK_TABLE(table), rep_label, 0, 1, 1, 2);
168 gtk_table_attach_defaults (GTK_TABLE(table), replace, 1, 2, 1, 2);
169 gtk_table_attach_defaults (GTK_TABLE(table), all, 2, 3, 1, 2);
170
171 gtk_table_attach_defaults (GTK_TABLE(table), check1, 1, 2, 2, 3);
172 gtk_table_attach_defaults (GTK_TABLE(table), cancel, 2, 3, 2, 3);
173
174 gtk_table_attach_defaults (GTK_TABLE(table), check2, 1, 2, 3, 4);
175 gtk_table_attach_defaults (GTK_TABLE(table), check3, 1, 2, 4, 5);
176 gtk_table_attach_defaults (GTK_TABLE(table), check4, 1, 2, 5, 6);
177 gtk_table_attach_defaults (GTK_TABLE(table), check5, 1, 2, 6, 7);
178
179 gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), table);
180 gtk_widget_show_all(table);
181 gtk_dialog_run(GTK_DIALOG(dialog));
182 }
183
184 static void search_current_buffer() {
185 GtkWidget *dialog = gtk_dialog_new_with_buttons ("������������������", GTK_WINDOW(Shiki_EDITOR_WINDOW), GTK_DIALOG_DESTROY_WITH_PARENT, NULL);
186 GtkWidget *table = gtk_table_new(6, 3, FALSE);
187 GtkWidget *label = gtk_label_new("������ : ");
188 GtkWidget *input = gtk_entry_new();
189 GtkWidget *check1 = gtk_check_button_new_with_label("���������������������������������");
190 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check1), TRUE);
191 GtkWidget *check2 = gtk_check_button_new_with_label("���������������������������");
192 GtkWidget *check3 = gtk_check_button_new_with_label("������������");
193 GtkWidget *check4 = gtk_check_button_new_with_label("���������������������������������������������");
194 GtkWidget *check5 = gtk_check_button_new_with_label("���������������������������������");
195 GtkWidget *prev = gtk_button_new_with_label ("���������");
196 GtkWidget *next = gtk_button_new_with_label ("���������");
197 GtkWidget *cancel = gtk_button_new_with_label ("���������������");
198 gboolean f1 = TRUE, f2, f3, f4, f5;
199 f2 = f3 = f4 = f5 = FALSE;
200 g_signal_connect (check1, "toggled", G_CALLBACK (toggled_handler), &f1);
201 g_signal_connect (check2, "toggled", G_CALLBACK (toggled_handler), &f2);
202 g_signal_connect (check3, "toggled", G_CALLBACK (toggled_handler), &f3);
203 g_signal_connect (check4, "toggled", G_CALLBACK (toggled_handler), &f4);
204 g_signal_connect (check5, "toggled", G_CALLBACK (toggled_handler), &f5);
205
206 g_signal_connect (G_OBJECT(dialog), "delete_event", G_CALLBACK(gtk_widget_destroy), NULL);
207 g_signal_connect (G_OBJECT(cancel), "clicked", G_CALLBACK(destroy_handler), dialog);
208 gtk_table_set_row_spacings(GTK_TABLE(table), 10);
209 gtk_table_set_col_spacings(GTK_TABLE(table), 10);
210 gtk_container_border_width (GTK_CONTAINER (dialog), 10);
211 gtk_table_attach_defaults (GTK_TABLE(table), label, 0, 1, 0, 1);
212 gtk_table_attach_defaults (GTK_TABLE(table), input, 1, 2, 0, 1);
213 gtk_table_attach_defaults (GTK_TABLE(table), prev, 2, 3, 0, 1);
214 gtk_table_attach_defaults (GTK_TABLE(table), check1, 1, 2, 1, 2);
215 gtk_table_attach_defaults (GTK_TABLE(table), check2, 1, 2, 2, 3);
216 gtk_table_attach_defaults (GTK_TABLE(table), check3, 1, 2, 3, 4);
217 gtk_table_attach_defaults (GTK_TABLE(table), check4, 1, 2, 4, 5);
218 gtk_table_attach_defaults (GTK_TABLE(table), check5, 1, 2, 5, 6);
219 gtk_table_attach_defaults (GTK_TABLE(table), next, 2, 3, 1, 2);
220 gtk_table_attach_defaults (GTK_TABLE(table), cancel, 2, 3, 2, 3);
221 gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), table);
222 gtk_widget_show_all(table);
223 gtk_dialog_run(GTK_DIALOG(dialog));
224 }
225
226 static gboolean is_not_scheme_delimita_p(gunichar ch, gpointer user_data) {
227 return ch != '(' && ch != ')' && !g_unichar_isspace(ch);
228 }
229
230 static gboolean is_double_quote(gunichar ch, gpointer user_data) {
231 return ch == '\"';
232 }
233
234 static gboolean is_scheme_delimita_p(gunichar ch, gpointer user_data) {
235 return ch == ' ' || ch == '(' || ch == ')' || ch == '\"' || g_unichar_isspace(ch);
236 }
237
238 /* ������������������������������ */
239 static void scheme_keyword_highlighting_current_buffer() {
240 GtkTextIter s, e;
241 HIGHILIGHT_COLOR c;
242 gchar *word;
243 gboolean is_comment, is_string;
244 gunichar ch;
245
246 gtk_text_buffer_get_start_iter(Shiki_CURRENT_TEXT_BUFFER, &s);
247
248 /* ��������� Scheme ������������������������������ */
249 while(TRUE) {
250 is_comment = FALSE;
251 is_string = FALSE;
252 if((ch = gtk_text_iter_get_char(&s)) != ';' && ch != '\"')
253 gtk_text_iter_forward_find_char(&s, is_not_scheme_delimita_p, NULL, NULL);
254 e = s;
255 if(gtk_text_iter_get_char(&s) == ';') {
256 gtk_text_iter_forward_line(&e);
257 gtk_text_iter_backward_char(&e);
258 is_comment = TRUE;
259 } else if(gtk_text_iter_get_char(&s) == '\"') {
260 while(TRUE) {
261 gtk_text_iter_forward_find_char(&e, is_double_quote, NULL, NULL);
262 gtk_text_iter_backward_char(&e);
263 if(gtk_text_iter_get_char(&e) != '\\') {
264 is_string = TRUE;
265 gtk_text_iter_forward_char(&e);
266 gtk_text_iter_forward_char(&e);
267 break;
268 }
269 gtk_text_iter_forward_char(&e);
270 gtk_text_iter_forward_char(&e);
271 }
272
273 } else
274 gtk_text_iter_forward_find_char(&e, is_scheme_delimita_p, NULL, NULL);
275
276 word = gtk_text_buffer_get_text(Shiki_CURRENT_TEXT_BUFFER, &s, &e, FALSE);
277
278 /* ������������������������������������������������������������ */
279 if(is_comment) /* ������������ */
280 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "comment_highlighting", &s, &e);
281 else if(is_string) /* ��������� */
282 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "string_highlighting", &s, &e);
283 else if(R5RS_KEYWORD_COLOR == (c = GPOINTER_TO_INT(g_hash_table_lookup(keywords_hash, word)))) /* R5RS ��������������� */
284 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "keyword_highlighting", &s, &e);
285 else if(R5RS_FUNCTION_COLOR == c) /* R5RS ������ */
286 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "function_highlighting", &s, &e);
287
288 /* XXX : get_text() ������������������������������������������������������������������������������������������������������ GtkTextBuffer ������ const gchar * ������������������������������ */
289 g_free(word);
290
291 if(gtk_text_iter_is_end(&e)) break;
292 s = e;
293 }
294 }
295
296 /* ������������������������������������������������������������ */
297 static void switch_tabpage_handler(GtkNotebook *notebook, GtkNotebookPage *page, guint pagenum) {
298 /* ��������������������������������������������������������������������� */
299 Shiki_CURRENT_TAB_INFO = (ShikiBuffer *)g_list_nth_data(Shiki_EDITOR_BUFFER_LIST, pagenum);
300
301 /* ������������������������������������ */
302 Shiki_CURRENT_TAB_NUM = pagenum;
303
304 /* ������������������������������������������������������ */
305 if(!Shiki_CURRENT_TAB_INFO) return;
306 gtk_window_set_title (GTK_WINDOW(Shiki_EDITOR_WINDOW), Shiki_CURRENT_FILENAME);
307
308 update_modeline_label();
309 }
310
311 static void undo() {
312 g_print("Undo\n");
313 GtkTextIter start, end;
314 ShikiUndoInfo *undoInfo = g_list_nth_data(Shiki_CURRENT_UNDO_INFO_LIST, 0);
315 if(!undoInfo) {
316 g_print("������������ Undo ���������������\n");
317 return;
318 }
319 gtk_text_buffer_get_iter_at_offset(Shiki_CURRENT_TEXT_BUFFER, &start, undoInfo->start);
320 gtk_text_buffer_get_iter_at_offset(Shiki_CURRENT_TEXT_BUFFER, &end, undoInfo->end);
321
322 if(undoInfo->action == SHIKI_UNDO_INSERT) {
323 Shiki_CURRENT_UNDO_INFO_LIST = g_list_delete_link(Shiki_CURRENT_UNDO_INFO_LIST, g_list_first(Shiki_CURRENT_UNDO_INFO_LIST));
324 gtk_text_buffer_delete(Shiki_CURRENT_TEXT_BUFFER, &start, &end);
325 g_free(undoInfo->str);
326 g_free(undoInfo);
327 } else if(undoInfo->action == SHIKI_UNDO_DELETE) {
328 Shiki_CURRENT_UNDO_INFO_LIST = g_list_delete_link(Shiki_CURRENT_UNDO_INFO_LIST, g_list_first(Shiki_CURRENT_UNDO_INFO_LIST));
329 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &start, undoInfo->str, -1);
330 g_free(undoInfo->str);
331 g_free(undoInfo);
332 }
333
334 }
335
336 /* ��������������������� */
337 static gboolean signal_key_press_handler (GtkWidget *notebook, GdkEventKey *event, gpointer contextid) {
338 GtkTextIter start, end;
339
340 /* ������������������������������������ */
341 gtk_text_buffer_get_start_iter(Shiki_CURRENT_TEXT_BUFFER, &start);
342 gtk_text_buffer_get_end_iter(Shiki_CURRENT_TEXT_BUFFER, &end);
343 gtk_text_buffer_remove_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "parent_emphasis_background", &start, &end);
344
345 if(event->state & GDK_CONTROL_MASK && event->state & GDK_MOD1_MASK) {
346 switch(event->keyval) {
347 case GDK_at : /* C-M-SPC */
348 { GtkTextIter start, end;
349 if(!search_sexp(&start, &end)) return FALSE;
350 gtk_text_buffer_select_range(Shiki_CURRENT_TEXT_BUFFER, &start, &end);
351 }
352 break;
353 case GDK_space : /* C-M-SPC */
354 { GtkTextIter start, end;
355 if(!search_last_sexp(&start, &end)) return FALSE;
356 gtk_text_buffer_select_range(Shiki_CURRENT_TEXT_BUFFER, &start, &end);
357 }
358 break;
359 }
360 } else if(event->state & GDK_CONTROL_MASK) {
361 switch(event->keyval) {
362 case GDK_f : /* Ctrl + f : forward */
363 Shiki_forward_char();
364 break;
365 case GDK_b : /* Ctrl + b : backward */
366 Shiki_backward_char();
367 break;
368 case GDK_n : /* Ctrl + n : next line */
369 Shiki_forward_line(1);
370 break;
371 case GDK_p : /* Ctrl + p : previous line */
372 Shiki_forward_line(-1);
373 break;
374 case GDK_h :
375 { GtkTextIter p;
376 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER,&p, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
377 gtk_text_buffer_backspace(Shiki_CURRENT_TEXT_BUFFER, &p, FALSE, TRUE);
378 }
379 break;
380
381 case GDK_e : /* Ctrl + e : eval-expression */
382 {
383 gchar *code, *result;
384 GtkTextIter start, end;
385
386 if(!search_sexp(&start, &end)) return FALSE;
387
388 code = gtk_text_buffer_get_text(Shiki_CURRENT_TEXT_BUFFER, &start, &end, FALSE);
389 result = eval_cstring_by_gauche(code);
390 g_free(code);
391
392 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, &end, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
393
394 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, "\n", -1);
395 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, result, -1);
396 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, "\n", -1);
397 }
398 break;
399
400 case GDK_j : /* Ctrl + j : eval-last-sexp */
401 {
402 gchar *code, *result;
403 GtkTextIter start, end;
404
405 if(!search_last_sexp(&start, &end)) return FALSE;
406
407 code = gtk_text_buffer_get_text(Shiki_CURRENT_TEXT_BUFFER, &start, &end, FALSE);
408 result = eval_cstring_by_gauche(code);
409 g_free(code);
410
411 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, &end, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
412
413 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, "\n", -1);
414 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, result, -1);
415 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &end, "\n", -1);
416 }
417 break;
418
419 case GDK_underscore : /* Ctrl + _ : Undo */
420 undo();
421 break;
422
423 case GDK_t : /* Ctrl + t : ��������������� */
424 append_default_tabpage_handler();
425 break;
426
427 case GDK_k : /* Ctrl + k : ������������������ */
428 kill_buffer_handler();
429 break;
430
431 case GDK_w : /* Ctrl + w : ��������� */
432 gtk_text_buffer_cut_clipboard(Shiki_CURRENT_TEXT_BUFFER, Shiki_EDITOR_CLIPBOARD, TRUE);
433 break;
434
435 case GDK_y : /* Ctrl + y : ��������� */
436 {GtkTextIter p;
437 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER,&p, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
438 gtk_text_buffer_paste_clipboard(Shiki_CURRENT_TEXT_BUFFER, Shiki_EDITOR_CLIPBOARD, &p, TRUE);
439 }
440 break;
441 }
442 }
443 return FALSE;
444 }
445
446 static void append_default_tabpage_handler() {
447 Shiki_new_buffer_create(g_strdup("*scratch*"));
448 }
449
450 /* ������������������������������������������ (������������) ��������� */
451 static void kill_buffer_handler() {
452 Shiki_kill_buffer(Shiki_CURRENT_TEXT_BUFFER);
453 }
454
455 /* ������������������������������������ */
456 static void load_buffer_by_gauche() {
457 GtkTextIter p;
458 gtk_text_buffer_get_end_iter(Shiki_CURRENT_TEXT_BUFFER, &p);
459 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, "\n\n", -1);
460 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, load_cstring_by_gauche(get_all_buffer_contents(Shiki_CURRENT_TEXT_BUFFER)), -1);
461 }
462
463 /* ������������������������ */
464 static void load_scheme_file_by_gauche() {
465 gchar *contents, *text = NULL;
466 gsize br, bw, len;
467 GError *err = NULL;
468 const gchar *filename = Shiki_file_name_dialog("File Selection");
469 GtkTextIter p;
470
471 if(!filename) return;
472
473 gtk_text_buffer_get_end_iter(Shiki_CURRENT_TEXT_BUFFER, &p);
474 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, "\n\n", -1);
475
476 if(g_file_get_contents(filename, &contents, &len, NULL)) {
477 if(!(text = g_locale_to_utf8(contents, -1, &br, &bw, &err)))
478 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, load_cstring_by_gauche(text), -1);
479 else
480 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, load_cstring_by_gauche(contents), -1);
481 }
482 g_free(text); g_free(contents);
483 }
484
485 /* gauche ������������������������������������ */
486 static gchar *load_cstring_by_gauche(gchar *s) {
487 gchar *msg;
488
489 ScmObj result, error;
490 /* ��������������������������������� */
491 ScmObj is = Scm_MakeInputStringPort(SCM_STRING(SCM_MAKE_STR_COPYING(s)), TRUE);
492 /* ������������������������������ */
493 ScmObj os = Scm_MakeOutputStringPort(TRUE);
494
495 Scm_Define(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*input*")), is);
496 Scm_Define(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*error*")), SCM_FALSE);
497 /* Scheme ��������������������������������������������������������������������������������� S ��������������������������������������������������������������������������� *error* ������������������ */
498 result = Scm_EvalCString("(guard (e (else (set! *error* e) #f)) (eval (load-from-port *input*) (current-module)))", SCM_OBJ(Shiki_CURRENT_BUFFER_ENV));
499
500 error = Scm_GlobalVariableRef(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*error*")), 0);
501
502 /* ��������������������������������������������������������� */
503 if (!SCM_FALSEP(error))
504 Scm_Write(error, os, SCM_WRITE_DISPLAY);
505 else
506 Scm_Write(result, os, SCM_WRITE_DISPLAY);
507
508 msg = Scm_GetString(SCM_STRING(Scm_GetOutputString(SCM_PORT(os))));
509 /* ������������������ */
510 Scm_ClosePort(SCM_PORT(is));
511 Scm_ClosePort(SCM_PORT(os));
512
513 return msg;
514 }
515
516 static void font_selection_ok(GtkWidget *button, GtkWidget *font_dialog) {
517 gchar *font_name = gtk_font_selection_dialog_get_font_name (GTK_FONT_SELECTION_DIALOG (font_dialog));
518 if(font_name) {
519 GtkRcStyle *style = gtk_rc_style_new ();
520 pango_font_description_free(style->font_desc);
521 style->font_desc = pango_font_description_from_string(font_name);
522 gtk_widget_modify_style (GTK_WIDGET(Shiki_CURRENT_TEXT_VIEW), style);
523 gtk_rc_style_unref (style);
524 g_free (font_name);
525 }
526 }
527
528 /* ������������������������������������������������������ */
529 static void select_font(){
530 GtkWidget *font_dialog = gtk_font_selection_dialog_new("Font Selection Dialog");
531 g_signal_connect (GTK_FONT_SELECTION_DIALOG (font_dialog)->ok_button, "clicked", G_CALLBACK(font_selection_ok), font_dialog);
532 gtk_dialog_run(GTK_DIALOG(font_dialog));
533 gtk_widget_destroy(font_dialog);
534 }
535
536 /* ������������������������������������������ */
537 static void about_this_application() {
538 GtkAboutDialog *about = GTK_ABOUT_DIALOG(gtk_about_dialog_new());
539 const gchar *authors[] = {
540 "������������ (���������) <alohakun@gmail.com>\n",
541 "Contribute : tkng ������",
542 "(http://d.hatena.ne.jp/tkng/20061113)", NULL
543 };
544 gtk_about_dialog_set_authors(about, authors);
545 gtk_about_dialog_set_copyright(about, "Copyright(C)2006 WAKATSUKI Toshihiro");
546 gtk_about_dialog_set_name(about, "��� (SHIKI)");
547 gtk_about_dialog_set_website_label(about, "���������30������������������������������������������������������������Blog");
548 gtk_about_dialog_set_website(about, "http://alohakun.blog7.fc2.com/blog-category-29.html");
549 gtk_dialog_run(GTK_DIALOG(about));
550 gtk_widget_destroy(GTK_WIDGET(about));
551 }
552
553 /* ��������� */
554 static void dummy_handler() {
555 GtkWidget *dummy = gtk_message_dialog_new(GTK_WINDOW(Shiki_EDITOR_WINDOW),
556 GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_WARNING, GTK_BUTTONS_CLOSE,
557 "Sorry... This Button is Dummy.");
558 gtk_dialog_run(GTK_DIALOG(dummy));
559 gtk_widget_destroy(dummy);
560 }
561
562 /* ��������������������������������������������������������������� */
563 static gchar* get_all_buffer_contents(GtkTextBuffer *buffer) {
564 GtkTextIter start, end;
565 gtk_text_buffer_get_start_iter(buffer, &start);
566 gtk_text_buffer_get_end_iter(buffer, &end);
567 return gtk_text_buffer_get_text(buffer, &start, &end, FALSE);
568 }
569
570 /* buffer ������������������������ filename ��������� */
571 static gboolean save_text_buffer(const gchar *filename, GtkTextBuffer *buffer) {
572 gchar *contents, *text;
573 gsize br, bw;
574 GError *err = NULL;
575
576 if(!filename) return FALSE;
577 contents = get_all_buffer_contents(buffer);
578 text = g_locale_from_utf8(contents, -1, &br, &bw, &err);
579 /* ��������������������������������� */
580 g_file_set_contents(filename, text, -1, NULL);
581 gtk_text_buffer_set_modified(buffer, FALSE);
582 update_modeline_label();
583 g_free(contents); g_free(text);
584 return TRUE;
585 }
586
587 /* ��������������������������������������������������������������������� */
588 static void save_file() {
589
590 /* ��������������������������������������� */
591 if(strcmp("*help*", Shiki_CURRENT_TAB_TITLE) == 0) return;
592
593 /* ������������������������������������ */
594 if(!gtk_text_buffer_get_modified(Shiki_CURRENT_TEXT_BUFFER)) return;
595
596 /* ������������������������������������������������������������������������������������������������������ */
597 if(strcmp("*scratch*", Shiki_CURRENT_FILENAME) == 0) {
598 const gchar *filename = Shiki_file_name_dialog("Save File As ...");
599 if(!filename) return;
600 if(!save_text_buffer(filename, Shiki_CURRENT_TEXT_BUFFER)) return;
601 gtk_notebook_set_tab_label_text(Shiki_EDITOR_NOTEBOOK, GTK_WIDGET(Shiki_CURRENT_TAB), filename);
602 gtk_window_set_title (GTK_WINDOW(Shiki_EDITOR_WINDOW), filename);
603 } else
604 save_text_buffer(Shiki_CURRENT_TAB_TITLE, Shiki_CURRENT_TEXT_BUFFER);
605 }
606
607 /* ��������������������������������������������������������������������������� */
608 static void save_file_as() {
609 const gchar *filename = Shiki_file_name_dialog("Save File As ...");
610
611 if(!filename) return;
612 if(!save_text_buffer(filename, Shiki_CURRENT_TEXT_BUFFER)) return;
613
614 gtk_notebook_set_tab_label_text(Shiki_EDITOR_NOTEBOOK, GTK_WIDGET(Shiki_CURRENT_TAB), filename);
615 gtk_window_set_title (GTK_WINDOW (Shiki_EDITOR_WINDOW), filename);
616 }
617
618 /* gauche ��������������������������������� */
619 static gchar *eval_cstring_by_gauche(gchar *s) {
620 gchar *msg;
621
622 ScmObj result, error;
623 /* ������������������������������ */
624 ScmObj os = Scm_MakeOutputStringPort(TRUE);
625
626 /* Scheme ��������������������������������������� */
627 /* http://alohakun.blog7.fc2.com/blog-entry-517.html */
628 Scm_Define(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*input*")), SCM_MAKE_STR_COPYING(s));
629 Scm_Define(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*error*")), SCM_FALSE);
630
631 result = Scm_EvalCString("(guard (e (else (set! *error* e) #f)) (eval (read-from-string *input*) (current-module)))", SCM_OBJ(Shiki_CURRENT_BUFFER_ENV));
632
633 error = Scm_GlobalVariableRef(SCM_MODULE(Shiki_CURRENT_BUFFER_ENV), SCM_SYMBOL(SCM_INTERN("*error*")), 0);
634
635 /* ��������������������������������������������������������� */
636 if (!SCM_FALSEP(error))
637 Scm_Write(error, os, SCM_WRITE_DISPLAY);
638 else
639 Scm_Write(result, os, SCM_WRITE_DISPLAY);
640
641 msg = Scm_GetString(SCM_STRING(Scm_GetOutputString(SCM_PORT(os))));
642 /* ������������������ */
643 Scm_ClosePort(SCM_PORT(os));
644
645 return msg;
646 }
647
648 /* ������������������������������������������������������������������ S ��������������� (������������) */
649 static void load_region_by_gauche() {
650
651 GtkTextIter start, end, p;
652 gchar *code;
653 gtk_text_buffer_get_end_iter(Shiki_CURRENT_TEXT_BUFFER, &p);
654 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, "\n\n", -1);
655
656 /* ������������������������������������������������������������ */
657 if(gtk_text_buffer_get_selection_bounds(Shiki_CURRENT_TEXT_BUFFER, &start, &end)) {
658 code = gtk_text_buffer_get_text(Shiki_CURRENT_TEXT_BUFFER, &start, &end, FALSE);
659 gtk_text_buffer_insert(Shiki_CURRENT_TEXT_BUFFER, &p, load_cstring_by_gauche(code), -1);
660 g_free(code);
661 }
662 }
663
664 // GtkTextCharPredicate
665 static gboolean is_kakko_or_kokka(gunichar ch, gpointer p) {
666 return ch == '(' || ch == ')';
667 }
668 static gboolean is_kakko(gunichar ch, gpointer p) {return ch == '(';}
669 static gboolean is_kokka(gunichar ch, gpointer p) {return ch == ')';}
670
671 /* ��������������������� '(' ��������������� ')' ������������������ (S ���) ��������������� */
672 static gboolean search_sexp(GtkTextIter *start, GtkTextIter *end) {
673
674 /* ������������������������������ */
675 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, start, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
676
677 if(gtk_text_iter_get_char(start) != '(')
678 gtk_text_iter_forward_find_char(start, is_kakko, NULL, NULL);
679
680 *end = *start;
681
682 /* ��������������������������������� S ������������������ */
683 if(!search_sexp_kokka(end)) return FALSE;
684 gtk_text_iter_forward_char(end);
685 return TRUE;
686 }
687
688 static gboolean search_sexp_kokka(GtkTextIter *end) {
689 gint nest_level = 0;
690
691 /* ������������ ')' ��������� */
692 while(1) {
693 if(!gtk_text_iter_forward_find_char(end, is_kakko_or_kokka, NULL, NULL))
694 return FALSE;
695
696 if(gtk_text_iter_get_char(end) == '(')
697 nest_level++;
698 else {
699 if(!nest_level)
700 break;
701 else
702 nest_level--;
703 }
704 }
705 return TRUE;
706 }
707
708 /* ��������������������� ')' ��������������� '(' ������������������ (S ���) ��������������� */
709 static gboolean search_last_sexp(GtkTextIter *start, GtkTextIter *end) {
710
711 /* ������������������������������ */
712 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, end, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
713
714 gtk_text_iter_backward_char(end);
715
716 if(gtk_text_iter_get_char(end) != ')')
717 gtk_text_iter_backward_find_char(end, is_kokka, NULL, NULL);
718 *start = *end;
719 gtk_text_iter_forward_char(end);
720
721 /* ��������������������������������� S ������������������ */
722 if(!search_last_sexp_kakko(start)) return FALSE;
723
724 return TRUE;
725 }
726
727 /* ')' ��������������� '(' ������������������ (S ���) ��������������� */
728 static gboolean search_last_sexp_kakko(GtkTextIter *start) {
729 gint nest_level = 0;
730 /* ��������������������������������������� ')' ��������� */
731 while(1) {
732 if(!gtk_text_iter_backward_find_char(start, is_kakko_or_kokka, NULL, NULL))
733 return FALSE;
734
735 if(gtk_text_iter_get_char(start) == ')')
736 nest_level++;
737 else {
738 if(!nest_level)
739 break;
740 else
741 nest_level--;
742 }
743 }
744 return TRUE;
745 }
746
747 /* ������������������������������������������������ */
748 static gint get_parent_nest_level_at_cursor(GtkTextBuffer *buffer) {
749 gint nest_level = 0;
750 GtkTextIter start, end;
751 gtk_text_buffer_get_start_iter(buffer, &start);
752 if(gtk_text_iter_get_char(&start) == '(') nest_level++;
753
754 /* ��������������������� (= end) ��������� */
755 gtk_text_buffer_get_iter_at_mark(buffer,&end, gtk_text_buffer_get_insert(buffer));
756
757 while(1) {
758 /* end ������ '(' ��� ')' ��������������������������������������������� */
759 if(!gtk_text_iter_forward_find_char(&start, is_kakko_or_kokka, NULL, &end))
760 return nest_level;
761
762 if(gtk_text_iter_get_char(&start) == '(')
763 nest_level++;
764 else
765 nest_level--;
766 }
767 }
768
769 /* ��������������������������������� on/off */
770 static void tabsborder_on_off(GtkButton *button, GtkNotebook *notebook) {
771 gint tval = FALSE;
772 gint bval = FALSE;
773 if(notebook->show_tabs == FALSE)
774 tval = TRUE;
775 if(notebook->show_border == FALSE)
776 bval = TRUE;
777
778 gtk_notebook_set_show_tabs(notebook, tval);
779 gtk_notebook_set_show_border(notebook, bval);
780 }
781
782 /* ������������������������ */
783 static void rotate_tab_position(GtkButton *button, GtkNotebook *notebook ) {
784 gtk_notebook_set_tab_pos(notebook, (notebook->tab_pos + 1) % 4);
785 }
786
787 /* ��������������������� */
788 static gboolean signal_key_release_handler (GtkWidget *notebook, GdkEventKey *event, gpointer contextid) {
789 static gint metakey_pressed = 0;
790 static gint controlx_pressed = 0;
791
792 if(event->keyval == GDK_parenright && event->state & GDK_SHIFT_MASK) {
793 GtkTextIter start, end;
794
795 /* ������������������������������ */
796 gtk_text_buffer_get_iter_at_mark(Shiki_CURRENT_TEXT_BUFFER, &end, gtk_text_buffer_get_insert(Shiki_CURRENT_TEXT_BUFFER));
797
798 start = end;
799 gtk_text_iter_backward_char(&start);
800
801 /* ��������������������������������� S ������������������ */
802 if(!search_last_sexp_kakko(&start)) return FALSE;
803
804 gtk_text_buffer_apply_tag_by_name(Shiki_CURRENT_TEXT_BUFFER, "parent_emphasis_background", &start, &end);
805 }
806
807 /* ������������������������������������������������������������������������������������������������ (���������������) ������������������ */
808 if(event->keyval == GDK_Return) {
809 gint indentWidth = get_parent_nest_level_at_cursor(Shiki_CURRENT_TEXT_BUFFER) * editor_indent_width;
810 gchar *indent = g_strnfill(indentWidth, ' ');
811 gtk_text_buffer_insert_at_cursor(Shiki_CURRENT_TEXT_BUFFER, indent, -1);
812 g_free(indent);
813 }
814
815 /* C-x */
816 if(event->keyval == GDK_x && event->state & GDK_CONTROL_MASK) {
817 controlx_pressed++;
818 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-x -");
819 } else if(event->state & GDK_CONTROL_MASK) {
820
821 if(controlx_pressed > 0) {
822 switch(event->keyval) {
823 case GDK_c :/* C-x C-c : ������ */
824 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-c");
825 {/* "delete-event" ��������������������������������������� �� ������������������������������������ */
826 GdkEvent ev;
827
828 ev.any.type = GDK_DELETE;
829 ev.any.window = Shiki_EDITOR_WINDOW->window;
830 ev.any.send_event = FALSE;
831 gdk_event_put (&ev);
832 }
833 break;
834
835 case GDK_f : /* C-x C-f : ������������������ */
836 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-f");
837 Shiki_open_file_dialog();
838 break;
839
840 case GDK_s : /* C-x C-s : ������������������ */
841 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-s");
842 save_file();
843 break;
844
845 case GDK_w : /* C-x C-w : ������������������������ */
846 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "C-w");
847 save_file_as();
848 break;
849 }
850 controlx_pressed = 0;
851 }
852
853 switch(event->keyval) {
854 case GDK_g :/* C-g : ��������������� */
855 metakey_pressed = 0;
856 controlx_pressed = 0;
857
858 gtk_statusbar_push(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), GPOINTER_TO_INT(contextid), "Quit");
859 break;
860 }
861
862 }
863 return FALSE;
864 }
865 static void open_online_help() {
866 GtkTextIter p;
867 Shiki_new_buffer_create(g_strdup("*help*"));
868 gtk_text_buffer_set_text(Shiki_CURRENT_TEXT_BUFFER,
869 "������������������������������������������\n"
870 "$ ./shiki [file1 file2 ....]\n\n"
871 "[���������������������������] ��������������������� (C-x C-f)\n"
872 "[������������������������������������������] ��������������������� (C-x C-s)\n"
873 "[���������������������������������������������] ��������������������������� (C-x C-w)\n"
874 "[���������������������] ��������������� gauche ������������\n"
875 "[��������������� (���������) ���������������] ��������� on/off\n"
876 "[������ (���������) ���������������] ������������������������\n"
877 "[������������������] ��������������������������� (C-t)\n"
878 "[���������������������������] ������������ (C-_)\n"
879 "[���������������������] ������������������������\n"
880 "\n"
881 "(��� : ������������������������������������������������������������������������������������������������������������������������������������������������������������������������������������������������������ Undo ������������������)\n"
882 "\n"
883 "[�� ������������] ������������������������ (C-k)\n"
884 "[A ������������] ���������������������\n"
885 "[���������������������������] Scheme ������������������������\n"
886 "[���������������������������������] ���������������������������������������\n"
887 "[��������� (?) ������������] ���������������������������������������\n"
888 "[info ������������] ���������������������������������������������������\n"
889 "\n"
890 "C-f : ��� ��������� (forward)\n"
891 "C-b : ��� ��������� (backward)\n"
892 "C-n : ��� ��������� (next line)\n"
893 "C-p : ��� ��������� (previous line)\n"
894 "\n"
895 "C-h : ���������������������\n"
896 "C-w : ���������\n"
897 "C-y : ��������� (������������)\n"
898 "\n"
899 "C-e : ��������������������� S ������������ (eval-expression)\n"
900 "C-j : ��������������������� S ������������ (eval-last-sexp)\n"
901 "(emacs/xyzzy ��� *scratch* ���������������������)\n"
902 "\n"
903 "C-M-@ : ��������������������� S ������������ (mark-sexp)\n"
904 "C-M-SPC : ��������������������� S ������������ (mark-last-sexp)\n"
905 "C-x C-c : ��������������������������� �� ���������������������������������\n"
906 "\n"
907 "������������ API\n"
908 "(clear-current-buffer) : ��������������������������������� (������������)\n"
909 , -1);
910 gtk_text_buffer_set_modified(Shiki_CURRENT_TEXT_BUFFER, FALSE);
911 /* ������������������������������ */
912 gtk_text_buffer_get_start_iter(Shiki_CURRENT_TEXT_BUFFER, &p);
913 gtk_text_buffer_place_cursor(Shiki_CURRENT_TEXT_BUFFER, &p);
914 }
915
916 /* ��������������������������������������� */
917 static void Shiki_editor_window_init(int argc, char **argv) {
918 GtkWidget *vbox, *toolbar, *modeline_bg = gtk_event_box_new();
919 GtkToolItem *icon;
920 GtkIconSize iconsize;
921 GtkTooltips *toolbar_tips = gtk_tooltips_new();
922 /* ��������������������������������������������������������������������������������� */
923 GtkToolItem *oicon, *sicon, *saicon, *eicon;
924
925 gint contextid, i;
926
927 /* ������������������������������������������������������������������������������ */
928 keywords_hash = g_hash_table_new(g_str_hash, g_str_equal);
929 i = 0;
930 while(R5RS_keywords[i] != NULL)
931 g_hash_table_insert(keywords_hash, R5RS_keywords[i++], GINT_TO_POINTER(R5RS_KEYWORD_COLOR));
932 i = 0;
933 while(R5RS_functions[i] != NULL)
934 g_hash_table_insert(keywords_hash, R5RS_functions[i++], GINT_TO_POINTER(R5RS_FUNCTION_COLOR));
935
936 /* ������������ */
937 Shiki_EDITOR_WINDOW = gtk_window_new(GTK_WINDOW_TOPLEVEL);
938 g_signal_connect(G_OBJECT(Shiki_EDITOR_WINDOW), "destroy", G_CALLBACK(gtk_main_quit), NULL);
939
940 /* ������������������������������������������������������������ */
941 Shiki_EDITOR_CLIPBOARD = gtk_clipboard_get(GDK_SELECTION_PRIMARY);
942
943 /* ������������������������������������ */
944 vbox = gtk_vbox_new(FALSE, 0);
945 /* ��������������������� */
946 toolbar = gtk_toolbar_new();
947 gtk_box_pack_start(GTK_BOX(vbox), toolbar, FALSE, FALSE, 0);
948
949 Shiki_EDITOR_NOTEBOOK = GTK_NOTEBOOK(gtk_notebook_new());
950 g_signal_connect(G_OBJECT(Shiki_EDITOR_NOTEBOOK), "switch-page", GTK_SIGNAL_FUNC(switch_tabpage_handler), NULL);
951
952 /* ������������������������������������������������ */
953 gtk_toolbar_set_style(GTK_TOOLBAR (toolbar), GTK_TOOLBAR_ICONS);
954 iconsize = gtk_toolbar_get_icon_size (GTK_TOOLBAR (toolbar));
955
956 /* ������������������ */
957
958 /* ������������������ */
959 oicon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_OPEN, iconsize), "");
960 /* ������������������������������������������������������������������������������������ */
961 g_signal_connect(G_OBJECT(oicon), "clicked", G_CALLBACK(Shiki_open_file_dialog), NULL);
962 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(oicon));
963 gtk_tool_item_set_tooltip(oicon, toolbar_tips, "���������������������������",
964 "���������������������������������������������������������������������������������������");
965
966 /* ������������������ */
967 sicon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_SAVE, iconsize), "");
968 /* ������������������������������������������������������������������������������������ */
969 g_signal_connect(G_OBJECT(sicon), "clicked", G_CALLBACK(save_file), NULL);
970 gtk_container_add (GTK_CONTAINER (toolbar), GTK_WIDGET(sicon));
971 gtk_tool_item_set_tooltip(sicon, toolbar_tips, "������������������������������",
972 "������������������������������������������������������������������������������������������������������������������������������������");
973
974 /* ��������������������������� */
975 saicon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_SAVE_AS, iconsize), "");
976 /* ������������������������������������������������������������������������������������������������������������������ */
977 g_signal_connect(G_OBJECT(saicon), "clicked", G_CALLBACK(save_file_as), NULL);
978 gtk_container_add (GTK_CONTAINER (toolbar), GTK_WIDGET(saicon));
979 gtk_tool_item_set_tooltip(saicon, toolbar_tips, "������������������������������������",
980 "");
981
982 /* ������������������ */
983 eicon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_EXECUTE, iconsize), "");
984 /* ������������������������������������������ libgauche ������������������ */
985 g_signal_connect(G_OBJECT(eicon), "clicked", G_CALLBACK(load_region_by_gauche), NULL);
986 gtk_container_add (GTK_CONTAINER (toolbar), GTK_WIDGET(eicon));
987 gtk_tool_item_set_tooltip(eicon, toolbar_tips, "��������������� S ������������������������ (load-region-lisp)",
988 "Scheme (gauche) ������������������ S ������������������������");
989
990 gtk_container_add(GTK_CONTAINER(Shiki_EDITOR_WINDOW), vbox);
991 gtk_container_add(GTK_CONTAINER(vbox), GTK_WIDGET(Shiki_EDITOR_NOTEBOOK));
992
993 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_NEW, iconsize), "");
994 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(append_default_tabpage_handler), NULL);
995 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
996 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������������������", "");
997
998 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_UNDO, iconsize), "");
999 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(undo), G_OBJECT(Shiki_EDITOR_NOTEBOOK));
1000 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1001 gtk_tool_item_set_tooltip(icon, toolbar_tips, "Undo","");
1002
1003 /* XXX : TODO */
1004 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_REDO, iconsize), "");
1005 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(dummy_handler), NULL);
1006 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1007 gtk_tool_item_set_tooltip(icon, toolbar_tips, "Redo", "");
1008
1009 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_FIND, iconsize), "");
1010 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(search_current_buffer), NULL);
1011 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1012 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������", "");
1013
1014
1015 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_FIND_AND_REPLACE, iconsize), "");
1016 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(replace_current_buffer), NULL);
1017 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1018 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������", "");
1019
1020 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_REFRESH, iconsize), "");
1021 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(dummy_handler), NULL);
1022 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1023 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������", "");
1024
1025 /* TODO ������������ */
1026
1027 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_DELETE, iconsize), "");
1028 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(clear_current_buffer_handler), NULL);
1029 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1030 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������������������",
1031 "���������������������������������������������������������������������");
1032
1033 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_CLOSE, iconsize), "");
1034 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(kill_buffer_handler), NULL);
1035 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1036 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������������",
1037 "���������������������������������������������������������������");
1038
1039 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_CONNECT, iconsize), "");
1040 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(load_scheme_file_by_gauche), NULL);
1041 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1042 gtk_tool_item_set_tooltip(icon, toolbar_tips, "Scheme ������������������������", "");
1043
1044 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_CONVERT, iconsize), "");
1045 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(load_buffer_by_gauche), G_OBJECT(Shiki_EDITOR_NOTEBOOK));
1046 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1047 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������������������", "");
1048
1049 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_SPELL_CHECK, iconsize), "");
1050 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(scheme_keyword_highlighting_current_buffer), NULL);
1051 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1052 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������������������������������������", "");
1053
1054
1055 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_DIALOG_QUESTION, iconsize), "");
1056 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(open_online_help), NULL);
1057 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1058 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������", "");
1059
1060 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_BOLD, iconsize), "");
1061 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(select_font), NULL);
1062 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1063 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������", "");
1064
1065 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_APPLY, iconsize), "");
1066 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(tabsborder_on_off), G_OBJECT(Shiki_EDITOR_NOTEBOOK));
1067 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1068 gtk_tool_item_set_tooltip(icon, toolbar_tips, "��������� on/off", "");
1069
1070 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_PREFERENCES, iconsize), "");
1071 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(rotate_tab_position), G_OBJECT(Shiki_EDITOR_NOTEBOOK));
1072 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1073 gtk_tool_item_set_tooltip(icon, toolbar_tips, "���������������������", "");
1074
1075 icon = gtk_tool_button_new(gtk_image_new_from_stock (GTK_STOCK_DIALOG_INFO, iconsize), "");
1076 g_signal_connect(G_OBJECT(icon), "clicked", G_CALLBACK(about_this_application), NULL);
1077 gtk_container_add(GTK_CONTAINER (toolbar), GTK_WIDGET(icon));
1078 gtk_tool_item_set_tooltip(icon, toolbar_tips, "������������������������������������������", "");
1079
1080 /* ������������������������ */
1081 Shiki_EDITOR_MODELINE_LABEL = gtk_label_new(NULL);
1082 gtk_container_add(GTK_CONTAINER (modeline_bg), Shiki_EDITOR_MODELINE_LABEL);
1083
1084 gdk_color_parse("black", &COLOR_BLACK);
1085 gdk_color_parse("green", &COLOR_GREEN);
1086
1087 gtk_widget_modify_fg(Shiki_EDITOR_MODELINE_LABEL, GTK_STATE_NORMAL, &COLOR_GREEN);
1088 gtk_widget_modify_bg(modeline_bg, GTK_STATE_NORMAL, &COLOR_BLACK);
1089
1090 gtk_box_pack_start(GTK_BOX(vbox), modeline_bg, TRUE, TRUE, 0);
1091
1092 /* C-x C-s ��������������������������������������������������������������������������������������� */
1093 Shiki_EDITOR_STATUSBAR = gtk_statusbar_new();
1094 gtk_box_pack_start(GTK_BOX(vbox), Shiki_EDITOR_STATUSBAR, TRUE, TRUE, 0);
1095 contextid = gtk_statusbar_get_context_id(GTK_STATUSBAR(Shiki_EDITOR_STATUSBAR), "");
1096
1097 /* ������������������������������������������������ */
1098 g_signal_connect(G_OBJECT(Shiki_EDITOR_NOTEBOOK), "key-press-event", G_CALLBACK (signal_key_press_handler), GINT_TO_POINTER(contextid));
1099 g_signal_connect(G_OBJECT(Shiki_EDITOR_NOTEBOOK), "key-release-event", G_CALLBACK (signal_key_release_handler), GINT_TO_POINTER(contextid));
1100
1101 /* ��������������������������������������������� */
1102 if(argc >= 2) {
1103 int i;
1104 for(i = 1; i < argc; i++)
1105 Shiki_create_file_buffer(argv[i]);
1106 } else /* ������������������������������������������������������������������ */
1107 open_online_help(Shiki_EDITOR_NOTEBOOK);
1108
1109 gtk_widget_grab_focus(GTK_WIDGET(Shiki_EDITOR_NOTEBOOK));
1110 gtk_widget_show_all(Shiki_EDITOR_WINDOW);
1111 }
1112
1113 int main(int argc, char *argv[]) {
1114 /* ������������������������������������ */
1115 Shiki_EDITOR_DEFAULT_LOCALE = g_locale_to_utf8(gtk_set_locale(), -1, NULL, NULL, NULL);
1116 gtk_init(&argc, &argv);
1117 GC_INIT(); Scm_Init(GAUCHE_SIGNATURE);
1118 Scm_Load("gauche-init.scm", 0);
1119 Shiki_editor_window_init(argc, argv);
1120 gtk_main();
1121 Scm_Exit(0);
1122 return 0;
1123 }

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