123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256 |
- /*
- * Copyright (C)2005-2019 Haxe Foundation
- *
- * Permission is hereby granted, free of charge, to any person obtaining a
- * copy of this software and associated documentation files (the "Software"),
- * to deal in the Software without restriction, including without limitation
- * the rights to use, copy, modify, merge, publish, distribute, sublicense,
- * and/or sell copies of the Software, and to permit persons to whom the
- * Software is furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in
- * all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
- * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
- * DEALINGS IN THE SOFTWARE.
- */
- @:coreApi
- class String {
- var bytes:hl.Bytes;
- public var length(default, null):Int;
- public function new(string:String):Void {
- bytes = string.bytes;
- length = string.length;
- }
- public function toUpperCase():String {
- return __alloc__(@:privateAccess bytes.ucs2Upper(0, length), length);
- }
- public function toLowerCase():String {
- return __alloc__(@:privateAccess bytes.ucs2Lower(0, length), length);
- }
- public function charAt(index:Int):String {
- if ((index : UInt) >= (length : UInt))
- return "";
- var b = new hl.Bytes(4);
- b.setUI16(0, bytes.getUI16(index << 1));
- b.setUI16(2, 0);
- return __alloc__(b, 1);
- }
- public function charCodeAt(index:Int):Null<Int> {
- var idx:UInt = index;
- if (idx >= (length : UInt))
- return null;
- return bytes.getUI16(index << 1);
- }
- inline function findChar(start:Int, len:Int, src:hl.Bytes, srcLen:Int):Int {
- var p = 0;
- while (true) {
- p = bytes.find(start, len - start, src, 0, srcLen);
- if (p < 0 || p & 1 == 0)
- break;
- start = p + 1;
- }
- return p;
- }
- public function indexOf(str:String, ?startIndex:Int):Int {
- var startByte = 0;
- if (startIndex != null && startIndex > 0) {
- if (startIndex >= length)
- return str == '' ? length : -1;
- startByte = startIndex << 1;
- }
- var p = findChar(startByte, length << 1, str.bytes, str.length << 1);
- if (p > 0)
- p >>= 1;
- return p;
- }
- public function lastIndexOf(str:String, ?startIndex:Int):Int {
- var max = this.length;
- if (startIndex != null) {
- max = startIndex + str.length;
- if (max < 0)
- max = 0;
- if (max > this.length)
- max = this.length;
- }
- var pos = max - str.length;
- var slen = str.length << 1;
- while (pos >= 0) {
- if (bytes.compare(pos << 1, str.bytes, 0, slen) == 0)
- return pos;
- pos--;
- }
- return -1;
- }
- public function split(delimiter:String):Array<String> {
- var out = [];
- if (length == 0) {
- out.push("");
- return out;
- }
- if (delimiter.length == 0) {
- for (i in 0...length)
- out.push(substr(i, 1));
- return out;
- }
- var pos = 0;
- var dlen = delimiter.length;
- while (true) {
- var p = findChar(pos << 1, length << 1, delimiter.bytes, dlen << 1);
- if (p < 0) {
- out.push(substr(pos, length - pos));
- break;
- }
- p >>= 1;
- out.push(substr(pos, p - pos));
- pos = p + dlen;
- }
- return out;
- }
- public function substr(pos:Int, ?len:Int):String@:privateAccess {
- var sl = length;
- var len:Int = if (len == null) sl else len;
- if (len == 0)
- return "";
- if (pos != 0 && len < 0)
- return "";
- if (pos < 0) {
- pos = sl + pos;
- if (pos < 0)
- pos = 0;
- } else if (len < 0) {
- len = sl + len - pos;
- if (len < 0)
- return "";
- }
- if (((pos + len) : UInt) > (sl : UInt))
- len = sl - pos;
- if (pos < 0 || len <= 0)
- return "";
- var b = new hl.Bytes((len + 1) << 1);
- b.blit(0, bytes, pos << 1, len << 1);
- b.setUI16(len << 1, 0);
- return __alloc__(b, len);
- }
- public function substring(startIndex:Int, ?endIndex:Int):String {
- var end:Int;
- if (endIndex == null)
- end = length;
- else {
- end = endIndex;
- if (end < 0)
- end = 0;
- else if (end > length)
- end = length;
- }
- if (startIndex < 0)
- startIndex = 0;
- else if (startIndex > length)
- startIndex = length;
- if (startIndex > end) {
- var tmp = startIndex;
- startIndex = end;
- end = tmp;
- }
- return substr(startIndex, end - startIndex);
- }
- public function toString():String {
- return this;
- }
- public static function fromCharCode(code:Int):String {
- if (code >= 0 && code < 0x10000) {
- if (code >= 0xD800 && code <= 0xDFFF)
- throw "Invalid unicode char " + code;
- var b = new hl.Bytes(4);
- b.setUI16(0, code);
- b.setUI16(2, 0);
- return __alloc__(b, 1);
- } else if (code < 0x110000) {
- var b = new hl.Bytes(6);
- code -= 0x10000;
- b.setUI16(0, (code >> 10) + 0xD800);
- b.setUI16(2, (code & 1023) + 0xDC00);
- b.setUI16(4, 0);
- return __alloc__(b, 2); // UTF16 encoding but UCS2 API (same as JS)
- } else
- throw "Invalid unicode char " + code;
- }
- function toUtf8():hl.Bytes {
- return bytes.utf16ToUtf8(0, null);
- }
- @:keep function __string():hl.Bytes {
- return bytes;
- }
- @:keep function __compare(v:Dynamic):Int {
- var s = Std.downcast(v, String);
- if (s == null)
- return hl.Api.comparePointer(this, v);
- #if (hl_ver >= version("1.10.0"))
- var v = bytes.compare16(s.bytes, length < s.length ? length : s.length);
- #else
- var v = bytes.compare(0, s.bytes, 0, (length < s.length ? length : s.length) << 1);
- #end
- return v == 0 ? length - s.length : v;
- }
- @:keep static inline function __alloc__(b:hl.Bytes, length:Int):String {
- var s:String = untyped $new(String);
- s.bytes = b;
- s.length = length;
- return s;
- }
- @:keep static function call_toString(v:Dynamic):hl.Bytes {
- var s:String = v.toString();
- return s.bytes;
- }
- inline static function fromUCS2(b:hl.Bytes):String {
- var s:String = untyped $new(String);
- s.bytes = b;
- s.length = @:privateAccess b.ucs2Length(0);
- return s;
- }
- @:keep static function fromUTF8(b:hl.Bytes):String {
- var outLen = 0;
- var b2 = @:privateAccess b.utf8ToUtf16(0, outLen);
- return __alloc__(b2, outLen >> 1);
- }
- @:keep static function __add__(a:String, b:String):String {
- if (a == null)
- a = "null";
- if (b == null)
- b = "null";
- var asize = a.length << 1, bsize = b.length << 1, tot = asize + bsize;
- var bytes = new hl.Bytes(tot + 2);
- bytes.blit(0, a.bytes, 0, asize);
- bytes.blit(asize, b.bytes, 0, bsize);
- bytes.setUI16(tot, 0);
- return __alloc__(bytes, tot >> 1);
- }
- }
|