Hello all!

    I didn't thought it was so easy to extend `dmenu` but it seems it
is. So will just to fill in the context of this patch: yesterday I've
asked on the mailing list if there is a `dmenu` like replacement for
xmessage. (See the archive at:
http://lists.suckless.org/dev/1011/6486.html ) And I've received a lot
of feedback for which I thank to the posters.

    Unfortunately none of the tools did what I needed... I figured out
that I actually need `dmenu` to be able to display some text message
(explaining the user what the input was for -- like a help message),
and just behave normally by letting the user select an item from the
list. (I think of it as a `dialog` replacement.) (I want to use it to
allow the user to confirm the actions of a background daemon, and the
message is longer than a single line.)

    Thus I've made the following modifications to `dmenu`, which I
hereby place in the public domain (in the hope that it is going to be
accepted by the `dmenu` team and incorporated in the tool), and for
which I want to receive comments.

    In summary, I've added two new arguments:
    * `-of message-file` which gives the path to a file that contains
the message to be displayed; (it can be a pipe, nomal file, etc.);
    * `-ol message-lines` which should limit the number of lines displayed;

    I've also refactored the `readstdin` function which I've made to
delegate work to the `readfile` function, because I reuse it inside
the newly created function `readmsgfile`. (I've been careful to keep
inside `readstdin` the `inputw` computation and thus it is not
influenced by the `readmsgfile` function.)

    Then I've added three new global variables: `msgfile`, `msglines`,
and `msgitems`, plus I've updated the `drawmenu` function (just before
the prompt) to include the drawing of the message lines.

    (I've also added a sanity check to the `main` function to check if
the number of lines isn't negative, or if `-ol` was specified but not
`-of`.)

    So any thoughts?
    Ciprian.
diff -r a79e4a9cb167 dmenu.1
--- a/dmenu.1	Sat Nov 20 09:25:08 2010 +0000
+++ b/dmenu.1	Wed Nov 24 20:59:28 2010 +0200
@@ -11,6 +11,10 @@
 .IR monitor ]
 .RB [ \-p
 .IR prompt ]
+.RB [ \-of
+.IR message-file ]
+.RB [ \-ol
+.IR message-lines ]
 .RB [ \-fn
 .IR font ]
 .RB [ \-nb
@@ -59,6 +63,12 @@
 .BI \-p " prompt"
 defines the prompt to be displayed to the left of the input field.
 .TP
+.BI \-of " message-file"
+defines the file from which a message is going to be displayed before the prompt.
+.TP
+.BI \-ol " message-lines"
+defines the maximum number of lines of message to display before the prompt.
+.TP
 .BI \-fn " font"
 defines the font or font set used.
 .TP
diff -r a79e4a9cb167 dmenu.c
--- a/dmenu.c	Sat Nov 20 09:25:08 2010 +0000
+++ b/dmenu.c	Wed Nov 24 20:59:28 2010 +0200
@@ -34,6 +34,8 @@
 static size_t nextrune(int incr);
 static void paste(void);
 static void readstdin(void);
+static void readmsgfile(void);
+static void readfile(Item **items, int maxitems, FILE *stream);
 static void run(void);
 static void setup(void);
 static void usage(void);
@@ -51,12 +53,15 @@
 static const char *normfgcolor = "#000000";
 static const char *selbgcolor  = "#0066ff";
 static const char *selfgcolor  = "#ffffff";
+static const char *msgfile = NULL;
+static int msglines = 0;
 static unsigned long normcol[ColLast];
 static unsigned long selcol[ColLast];
 static Atom utf8;
 static Bool topbar = True;
 static DC *dc;
 static Item *items = NULL;
+static Item *msgitems = NULL;
 static Item *matches, *sel;
 static Item *prev, *curr, *next;
 static Window root, win;
@@ -97,12 +102,21 @@
 			selbgcolor = argv[++i];
 		else if(!strcmp(argv[i], "-sf"))
 			selfgcolor = argv[++i];
+		else if(!strcmp(argv[i], "-ol"))
+			msglines = atoi(argv[++i]);
+		else if(!strcmp(argv[i], "-of"))
+			msgfile = argv[++i];
 		else
 			usage();
 
+	if((!msgfile && msglines!=0) || (lines < 0) || (monitor < -1) || (msglines < 0))
+		usage();
+
 	dc = initdc();
 	initfont(dc, font);
 	readstdin();
+	if(msgfile)
+		readmsgfile();
 	setup();
 	run();
 
@@ -147,6 +161,13 @@
 	dc->h = bh;
 	drawrect(dc, 0, 0, mw, mh, True, BG(dc, normcol));
 
+	if(msglines) {
+		dc->w = mw;
+		for(item = msgitems, curpos = 0; item && curpos < msglines; item = item->next, curpos++) {
+			drawtext(dc, item->text, selcol);
+			dc->y += dc->h;
+		}
+	}
 	if(prompt) {
 		dc->w = promptw;
 		drawtext(dc, prompt, selcol);
@@ -432,10 +453,34 @@
 
 void
 readstdin(void) {
+	Item *item;
+	readfile(&items, 0, stdin);
+	for(item = items; item; item = item->next)
+		inputw = MAX(inputw, textw(dc, item->text));
+}
+
+void
+readmsgfile(void) {
+	Item *item;
+	FILE *stream;
+	stream = fopen (msgfile, "r");
+	if (!stream)
+		eprintf("cannot open message file %s\n", msgfile);
+	readfile(&msgitems, msglines, stream);
+	fclose(stream);
+	
+	msglines = 0;
+	for(item = msgitems; item; item = item->next)
+		msglines++;
+}
+
+void
+readfile(Item **items, int maxlines, FILE *stream) {
 	char buf[sizeof text], *p;
 	Item *item, **end;
+	int count;
 
-	for(end = &items; fgets(buf, sizeof buf, stdin); *end = item, end = &item->next) {
+	for(end = items, count = 0; (!maxlines || count < maxlines) && fgets(buf, sizeof buf, stream); *end = item, end = &item->next, count++) {
 		if((p = strchr(buf, '\n')))
 			*p = '\0';
 		if(!(item = malloc(sizeof *item)))
@@ -443,7 +488,6 @@
 		if(!(item->text = strdup(buf)))
 			eprintf("cannot strdup %u bytes\n", strlen(buf)+1);
 		item->next = item->left = item->right = NULL;
-		inputw = MAX(inputw, textw(dc, item->text));
 	}
 }
 
@@ -492,7 +536,7 @@
 	/* menu geometry */
 	bh = dc->font.height + 2;
 	lines = MAX(lines, 0);
-	mh = (lines + 1) * bh;
+	mh = (msglines + lines + 1) * bh;
 #ifdef XINERAMA
 	if((info = XineramaQueryScreens(dc->dpy, &n))) {
 		int i, di;
@@ -536,7 +580,8 @@
 
 void
 usage(void) {
-	fputs("usage: dmenu [-b] [-i] [-l lines] [-m monitor] [-p prompt] [-fn font]\n"
+	fputs("usage: dmenu [-b] [-i] [-l lines] [-m monitor] [-p prompt]\n"
+	      "             [-of message-file] [-ol message-lines] [-fn font]\n"
 	      "             [-nb color] [-nf color] [-sb color] [-sf color] [-v]\n", stderr);
 	exit(EXIT_FAILURE);
 }

Reply via email to