utilitiec/src/StringView/StringView.c
2024-06-13 15:28:21 +02:00

206 lines
4.6 KiB
C

/*
* This code is part of the programming language Ivy.
* Ivy comes with ABSOLUTELY NO WARRANTY and is licensed under AGPL-3.0 or later.
* Copyright (C) 2024 VegOwOtenks
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include "StringView.h"
#include <ctype.h>
#include <string.h>
#include <stdint.h>
StringView StringView_FromString(const char* source)
{
return StringView_FromStringSized(source, strlen(source));
}
StringView StringView_FromStringSized(const char* source, size_t length)
{
StringView view = {
.source = source,
.length = length,
};
return view;
}
bool StringView_Equal(StringView first, StringView second)
{
return first.length == second.length
&&
strncmp(first.source, second.source, first.length) == 0;
}
bool StringView_StartsWith(StringView string, StringView start)
{
return strncmp(string.source, start.source, start.length) == 0;
}
bool StringView_EndsWith(StringView string, StringView end)
{
if (end.length > string.length) {
return false;
}
return StringView_Equal(
end,
StringView_Slice(
string,
string.length - end.length,
string.length
)
);
}
bool StringView_Contains(StringView string, StringView find)
{
return StringView_FindString(string, find).source != NULL;
}
size_t StringView_Count(StringView string, StringView find)
{
size_t count = 0;
while ((find = StringView_FindString(string, find)).source != NULL) {
size_t offset = (uintptr_t) (find.source) - (uintptr_t) (string.source);
string = (StringView) {string.source + offset, string.length - offset};
}
return count;
}
size_t StringView_FindStringOffset(StringView haystack, StringView needle)
{
size_t offset = 0;
size_t index = 0;
while (offset < haystack.length) {
while (index < needle.length && needle.source[index] == haystack.source[offset + index]) {
index++;
}
if (index == needle.length) {
return offset;
}
offset++;
}
return SIZE_MAX;
}
StringView StringView_FindString(StringView haystack, StringView needle)
{
size_t offset = 0;
size_t index = 0;
while (offset < haystack.length) {
while (index < needle.length && needle.source[index] == haystack.source[offset + index]) {
index++;
}
if (index == needle.length) {
return (StringView) {haystack.source + offset, needle.length};
}
offset++;
}
return (StringView) {NULL, 0};
}
StringView StringView_Slice(StringView string, size_t start, size_t end)
{
if (end > string.length || start > string.length) {
return (StringView) {NULL, 0};
}
StringView slice = {
.length = end - start,
.source = string.source + start,
};
return slice;
}
bool StringView_NextSplit(StringView* dest, StringView* source, StringView delim)
{
if (source->length == 0) return false;
size_t offset = StringView_FindStringOffset(*source, delim);
if (offset == SIZE_MAX) {
// No more delimiters, return entire source
offset = source->length;
}
*dest = StringView_Slice(*source, 0, offset);
*source = StringView_Slice(*source, offset + delim.length, source->length);
return true;
}
StringView StringView_StripLeft(StringView sv, StringView strip)
{
while (StringView_StartsWith(sv, strip)) {
sv = StringView_Slice(sv, strip.length, sv.length);
}
return sv;
}
StringView StringView_StripRight(StringView sv, StringView strip)
{
while (StringView_EndsWith(sv, strip)) {
sv = StringView_Slice(sv, 0, sv.length - strip.length);
}
return sv;
}
void StringView_Paste(char* destination, StringView source)
{
memcpy(destination, source.source, source.length);
}
int StringView_ParseInt(StringView source)
{
int value = 0;
int sign = 1;
switch (source.source[0]) {
case '+':
sign = 1;
source.source++;
source.length--;
break;
case '-':
sign = -1;
source.source++;
source.length--;
break;
default:
break;
}
while (source.length && isdigit(source.source[0])) {
int digit = source.source[0] - '0';
value *= 10;
value += digit;
source.source++;
source.length--;
}
return value * sign;
}