/* $Id: widget_text.c,v 1.5 2004/01/15 04:29:45 reinelt Exp $ * * simple text widget handling * * Copyright 2003,2004 Michael Reinelt * Copyright 2004 The LCD4Linux Team * * 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, 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. * * * $Log: widget_text.c,v $ * Revision 1.5 2004/01/15 04:29:45 reinelt * moved lcd4linux.conf.sample to *.old * lcd4linux.conf.sample with new layout * new plugins 'loadavg' and 'meminfo' * text widget have pre- and postfix * * Revision 1.4 2004/01/14 11:33:00 reinelt * new plugin 'uname' which does what it's called * text widget nearly finished * first results displayed on MatrixOrbital * * Revision 1.3 2004/01/13 08:18:20 reinelt * timer queues added * liblcd4linux deactivated turing transformation to new layout * * Revision 1.2 2004/01/11 18:26:02 reinelt * further widget and layout processing * * Revision 1.1 2004/01/10 20:22:33 reinelt * added new function 'cfg_list()' (not finished yet) * added layout.c (will replace processor.c someday) * added widget_text.c (will be the first and most important widget) * modified lcd4linux.c so that old-style configs should work, too * */ /* * exported functions: * * WIDGET_CLASS Widget_Text * a simple text widget which * must be supported by all displays * */ #include #include #include #include #include "debug.h" #include "cfg.h" #include "evaluator.h" #include "timer.h" #include "widget.h" #include "widget_text.h" void widget_text_scroll (void *Self) { WIDGET *W = (WIDGET*)Self; WIDGET_TEXT *T = W->data; int num, len, pad; char *src, *dst; src=T->value; dst=T->buffer; num=0; len=strlen(T->value); switch (T->align) { case ALIGN_LEFT: pad=0; break; case ALIGN_CENTER: pad=(T->width - len)/2; if (pad<0) pad=0; break; case ALIGN_RIGHT: pad=T->width - len; if (pad<0) pad=0; break; case ALIGN_MARQUEE: pad=T->width - T->scroll; T->scroll++; if (T->scroll >= T->width+len) T->scroll=0; break; default: // not reached pad=0; } // pad blanks on the beginning while (pad>0) { *(dst++)=' '; num++; pad--; } // overread src (marquee) while (pad<0) { src++; pad++; } // copy content while (num < T->width) { if (*src=='\0') break; *(dst++)=*(src++); num++; } // pad blanks on the end while (num < T->width) { *(dst++)=' '; num++; } *dst='\0'; // finally, draw it! if (W->class->draw) W->class->draw(W); } void widget_text_update (void *Self) { WIDGET *W = (WIDGET*)Self; WIDGET_TEXT *T = W->data; RESULT result = {0, 0.0, NULL}; char *prefix, *postfix, *value; // evaluate prefix if (T->prefix!=NULL && strlen(T->prefix)>0) { Eval(T->prefix, &result); prefix=strdup(R2S(&result)); DelResult (&result); } else { prefix=strdup(""); } // evaluate postfix if (T->postfix!=NULL && *(T->postfix)!='\0') { debug ("Eval_postfix(%s)", T->postfix); Eval(T->postfix, &result); postfix=strdup(R2S(&result)); DelResult (&result); } else { postfix=strdup(""); } // evaluate expression Eval(T->expression, &result); // string or number? if (T->precision==0xC0DE) { value=strdup(R2S(&result)); } else { double number=R2N(&result); int width=T->width; int precision=T->precision; // print zero bytes so we can specify NULL as target // and get the length of the resulting string int size=snprintf (NULL, 0, "%.*f", precision, number); // number does not fit into field width: try to reduce precision if (size>width && precision>0) { int delta=size-width; if (delta>precision) delta=precision; precision-=delta; size-=delta; // zero precision: omit decimal point, too if (precision==0) size--; } // number still doesn't fit: display '*****' if (size>width) { debug ("overflow"); value=malloc(width+1); memset (value, '*', width); *(value+width)='\0'; } else { value=malloc(size+1); snprintf (value, size+1, "%.*f", precision, number); } } DelResult (&result); // has value changed? if (T->value == NULL || strcmp(T->value, value)!=0) { // free old value if (T->value) free (T->value); T->value=value; // reset marquee counter T->scroll=0; // if there's a marquee scroller active, it has its own // update callback timer, so we do nothing here; otherwise // we simply call this scroll callback directly if (T->align!=ALIGN_MARQUEE) { widget_text_scroll (Self); } } else { // value is the same, so free our buffer free (value); } } int widget_text_init (WIDGET *Self) { char *section; char *c; WIDGET_TEXT *T; // prepare config section // strlen("Widget:")=7 section=malloc(strlen(Self->name)+8); strcpy(section, "Widget:"); strcat(section, Self->name); T=malloc(sizeof(WIDGET_TEXT)); memset (T, 0, sizeof(WIDGET_TEXT)); // get raw pre- and postfix (we evaluate it ourselves) T->prefix = cfg_get_raw (section, "prefix", NULL); T->postfix = cfg_get_raw (section, "prefix", NULL); // get raw expression (we evaluate it ourselves) T->expression = cfg_get_raw (section, "expression", "''"); // field width, default 10 cfg_number (section, "width", 10, 0, 99999, &(T->width)); // precision: number of digits after the decimal point (default: none) // Note: this is the *maximum* precision on small values, // for larger values the precision may be reduced to fit into the field width. // The default value 0xC0DE is used to distinguish between numbers and strings: // if no precision is given, the result is always treated as a string. If a // precision is specified, the result is treated as a number. cfg_number (section, "precision", 0xC0DE, 0, 80, &(T->precision)); // field alignment: Left (default), Center, Right or Marquee c = cfg_get (section, "align", "L"); switch (toupper(*c)) { case 'L': T->align=ALIGN_LEFT; break; case 'C': T->align=ALIGN_CENTER; break; case 'R': T->align=ALIGN_RIGHT; break; case 'M': T->align=ALIGN_MARQUEE; break; default: error ("widget %s has unknown alignment '%s', using 'Left'", section, c); T->align=ALIGN_LEFT; } free (c); // update interval (msec), default 1 sec cfg_number (section, "update", 1000, 10, 99999, &(T->update)); // marquee scroller speed: interval (msec), default 500msec if (T->align==ALIGN_MARQUEE) { cfg_number (section, "speed", 500, 10, 99999, &(T->speed)); } // buffer T->buffer=malloc(T->width+1); free (section); Self->data=T; timer_add (widget_text_update, Self, T->update, 0); // a marquee scroller has its own timer and callback if (T->align==ALIGN_MARQUEE) { timer_add (widget_text_scroll, Self, T->speed, 0); } return 0; } int widget_text_quit (WIDGET *Self) { if (Self->data) { free (Self->data); Self->data=NULL; } return 0; } WIDGET_CLASS Widget_Text = { name: "text", init: widget_text_init, draw: NULL, quit: widget_text_quit, };