2020-02-03 07:00:36 +03:00
|
|
|
// Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
|
2019-06-23 05:21:30 +03:00
|
|
|
// Use of this source code is governed by an MIT license
|
|
|
|
// that can be found in the LICENSE file.
|
2019-09-14 23:48:30 +03:00
|
|
|
module strings
|
2019-06-22 21:20:28 +03:00
|
|
|
|
2019-10-24 12:47:21 +03:00
|
|
|
pub struct Builder {
|
2020-05-09 16:16:48 +03:00
|
|
|
// TODO
|
|
|
|
pub mut:
|
2019-12-20 00:29:37 +03:00
|
|
|
buf []byte
|
2020-04-28 11:03:37 +03:00
|
|
|
str_calls int
|
2019-12-20 00:29:37 +03:00
|
|
|
len int
|
2020-04-06 15:45:28 +03:00
|
|
|
initial_size int = 1
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
2019-07-03 23:11:27 +03:00
|
|
|
pub fn new_builder(initial_size int) Builder {
|
2019-12-20 00:29:37 +03:00
|
|
|
return Builder{
|
2020-04-26 18:52:27 +03:00
|
|
|
//buf: make(0, initial_size)
|
|
|
|
buf: []byte{cap: initial_size}
|
2020-05-18 22:38:06 +03:00
|
|
|
str_calls: 0
|
|
|
|
len: 0
|
2019-12-11 17:32:54 +03:00
|
|
|
initial_size: initial_size
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-17 14:51:18 +03:00
|
|
|
pub fn (mut b Builder) write_bytes(bytes byteptr, howmany int) {
|
2019-12-11 19:20:46 +03:00
|
|
|
b.buf.push_many(bytes, howmany)
|
|
|
|
b.len += howmany
|
|
|
|
}
|
|
|
|
|
2020-05-17 14:51:18 +03:00
|
|
|
pub fn (mut b Builder) write_b(data byte) {
|
2019-12-06 23:02:09 +03:00
|
|
|
b.buf << data
|
2019-12-08 14:11:19 +03:00
|
|
|
b.len++
|
2019-12-06 23:02:09 +03:00
|
|
|
}
|
|
|
|
|
2020-05-17 14:51:18 +03:00
|
|
|
pub fn (mut b Builder) write(s string) {
|
2019-12-24 20:54:43 +03:00
|
|
|
if s == '' {
|
|
|
|
return
|
|
|
|
}
|
2019-10-10 01:15:19 +03:00
|
|
|
b.buf.push_many(s.str, s.len)
|
2019-12-20 00:29:37 +03:00
|
|
|
// for c in s {
|
|
|
|
// b.buf << c
|
|
|
|
// }
|
|
|
|
// b.buf << []byte(s) // TODO
|
2019-06-22 21:20:28 +03:00
|
|
|
b.len += s.len
|
|
|
|
}
|
|
|
|
|
2020-05-17 14:51:18 +03:00
|
|
|
pub fn (mut b Builder) go_back(n int) {
|
2020-03-11 01:21:26 +03:00
|
|
|
b.buf.trim(b.buf.len-n)
|
2020-03-11 03:31:24 +03:00
|
|
|
b.len -= n
|
|
|
|
}
|
2020-03-11 01:21:26 +03:00
|
|
|
|
2020-07-01 01:53:53 +03:00
|
|
|
fn bytes2string(b []byte) string {
|
|
|
|
mut copy := b.clone()
|
|
|
|
copy << `\0`
|
|
|
|
res := tos(copy.data, copy.len-1)
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
2020-06-17 01:59:33 +03:00
|
|
|
pub fn (mut b Builder) cut_last(n int) string {
|
2020-07-01 01:53:53 +03:00
|
|
|
res := bytes2string( b.buf[b.len-n..] )
|
2020-06-17 01:59:33 +03:00
|
|
|
b.buf.trim(b.buf.len-n)
|
|
|
|
b.len -= n
|
2020-07-01 01:53:53 +03:00
|
|
|
return res
|
2020-06-17 01:59:33 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
pub fn (mut b Builder) cut_to(pos int) string {
|
2020-07-01 01:53:53 +03:00
|
|
|
res := bytes2string( b.buf[pos..] )
|
2020-06-17 01:59:33 +03:00
|
|
|
b.buf.trim(pos)
|
|
|
|
b.len = pos
|
2020-07-01 01:53:53 +03:00
|
|
|
return res
|
2020-06-17 01:59:33 +03:00
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
2020-05-17 14:51:18 +03:00
|
|
|
pub fn (mut b Builder) go_back_to(pos int) {
|
2020-04-28 14:15:37 +03:00
|
|
|
b.buf.trim(pos)
|
|
|
|
b.len = pos
|
|
|
|
}
|
|
|
|
|
2020-05-17 14:51:18 +03:00
|
|
|
pub fn (mut b Builder) writeln(s string) {
|
2019-12-20 00:29:37 +03:00
|
|
|
// for c in s {
|
|
|
|
// b.buf << c
|
|
|
|
// }
|
2019-10-10 01:15:19 +03:00
|
|
|
b.buf.push_many(s.str, s.len)
|
2019-12-20 00:29:37 +03:00
|
|
|
// b.buf << []byte(s) // TODO
|
2019-06-22 21:20:28 +03:00
|
|
|
b.buf << `\n`
|
|
|
|
b.len += s.len + 1
|
|
|
|
}
|
|
|
|
|
2020-03-21 09:01:06 +03:00
|
|
|
// buf == 'hello world'
|
|
|
|
// last_n(5) returns 'world'
|
2020-03-21 09:04:53 +03:00
|
|
|
pub fn (b &Builder) last_n(n int) string {
|
2020-03-21 09:01:06 +03:00
|
|
|
if n > b.len {
|
|
|
|
return ''
|
|
|
|
}
|
2020-07-01 01:53:53 +03:00
|
|
|
return bytes2string( b.buf[b.len-n..] )
|
2020-03-21 09:01:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// buf == 'hello world'
|
|
|
|
// after(6) returns 'world'
|
2020-03-21 09:04:53 +03:00
|
|
|
pub fn (b &Builder) after(n int) string {
|
2020-03-21 09:01:06 +03:00
|
|
|
if n >= b.len {
|
|
|
|
return ''
|
|
|
|
}
|
2020-07-01 01:53:53 +03:00
|
|
|
return bytes2string( b.buf[n..] )
|
2020-03-21 09:01:06 +03:00
|
|
|
}
|
|
|
|
|
2020-04-28 11:03:37 +03:00
|
|
|
// NB: in order to avoid memleaks and additional memory copies, after a call to b.str(),
|
|
|
|
// the builder b will be empty. The returned string *owns* the accumulated data so far.
|
2020-05-17 14:51:18 +03:00
|
|
|
pub fn (mut b Builder) str() string {
|
2020-04-28 11:03:37 +03:00
|
|
|
b.str_calls++
|
|
|
|
if b.str_calls > 1 {
|
|
|
|
panic('builder.str() should be called just once.\n' +
|
|
|
|
'If you want to reuse a builder, call b.free() first.')
|
|
|
|
}
|
2019-11-17 02:22:43 +03:00
|
|
|
b.buf << `\0`
|
2020-04-28 11:03:37 +03:00
|
|
|
s := string(b.buf,b.len)
|
|
|
|
bis := b.initial_size
|
2020-06-28 14:56:38 +03:00
|
|
|
//free(b.buf.data)
|
2020-04-28 11:03:37 +03:00
|
|
|
b.buf = []byte{cap: bis}
|
|
|
|
b.len = 0
|
|
|
|
return s
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
2020-05-17 14:51:18 +03:00
|
|
|
pub fn (mut b Builder) free() {
|
2019-12-20 00:29:37 +03:00
|
|
|
unsafe{
|
|
|
|
free(b.buf.data)
|
|
|
|
}
|
2020-05-06 14:44:08 +03:00
|
|
|
//b.buf = []byte{cap: b.initial_size}
|
2019-11-19 00:13:14 +03:00
|
|
|
b.len = 0
|
2020-04-28 11:03:37 +03:00
|
|
|
b.str_calls = 0
|
2019-06-24 23:34:21 +03:00
|
|
|
}
|