summaryrefslogtreecommitdiff
path: root/source3/utils
diff options
context:
space:
mode:
authorC. Davis <cd.rattan@gmail.com>2012-08-08 17:49:06 -0700
committerMichael Adam <obnox@samba.org>2013-04-29 13:06:13 +0200
commitf723fc9619fa7cf69888680731b4329832659ff4 (patch)
tree59bbc4f0d094ac7e0808f1a4baabbefb3a203bfa /source3/utils
parente4ed97f9b36a562815fe370c2b0b60105d7f5eba (diff)
downloadsamba-f723fc9619fa7cf69888680731b4329832659ff4.tar.gz
samba-f723fc9619fa7cf69888680731b4329832659ff4.tar.bz2
samba-f723fc9619fa7cf69888680731b4329832659ff4.zip
regedit: Add an input dialog.
Reviewed-by: Andreas Schneider <asn@samba.org> Reviewed-by: Michael Adam <obnox@samba.org>
Diffstat (limited to 'source3/utils')
-rw-r--r--source3/utils/regedit_dialog.c242
-rw-r--r--source3/utils/regedit_dialog.h3
2 files changed, 179 insertions, 66 deletions
diff --git a/source3/utils/regedit_dialog.c b/source3/utils/regedit_dialog.c
index 94c3f362bd..4065bf04f1 100644
--- a/source3/utils/regedit_dialog.c
+++ b/source3/utils/regedit_dialog.c
@@ -25,6 +25,32 @@
#include <stdarg.h>
#include <form.h>
+static char *string_trim_n(TALLOC_CTX *ctx, const char *buf, size_t n)
+{
+ char *str;
+
+ str = talloc_strndup(ctx, buf, n);
+
+ if (str) {
+ trim_string(str, " ", " ");
+ }
+
+ return str;
+}
+
+static char *string_trim(TALLOC_CTX *ctx, const char *buf)
+{
+ char *str;
+
+ str = talloc_strdup(ctx, buf);
+
+ if (str) {
+ trim_string(str, " ", " ");
+ }
+
+ return str;
+}
+
static int dialog_free(struct dialog *dia)
{
if (dia->window) {
@@ -217,6 +243,34 @@ static int handle_menu_input(MENU *menu, int c)
return -1;
}
+static void handle_form_input(FORM *frm, int c)
+{
+ switch (c) {
+ case '\n':
+ form_driver(frm, REQ_NEW_LINE);
+ break;
+ case KEY_UP:
+ form_driver(frm, REQ_UP_CHAR);
+ break;
+ case KEY_DOWN:
+ form_driver(frm, REQ_DOWN_CHAR);
+ break;
+ case '\b':
+ case KEY_BACKSPACE:
+ form_driver(frm, REQ_DEL_PREV);
+ break;
+ case KEY_LEFT:
+ form_driver(frm, REQ_LEFT_CHAR);
+ break;
+ case KEY_RIGHT:
+ form_driver(frm, REQ_RIGHT_CHAR);
+ break;
+ default:
+ form_driver(frm, c);
+ break;
+ }
+}
+
static int modal_loop(struct dialog *dia)
{
int c;
@@ -238,71 +292,153 @@ static int modal_loop(struct dialog *dia)
return selection;
}
-int dialog_notice(TALLOC_CTX *ctx, enum dialog_type type,
- const char *title, WINDOW *below,
- const char *msg, ...)
+static struct dialog *dialog_msg_new(TALLOC_CTX *ctx, const char *title,
+ const char **choices, WINDOW *below,
+ int nlines, const char *msg, va_list ap)
{
- va_list ap;
struct dialog *dia;
char *str;
+ int width;
+#define MIN_WIDTH 20
+
+ str = talloc_vasprintf(ctx, msg, ap);
+ if (str == NULL) {
+ return NULL;
+ }
+
+ width = strlen(str) + 2;
+ if (width < MIN_WIDTH) {
+ width = MIN_WIDTH;
+ }
+ dia = dialog_choice_center_new(ctx, title, choices, nlines, width, below);
+ if (dia == NULL) {
+ return NULL;
+ }
+
+ waddstr(dia->sub_window, str);
+ talloc_free(str);
+
+ return dia;
+}
+
+int dialog_input(TALLOC_CTX *ctx, char **output, const char *title,
+ WINDOW *below, const char *msg, ...)
+{
+ va_list ap;
+ struct dialog *dia;
const char *choices[] = {
"Ok",
"Cancel",
NULL
};
- int width;
+ FIELD *field[2] = {0};
+ FORM *input;
+ WINDOW *input_win;
+ int y, x;
+ int rv = -1;
+ bool input_section = true;
va_start(ap, msg);
- str = talloc_vasprintf(ctx, msg, ap);
+ dia = dialog_msg_new(ctx, title, choices, below, 7, msg, ap);
va_end(ap);
- if (str == NULL) {
+ if (dia == NULL) {
return -1;
}
- width = strlen(str) + 2;
+ getmaxyx(dia->sub_window, y, x);
+ input_win = derwin(dia->sub_window, 1, x - 2, 2, 1);
+ if (input_win == NULL) {
+ goto finish;
+ }
+ field[0] = new_field(1, x - 2, 0, 0, 0, 0);
+ if (field[0] == NULL) {
+ goto finish;
+ }
+
+ field_opts_off(field[0], O_BLANK | O_AUTOSKIP | O_STATIC);
+ set_field_back(field[0], A_REVERSE);
+
+ input = new_form(field);
+ form_opts_off(input, O_NL_OVERLOAD | O_BS_OVERLOAD);
+ set_form_win(input, dia->sub_window);
+ set_form_sub(input, input_win);
+ set_current_field(input, field[0]);
+ post_form(input);
+ *output = NULL;
+
+ keypad(dia->window, true);
+ update_panels();
+ doupdate();
+
+ while (rv == -1) {
+ int c = wgetch(dia->window);
+
+ if (c == '\t' || c == KEY_BTAB) {
+ if (input_section) {
+ if (form_driver(input,REQ_VALIDATION) == E_OK) {
+ input_section = false;
+ menu_driver(dia->choices, REQ_FIRST_ITEM);
+ }
+ } else {
+ input_section = true;
+ set_current_field(input, field[0]);
+ }
+ continue;
+ }
+
+ if (input_section) {
+ handle_form_input(input, c);
+ } else {
+ rv = handle_menu_input(dia->choices, c);
+ if (rv == DIALOG_OK) {
+ const char *buf = field_buffer(field[0], 0);
+ *output = string_trim(ctx, buf);
+ }
+ }
+ }
+
+finish:
+ if (input) {
+ unpost_form(input);
+ free_form(input);
+ }
+ if (field[0]) {
+ free_field(field[0]);
+ }
+ if (input_win) {
+ delwin(input_win);
+ }
+ talloc_free(dia);
+
+ return rv;
+}
+
+int dialog_notice(TALLOC_CTX *ctx, enum dialog_type type,
+ const char *title, WINDOW *below,
+ const char *msg, ...)
+{
+ va_list ap;
+ struct dialog *dia;
+ const char *choices[] = {
+ "Ok",
+ "Cancel",
+ NULL
+ };
if (type == DIA_ALERT) {
choices[1] = NULL;
}
- dia = dialog_choice_center_new(ctx, title, choices, 5, width, below);
+
+ va_start(ap, msg);
+ dia = dialog_msg_new(ctx, title, choices, below, 5, msg, ap);
+ va_end(ap);
if (dia == NULL) {
return -1;
}
- waddstr(dia->sub_window, str);
- talloc_free(str);
-
return modal_loop(dia);
}
-static void handle_form_input(FORM *frm, int c)
-{
- switch (c) {
- case '\n':
- form_driver(frm, REQ_NEW_LINE);
- break;
- case KEY_UP:
- form_driver(frm, REQ_UP_CHAR);
- break;
- case KEY_DOWN:
- form_driver(frm, REQ_DOWN_CHAR);
- break;
- case '\b':
- case KEY_BACKSPACE:
- form_driver(frm, REQ_DEL_PREV);
- break;
- case KEY_LEFT:
- form_driver(frm, REQ_LEFT_CHAR);
- break;
- case KEY_RIGHT:
- form_driver(frm, REQ_RIGHT_CHAR);
- break;
- default:
- form_driver(frm, c);
- break;
- }
-}
-
#define MAX_FIELDS 8
enum input_section {
@@ -390,32 +526,6 @@ static WERROR fill_value_buffer(struct edit_dialog *edit,
return WERR_OK;
}
-static char *string_trim_n(TALLOC_CTX *ctx, const char *buf, size_t n)
-{
- char *str;
-
- str = talloc_strndup(ctx, buf, n);
-
- if (str) {
- trim_string(str, " ", " ");
- }
-
- return str;
-}
-
-static char *string_trim(TALLOC_CTX *ctx, const char *buf)
-{
- char *str;
-
- str = talloc_strdup(ctx, buf);
-
- if (str) {
- trim_string(str, " ", " ");
- }
-
- return str;
-}
-
static bool value_exists(TALLOC_CTX *ctx, const struct registry_key *key,
const char *name)
{
diff --git a/source3/utils/regedit_dialog.h b/source3/utils/regedit_dialog.h
index 132e48b3e5..970cb7c42a 100644
--- a/source3/utils/regedit_dialog.h
+++ b/source3/utils/regedit_dialog.h
@@ -61,6 +61,9 @@ int dialog_notice(TALLOC_CTX *ctx, enum dialog_type type,
const char *title, WINDOW *below,
const char *msg, ...);
+int dialog_input(TALLOC_CTX *ctx, char **output, const char *title,
+ WINDOW *below, const char *msg, ...);
+
struct registry_key;
struct value_item;