Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(220)

Unified Diff: sdk/lib/async/string_transform.dart

Issue 11783009: Big merge from experimental to bleeding edge. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Created 7 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « sdk/lib/async/stream_pipe.dart ('k') | sdk/lib/async/timer.dart » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: sdk/lib/async/string_transform.dart
diff --git a/sdk/lib/async/string_transform.dart b/sdk/lib/async/string_transform.dart
new file mode 100644
index 0000000000000000000000000000000000000000..bd8f4cb3f497dad2897f83ace76e2d209f01b094
--- /dev/null
+++ b/sdk/lib/async/string_transform.dart
@@ -0,0 +1,127 @@
+// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+
+// part of dart.async;
+
+abstract class _StringDecoder extends _StreamTransformer<List<int>, String> {
+
+ handleData(List<int> bytes, StreamSink<String> sink) {
+ var data = _carry;
+ data.addAll(bytes);
+ _carry = [];
+ var buffer = new StringBuffer();
+ int pos = 0;
+ while (pos < data.length) {
+ int currentPos = pos;
+ int getNext() {
+ if (pos < data.length) {
+ return data[pos++];
+ }
+ return -1;
+ }
+ _chars = [];
+ if (_processByte(data[pos++], getNext)) {
+ _chars.forEach(buffer.addCharCode);
+ } else {
+ _carry = data.getRange(currentPos, data.length - currentPos);
+ break;
+ }
+ }
+ sink.add(buffer.toString());
+ }
+
+ void handleDone(StreamSink<String> sink) {
+ if (!_carry.isEmpty) {
+ sink.signalError(new AsyncError(
+ new StateError("Unhandled tailing utf8 chars")));
+ }
+ sink.close();
+ }
+
+ bool _processByte(int byte, int getNext());
+
+ void addChar(int char) {
+ _chars.add(char);
+ }
+
+ List<int> _carry = [];
+ List<int> _chars;
+}
+
+/**
+ * StringTransformer class that decodes a utf8 encoded bytes.
+ */
+class Utf8DecoderTransformer extends _StringDecoder {
+ bool _processByte(int byte, int getNext()) {
+ int value = byte & 0xFF;
+ if ((value & 0x80) == 0x80) {
+ int additionalBytes;
+ if ((value & 0xe0) == 0xc0) { // 110xxxxx
+ value = value & 0x1F;
+ additionalBytes = 1;
+ } else if ((value & 0xf0) == 0xe0) { // 1110xxxx
+ value = value & 0x0F;
+ additionalBytes = 2;
+ } else { // 11110xxx
+ value = value & 0x07;
+ additionalBytes = 3;
+ }
+ for (int i = 0; i < additionalBytes; i++) {
+ int next = getNext();
+ if (next < 0) return false;
+ value = value << 6 | (next & 0x3F);
+ }
+ }
+ addChar(value);
+ return true;
+ }
+}
+
+
+abstract class _StringEncoder extends _StreamTransformer<String, List<int>> {
+ handleData(String string, StreamSink<List<int>> sink) {
+ sink.add(_processString(string));
+ }
+
+ List<int> _processString(String string);
+}
+
+/**
+ * StringTransformer class that utf8 encodes a string.
+ */
+class Utf8EncoderTransformer extends _StringEncoder {
+ List<int> _processString(String string) {
+ var bytes = [];
+ int pos = 0;
+ int length = string.length;
+ for (int i = 0; i < length; i++) {
+ int additionalBytes;
+ int charCode = string.charCodeAt(i);
+ if (charCode <= 0x007F) {
+ additionalBytes = 0;
+ bytes.add(charCode);
+ } else if (charCode <= 0x07FF) {
+ // 110xxxxx (xxxxx is top 5 bits).
+ bytes.add(((charCode >> 6) & 0x1F) | 0xC0);
+ additionalBytes = 1;
+ } else if (charCode <= 0xFFFF) {
+ // 1110xxxx (xxxx is top 4 bits)
+ bytes.add(((charCode >> 12) & 0x0F)| 0xE0);
+ additionalBytes = 2;
+ } else {
+ // 11110xxx (xxx is top 3 bits)
+ bytes.add(((charCode >> 18) & 0x07) | 0xF0);
+ additionalBytes = 3;
+ }
+ for (int i = additionalBytes; i > 0; i--) {
+ // 10xxxxxx (xxxxxx is next 6 bits from the top).
+ bytes.add(((charCode >> (6 * (i - 1))) & 0x3F) | 0x80);
+ }
+ pos += additionalBytes + 1;
+ }
+ return bytes;
+ }
+}
+
+
« no previous file with comments | « sdk/lib/async/stream_pipe.dart ('k') | sdk/lib/async/timer.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698