| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225 | /* * Copyright (C)2005-2018 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. */@:coreApiclass 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);	}	public function indexOf( str : String, ?startIndex : Int ) : Int {		var startByte = 0;		if( startIndex != null && startIndex > 0 ) {			if( startIndex >= length )				return -1;			startByte = startIndex << 1;		}		var p = bytes.find(startByte, (length << 1) - startByte, str.bytes, 0, str.length << 1);		if( p > 0 ) p >>= 1;		return p;	}	public function lastIndexOf( str : String, ?startIndex : Int ) : Int {		var last = 0;		var start = this.length;		if( startIndex != null )			start = startIndex;		start <<= 1;		while( true ) {			var p = bytes.find(last, (length << 1) - last, str.bytes, 0, str.length << 1);			if( p < 0 || p > start )				return (last >> 1) - 1;			last = p + 2;		}		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 = bytes.find(pos << 1, (length - pos) << 1, delimiter.bytes, 0, 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( s : String ) : Int {		var v = bytes.compare(0, s.bytes, 0, (length < s.length ? length : s.length) << 1);		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);	}}
 |