Use pointer to memos instead of accessing the array.
[kugel-rb.git] / apps / player / keyboard.c
blobe657f366971c35b262581745ba8a350a7ff0b070
1 /***************************************************************************
2 * __________ __ ___.
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
7 * \/ \/ \/ \/ \/
8 * $Id$
10 * Copyright (C) 2002 by Björn Stenberg
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation; either version 2
15 * of the License, or (at your option) any later version.
17 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
18 * KIND, either express or implied.
20 ****************************************************************************/
21 #include "lcd.h"
22 #include "button.h"
23 #include "kernel.h"
24 #include "system.h"
25 #include "version.h"
26 #include "sprintf.h"
27 #include <string.h>
28 #include "settings.h"
29 #include "statusbar.h"
30 #include "talk.h"
31 #include "misc.h"
32 #include "rbunicode.h"
33 #include "lang.h"
34 #include "keyboard.h"
35 #include "splash.h"
37 #define KBD_BUF_SIZE 64
38 #define KEYBOARD_PAGES 3
40 static unsigned short *kbd_setupkeys(int page, int* len)
42 static unsigned short kbdline[KBD_BUF_SIZE];
43 const unsigned char *p;
44 int i = 0;
46 switch (page)
48 case 0: /* Capitals */
49 p = "ABCDEFGHIJKLMNOPQRSTUVWXYZÀÁÂÃÄÅ"
50 "ÆÇÈÉÊËÌÍÎÏÐÑÒÓÔÕÖØÙÚÛÜÝ";
51 break;
53 case 1: /* Small */
54 p = "abcdefghijklmnopqrstuvwxyzßàáâãä"
55 "åçèéêëìíîïñòóôõöøùúûüýÿ";
56 break;
58 default: /* Others */
59 p = " !\"#$%&'()*+,-./0123456789:;<=>?@[]_{}~";
60 break;
63 while (*p)
64 p = utf8decode(p, &kbdline[i++]);
66 *len = i;
68 return kbdline;
71 /* Delimiters for highlighting the character selected for insertion */
72 #define KEYBOARD_INSERT_LEFT 0xe110
73 #define KEYBOARD_INSERT_RIGHT 0xe10f
75 #define KEYBOARD_CURSOR 0x7f
76 #define KEYBOARD_ARROW 0xe10c
78 /* helper function to spell a char if voice UI is enabled */
79 static void kbd_spellchar(unsigned short c)
81 if (global_settings.talk_menu) /* voice UI? */
83 unsigned char tmp[5];
84 /* store char to pass to talk_spell */
85 unsigned char* utf8 = utf8encode(c, tmp);
86 *utf8 = 0;
88 if(c == ' ')
89 talk_id(VOICE_BLANK, false);
90 else
91 talk_spell(tmp, false);
95 static void say_edit(void)
97 if (global_settings.talk_menu)
98 talk_id(VOICE_EDIT, false);
101 int kbd_input(char* text, int buflen)
103 bool done = false;
104 bool redraw = true;
105 bool line_edit = false;
106 int page = 0, x = 0;
107 int linelen;
109 int len, len_utf8, i, j;
110 int editpos, curpos, leftpos;
111 unsigned short *line = kbd_setupkeys(page, &linelen);
112 unsigned char temptext[36];
113 unsigned char *utf8;
115 int button, lastbutton = 0;
116 int ret = 0; /* assume success */
118 editpos = utf8length(text);
120 if (global_settings.talk_menu) /* voice UI? */
121 talk_spell(text, true); /* spell initial text */
123 while (!done)
125 len = strlen(text);
126 len_utf8 = utf8length(text);
128 if (redraw)
130 if (line_edit)
132 lcd_putc(0, 0, ' ');
133 lcd_putc(0, 1, KEYBOARD_ARROW);
135 else
137 lcd_putc(0, 0, KEYBOARD_ARROW);
138 lcd_putc(0, 1, ' ');
141 lcd_putc(1, 0, KEYBOARD_INSERT_LEFT);
142 lcd_putc(2, 0, line[x]);
143 lcd_putc(3, 0, KEYBOARD_INSERT_RIGHT);
144 for (i = 1; i < 8; i++)
146 lcd_putc(i + 3, 0, line[(x+i)%linelen]);
149 /* write out the text */
150 curpos = MIN(MIN(editpos, 10 - MIN(len_utf8 - editpos, 3)), 9);
151 leftpos = editpos - curpos;
152 if (!leftpos) {
153 utf8 = text + utf8seek(text, leftpos);
154 i = 0;
155 j = 0;
156 } else {
157 temptext[0] = '<';
158 i = 1;
159 j = 1;
160 utf8 = text + utf8seek(text, leftpos+1);
162 while (*utf8 && i < 10) {
163 temptext[j++] = *utf8++;
164 if ((*utf8 & MASK) != COMP)
165 i++;
167 temptext[j] = 0;
170 if (len_utf8 - leftpos > 10) {
171 utf8 = temptext + utf8seek(temptext, 9);
172 *utf8++ = '>';
173 *utf8 = 0;
176 lcd_remove_cursor();
177 lcd_puts(1, 1, temptext);
178 lcd_put_cursor(curpos + 1, 1, KEYBOARD_CURSOR);
180 gui_syncstatusbar_draw(&statusbars, true);
183 /* The default action is to redraw */
184 redraw = true;
186 button = button_get_w_tmo(HZ/2);
187 switch (button)
189 case BUTTON_STOP: /* abort */
190 ret = -1; done = true;
191 break;
193 case BUTTON_MENU: /* page flip */
194 if (++page == KEYBOARD_PAGES)
195 page = 0;
196 line = kbd_setupkeys(page, &linelen);
197 if (x > linelen - 1)
198 x = linelen - 1;
199 kbd_spellchar(line[x]);
200 break;
202 case BUTTON_ON: /* toggle mode */
203 line_edit = !line_edit;
204 if (line_edit)
205 say_edit();
206 else
207 kbd_spellchar(line[x]);
208 break;
210 case BUTTON_RIGHT:
211 case BUTTON_RIGHT | BUTTON_REPEAT:
212 if (line_edit)
214 if (editpos < len_utf8)
216 editpos++;
217 int c = utf8seek(text, editpos);
218 kbd_spellchar(text[c]);
221 else
223 if (++x >= linelen)
224 x = 0;
225 kbd_spellchar(line[x]);
227 break;
229 case BUTTON_LEFT:
230 case BUTTON_LEFT | BUTTON_REPEAT:
231 if (line_edit)
233 if (editpos)
235 editpos--;
236 int c = utf8seek(text, editpos);
237 kbd_spellchar(text[c]);
240 else
242 if (--x < 0)
243 x = linelen - 1;
244 kbd_spellchar(line[x]);
246 break;
248 case BUTTON_PLAY | BUTTON_REPEAT:
249 /* accepts what was entered and continues */
250 ret = 0; done = true;
251 break;
253 case BUTTON_PLAY | BUTTON_REL:
254 if (lastbutton != BUTTON_PLAY)
255 break;
256 if (line_edit) /* backspace in line_edit */
258 if (editpos > 0)
260 utf8 = text + utf8seek(text, editpos);
261 i = 0;
262 do {
263 i++;
264 utf8--;
265 } while ((*utf8 & MASK) == COMP);
266 while (utf8[i]) {
267 *utf8 = utf8[i];
268 utf8++;
270 *utf8 = 0;
271 editpos--;
274 else /* inserts the selected char */
276 utf8 = utf8encode(line[x], temptext);
277 *utf8 = 0;
278 j = strlen(temptext);
279 if (len + j < buflen)
281 int k = len_utf8;
282 for (i = len+j; k >= editpos; i--) {
283 text[i] = text[i-j];
284 if ((text[i] & MASK) != COMP)
285 k--;
287 while (j--)
288 text[i--] = temptext[j];
289 editpos++;
292 if (global_settings.talk_menu) /* voice UI? */
293 talk_spell(text, false); /* speak revised text */
294 break;
296 case BUTTON_NONE:
297 gui_syncstatusbar_draw(&statusbars, false);
298 redraw = false;
299 break;
301 default:
302 default_event_handler(button);
303 break;
305 if (button != BUTTON_NONE)
306 lastbutton = button;
309 if (ret < 0)
310 splash(HZ/2, ID2P(LANG_CANCEL));
311 return ret;