Files
kraken/stdlib/str.krak

338 lines
8.9 KiB
Plaintext
Raw Normal View History

2018-05-22 19:43:54 -04:00
import vec
import util
import mem
import serialize
import io
ext fun snprintf(to_str: *char, num: ulong, format: *char, ...): int
2018-05-22 19:43:54 -04:00
fun to_string(in: float): str
return to_string((in) cast double)
2018-05-22 19:43:54 -04:00
fun to_string(in: double): str {
var how_much = snprintf(mem::null<char>(), (0) cast ulong, "%f", in)
var int_str = mem::new<char>(how_much+2)
snprintf(int_str, (how_much+1) cast ulong, "%f", in)
2018-05-22 19:43:54 -04:00
var to_ret = str(int_str)
mem::delete(int_str)
return to_ret
}
2018-05-22 19:43:54 -04:00
fun to_string(in: bool): str
if (in) return str("true")
else return str("false")
fun to_string(in: char): str
return str(in)
fun to_string(in: uchar): str
return to_string_num(in)
2018-05-22 19:43:54 -04:00
fun to_string(in: short): str
return to_string_num(in)
2018-05-22 19:43:54 -04:00
fun to_string(in: ushort): str
return to_string_num(in)
2018-05-22 19:43:54 -04:00
fun to_string(in: int): str
return to_string_num(in)
2018-05-22 19:43:54 -04:00
fun to_string(in: uint): str
return to_string_num(in)
2018-05-22 19:43:54 -04:00
fun to_string(in: long): str
return to_string_num(in)
2018-05-22 19:43:54 -04:00
fun to_string(in: ulong): str
return to_string_num(in)
2018-05-22 19:43:54 -04:00
fun to_string<T>(in: *T): str
return str("ptr:<") + to_string_num((in) cast ulong) + ">"
2018-10-08 00:28:42 -04:00
/*fun deref_to_string<T>(in: *T): str*/
/*if (in == mem::null<T>())*/
/*return str("null")*/
/*else*/
/*return to_string(in)*/
/*return in->to_string()*/
2018-06-22 23:13:08 -04:00
2018-05-22 19:43:54 -04:00
fun string_to_num<T>(it: str): T {
var is_negative = false
if (it[0] == '-') {
is_negative = true
it = it.slice(1,-1)
}
var result:T = 0
var power:T = 1
while (it.length()) {
result += power * (it.last() - '0')
it = it.slice(0,-2)
power *= 10
}
if (is_negative)
return -result
return result
}
2018-05-22 19:43:54 -04:00
fun string_to_double(it: str): double {
var parts = it.split('.')
var divisor = 1
for (var i = 0; i < parts[1].length(); i++;) divisor *= 10
return string_to_num<long>(parts[0]) + (string_to_num<long>(parts[1])) cast double / divisor
}
2018-05-22 19:43:54 -04:00
fun to_string_num<T>(in: T): str {
if (in == 0)
2018-05-22 19:43:54 -04:00
return str("0")
var ret = str()
var pos = 0
var is_neg = false
if (in > 0) {
pos = in
} else {
pos = -in
is_neg = true
}
while (pos) {
2018-05-22 19:43:54 -04:00
ret = str((pos%10 + '0') cast char) + ret
pos = pos / 10
}
if (is_neg)
2018-05-22 19:43:54 -04:00
return str("-") + ret
return ret
}
2018-05-22 19:43:54 -04:00
fun operator+(first: *char, second: ref str): str {
return str(first) + second
}
2018-05-22 19:43:54 -04:00
fun operator+(first: int, second: ref str): str {
return to_string(first) + second
}
2018-06-18 19:04:24 -04:00
fun operator*(first: *char, second: int): str {
return str(first) * second
}
fun operator*(first: int, second: *char): str {
return str(second) * first
}
fun operator*(first: int, second: ref str): str {
return second * first
}
2018-05-22 19:43:54 -04:00
fun str(in:*char):str {
var out.construct(in):str
return out
}
2018-05-22 19:43:54 -04:00
fun str(in:char):str {
var out.construct():str
out += in
return out
}
2018-05-22 19:43:54 -04:00
fun str():str {
return str("")
}
2018-05-22 19:43:54 -04:00
obj str (Object, Serializable, Hashable) {
var data: vec::vec<char>;
fun construct(): *str {
data.construct();
return this;
}
2018-05-22 19:43:54 -04:00
fun construct(ammt: int): *str {
data.construct(ammt);
return this;
}
2018-05-22 19:43:54 -04:00
fun construct(str: *char): *str {
2016-06-11 00:45:18 -07:00
var len = 0
while (str[len] != 0) len++
data.construct(len);
data.set_size(len);
mem::memmove((data.getBackingMemory()) cast *void, (str) cast *void, (len) cast ulong)
// no null terminator
return this;
}
2018-05-22 19:43:54 -04:00
fun construct(vec: ref vec::vec<char>): *str {
data.copy_construct(&vec);
return this;
}
2018-05-22 19:43:54 -04:00
fun construct(str: ref str): *str {
return construct(str.data);
}
2018-05-22 19:43:54 -04:00
fun copy_construct(old: *str): void {
data.copy_construct(&old->data)
}
fun copy_construct(old: **char): void {
construct(*old)
}
2017-10-28 15:28:34 -04:00
fun hash(): ulong {
var hash: ulong = 7
for (var i = 0; i < data.size; i++;)
hash = hash*31 + data[i]
return hash
}
fun operator=(str: *char): void {
destruct();
construct(str)
}
2018-05-22 19:43:54 -04:00
fun operator=(str: ref str): void {
2016-05-05 04:51:10 -04:00
data = str.data
}
fun destruct():void {
data.destruct()
}
2018-05-22 19:43:54 -04:00
fun serialize(): vec::vec<char> {
return serialize::serialize(data)
}
2018-05-22 19:43:54 -04:00
fun unserialize(it: ref vec::vec<char>, pos: int): int {
return data.unserialize(it, pos)
}
fun operator[](index: int): ref char { return data[index]; }
2018-05-22 19:43:54 -04:00
fun slice(first: int, second: int): str {
var new.construct(data.slice(first,second)): str
return new
}
fun set(index: int, toSet: char) {
data.set(index, toSet)
}
fun length():int { return data.size; }
2018-05-22 19:43:54 -04:00
fun operator!=(other: ref str): bool return !(*this ==other)
2015-12-06 15:15:33 -05:00
fun operator!=(other: *char): bool return !(*this==other)
2018-05-22 19:43:54 -04:00
fun operator==(other: ref str): bool {
// you were too good for this world
//return length() == other.length() && !util::range(length()).any_true(fun(i: int):bool { return data[i] != other[i]; } )
if (data.size != other.data.size)
return false
for (var i = 0; i < data.size; i++;)
if (data.data[i] != other.data.data[i])
return false
return true
}
2015-07-05 02:34:45 -04:00
fun operator==(other: *char): bool {
2018-05-22 19:43:54 -04:00
var str.construct(other) : str
2015-07-05 02:34:45 -04:00
return *this == str
}
fun operator<=(other: ref str): bool {
var l = 0
var r = 0
while (l < length() || r < other.length()) {
if l == length() {
return true
} else if r == other.length() {
return false
} else if (*this)[l] < other[r] {
return true
} else if (*this)[l] > other[r] {
return false
}
l++
r++
}
}
fun operator<(other: ref str): bool {
return *this <= other && *this != other
}
2018-06-18 19:04:24 -04:00
fun operator*(n: int): str {
var to_ret.construct(): str
while (n-- > 0)
to_ret += *this
return to_ret
}
2018-05-22 19:43:54 -04:00
fun operator+(c: char): str {
var to_ret = *this
to_ret += c
return to_ret
}
2018-05-22 19:43:54 -04:00
fun operator+(integer: int): str return *this + to_string(integer);
fun operator+(integer: long): str return *this + to_string(integer);
fun operator+(integer: ulong): str return *this + to_string(integer);
/*fun operator+(b: bool): str return *this + to_string(b);*/
2018-05-22 19:43:54 -04:00
fun operator+(str: *char): str {
var newStr.construct(str):str
var ret.construct(data + newStr.data):str
return ret
}
2018-05-22 19:43:54 -04:00
fun operator+(str: ref str): str {
var ret.construct(data + str.data):str
return ret
}
fun operator+=(integer: int) *this += to_string(integer);
fun operator+=(integer: ulong) *this += to_string(integer);
2016-06-16 09:53:15 -07:00
/*fun operator+=(b: bool) *this += to_string(b);*/
fun operator+=(character: char): void {
data += character
}
fun operator+=(str: *char): void {
2018-05-22 19:43:54 -04:00
var newStr.construct(str):str
data += newStr.data
}
2018-05-22 19:43:54 -04:00
fun operator+=(str: ref str): void {
//var newStr.construct(str):str
//data += newStr.data
data += str.data
}
fun toCharArray(): *char {
var out: *char = mem::new<char>(data.size+1);
for (var i: int = 0; i < data.size; i++;)
out[i] = data.get(i);
// null terminator
out[data.size] = 0
return out;
}
fun getBackingMemory(): *char return data.getBackingMemory();
2018-05-22 19:43:54 -04:00
fun split(delim: *char): vec::vec<str> return split(str(delim))
fun split(delim: str): vec::vec<str> {
var out.construct(): vec::vec<str>
var current = str("")
for (var i = 0; i < data.size; i++;) {
if (i < data.size-delim.length() && slice(i, i+delim.length()) == delim) {
out.add(current)
2018-05-22 19:43:54 -04:00
current = str("")
i += delim.length()-1
} else {
current += data[i]
}
}
out.add(current)
return out
}
fun first(): char return data.first()
fun last(): char return data.last()
2018-05-22 19:43:54 -04:00
fun lines(): vec::vec<str> return split('\n')
fun split(delim: char): vec::vec<str> {
var out.construct(): vec::vec<str>
var current = str("")
for (var i = 0; i < data.size; i++;) {
if (data[i] == delim) {
out.add(current)
2018-05-22 19:43:54 -04:00
current = str("")
} else {
current += data[i]
}
}
out.add(current)
return out
}
2018-05-22 19:43:54 -04:00
fun join(to_join: ref vec::vec<str>): str {
if (to_join.size != 0) {
var to_ret = to_join.first()
for (var i = 1; i < to_join.size; i++;)
to_ret += *this + to_join[i]
return to_ret
} else {
return str("")
}
}
fun for_each(func: fun(char):void) {
data.for_each(func)
}
2018-09-22 20:10:56 -04:00
fun contains(c: char): bool {
return data.contains(c)
}
};