mirror of
https://github.com/ultimatepp/ultimatepp.git
synced 2026-05-15 14:16:07 -06:00
This might bring some incompatibilities in the code that expects wchar to be 16 bit, which escpecially involves dealing with Win32 (and to lesser extend MacOS) APIs, so if your application is doing that, please check all instances of WCHAR (UniChar on MacOS) or even wchar especially type casts. To support host APIs, char16 is introduced (but there is no 16-bit String varian). Use ToSystemCharsetW, FromSystemCharsetW to convert texts to Win32 API. - Support of drawing non-BMP characters in GUI - Vastly improved character font replacement code (when drawing characters missing with requested font, replacement font is used) - Last instances of Win32 ANSI calls (those ending with A) are removed - UTF handling routines are refactored and their's naming is unified - RTF is now being able to handle non-BMP characters (RTF is used as clipboard format for RichText) Other minor changes: - fixed TryRealloc issue - improved MemoryCheck - Removed MemoryAlloc48/MemoryFree48 - In theide Background parsing should less often cause delays in the main thread
187 lines
5 KiB
C++
187 lines
5 KiB
C++
#include "Core.h"
|
|
|
|
namespace Upp {
|
|
|
|
template <class S, class Char, class F>
|
|
Vector<S> SplitGeneric(int maxcount, const F& delim, const Char *s, bool ignoreempty = true)
|
|
{
|
|
Vector<S> r;
|
|
r.Reserve(min(maxcount, 8));
|
|
const Char *t = s;
|
|
while(*t && r.GetCount() < maxcount) {
|
|
const Char *q = delim(t);
|
|
if(q) {
|
|
if(!ignoreempty || t > s)
|
|
r.Add().Set(s, int(t - s)); // This is faster than r.Add(String(s, t))...
|
|
t = s = q;
|
|
}
|
|
else
|
|
t++;
|
|
}
|
|
if((!ignoreempty || t > s) && r.GetCount() < maxcount)
|
|
r.Add().Set(s, int(t - s));
|
|
return r;
|
|
}
|
|
|
|
Vector<String> Split(int maxcount, const char *s, const char * (*text_filter)(const char *), bool ignoreempty)
|
|
{
|
|
return SplitGeneric<String>(maxcount, text_filter, s, ignoreempty);
|
|
}
|
|
|
|
struct SplitDelimFilter__ {
|
|
int (*filter)(int);
|
|
const char *operator()(const char *s) const { return (*filter)((byte)*s) ? s + 1 : NULL; }
|
|
};
|
|
|
|
Vector<String> Split(int maxcount, const char *s, int (*filter)(int), bool ignoreempty)
|
|
{
|
|
SplitDelimFilter__ delim;
|
|
delim.filter = filter;
|
|
return SplitGeneric<String>(maxcount, delim, s, ignoreempty);
|
|
}
|
|
|
|
struct SplitDelimChar__ {
|
|
int chr;
|
|
const char *operator()(const char *s) const { return *s == chr ? s + 1 : NULL; }
|
|
};
|
|
|
|
Vector<String> Split(int maxcount, const char *s, int chr, bool ignoreempty)
|
|
{
|
|
SplitDelimChar__ delim;
|
|
delim.chr = chr;
|
|
return SplitGeneric<String>(maxcount, delim, s, ignoreempty);
|
|
}
|
|
|
|
struct SplitDelimText__ {
|
|
const char *ds;
|
|
int l;
|
|
const char *operator()(const char *s) const { return strncmp(s, ds, l) == 0 ? s + l : NULL; }
|
|
};
|
|
|
|
Vector<String> Split(int maxcount, const char *s, const char *text, bool ignoreempty)
|
|
{
|
|
SplitDelimText__ delim;
|
|
delim.ds = text;
|
|
delim.l = (int)strlen(text);
|
|
return delim.l ? SplitGeneric<String>(maxcount, delim, s, ignoreempty) : Vector<String>();
|
|
}
|
|
|
|
Vector<String> Split(const char *s, const char * (*text_filter)(const char *), bool ignoreempty)
|
|
{
|
|
return Split(INT_MAX, s, text_filter, ignoreempty);
|
|
}
|
|
|
|
Vector<String> Split(const char *s, int (*filter)(int), bool ignoreempty)
|
|
{
|
|
return Split(INT_MAX, s, filter, ignoreempty);
|
|
}
|
|
|
|
Vector<String> Split(const char *s, int chr, bool ignoreempty)
|
|
{
|
|
return Split(INT_MAX, s, chr, ignoreempty);
|
|
}
|
|
|
|
Vector<String> Split(const char *s, const char *text, bool ignoreempty)
|
|
{
|
|
return Split(INT_MAX, s, text, ignoreempty);
|
|
}
|
|
|
|
Vector<WString> Split(int maxcount, const wchar *s, const wchar * (*text_filter)(const wchar *), bool ignoreempty)
|
|
{
|
|
return SplitGeneric<WString>(maxcount, text_filter, s, ignoreempty);
|
|
}
|
|
|
|
struct SplitDelimWFilter__ {
|
|
int (*filter)(int);
|
|
const wchar *operator()(const wchar *s) const { return (*filter)((byte)*s) ? s + 1 : NULL; }
|
|
};
|
|
|
|
Vector<WString> Split(int maxcount, const wchar *s, int (*filter)(int), bool ignoreempty)
|
|
{
|
|
SplitDelimWFilter__ delim;
|
|
delim.filter = filter;
|
|
return SplitGeneric<WString>(maxcount, delim, s, ignoreempty);
|
|
}
|
|
|
|
struct SplitDelimWChar__ {
|
|
int chr;
|
|
const wchar *operator()(const wchar *s) const { return *s == chr ? s + 1 : NULL; }
|
|
};
|
|
|
|
Vector<WString> Split(int maxcount, const wchar *s, int chr, bool ignoreempty)
|
|
{
|
|
SplitDelimWChar__ delim;
|
|
delim.chr = chr;
|
|
return SplitGeneric<WString>(maxcount, delim, s, ignoreempty);
|
|
}
|
|
|
|
int w_strncmp(const wchar *s, const wchar *t, int n)
|
|
{
|
|
while(*s && *t && n > 0) {
|
|
int q = (int)(uint16)*s - (int)(uint16)*t;
|
|
if(q)
|
|
return q;
|
|
s++;
|
|
t++;
|
|
n--;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
struct SplitDelimWText {
|
|
const wchar *ds;
|
|
int l;
|
|
const wchar *operator()(const wchar *s) const { return w_strncmp(s, ds, l) == 0 ? s + l : NULL; }
|
|
};
|
|
|
|
Vector<WString> Split(int maxcount, const wchar *s, const wchar *text, bool ignoreempty)
|
|
{
|
|
SplitDelimWText delim;
|
|
delim.ds = text;
|
|
delim.l = strlen__(text);
|
|
return delim.l ? SplitGeneric<WString>(maxcount, delim, s, ignoreempty) : Vector<WString>();
|
|
}
|
|
|
|
Vector<WString> Split(const wchar *s, int (*filter)(int), bool ignoreempty)
|
|
{
|
|
return Split(INT_MAX, s, filter, ignoreempty);
|
|
}
|
|
|
|
Vector<WString> Split(const wchar *s, int chr, bool ignoreempty)
|
|
{
|
|
return Split(INT_MAX, s, chr, ignoreempty);
|
|
}
|
|
|
|
Vector<WString> Split(const wchar *s, const wchar *text, bool ignoreempty)
|
|
{
|
|
return Split(INT_MAX, s, text, ignoreempty);
|
|
}
|
|
|
|
Vector<WString> Split(const wchar *s, const wchar * (*text_filter)(const wchar *), bool ignoreempty)
|
|
{
|
|
return SplitGeneric<WString>(INT_MAX, text_filter, s, ignoreempty);
|
|
}
|
|
|
|
template <class B, class T>
|
|
T Join_(const Vector<T>& im, const T& delim, bool ignoreempty) {
|
|
B r;
|
|
bool next = false;
|
|
for(int i = 0; i < im.GetCount(); i++)
|
|
if(!ignoreempty || im[i].GetCount()) {
|
|
if(next)
|
|
r.Cat(delim);
|
|
r.Cat(im[i]);
|
|
next = true;
|
|
}
|
|
return T(r);
|
|
}
|
|
|
|
String Join(const Vector<String>& im, const String& delim, bool ignoreempty) {
|
|
return Join_<StringBuffer, String>(im, delim, ignoreempty);
|
|
}
|
|
|
|
WString Join(const Vector<WString>& im, const WString& delim, bool ignoreempty) {
|
|
return Join_<WStringBuffer, WString>(im, delim, ignoreempty);
|
|
}
|
|
|
|
}
|