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"
|
2016-02-02 14:53:02 +00:00
|
|
|
#include "XAlloc.h"
|
2006-03-04 18:16:49 +00:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2011-12-26 21:35:57 +00:00
|
|
|
|
2014-11-20 01:17:52 +00:00
|
|
|
#define RICHSTRING_MAXLEN 350
|
2011-12-26 21:35:57 +00:00
|
|
|
|
|
|
|
/*{
|
|
|
|
#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
|
|
|
|
2015-08-25 13:23:50 +00:00
|
|
|
#ifdef HAVE_LIBNCURSESW
|
|
|
|
#include <wctype.h>
|
|
|
|
#endif
|
|
|
|
|
2010-11-22 12:40:20 +00:00
|
|
|
#define RichString_size(this) ((this)->chlen)
|
|
|
|
#define RichString_sizeVal(this) ((this).chlen)
|
|
|
|
|
2015-08-20 03:31:48 +00:00
|
|
|
#define RichString_begin(this) RichString (this); memset(&this, 0, sizeof(RichString)); (this).chptr = (this).chstr;
|
|
|
|
#define RichString_beginAllocated(this) memset(&this, 0, sizeof(RichString)); (this).chptr = (this).chstr;
|
2010-11-22 12:40:20 +00:00
|
|
|
#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)
|
2015-08-28 00:45:02 +00:00
|
|
|
#define RichString_setChar(this, at, ch) do{ (this)->chptr[(at)] = (CharType) { .chars = { ch, 0 } }; } while(0)
|
2010-11-22 12:40:20 +00:00
|
|
|
#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* chptr;
|
2014-11-20 01:17:52 +00:00
|
|
|
CharType chstr[RICHSTRING_MAXLEN+1];
|
2006-03-04 18:16:49 +00:00
|
|
|
} RichString;
|
|
|
|
|
|
|
|
}*/
|
|
|
|
|
2018-02-26 14:05:12 +00:00
|
|
|
#ifndef CLAMP
|
|
|
|
#define CLAMP(x,low,high) (((x)>(high))?(high):(((x)<(low))?(low):(x)))
|
|
|
|
#endif
|
|
|
|
|
2019-10-31 16:39:12 +00:00
|
|
|
#define charBytes(n) (sizeof(CharType) * (n))
|
2010-11-22 12:40:20 +00:00
|
|
|
|
2014-11-20 01:17:52 +00:00
|
|
|
static void RichString_extendLen(RichString* this, int len) {
|
2010-11-22 12:40:20 +00:00
|
|
|
if (this->chlen <= RICHSTRING_MAXLEN) {
|
|
|
|
if (len > RICHSTRING_MAXLEN) {
|
2016-02-02 14:53:02 +00:00
|
|
|
this->chptr = xMalloc(charBytes(len + 1));
|
2015-08-28 00:45:02 +00:00
|
|
|
memcpy(this->chptr, this->chstr, charBytes(this->chlen));
|
2010-11-22 12:40:20 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (len <= RICHSTRING_MAXLEN) {
|
2015-08-28 00:45:02 +00:00
|
|
|
memcpy(this->chstr, this->chptr, charBytes(len));
|
2010-11-22 12:40:20 +00:00
|
|
|
free(this->chptr);
|
|
|
|
this->chptr = this->chstr;
|
|
|
|
} else {
|
2016-02-02 14:53:02 +00:00
|
|
|
this->chptr = xRealloc(this->chptr, charBytes(len + 1));
|
2010-11-22 12:40:20 +00:00
|
|
|
}
|
|
|
|
}
|
2015-08-20 03:31:48 +00:00
|
|
|
|
2010-11-22 12:40:20 +00:00
|
|
|
RichString_setChar(this, len, 0);
|
|
|
|
this->chlen = len;
|
|
|
|
}
|
|
|
|
|
2015-08-20 04:13:20 +00:00
|
|
|
#define RichString_setLen(this, len) do{ if(len < RICHSTRING_MAXLEN && this->chlen < RICHSTRING_MAXLEN) { RichString_setChar(this,len,0); this->chlen=len; } else RichString_extendLen(this,len); }while(0)
|
2014-11-20 01:17:52 +00:00
|
|
|
|
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);
|
2015-08-28 00:45:02 +00:00
|
|
|
if (len < 0)
|
2008-03-09 02:33:23 +00:00
|
|
|
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
|
|
|
for (int i = from, j = 0; i < newLen; i++, j++) {
|
2015-11-23 05:46:43 +00:00
|
|
|
this->chptr[i] = (CharType) { .attr = attrs & 0xffffff, .chars = { (iswprint(data[j]) ? data[j] : '?') } };
|
2008-03-09 02:33:23 +00:00
|
|
|
}
|
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;
|
2018-02-26 14:05:12 +00:00
|
|
|
finish = CLAMP(finish, 0, this->chlen - 1);
|
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++)
|
2015-03-23 01:56:28 +00:00
|
|
|
this->chptr[i] = (data_c[j] >= 32 ? data_c[j] : '?') | attrs;
|
2010-11-22 12:40:20 +00:00
|
|
|
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;
|
2018-02-26 14:05:12 +00:00
|
|
|
finish = CLAMP(finish, 0, this->chlen - 1);
|
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);
|
2015-08-20 03:31:48 +00:00
|
|
|
memset(this, 0, sizeof(RichString));
|
2010-11-22 12:40:20 +00:00
|
|
|
this->chptr = this->chstr;
|
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));
|
2014-11-24 20:55:03 +00:00
|
|
|
}
|