2011-12-26 21:35:57 +00:00
|
|
|
/*
|
|
|
|
htop - RichString.c
|
|
|
|
(C) 2004,2011 Hisham H. Muhammad
|
2020-10-05 07:51:32 +00:00
|
|
|
Released under the GNU GPLv2, see the COPYING file
|
2011-12-26 21:35:57 +00:00
|
|
|
in the source distribution for its full text.
|
|
|
|
*/
|
2006-03-04 18:16:49 +00:00
|
|
|
|
|
|
|
#include "RichString.h"
|
|
|
|
|
2020-12-04 13:44:57 +00:00
|
|
|
#include <ctype.h>
|
2006-03-04 18:16:49 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2011-12-26 21:35:57 +00:00
|
|
|
|
2020-09-19 11:55:23 +00:00
|
|
|
#include "Macros.h"
|
|
|
|
#include "XUtils.h"
|
|
|
|
|
2020-10-14 22:56:22 +00:00
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2020-10-27 20:26:35 +00:00
|
|
|
static void RichString_setLen(RichString* this, int len) {
|
|
|
|
if (len < RICHSTRING_MAXLEN && this->chlen < RICHSTRING_MAXLEN) {
|
|
|
|
RichString_setChar(this, len, 0);
|
|
|
|
this->chlen = len;
|
|
|
|
} else {
|
|
|
|
RichString_extendLen(this, len);
|
|
|
|
}
|
|
|
|
}
|
2014-11-20 01:17:52 +00:00
|
|
|
|
2008-03-09 02:33:23 +00:00
|
|
|
#ifdef HAVE_LIBNCURSESW
|
|
|
|
|
2020-12-04 13:44:57 +00:00
|
|
|
static inline void RichString_writeFromWide(RichString* this, int attrs, const char* data_c, int from, int len) {
|
2020-10-31 22:28:02 +00:00
|
|
|
wchar_t data[len + 1];
|
2010-11-22 12:40:20 +00:00
|
|
|
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;
|
2020-11-01 00:09:51 +00:00
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2020-12-04 13:44:57 +00:00
|
|
|
static inline void RichString_writeFromAscii(RichString* this, int attrs, const char* data, int from, int len) {
|
|
|
|
int newLen = from + len;
|
|
|
|
RichString_setLen(this, newLen);
|
|
|
|
for (int i = from, j = 0; i < newLen; i++, j++) {
|
|
|
|
this->chptr[i] = (CharType) { .attr = attrs & 0xffffff, .chars = { (isprint(data[j]) ? data[j] : '?') } };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2020-09-08 11:59:30 +00:00
|
|
|
#else /* HAVE_LIBNCURSESW */
|
2008-03-09 02:33:23 +00:00
|
|
|
|
2020-12-04 13:44:57 +00:00
|
|
|
static inline void RichString_writeFromWide(RichString* this, int attrs, const char* data_c, int from, int len) {
|
2012-11-10 00:31:37 +00:00
|
|
|
int newLen = from + len;
|
2010-11-22 12:40:20 +00:00
|
|
|
RichString_setLen(this, newLen);
|
2020-11-01 00:09:51 +00:00
|
|
|
for (int i = from, j = 0; i < newLen; i++, j++) {
|
2020-12-04 15:10:40 +00:00
|
|
|
this->chptr[i] = (((unsigned char)data_c[j]) >= 32 ? ((unsigned char)data_c[j]) : '?') | attrs;
|
2020-11-01 00:09:51 +00:00
|
|
|
}
|
2010-11-22 12:40:20 +00:00
|
|
|
this->chptr[newLen] = 0;
|
2008-03-09 02:33:23 +00:00
|
|
|
}
|
|
|
|
|
2020-12-04 13:44:57 +00:00
|
|
|
static inline void RichString_writeFromAscii(RichString* this, int attrs, const char* data_c, int from, int len) {
|
|
|
|
RichString_writeFromWide(this, attrs, data_c, from, len);
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2020-09-08 11:59:30 +00:00
|
|
|
#endif /* HAVE_LIBNCURSESW */
|
2008-03-09 02:33:23 +00:00
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2020-11-23 19:12:11 +00:00
|
|
|
void RichString_appendChr(RichString* this, char c, int count) {
|
|
|
|
int from = this->chlen;
|
|
|
|
int newLen = from + count;
|
|
|
|
RichString_setLen(this, newLen);
|
|
|
|
for (int i = from; i < newLen; i++) {
|
|
|
|
RichString_setChar(this, i, c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2020-12-04 13:44:57 +00:00
|
|
|
void RichString_appendWide(RichString* this, int attrs, const char* data) {
|
|
|
|
RichString_writeFromWide(this, attrs, data, this->chlen, strlen(data));
|
|
|
|
}
|
|
|
|
|
|
|
|
void RichString_appendnWide(RichString* this, int attrs, const char* data, int len) {
|
|
|
|
RichString_writeFromWide(this, attrs, data, this->chlen, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RichString_writeWide(RichString* this, int attrs, const char* data) {
|
|
|
|
RichString_writeFromWide(this, attrs, data, 0, strlen(data));
|
|
|
|
}
|
|
|
|
|
|
|
|
void RichString_appendAscii(RichString* this, int attrs, const char* data) {
|
|
|
|
RichString_writeFromAscii(this, attrs, data, this->chlen, strlen(data));
|
2012-11-10 00:31:37 +00:00
|
|
|
}
|
|
|
|
|
2020-12-04 13:44:57 +00:00
|
|
|
void RichString_appendnAscii(RichString* this, int attrs, const char* data, int len) {
|
|
|
|
RichString_writeFromAscii(this, attrs, data, this->chlen, len);
|
2008-03-09 02:33:23 +00:00
|
|
|
}
|
|
|
|
|
2020-12-04 13:44:57 +00:00
|
|
|
void RichString_writeAscii(RichString* this, int attrs, const char* data) {
|
|
|
|
RichString_writeFromAscii(this, attrs, data, 0, strlen(data));
|
2014-11-24 20:55:03 +00:00
|
|
|
}
|