2011-12-26 21:35:57 +00:00
|
|
|
/*
|
|
|
|
htop - RichString.c
|
|
|
|
(C) 2004,2011 Hisham H. Muhammad
|
|
|
|
Released under the GNU GPL, see the COPYING file
|
|
|
|
in the source distribution for its full text.
|
|
|
|
*/
|
2006-03-04 18:16:49 +00:00
|
|
|
|
|
|
|
#include "RichString.h"
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2011-12-26 21:35:57 +00:00
|
|
|
|
|
|
|
#define RICHSTRING_MAXLEN 300
|
|
|
|
|
|
|
|
/*{
|
|
|
|
#include "config.h"
|
2009-03-11 13:51:50 +00:00
|
|
|
#include <ctype.h>
|
2006-03-04 18:16:49 +00:00
|
|
|
|
|
|
|
#include <assert.h>
|
2011-11-03 22:01:54 +00:00
|
|
|
#ifdef HAVE_NCURSESW_CURSES_H
|
2011-09-08 01:45:16 +00:00
|
|
|
#include <ncursesw/curses.h>
|
|
|
|
#elif HAVE_NCURSES_NCURSES_H
|
|
|
|
#include <ncurses/ncurses.h>
|
2013-02-26 17:10:11 +00:00
|
|
|
#elif HAVE_NCURSES_CURSES_H
|
|
|
|
#include <ncurses/curses.h>
|
2011-11-03 22:01:54 +00:00
|
|
|
#elif HAVE_NCURSES_H
|
|
|
|
#include <ncurses.h>
|
|
|
|
#elif HAVE_CURSES_H
|
|
|
|
#include <curses.h>
|
2008-03-09 02:33:23 +00:00
|
|
|
#endif
|
2006-03-04 18:16:49 +00:00
|
|
|
|
2010-11-22 12:40:20 +00:00
|
|
|
#define RichString_size(this) ((this)->chlen)
|
|
|
|
#define RichString_sizeVal(this) ((this).chlen)
|
|
|
|
|
|
|
|
#define RichString_begin(this) RichString (this); (this).chlen = 0; (this).chptr = (this).chstr;
|
|
|
|
#define RichString_beginAllocated(this) (this).chlen = 0; (this).chptr = (this).chstr;
|
|
|
|
#define RichString_end(this) RichString_prune(&(this));
|
2006-07-12 01:16:03 +00:00
|
|
|
|
2008-03-09 02:33:23 +00:00
|
|
|
#ifdef HAVE_LIBNCURSESW
|
2010-11-22 12:40:20 +00:00
|
|
|
#define RichString_printVal(this, y, x) mvadd_wchstr(y, x, (this).chptr)
|
|
|
|
#define RichString_printoffnVal(this, y, x, off, n) mvadd_wchnstr(y, x, (this).chptr + off, n)
|
|
|
|
#define RichString_getCharVal(this, i) ((this).chptr[i].chars[0] & 255)
|
|
|
|
#define RichString_setChar(this, at, ch) do{ (this)->chptr[(at)].chars[0] = ch; } while(0)
|
|
|
|
#define CharType cchar_t
|
2008-03-09 02:33:23 +00:00
|
|
|
#else
|
2010-11-22 12:40:20 +00:00
|
|
|
#define RichString_printVal(this, y, x) mvaddchstr(y, x, (this).chptr)
|
|
|
|
#define RichString_printoffnVal(this, y, x, off, n) mvaddchnstr(y, x, (this).chptr + off, n)
|
|
|
|
#define RichString_getCharVal(this, i) ((this).chptr[i])
|
|
|
|
#define RichString_setChar(this, at, ch) do{ (this)->chptr[(at)] = ch; } while(0)
|
|
|
|
#define CharType chtype
|
2008-03-09 02:33:23 +00:00
|
|
|
#endif
|
|
|
|
|
2006-03-04 18:16:49 +00:00
|
|
|
typedef struct RichString_ {
|
2010-11-22 12:40:20 +00:00
|
|
|
int chlen;
|
|
|
|
CharType chstr[RICHSTRING_MAXLEN+1];
|
|
|
|
CharType* chptr;
|
2006-03-04 18:16:49 +00:00
|
|
|
} RichString;
|
|
|
|
|
|
|
|
}*/
|
|
|
|
|
|
|
|
#ifndef MIN
|
|
|
|
#define MIN(a,b) ((a)<(b)?(a):(b))
|
|
|
|
#endif
|
|
|
|
|
2010-11-22 12:40:20 +00:00
|
|
|
#define charBytes(n) (sizeof(CharType) * (n))
|
|
|
|
|
|
|
|
static inline void RichString_setLen(RichString* this, int len) {
|
|
|
|
if (this->chlen <= RICHSTRING_MAXLEN) {
|
|
|
|
if (len > RICHSTRING_MAXLEN) {
|
|
|
|
this->chptr = malloc(charBytes(len+1));
|
|
|
|
memcpy(this->chptr, this->chstr, charBytes(this->chlen+1));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (len <= RICHSTRING_MAXLEN) {
|
|
|
|
memcpy(this->chstr, this->chptr, charBytes(this->chlen));
|
|
|
|
free(this->chptr);
|
|
|
|
this->chptr = this->chstr;
|
|
|
|
} else {
|
|
|
|
this->chptr = realloc(this->chptr, charBytes(len+1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
RichString_setChar(this, len, 0);
|
|
|
|
this->chlen = len;
|
|
|
|
}
|
|
|
|
|
2008-03-09 02:33:23 +00:00
|
|
|
#ifdef HAVE_LIBNCURSESW
|
|
|
|
|
2012-11-10 00:31:37 +00:00
|
|
|
static inline void RichString_writeFrom(RichString* this, int attrs, const char* data_c, int from, int len) {
|
2010-11-22 12:40:20 +00:00
|
|
|
wchar_t data[len+1];
|
|
|
|
len = mbstowcs(data, data_c, len);
|
2008-03-09 02:33:23 +00:00
|
|
|
if (len<0)
|
|
|
|
return;
|
2012-11-10 00:31:37 +00:00
|
|
|
int newLen = from + len;
|
2010-11-22 12:40:20 +00:00
|
|
|
RichString_setLen(this, newLen);
|
2012-11-10 00:31:37 +00:00
|
|
|
memset(&this->chptr[from], 0, sizeof(CharType) * (newLen - from));
|
|
|
|
for (int i = from, j = 0; i < newLen; i++, j++) {
|
2010-11-22 12:40:20 +00:00
|
|
|
this->chptr[i].chars[0] = data[j];
|
|
|
|
this->chptr[i].attr = attrs;
|
2008-03-09 02:33:23 +00:00
|
|
|
}
|
2010-11-22 12:40:20 +00:00
|
|
|
this->chptr[newLen].chars[0] = 0;
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
|
|
|
|
2010-11-22 12:40:20 +00:00
|
|
|
inline void RichString_setAttrn(RichString* this, int attrs, int start, int finish) {
|
|
|
|
cchar_t* ch = this->chptr + start;
|
2008-03-09 02:33:23 +00:00
|
|
|
for (int i = start; i <= finish; i++) {
|
|
|
|
ch->attr = attrs;
|
|
|
|
ch++;
|
|
|
|
}
|
2007-11-09 00:40:59 +00:00
|
|
|
}
|
|
|
|
|
2010-11-22 12:40:20 +00:00
|
|
|
int RichString_findChar(RichString* this, char c, int start) {
|
2008-03-09 02:33:23 +00:00
|
|
|
wchar_t wc = btowc(c);
|
2010-11-22 12:40:20 +00:00
|
|
|
cchar_t* ch = this->chptr + start;
|
|
|
|
for (int i = start; i < this->chlen; i++) {
|
2008-03-09 02:33:23 +00:00
|
|
|
if (ch->chars[0] == wc)
|
|
|
|
return i;
|
|
|
|
ch++;
|
|
|
|
}
|
|
|
|
return -1;
|
2007-11-09 00:40:59 +00:00
|
|
|
}
|
|
|
|
|
2008-03-09 02:33:23 +00:00
|
|
|
#else
|
|
|
|
|
2012-11-10 00:31:37 +00:00
|
|
|
static inline void RichString_writeFrom(RichString* this, int attrs, const char* data_c, int from, int len) {
|
|
|
|
int newLen = from + len;
|
2010-11-22 12:40:20 +00:00
|
|
|
RichString_setLen(this, newLen);
|
2012-11-10 00:31:37 +00:00
|
|
|
for (int i = from, j = 0; i < newLen; i++, j++)
|
2010-11-22 12:40:20 +00:00
|
|
|
this->chptr[i] = (isprint(data_c[j]) ? data_c[j] : '?') | attrs;
|
|
|
|
this->chptr[newLen] = 0;
|
2008-03-09 02:33:23 +00:00
|
|
|
}
|
|
|
|
|
2010-11-22 12:40:20 +00:00
|
|
|
void RichString_setAttrn(RichString* this, int attrs, int start, int finish) {
|
|
|
|
chtype* ch = this->chptr + start;
|
2008-03-09 02:33:23 +00:00
|
|
|
for (int i = start; i <= finish; i++) {
|
2006-07-12 01:16:03 +00:00
|
|
|
*ch = (*ch & 0xff) | attrs;
|
|
|
|
ch++;
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-22 12:40:20 +00:00
|
|
|
int RichString_findChar(RichString* this, char c, int start) {
|
|
|
|
chtype* ch = this->chptr + start;
|
|
|
|
for (int i = start; i < this->chlen; i++) {
|
2010-02-25 01:43:18 +00:00
|
|
|
if ((*ch & 0xff) == (chtype) c)
|
2008-03-09 02:33:23 +00:00
|
|
|
return i;
|
2006-07-12 01:16:03 +00:00
|
|
|
ch++;
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
2008-03-09 02:33:23 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2009-06-02 04:51:23 +00:00
|
|
|
void RichString_prune(RichString* this) {
|
2010-11-22 12:40:20 +00:00
|
|
|
if (this->chlen > RICHSTRING_MAXLEN)
|
|
|
|
free(this->chptr);
|
|
|
|
this->chptr = this->chstr;
|
|
|
|
this->chlen = 0;
|
2009-06-02 04:51:23 +00:00
|
|
|
}
|
|
|
|
|
2010-11-22 12:40:20 +00:00
|
|
|
void RichString_setAttr(RichString* this, int attrs) {
|
|
|
|
RichString_setAttrn(this, attrs, 0, this->chlen - 1);
|
2008-03-09 02:33:23 +00:00
|
|
|
}
|
|
|
|
|
2012-11-10 00:31:37 +00:00
|
|
|
void RichString_append(RichString* this, int attrs, const char* data) {
|
|
|
|
RichString_writeFrom(this, attrs, data, this->chlen, strlen(data));
|
|
|
|
}
|
|
|
|
|
|
|
|
void RichString_appendn(RichString* this, int attrs, const char* data, int len) {
|
|
|
|
RichString_writeFrom(this, attrs, data, this->chlen, len);
|
2008-03-09 02:33:23 +00:00
|
|
|
}
|
|
|
|
|
2010-02-25 01:43:18 +00:00
|
|
|
void RichString_write(RichString* this, int attrs, const char* data) {
|
2012-11-10 00:31:37 +00:00
|
|
|
RichString_writeFrom(this, attrs, data, 0, strlen(data));
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|