2024-06-01 18:18:38 +08:00

234 lines
6.6 KiB
C++

#pragma once
#include "common.h"
#include "memory.h"
#include "vector.h"
namespace pkpy {
int utf8len(unsigned char c, bool suppress=false);
struct SStream;
struct Str{
int size;
bool is_ascii;
char* data;
char _inlined[16];
bool is_inlined() const { return data == _inlined; }
Str();
Str(int size, bool is_ascii);
Str(const std::string& s);
Str(std::string_view s);
Str(const char* s);
Str(const char* s, int len);
Str(std::pair<char *, int>);
Str(const Str& other);
Str(Str&& other);
operator std::string_view() const { return sv(); }
const char* begin() const { return data; }
const char* end() const { return data + size; }
char operator[](int idx) const { return data[idx]; }
int length() const { return size; }
bool empty() const { return size == 0; }
size_t hash() const{ return std::hash<std::string_view>()(sv()); }
Str& operator=(const Str&);
Str operator+(const Str&) const;
Str operator+(const char*) const;
friend Str operator+(const char*, const Str&);
bool operator==(const std::string_view other) const;
bool operator!=(const std::string_view other) const;
bool operator<(const std::string_view other) const;
friend bool operator<(const std::string_view other, const Str& str);
bool operator==(const char* p) const;
bool operator!=(const char* p) const;
bool operator==(const Str& other) const;
bool operator!=(const Str& other) const;
bool operator<(const Str& other) const;
bool operator>(const Str& other) const;
bool operator<=(const Str& other) const;
bool operator>=(const Str& other) const;
~Str();
friend std::ostream& operator<<(std::ostream& os, const Str& str);
const char* c_str() const { return data; }
std::string_view sv() const { return std::string_view(data, size); }
std::string str() const { return std::string(data, size); }
Str substr(int start, int len) const;
Str substr(int start) const;
Str strip(bool left, bool right, const Str& chars) const;
Str strip(bool left=true, bool right=true) const;
Str lstrip() const { return strip(true, false); }
Str rstrip() const { return strip(false, true); }
Str lower() const;
Str upper() const;
Str escape(bool single_quote=true) const;
void escape_(SStream& ss, bool single_quote=true) const;
int index(const Str& sub, int start=0) const;
Str replace(char old, char new_) const;
Str replace(const Str& old, const Str& new_, int count=-1) const;
vector<std::string_view> split(const Str& sep) const;
vector<std::string_view> split(char sep) const;
int count(const Str& sub) const;
/*************unicode*************/
int _unicode_index_to_byte(int i) const;
int _byte_index_to_unicode(int n) const;
Str u8_getitem(int i) const;
Str u8_slice(int start, int stop, int step) const;
int u8_length() const;
};
struct StrName {
uint16_t index;
StrName(): index(0) {}
explicit StrName(uint16_t index): index(index) {}
StrName(const char* s): index(get(s).index) {}
StrName(const Str& s): index(get(s.sv()).index) {}
std::string_view sv() const { return _r_interned()[index];}
const char* c_str() const { return _r_interned()[index].c_str(); }
bool empty() const { return index == 0; }
Str escape() const { return Str(sv()).escape(); }
bool operator==(const StrName& other) const noexcept {
return this->index == other.index;
}
bool operator!=(const StrName& other) const noexcept {
return this->index != other.index;
}
bool operator<(const StrName& other) const noexcept {
return sv() < other.sv();
}
bool operator>(const StrName& other) const noexcept {
return sv() > other.sv();
}
static StrName get(std::string_view s);
static std::map<std::string_view, uint16_t>& _interned();
static std::map<uint16_t, std::string>& _r_interned();
static uint32_t _pesudo_random_index;
};
struct SStream{
PK_ALWAYS_PASS_BY_POINTER(SStream)
vector<char> buffer;
int _precision = -1;
bool empty() const { return buffer.empty(); }
void setprecision(int precision) { _precision = precision; }
SStream() {}
SStream(int guess_size) { buffer.reserve(guess_size);}
Str str();
SStream& operator<<(const Str&);
SStream& operator<<(const char*);
SStream& operator<<(int);
SStream& operator<<(size_t);
SStream& operator<<(i64);
SStream& operator<<(f64);
SStream& operator<<(const std::string&);
SStream& operator<<(std::string_view);
SStream& operator<<(char);
SStream& operator<<(StrName);
void write_hex(unsigned char, bool non_zero=false);
void write_hex(void*);
void write_hex(i64);
};
#ifdef _S
#undef _S
#endif
template<typename... Args>
Str _S(Args&&... args) {
SStream ss;
(ss << ... << args);
return ss.str();
}
struct CString{
const char* ptr;
CString(const char* ptr): ptr(ptr) {}
operator const char*() const { return ptr; }
};
// unary operators
extern const StrName __repr__;
extern const StrName __str__;
extern const StrName __hash__;
extern const StrName __len__;
extern const StrName __iter__;
extern const StrName __next__;
extern const StrName __neg__;
// logical operators
extern const StrName __eq__;
extern const StrName __lt__;
extern const StrName __le__;
extern const StrName __gt__;
extern const StrName __ge__;
extern const StrName __contains__;
// binary operators
extern const StrName __add__;
extern const StrName __radd__;
extern const StrName __sub__;
extern const StrName __rsub__;
extern const StrName __mul__;
extern const StrName __rmul__;
extern const StrName __truediv__;
extern const StrName __floordiv__;
extern const StrName __mod__;
extern const StrName __pow__;
extern const StrName __matmul__;
extern const StrName __lshift__;
extern const StrName __rshift__;
extern const StrName __and__;
extern const StrName __or__;
extern const StrName __xor__;
extern const StrName __invert__;
// indexer
extern const StrName __getitem__;
extern const StrName __setitem__;
extern const StrName __delitem__;
// specials
extern const StrName __new__;
extern const StrName __init__;
extern const StrName __call__;
extern const StrName __divmod__;
extern const StrName __enter__;
extern const StrName __exit__;
extern const StrName __name__;
extern const StrName __all__;
extern const StrName __package__;
extern const StrName __path__;
extern const StrName __class__;
extern const StrName __missing__;
extern const StrName pk_id_add;
extern const StrName pk_id_set;
extern const StrName pk_id_long;
extern const StrName pk_id_complex;
#define DEF_SNAME(name) const static StrName name(#name)
} // namespace pkpy