Changes in 4.9.173 usbnet: ipheth: prevent TX queue timeouts when device not ready usbnet: ipheth: fix potential null pointer dereference in ipheth_carrier_set media: vivid: check if the cec_adapter is valid ARM: dts: bcm283x: Fix hdmi hpd gpio pull s390: limit brk randomization to 32MB qlcnic: Avoid potential NULL pointer dereference netfilter: nft_set_rbtree: check for inactive element after flag mismatch netfilter: bridge: set skb transport_header before entering NF_INET_PRE_ROUTING sc16is7xx: missing unregister/delete driver on error in sc16is7xx_init() serial: ar933x_uart: Fix build failure with disabled console usb: gadget: net2280: Fix overrun of OUT messages usb: gadget: net2280: Fix net2280_dequeue() usb: gadget: net2272: Fix net2272_dequeue() ARM: dts: pfla02: increase phy reset duration net: ks8851: Dequeue RX packets explicitly net: ks8851: Reassert reset pin if chip ID check fails net: ks8851: Delay requesting IRQ until opened net: ks8851: Set initial carrier state to down staging: rtl8712: uninitialized memory in read_bbreg_hdl() NFS: Fix a typo in nfs_init_timeout_values() net: xilinx: fix possible object reference leak net: ibm: fix possible object reference leak net: ethernet: ti: fix possible object reference leak scsi: qla4xxx: fix a potential NULL pointer dereference usb: u132-hcd: fix resource leak ceph: fix use-after-free on symlink traversal scsi: zfcp: reduce flood of fcrscn1 trace records on multi-element RSCN libata: fix using DMA buffers on stack gpio: of: Fix of_gpiochip_add() error path kconfig/[mn]conf: handle backspace (^H) key leds: pca9532: fix a potential NULL pointer dereference vfio/type1: Limit DMA mappings per container Linux 4.9.173 Signed-off-by: Greg Kroah-Hartman <gregkh@google.com>
303 lines
7.0 KiB
C
303 lines
7.0 KiB
C
/*
|
|
* inputbox.c -- implements the input box
|
|
*
|
|
* ORIGINAL AUTHOR: Savio Lam (lam836@cs.cuhk.hk)
|
|
* MODIFIED FOR LINUX KERNEL CONFIG BY: William Roadcap (roadcap@cfw.com)
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
#include "dialog.h"
|
|
|
|
char dialog_input_result[MAX_LEN + 1];
|
|
|
|
/*
|
|
* Print the termination buttons
|
|
*/
|
|
static void print_buttons(WINDOW * dialog, int height, int width, int selected)
|
|
{
|
|
int x = width / 2 - 11;
|
|
int y = height - 2;
|
|
|
|
print_button(dialog, gettext(" Ok "), y, x, selected == 0);
|
|
print_button(dialog, gettext(" Help "), y, x + 14, selected == 1);
|
|
|
|
wmove(dialog, y, x + 1 + 14 * selected);
|
|
wrefresh(dialog);
|
|
}
|
|
|
|
/*
|
|
* Display a dialog box for inputing a string
|
|
*/
|
|
int dialog_inputbox(const char *title, const char *prompt, int height, int width,
|
|
const char *init)
|
|
{
|
|
int i, x, y, box_y, box_x, box_width;
|
|
int input_x = 0, key = 0, button = -1;
|
|
int show_x, len, pos;
|
|
char *instr = dialog_input_result;
|
|
WINDOW *dialog;
|
|
|
|
if (!init)
|
|
instr[0] = '\0';
|
|
else
|
|
strcpy(instr, init);
|
|
|
|
do_resize:
|
|
if (getmaxy(stdscr) <= (height - INPUTBOX_HEIGTH_MIN))
|
|
return -ERRDISPLAYTOOSMALL;
|
|
if (getmaxx(stdscr) <= (width - INPUTBOX_WIDTH_MIN))
|
|
return -ERRDISPLAYTOOSMALL;
|
|
|
|
/* center dialog box on screen */
|
|
x = (getmaxx(stdscr) - width) / 2;
|
|
y = (getmaxy(stdscr) - height) / 2;
|
|
|
|
draw_shadow(stdscr, y, x, height, width);
|
|
|
|
dialog = newwin(height, width, y, x);
|
|
keypad(dialog, TRUE);
|
|
|
|
draw_box(dialog, 0, 0, height, width,
|
|
dlg.dialog.atr, dlg.border.atr);
|
|
wattrset(dialog, dlg.border.atr);
|
|
mvwaddch(dialog, height - 3, 0, ACS_LTEE);
|
|
for (i = 0; i < width - 2; i++)
|
|
waddch(dialog, ACS_HLINE);
|
|
wattrset(dialog, dlg.dialog.atr);
|
|
waddch(dialog, ACS_RTEE);
|
|
|
|
print_title(dialog, title, width);
|
|
|
|
wattrset(dialog, dlg.dialog.atr);
|
|
print_autowrap(dialog, prompt, width - 2, 1, 3);
|
|
|
|
/* Draw the input field box */
|
|
box_width = width - 6;
|
|
getyx(dialog, y, x);
|
|
box_y = y + 2;
|
|
box_x = (width - box_width) / 2;
|
|
draw_box(dialog, y + 1, box_x - 1, 3, box_width + 2,
|
|
dlg.dialog.atr, dlg.border.atr);
|
|
|
|
print_buttons(dialog, height, width, 0);
|
|
|
|
/* Set up the initial value */
|
|
wmove(dialog, box_y, box_x);
|
|
wattrset(dialog, dlg.inputbox.atr);
|
|
|
|
len = strlen(instr);
|
|
pos = len;
|
|
|
|
if (len >= box_width) {
|
|
show_x = len - box_width + 1;
|
|
input_x = box_width - 1;
|
|
for (i = 0; i < box_width - 1; i++)
|
|
waddch(dialog, instr[show_x + i]);
|
|
} else {
|
|
show_x = 0;
|
|
input_x = len;
|
|
waddstr(dialog, instr);
|
|
}
|
|
|
|
wmove(dialog, box_y, box_x + input_x);
|
|
|
|
wrefresh(dialog);
|
|
|
|
while (key != KEY_ESC) {
|
|
key = wgetch(dialog);
|
|
|
|
if (button == -1) { /* Input box selected */
|
|
switch (key) {
|
|
case TAB:
|
|
case KEY_UP:
|
|
case KEY_DOWN:
|
|
break;
|
|
case KEY_BACKSPACE:
|
|
case 8: /* ^H */
|
|
case 127: /* ^? */
|
|
if (pos) {
|
|
wattrset(dialog, dlg.inputbox.atr);
|
|
if (input_x == 0) {
|
|
show_x--;
|
|
} else
|
|
input_x--;
|
|
|
|
if (pos < len) {
|
|
for (i = pos - 1; i < len; i++) {
|
|
instr[i] = instr[i+1];
|
|
}
|
|
}
|
|
|
|
pos--;
|
|
len--;
|
|
instr[len] = '\0';
|
|
wmove(dialog, box_y, box_x);
|
|
for (i = 0; i < box_width; i++) {
|
|
if (!instr[show_x + i]) {
|
|
waddch(dialog, ' ');
|
|
break;
|
|
}
|
|
waddch(dialog, instr[show_x + i]);
|
|
}
|
|
wmove(dialog, box_y, input_x + box_x);
|
|
wrefresh(dialog);
|
|
}
|
|
continue;
|
|
case KEY_LEFT:
|
|
if (pos > 0) {
|
|
if (input_x > 0) {
|
|
wmove(dialog, box_y, --input_x + box_x);
|
|
} else if (input_x == 0) {
|
|
show_x--;
|
|
wmove(dialog, box_y, box_x);
|
|
for (i = 0; i < box_width; i++) {
|
|
if (!instr[show_x + i]) {
|
|
waddch(dialog, ' ');
|
|
break;
|
|
}
|
|
waddch(dialog, instr[show_x + i]);
|
|
}
|
|
wmove(dialog, box_y, box_x);
|
|
}
|
|
pos--;
|
|
}
|
|
continue;
|
|
case KEY_RIGHT:
|
|
if (pos < len) {
|
|
if (input_x < box_width - 1) {
|
|
wmove(dialog, box_y, ++input_x + box_x);
|
|
} else if (input_x == box_width - 1) {
|
|
show_x++;
|
|
wmove(dialog, box_y, box_x);
|
|
for (i = 0; i < box_width; i++) {
|
|
if (!instr[show_x + i]) {
|
|
waddch(dialog, ' ');
|
|
break;
|
|
}
|
|
waddch(dialog, instr[show_x + i]);
|
|
}
|
|
wmove(dialog, box_y, input_x + box_x);
|
|
}
|
|
pos++;
|
|
}
|
|
continue;
|
|
default:
|
|
if (key < 0x100 && isprint(key)) {
|
|
if (len < MAX_LEN) {
|
|
wattrset(dialog, dlg.inputbox.atr);
|
|
if (pos < len) {
|
|
for (i = len; i > pos; i--)
|
|
instr[i] = instr[i-1];
|
|
instr[pos] = key;
|
|
} else {
|
|
instr[len] = key;
|
|
}
|
|
pos++;
|
|
len++;
|
|
instr[len] = '\0';
|
|
|
|
if (input_x == box_width - 1) {
|
|
show_x++;
|
|
} else {
|
|
input_x++;
|
|
}
|
|
|
|
wmove(dialog, box_y, box_x);
|
|
for (i = 0; i < box_width; i++) {
|
|
if (!instr[show_x + i]) {
|
|
waddch(dialog, ' ');
|
|
break;
|
|
}
|
|
waddch(dialog, instr[show_x + i]);
|
|
}
|
|
wmove(dialog, box_y, input_x + box_x);
|
|
wrefresh(dialog);
|
|
} else
|
|
flash(); /* Alarm user about overflow */
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
switch (key) {
|
|
case 'O':
|
|
case 'o':
|
|
delwin(dialog);
|
|
return 0;
|
|
case 'H':
|
|
case 'h':
|
|
delwin(dialog);
|
|
return 1;
|
|
case KEY_UP:
|
|
case KEY_LEFT:
|
|
switch (button) {
|
|
case -1:
|
|
button = 1; /* Indicates "Help" button is selected */
|
|
print_buttons(dialog, height, width, 1);
|
|
break;
|
|
case 0:
|
|
button = -1; /* Indicates input box is selected */
|
|
print_buttons(dialog, height, width, 0);
|
|
wmove(dialog, box_y, box_x + input_x);
|
|
wrefresh(dialog);
|
|
break;
|
|
case 1:
|
|
button = 0; /* Indicates "OK" button is selected */
|
|
print_buttons(dialog, height, width, 0);
|
|
break;
|
|
}
|
|
break;
|
|
case TAB:
|
|
case KEY_DOWN:
|
|
case KEY_RIGHT:
|
|
switch (button) {
|
|
case -1:
|
|
button = 0; /* Indicates "OK" button is selected */
|
|
print_buttons(dialog, height, width, 0);
|
|
break;
|
|
case 0:
|
|
button = 1; /* Indicates "Help" button is selected */
|
|
print_buttons(dialog, height, width, 1);
|
|
break;
|
|
case 1:
|
|
button = -1; /* Indicates input box is selected */
|
|
print_buttons(dialog, height, width, 0);
|
|
wmove(dialog, box_y, box_x + input_x);
|
|
wrefresh(dialog);
|
|
break;
|
|
}
|
|
break;
|
|
case ' ':
|
|
case '\n':
|
|
delwin(dialog);
|
|
return (button == -1 ? 0 : button);
|
|
case 'X':
|
|
case 'x':
|
|
key = KEY_ESC;
|
|
break;
|
|
case KEY_ESC:
|
|
key = on_key_esc(dialog);
|
|
break;
|
|
case KEY_RESIZE:
|
|
delwin(dialog);
|
|
on_key_resize();
|
|
goto do_resize;
|
|
}
|
|
}
|
|
|
|
delwin(dialog);
|
|
return KEY_ESC; /* ESC pressed */
|
|
}
|