Remove ked ctrl-B hot key
[petitboot] / ui / ncurses / nc-ked.c
1 /*
2  *  Copyright (C) 2009 Sony Computer Entertainment Inc.
3  *  Copyright 2009 Sony Corp.
4  *
5  *  This program is free software; you can redistribute it and/or modify
6  *  it under the terms of the GNU General Public License as published by
7  *  the Free Software Foundation; version 2 of the License.
8  *
9  *  This program is distributed in the hope that it will be useful,
10  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
11  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  *  GNU General Public License for more details.
13  *
14  *  You should have received a copy of the GNU General Public License
15  *  along with this program; if not, write to the Free Software
16  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
17  */
18
19 #define _GNU_SOURCE
20
21 #include <assert.h>
22 #include <string.h>
23
24 #include "log/log.h"
25 #include "talloc/talloc.h"
26 #include "nc-ked.h"
27
28 static struct ked *ked_from_scr(struct nc_scr *scr)
29 {
30         struct ked *ked;
31
32         assert(scr->sig == pb_ked_sig);
33         ked = (struct ked *)((char *)scr - (size_t)&((struct ked *)0)->scr);
34         assert(ked->scr.sig == pb_ked_sig);
35         return ked;
36 }
37
38 static struct ked *ked_from_arg(void *arg)
39 {
40         struct ked *ked = arg;
41
42         assert(ked->scr.sig == pb_ked_sig);
43         return ked;
44 }
45
46 /**
47  * ked_move_cursor - Move the cursor, setting correct attributes.
48  * @req: An ncurses request or char to send to form_driver().
49  */
50
51 static int ked_move_cursor(struct ked *ked, int req)
52 {
53         int result;
54
55         wchgat(ked->scr.sub_ncw, 1, ked_attr_field_selected, 0, 0);
56         result = form_driver(ked->ncf, req);
57         wchgat(ked->scr.sub_ncw, 1, ked->attr_cursor, 0, 0);
58         wrefresh(ked->scr.main_ncw);
59         return result;
60 }
61
62 /**
63  * ked_insert_mode_set - Set the insert mode.
64  */
65
66 static void ked_insert_mode_set(struct ked *ked, int req)
67 {
68         switch (req) {
69         case REQ_INS_MODE:
70                 ked->attr_cursor = ked_attr_cursor_ins;
71                 break;
72         case REQ_OVL_MODE:
73                 ked->attr_cursor = ked_attr_cursor_ovl;
74                 break;
75         default:
76                 assert(0 && "bad req");
77                 break;
78         }
79         ked_move_cursor(ked, req);
80 }
81
82 /**
83  * ked_insert_mode_tog - Toggle the insert mode.
84  */
85
86 static void ked_insert_mode_tog(struct ked *ked)
87 {
88         if (ked->attr_cursor == ked_attr_cursor_ins)
89                 ked_insert_mode_set(ked, REQ_OVL_MODE);
90         else
91                 ked_insert_mode_set(ked, REQ_INS_MODE);
92 }
93
94 /**
95  * ked_move_field - Move selected field, setting correct attributes.
96  * @req: An ncurses request to send to form_driver().
97  */
98
99 static int ked_move_field(struct ked *ked, int req)
100 {
101         int result;
102
103         set_field_back(current_field(ked->ncf), ked_attr_field_normal);
104         result = form_driver(ked->ncf, req);
105         set_field_back(current_field(ked->ncf), ked_attr_field_selected);
106         ked_move_cursor(ked, REQ_END_FIELD);
107         return result;
108 }
109
110 static int ked_post(struct nc_scr *scr)
111 {
112         struct ked *ked = ked_from_scr(scr);
113
114         post_form(ked->ncf);
115
116         nc_scr_frame_draw(scr);
117         ked_move_field(ked, REQ_FIRST_FIELD);
118         ked_move_field(ked, REQ_END_FIELD);
119         ked_insert_mode_set(ked, REQ_INS_MODE);
120
121         redrawwin(ked->scr.main_ncw);
122         wrefresh(ked->scr.main_ncw);
123
124         return 0;
125 }
126
127 static int ked_unpost(struct nc_scr *scr)
128 {
129         return unpost_form(ked_from_scr(scr)->ncf);
130 }
131
132 static void ked_resize(struct nc_scr *scr)
133 {
134         /* FIXME: forms can't be resized, need to recreate here */
135         ked_unpost(scr);
136         ked_post(scr);
137 }
138
139 /**
140  * ked_chomp - Eat leading and trailing WS.
141  */
142
143 static char *ked_chomp(char *s)
144 {
145         char *start;
146         char *end;
147         char *const s_end = s + strlen(s);
148
149         for (; s < s_end; s++)
150                 if (*s != ' ' && *s != '\t')
151                         break;
152         start = s;
153
154         for (++s; s < s_end; s++)
155                 if (*s != ' ' && *s != '\t')
156                         end = s;
157         *(end + 1) = 0;
158         return start;
159 }
160
161 static struct pb_kexec_data *ked_prepare_data(struct ked *ked)
162 {
163         struct pb_kexec_data *kd;
164         char *s;
165
166         kd = talloc(ked, struct pb_kexec_data);
167
168         if (!kd)
169                 return NULL;
170
171         s = ked_chomp(field_buffer(ked->fields[0], 0));
172         kd->image = *s ? talloc_strdup(kd, s) : NULL;
173
174         s = ked_chomp(field_buffer(ked->fields[1], 0));
175         kd->initrd = *s ? talloc_strdup(kd, s) : NULL;
176
177         s = ked_chomp(field_buffer(ked->fields[2], 0));
178         kd->args = *s ? talloc_strdup(kd, s) : NULL;
179
180         return kd;
181 }
182
183 /**
184  * ked_process_key - Process a user keystroke.
185  *
186  * Called from the cui via the scr:process_key method.
187  */
188
189 static void ked_process_key(struct nc_scr *scr)
190 {
191         struct ked *ked = ked_from_scr(scr);
192         struct pb_kexec_data *kd;
193
194         while (1) {
195                 int c = getch();
196
197                 if (c == ERR)
198                         return;
199
200                 /* DBGS("%d (%o)\n", c, c); */
201
202                 switch (c) {
203                 default:
204                         ked_move_cursor(ked, c);
205                         break;
206
207                 /* hot keys */
208                 case 27: /* ESC */
209                         ked->on_exit(ked, ked_cancel, NULL);
210                         nc_flush_keys();
211                         return;
212                 case '\n':
213                 case '\r':
214                         form_driver(ked->ncf, REQ_VALIDATION);
215                         kd = ked_prepare_data(ked);
216                         ked->on_exit(ked, ked_update, kd);
217                         nc_flush_keys();
218                         return;
219
220                 /* insert mode */
221                 case KEY_IC:
222                         ked_insert_mode_tog(ked);
223                         break;
224
225                 /* form nav */
226                 case KEY_PPAGE:
227                         ked_move_field(ked, REQ_FIRST_FIELD);
228                         break;
229                 case KEY_NPAGE:
230                         ked_move_field(ked, REQ_LAST_FIELD);
231                         break;
232                 case KEY_DOWN:
233                         ked_move_field(ked, REQ_NEXT_FIELD);
234                         break;
235                 case KEY_UP:
236                         ked_move_field(ked, REQ_PREV_FIELD);
237                         break;
238
239                 /* field nav */
240                 case KEY_HOME:
241                         ked_move_cursor(ked, REQ_BEG_FIELD);
242                         break;
243                 case KEY_END:
244                         ked_move_cursor(ked, REQ_END_FIELD);
245                         break;
246                 case KEY_LEFT:
247                         ked_move_cursor(ked, REQ_LEFT_CHAR);
248                         break;
249                 case KEY_RIGHT:
250                         ked_move_cursor(ked, REQ_RIGHT_CHAR);
251                         break;
252                 case KEY_BACKSPACE:
253                         if (ked_move_cursor(ked, REQ_LEFT_CHAR) == E_OK)
254                                 ked_move_cursor(ked, REQ_DEL_CHAR);
255                         break;
256                 case KEY_DC:
257                         ked_move_cursor(ked, REQ_DEL_CHAR);
258                         break;
259                 }
260         }
261 }
262
263 /**
264  * ked_destructor - The talloc destructor for a ked.
265  */
266
267 static int ked_destructor(void *arg)
268 {
269         struct ked *ked = ked_from_arg(arg);
270         FIELD **f;
271
272         for (f = ked->fields; *f; f++)
273                 free_field(*f);
274
275         free_form(ked->ncf);
276         ked->scr.sig = pb_removed_sig;
277
278         return 0;
279 }
280
281 static FIELD *ked_setup_field(unsigned int y, unsigned int x, char *str)
282 {
283         FIELD *f;
284
285         f = new_field(1, COLS - 1 - x, y, x, 0, 0);
286         field_opts_off(f, O_STATIC | O_WRAP);
287         set_max_field(f, 256);
288         set_field_buffer(f, 0, str);
289         set_field_status(f, 0);
290         return f;
291 }
292
293 static FIELD *ked_setup_label(unsigned int y, unsigned int x, char *str)
294 {
295         FIELD *f;
296
297         f = new_field(1, strlen(str), y, x, 0, 0);
298         field_opts_off(f, O_ACTIVE);
299         set_field_buffer(f, 0, str);
300         return f;
301 }
302
303 struct ked *ked_init(void *ui_ctx, const struct pb_kexec_data *kd,
304         void (*on_exit)(struct ked *, enum ked_result, struct pb_kexec_data *))
305 {
306         struct ked *ked;
307
308         pb_log("%s: image:  '%s'\n", __func__, kd->image);
309         pb_log("%s: initrd: '%s'\n", __func__, kd->initrd);
310         pb_log("%s: args:   '%s'\n", __func__, kd->args);
311
312         assert(on_exit);
313
314         ked = talloc_zero(ui_ctx, struct ked);
315
316         if (!ked)
317                 return NULL;
318
319         talloc_set_destructor(ked, ked_destructor);
320
321         nc_scr_init(&ked->scr, pb_ked_sig, 0, ui_ctx, ked_process_key,
322                 ked_post, ked_unpost, ked_resize);
323
324         ked->scr.frame.title = talloc_strdup(ked, "Petitboot Option Editor");
325         ked->scr.frame.help = talloc_strdup(ked,
326                 "ESC=cancel, Enter=accept");
327
328         ked->on_exit = on_exit;
329
330         ked->fields = talloc_array(ked, FIELD *, 7);
331
332         ked->fields[0] = ked_setup_field(0, 9, kd->image);
333         ked->fields[1] = ked_setup_field(1, 9, kd->initrd);
334         ked->fields[2] = ked_setup_field(2, 9, kd->args);
335         ked->fields[3] = ked_setup_label(0, 1, "image:");
336         ked->fields[4] = ked_setup_label(1, 1, "initrd:");
337         ked->fields[5] = ked_setup_label(2, 1, "args:");
338         ked->fields[6] = NULL;
339
340         ked->ncf = new_form(ked->fields);
341
342         set_form_win(ked->ncf, ked->scr.main_ncw);
343         set_form_sub(ked->ncf, ked->scr.sub_ncw);
344
345         return ked;
346 }