Browse Source

PDF.js version 1.7.391 - See mozilla/pdf.js@cd5acf501e9899f386bb7efc9feded7b2c6e763b

pdfjsbot 8 years ago
parent
commit
75759fc128
100 changed files with 46776 additions and 56017 deletions
  1. 1 1
      bower.json
  2. 897 1183
      build/pdf.combined.js
  3. 897 1183
      build/pdf.js
  4. 0 0
      build/pdf.min.js
  5. 897 1183
      build/pdf.worker.js
  6. 0 0
      build/pdf.worker.min.js
  7. 558 600
      lib/core/annotation.js
  8. 325 371
      lib/core/arithmetic_decoder.js
  9. 195 708
      lib/core/bidi.js
  10. 1 393
      lib/core/cff_parser.js
  11. 1 488
      lib/core/charsets.js
  12. 441 439
      lib/core/chunked_stream.js
  13. 1 170
      lib/core/cmap.js
  14. 878 964
      lib/core/colorspace.js
  15. 339 763
      lib/core/crypto.js
  16. 437 451
      lib/core/document.js
  17. 1 1794
      lib/core/encodings.js
  18. 2739 2837
      lib/core/evaluator.js
  19. 686 735
      lib/core/font_renderer.js
  20. 10 269
      lib/core/fonts.js
  21. 921 950
      lib/core/function.js
  22. 4525 4524
      lib/core/glyphlist.js
  23. 449 445
      lib/core/image.js
  24. 900 1051
      lib/core/jbig2.js
  25. 845 891
      lib/core/jpg.js
  26. 1848 2025
      lib/core/jpx.js
  27. 2937 2936
      lib/core/metrics.js
  28. 111 113
      lib/core/murmurhash3.js
  29. 450 449
      lib/core/network.js
  30. 1358 1375
      lib/core/obj.js
  31. 902 1139
      lib/core/parser.js
  32. 726 785
      lib/core/pattern.js
  33. 153 154
      lib/core/pdf_manager.js
  34. 179 175
      lib/core/primitives.js
  35. 174 173
      lib/core/ps_parser.js
  36. 618 617
      lib/core/standard_fonts.js
  37. 231 953
      lib/core/stream.js
  38. 508 505
      lib/core/type1_parser.js
  39. 1579 1701
      lib/core/unicode.js
  40. 613 623
      lib/core/worker.js
  41. 542 549
      lib/display/annotation_layer.js
  42. 1259 1261
      lib/display/api.js
  43. 1654 1788
      lib/display/canvas.js
  44. 191 201
      lib/display/dom_utils.js
  45. 236 237
      lib/display/font_loader.js
  46. 40 39
      lib/display/global.js
  47. 60 52
      lib/display/metadata.js
  48. 336 359
      lib/display/pattern_helper.js
  49. 922 966
      lib/display/svg.js
  50. 454 491
      lib/display/text_layer.js
  51. 274 297
      lib/display/webgl.js
  52. 3 2
      lib/pdf.js
  53. 4 3
      lib/pdf.worker.js
  54. 1338 1344
      lib/shared/compatibility.js
  55. 889 1175
      lib/shared/util.js
  56. 1025 1150
      lib/test/unit/annotation_spec.js
  57. 972 1074
      lib/test/unit/api_spec.js
  58. 15 14
      lib/test/unit/bidi_spec.js
  59. 217 455
      lib/test/unit/cff_parser_spec.js
  60. 185 184
      lib/test/unit/cmap_spec.js
  61. 486 1191
      lib/test/unit/crypto_spec.js
  62. 14 12
      lib/test/unit/document_spec.js
  63. 38 44
      lib/test/unit/dom_utils_spec.js
  64. 247 250
      lib/test/unit/evaluator_spec.js
  65. 48 46
      lib/test/unit/fonts_spec.js
  66. 596 1082
      lib/test/unit/function_spec.js
  67. 80 102
      lib/test/unit/jasmine-boot.js
  68. 7 6
      lib/test/unit/metadata_spec.js
  69. 45 49
      lib/test/unit/murmurhash3_spec.js
  70. 129 135
      lib/test/unit/network_spec.js
  71. 111 134
      lib/test/unit/parser_spec.js
  72. 292 335
      lib/test/unit/primitives_spec.js
  73. 37 52
      lib/test/unit/stream_spec.js
  74. 27 26
      lib/test/unit/test_utils.js
  75. 56 58
      lib/test/unit/testreporter.js
  76. 72 84
      lib/test/unit/type1_parser_spec.js
  77. 110 137
      lib/test/unit/ui_utils_spec.js
  78. 89 88
      lib/test/unit/unicode_spec.js
  79. 25 24
      lib/test/unit/util_spec.js
  80. 57 57
      lib/web/annotation_layer_builder.js
  81. 1356 1381
      lib/web/app.js
  82. 1 0
      lib/web/chromecom.js
  83. 530 533
      lib/web/debugger.js
  84. 97 96
      lib/web/dom_events.js
  85. 44 44
      lib/web/download_manager.js
  86. 58 64
      lib/web/firefox_print_service.js
  87. 1 0
      lib/web/firefoxcom.js
  88. 117 122
      lib/web/grab_to_pan.js
  89. 47 50
      lib/web/hand_tool.js
  90. 24 46
      lib/web/interfaces.js
  91. 80 79
      lib/web/overlay_manager.js
  92. 49 48
      lib/web/password_prompt.js
  93. 98 97
      lib/web/pdf_attachment_viewer.js
  94. 119 120
      lib/web/pdf_document_properties.js
  95. 146 145
      lib/web/pdf_find_bar.js
  96. 344 343
      lib/web/pdf_find_controller.js
  97. 282 281
      lib/web/pdf_history.js
  98. 292 318
      lib/web/pdf_link_service.js
  99. 126 124
      lib/web/pdf_outline_viewer.js
  100. 452 457
      lib/web/pdf_page_view.js

+ 1 - 1
bower.json

@@ -1,6 +1,6 @@
 {
   "name": "pdfjs-dist",
-  "version": "1.7.389",
+  "version": "1.7.391",
   "main": [
     "build/pdf.js",
     "build/pdf.worker.js"

File diff suppressed because it is too large
+ 897 - 1183
build/pdf.combined.js


File diff suppressed because it is too large
+ 897 - 1183
build/pdf.js


File diff suppressed because it is too large
+ 0 - 0
build/pdf.min.js


File diff suppressed because it is too large
+ 897 - 1183
build/pdf.worker.js


File diff suppressed because it is too large
+ 0 - 0
build/pdf.worker.min.js


+ 558 - 600
lib/core/annotation.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var coreStream = require('./stream.js');
@@ -41,650 +42,607 @@ var Catalog = coreObj.Catalog;
 var ObjectLoader = coreObj.ObjectLoader;
 var FileSpec = coreObj.FileSpec;
 var OperatorList = coreEvaluator.OperatorList;
-function AnnotationFactory() {
-}
+function AnnotationFactory() {}
 AnnotationFactory.prototype = {
- create: function AnnotationFactory_create(xref, ref, pdfManager, idFactory) {
-  var dict = xref.fetchIfRef(ref);
-  if (!isDict(dict)) {
-   return;
-  }
-  var id = isRef(ref) ? ref.toString() : 'annot_' + idFactory.createObjId();
-  var subtype = dict.get('Subtype');
-  subtype = isName(subtype) ? subtype.name : null;
-  var parameters = {
-   xref: xref,
-   dict: dict,
-   ref: isRef(ref) ? ref : null,
-   subtype: subtype,
-   id: id,
-   pdfManager: pdfManager
-  };
-  switch (subtype) {
-  case 'Link':
-   return new LinkAnnotation(parameters);
-  case 'Text':
-   return new TextAnnotation(parameters);
-  case 'Widget':
-   var fieldType = Util.getInheritableProperty(dict, 'FT');
-   fieldType = isName(fieldType) ? fieldType.name : null;
-   switch (fieldType) {
-   case 'Tx':
-    return new TextWidgetAnnotation(parameters);
-   case 'Btn':
-    return new ButtonWidgetAnnotation(parameters);
-   case 'Ch':
-    return new ChoiceWidgetAnnotation(parameters);
-   }
-   warn('Unimplemented widget field type "' + fieldType + '", ' + 'falling back to base field type.');
-   return new WidgetAnnotation(parameters);
-  case 'Popup':
-   return new PopupAnnotation(parameters);
-  case 'Highlight':
-   return new HighlightAnnotation(parameters);
-  case 'Underline':
-   return new UnderlineAnnotation(parameters);
-  case 'Squiggly':
-   return new SquigglyAnnotation(parameters);
-  case 'StrikeOut':
-   return new StrikeOutAnnotation(parameters);
-  case 'FileAttachment':
-   return new FileAttachmentAnnotation(parameters);
-  default:
-   if (!subtype) {
-    warn('Annotation is missing the required /Subtype.');
-   } else {
-    warn('Unimplemented annotation type "' + subtype + '", ' + 'falling back to base annotation.');
-   }
-   return new Annotation(parameters);
+  create: function AnnotationFactory_create(xref, ref, pdfManager, idFactory) {
+    var dict = xref.fetchIfRef(ref);
+    if (!isDict(dict)) {
+      return;
+    }
+    var id = isRef(ref) ? ref.toString() : 'annot_' + idFactory.createObjId();
+    var subtype = dict.get('Subtype');
+    subtype = isName(subtype) ? subtype.name : null;
+    var parameters = {
+      xref: xref,
+      dict: dict,
+      ref: isRef(ref) ? ref : null,
+      subtype: subtype,
+      id: id,
+      pdfManager: pdfManager
+    };
+    switch (subtype) {
+      case 'Link':
+        return new LinkAnnotation(parameters);
+      case 'Text':
+        return new TextAnnotation(parameters);
+      case 'Widget':
+        var fieldType = Util.getInheritableProperty(dict, 'FT');
+        fieldType = isName(fieldType) ? fieldType.name : null;
+        switch (fieldType) {
+          case 'Tx':
+            return new TextWidgetAnnotation(parameters);
+          case 'Btn':
+            return new ButtonWidgetAnnotation(parameters);
+          case 'Ch':
+            return new ChoiceWidgetAnnotation(parameters);
+        }
+        warn('Unimplemented widget field type "' + fieldType + '", ' + 'falling back to base field type.');
+        return new WidgetAnnotation(parameters);
+      case 'Popup':
+        return new PopupAnnotation(parameters);
+      case 'Highlight':
+        return new HighlightAnnotation(parameters);
+      case 'Underline':
+        return new UnderlineAnnotation(parameters);
+      case 'Squiggly':
+        return new SquigglyAnnotation(parameters);
+      case 'StrikeOut':
+        return new StrikeOutAnnotation(parameters);
+      case 'FileAttachment':
+        return new FileAttachmentAnnotation(parameters);
+      default:
+        if (!subtype) {
+          warn('Annotation is missing the required /Subtype.');
+        } else {
+          warn('Unimplemented annotation type "' + subtype + '", ' + 'falling back to base annotation.');
+        }
+        return new Annotation(parameters);
+    }
   }
- }
 };
 var Annotation = function AnnotationClosure() {
- function getTransformMatrix(rect, bbox, matrix) {
-  var bounds = Util.getAxialAlignedBoundingBox(bbox, matrix);
-  var minX = bounds[0];
-  var minY = bounds[1];
-  var maxX = bounds[2];
-  var maxY = bounds[3];
-  if (minX === maxX || minY === maxY) {
-   return [
-    1,
-    0,
-    0,
-    1,
-    rect[0],
-    rect[1]
-   ];
-  }
-  var xRatio = (rect[2] - rect[0]) / (maxX - minX);
-  var yRatio = (rect[3] - rect[1]) / (maxY - minY);
-  return [
-   xRatio,
-   0,
-   0,
-   yRatio,
-   rect[0] - minX * xRatio,
-   rect[1] - minY * yRatio
-  ];
- }
- function Annotation(params) {
-  var dict = params.dict;
-  this.setFlags(dict.get('F'));
-  this.setRectangle(dict.getArray('Rect'));
-  this.setColor(dict.getArray('C'));
-  this.setBorderStyle(dict);
-  this.setAppearance(dict);
-  this.data = {};
-  this.data.id = params.id;
-  this.data.subtype = params.subtype;
-  this.data.annotationFlags = this.flags;
-  this.data.rect = this.rectangle;
-  this.data.color = this.color;
-  this.data.borderStyle = this.borderStyle;
-  this.data.hasAppearance = !!this.appearance;
- }
- Annotation.prototype = {
-  _hasFlag: function Annotation_hasFlag(flags, flag) {
-   return !!(flags & flag);
-  },
-  _isViewable: function Annotation_isViewable(flags) {
-   return !this._hasFlag(flags, AnnotationFlag.INVISIBLE) && !this._hasFlag(flags, AnnotationFlag.HIDDEN) && !this._hasFlag(flags, AnnotationFlag.NOVIEW);
-  },
-  _isPrintable: function AnnotationFlag_isPrintable(flags) {
-   return this._hasFlag(flags, AnnotationFlag.PRINT) && !this._hasFlag(flags, AnnotationFlag.INVISIBLE) && !this._hasFlag(flags, AnnotationFlag.HIDDEN);
-  },
-  get viewable() {
-   if (this.flags === 0) {
-    return true;
-   }
-   return this._isViewable(this.flags);
-  },
-  get printable() {
-   if (this.flags === 0) {
-    return false;
-   }
-   return this._isPrintable(this.flags);
-  },
-  setFlags: function Annotation_setFlags(flags) {
-   this.flags = isInt(flags) && flags > 0 ? flags : 0;
-  },
-  hasFlag: function Annotation_hasFlag(flag) {
-   return this._hasFlag(this.flags, flag);
-  },
-  setRectangle: function Annotation_setRectangle(rectangle) {
-   if (isArray(rectangle) && rectangle.length === 4) {
-    this.rectangle = Util.normalizeRect(rectangle);
-   } else {
-    this.rectangle = [
-     0,
-     0,
-     0,
-     0
-    ];
-   }
-  },
-  setColor: function Annotation_setColor(color) {
-   var rgbColor = new Uint8Array(3);
-   if (!isArray(color)) {
-    this.color = rgbColor;
-    return;
-   }
-   switch (color.length) {
-   case 0:
-    this.color = null;
-    break;
-   case 1:
-    ColorSpace.singletons.gray.getRgbItem(color, 0, rgbColor, 0);
-    this.color = rgbColor;
-    break;
-   case 3:
-    ColorSpace.singletons.rgb.getRgbItem(color, 0, rgbColor, 0);
-    this.color = rgbColor;
-    break;
-   case 4:
-    ColorSpace.singletons.cmyk.getRgbItem(color, 0, rgbColor, 0);
-    this.color = rgbColor;
-    break;
-   default:
-    this.color = rgbColor;
-    break;
-   }
-  },
-  setBorderStyle: function Annotation_setBorderStyle(borderStyle) {
-   this.borderStyle = new AnnotationBorderStyle();
-   if (!isDict(borderStyle)) {
-    return;
-   }
-   if (borderStyle.has('BS')) {
-    var dict = borderStyle.get('BS');
-    var dictType = dict.get('Type');
-    if (!dictType || isName(dictType, 'Border')) {
-     this.borderStyle.setWidth(dict.get('W'));
-     this.borderStyle.setStyle(dict.get('S'));
-     this.borderStyle.setDashArray(dict.getArray('D'));
+  function getTransformMatrix(rect, bbox, matrix) {
+    var bounds = Util.getAxialAlignedBoundingBox(bbox, matrix);
+    var minX = bounds[0];
+    var minY = bounds[1];
+    var maxX = bounds[2];
+    var maxY = bounds[3];
+    if (minX === maxX || minY === maxY) {
+      return [1, 0, 0, 1, rect[0], rect[1]];
     }
-   } else if (borderStyle.has('Border')) {
-    var array = borderStyle.getArray('Border');
-    if (isArray(array) && array.length >= 3) {
-     this.borderStyle.setHorizontalCornerRadius(array[0]);
-     this.borderStyle.setVerticalCornerRadius(array[1]);
-     this.borderStyle.setWidth(array[2]);
-     if (array.length === 4) {
-      this.borderStyle.setDashArray(array[3]);
-     }
-    }
-   } else {
-    this.borderStyle.setWidth(0);
-   }
-  },
-  setAppearance: function Annotation_setAppearance(dict) {
-   this.appearance = null;
-   var appearanceStates = dict.get('AP');
-   if (!isDict(appearanceStates)) {
-    return;
-   }
-   var normalAppearanceState = appearanceStates.get('N');
-   if (isStream(normalAppearanceState)) {
-    this.appearance = normalAppearanceState;
-    return;
-   }
-   if (!isDict(normalAppearanceState)) {
-    return;
-   }
-   var as = dict.get('AS');
-   if (!isName(as) || !normalAppearanceState.has(as.name)) {
-    return;
-   }
-   this.appearance = normalAppearanceState.get(as.name);
-  },
-  _preparePopup: function Annotation_preparePopup(dict) {
-   if (!dict.has('C')) {
-    this.data.color = null;
-   }
-   this.data.hasPopup = dict.has('Popup');
-   this.data.title = stringToPDFString(dict.get('T') || '');
-   this.data.contents = stringToPDFString(dict.get('Contents') || '');
-  },
-  loadResources: function Annotation_loadResources(keys) {
-   return new Promise(function (resolve, reject) {
-    this.appearance.dict.getAsync('Resources').then(function (resources) {
-     if (!resources) {
-      resolve();
-      return;
-     }
-     var objectLoader = new ObjectLoader(resources.map, keys, resources.xref);
-     objectLoader.load().then(function () {
-      resolve(resources);
-     }, reject);
-    }, reject);
-   }.bind(this));
-  },
-  getOperatorList: function Annotation_getOperatorList(evaluator, task, renderForms) {
-   if (!this.appearance) {
-    return Promise.resolve(new OperatorList());
-   }
-   var data = this.data;
-   var appearanceDict = this.appearance.dict;
-   var resourcesPromise = this.loadResources([
-    'ExtGState',
-    'ColorSpace',
-    'Pattern',
-    'Shading',
-    'XObject',
-    'Font'
-   ]);
-   var bbox = appearanceDict.getArray('BBox') || [
-    0,
-    0,
-    1,
-    1
-   ];
-   var matrix = appearanceDict.getArray('Matrix') || [
-    1,
-    0,
-    0,
-    1,
-    0,
-    0
-   ];
-   var transform = getTransformMatrix(data.rect, bbox, matrix);
-   var self = this;
-   return resourcesPromise.then(function (resources) {
-    var opList = new OperatorList();
-    opList.addOp(OPS.beginAnnotation, [
-     data.rect,
-     transform,
-     matrix
-    ]);
-    return evaluator.getOperatorList(self.appearance, task, resources, opList).then(function () {
-     opList.addOp(OPS.endAnnotation, []);
-     self.appearance.reset();
-     return opList;
-    });
-   });
+    var xRatio = (rect[2] - rect[0]) / (maxX - minX);
+    var yRatio = (rect[3] - rect[1]) / (maxY - minY);
+    return [xRatio, 0, 0, yRatio, rect[0] - minX * xRatio, rect[1] - minY * yRatio];
+  }
+  function Annotation(params) {
+    var dict = params.dict;
+    this.setFlags(dict.get('F'));
+    this.setRectangle(dict.getArray('Rect'));
+    this.setColor(dict.getArray('C'));
+    this.setBorderStyle(dict);
+    this.setAppearance(dict);
+    this.data = {};
+    this.data.id = params.id;
+    this.data.subtype = params.subtype;
+    this.data.annotationFlags = this.flags;
+    this.data.rect = this.rectangle;
+    this.data.color = this.color;
+    this.data.borderStyle = this.borderStyle;
+    this.data.hasAppearance = !!this.appearance;
   }
- };
- return Annotation;
+  Annotation.prototype = {
+    _hasFlag: function Annotation_hasFlag(flags, flag) {
+      return !!(flags & flag);
+    },
+    _isViewable: function Annotation_isViewable(flags) {
+      return !this._hasFlag(flags, AnnotationFlag.INVISIBLE) && !this._hasFlag(flags, AnnotationFlag.HIDDEN) && !this._hasFlag(flags, AnnotationFlag.NOVIEW);
+    },
+    _isPrintable: function AnnotationFlag_isPrintable(flags) {
+      return this._hasFlag(flags, AnnotationFlag.PRINT) && !this._hasFlag(flags, AnnotationFlag.INVISIBLE) && !this._hasFlag(flags, AnnotationFlag.HIDDEN);
+    },
+    get viewable() {
+      if (this.flags === 0) {
+        return true;
+      }
+      return this._isViewable(this.flags);
+    },
+    get printable() {
+      if (this.flags === 0) {
+        return false;
+      }
+      return this._isPrintable(this.flags);
+    },
+    setFlags: function Annotation_setFlags(flags) {
+      this.flags = isInt(flags) && flags > 0 ? flags : 0;
+    },
+    hasFlag: function Annotation_hasFlag(flag) {
+      return this._hasFlag(this.flags, flag);
+    },
+    setRectangle: function Annotation_setRectangle(rectangle) {
+      if (isArray(rectangle) && rectangle.length === 4) {
+        this.rectangle = Util.normalizeRect(rectangle);
+      } else {
+        this.rectangle = [0, 0, 0, 0];
+      }
+    },
+    setColor: function Annotation_setColor(color) {
+      var rgbColor = new Uint8Array(3);
+      if (!isArray(color)) {
+        this.color = rgbColor;
+        return;
+      }
+      switch (color.length) {
+        case 0:
+          this.color = null;
+          break;
+        case 1:
+          ColorSpace.singletons.gray.getRgbItem(color, 0, rgbColor, 0);
+          this.color = rgbColor;
+          break;
+        case 3:
+          ColorSpace.singletons.rgb.getRgbItem(color, 0, rgbColor, 0);
+          this.color = rgbColor;
+          break;
+        case 4:
+          ColorSpace.singletons.cmyk.getRgbItem(color, 0, rgbColor, 0);
+          this.color = rgbColor;
+          break;
+        default:
+          this.color = rgbColor;
+          break;
+      }
+    },
+    setBorderStyle: function Annotation_setBorderStyle(borderStyle) {
+      this.borderStyle = new AnnotationBorderStyle();
+      if (!isDict(borderStyle)) {
+        return;
+      }
+      if (borderStyle.has('BS')) {
+        var dict = borderStyle.get('BS');
+        var dictType = dict.get('Type');
+        if (!dictType || isName(dictType, 'Border')) {
+          this.borderStyle.setWidth(dict.get('W'));
+          this.borderStyle.setStyle(dict.get('S'));
+          this.borderStyle.setDashArray(dict.getArray('D'));
+        }
+      } else if (borderStyle.has('Border')) {
+        var array = borderStyle.getArray('Border');
+        if (isArray(array) && array.length >= 3) {
+          this.borderStyle.setHorizontalCornerRadius(array[0]);
+          this.borderStyle.setVerticalCornerRadius(array[1]);
+          this.borderStyle.setWidth(array[2]);
+          if (array.length === 4) {
+            this.borderStyle.setDashArray(array[3]);
+          }
+        }
+      } else {
+        this.borderStyle.setWidth(0);
+      }
+    },
+    setAppearance: function Annotation_setAppearance(dict) {
+      this.appearance = null;
+      var appearanceStates = dict.get('AP');
+      if (!isDict(appearanceStates)) {
+        return;
+      }
+      var normalAppearanceState = appearanceStates.get('N');
+      if (isStream(normalAppearanceState)) {
+        this.appearance = normalAppearanceState;
+        return;
+      }
+      if (!isDict(normalAppearanceState)) {
+        return;
+      }
+      var as = dict.get('AS');
+      if (!isName(as) || !normalAppearanceState.has(as.name)) {
+        return;
+      }
+      this.appearance = normalAppearanceState.get(as.name);
+    },
+    _preparePopup: function Annotation_preparePopup(dict) {
+      if (!dict.has('C')) {
+        this.data.color = null;
+      }
+      this.data.hasPopup = dict.has('Popup');
+      this.data.title = stringToPDFString(dict.get('T') || '');
+      this.data.contents = stringToPDFString(dict.get('Contents') || '');
+    },
+    loadResources: function Annotation_loadResources(keys) {
+      return new Promise(function (resolve, reject) {
+        this.appearance.dict.getAsync('Resources').then(function (resources) {
+          if (!resources) {
+            resolve();
+            return;
+          }
+          var objectLoader = new ObjectLoader(resources.map, keys, resources.xref);
+          objectLoader.load().then(function () {
+            resolve(resources);
+          }, reject);
+        }, reject);
+      }.bind(this));
+    },
+    getOperatorList: function Annotation_getOperatorList(evaluator, task, renderForms) {
+      if (!this.appearance) {
+        return Promise.resolve(new OperatorList());
+      }
+      var data = this.data;
+      var appearanceDict = this.appearance.dict;
+      var resourcesPromise = this.loadResources(['ExtGState', 'ColorSpace', 'Pattern', 'Shading', 'XObject', 'Font']);
+      var bbox = appearanceDict.getArray('BBox') || [0, 0, 1, 1];
+      var matrix = appearanceDict.getArray('Matrix') || [1, 0, 0, 1, 0, 0];
+      var transform = getTransformMatrix(data.rect, bbox, matrix);
+      var self = this;
+      return resourcesPromise.then(function (resources) {
+        var opList = new OperatorList();
+        opList.addOp(OPS.beginAnnotation, [data.rect, transform, matrix]);
+        return evaluator.getOperatorList(self.appearance, task, resources, opList).then(function () {
+          opList.addOp(OPS.endAnnotation, []);
+          self.appearance.reset();
+          return opList;
+        });
+      });
+    }
+  };
+  return Annotation;
 }();
 var AnnotationBorderStyle = function AnnotationBorderStyleClosure() {
- function AnnotationBorderStyle() {
-  this.width = 1;
-  this.style = AnnotationBorderStyleType.SOLID;
-  this.dashArray = [3];
-  this.horizontalCornerRadius = 0;
-  this.verticalCornerRadius = 0;
- }
- AnnotationBorderStyle.prototype = {
-  setWidth: function AnnotationBorderStyle_setWidth(width) {
-   if (width === (width | 0)) {
-    this.width = width;
-   }
-  },
-  setStyle: function AnnotationBorderStyle_setStyle(style) {
-   if (!style) {
-    return;
-   }
-   switch (style.name) {
-   case 'S':
+  function AnnotationBorderStyle() {
+    this.width = 1;
     this.style = AnnotationBorderStyleType.SOLID;
-    break;
-   case 'D':
-    this.style = AnnotationBorderStyleType.DASHED;
-    break;
-   case 'B':
-    this.style = AnnotationBorderStyleType.BEVELED;
-    break;
-   case 'I':
-    this.style = AnnotationBorderStyleType.INSET;
-    break;
-   case 'U':
-    this.style = AnnotationBorderStyleType.UNDERLINE;
-    break;
-   default:
-    break;
-   }
-  },
-  setDashArray: function AnnotationBorderStyle_setDashArray(dashArray) {
-   if (isArray(dashArray) && dashArray.length > 0) {
-    var isValid = true;
-    var allZeros = true;
-    for (var i = 0, len = dashArray.length; i < len; i++) {
-     var element = dashArray[i];
-     var validNumber = +element >= 0;
-     if (!validNumber) {
-      isValid = false;
-      break;
-     } else if (element > 0) {
-      allZeros = false;
-     }
-    }
-    if (isValid && !allZeros) {
-     this.dashArray = dashArray;
-    } else {
-     this.width = 0;
-    }
-   } else if (dashArray) {
-    this.width = 0;
-   }
-  },
-  setHorizontalCornerRadius: function AnnotationBorderStyle_setHorizontalCornerRadius(radius) {
-   if (radius === (radius | 0)) {
-    this.horizontalCornerRadius = radius;
-   }
-  },
-  setVerticalCornerRadius: function AnnotationBorderStyle_setVerticalCornerRadius(radius) {
-   if (radius === (radius | 0)) {
-    this.verticalCornerRadius = radius;
-   }
+    this.dashArray = [3];
+    this.horizontalCornerRadius = 0;
+    this.verticalCornerRadius = 0;
   }
- };
- return AnnotationBorderStyle;
+  AnnotationBorderStyle.prototype = {
+    setWidth: function AnnotationBorderStyle_setWidth(width) {
+      if (width === (width | 0)) {
+        this.width = width;
+      }
+    },
+    setStyle: function AnnotationBorderStyle_setStyle(style) {
+      if (!style) {
+        return;
+      }
+      switch (style.name) {
+        case 'S':
+          this.style = AnnotationBorderStyleType.SOLID;
+          break;
+        case 'D':
+          this.style = AnnotationBorderStyleType.DASHED;
+          break;
+        case 'B':
+          this.style = AnnotationBorderStyleType.BEVELED;
+          break;
+        case 'I':
+          this.style = AnnotationBorderStyleType.INSET;
+          break;
+        case 'U':
+          this.style = AnnotationBorderStyleType.UNDERLINE;
+          break;
+        default:
+          break;
+      }
+    },
+    setDashArray: function AnnotationBorderStyle_setDashArray(dashArray) {
+      if (isArray(dashArray) && dashArray.length > 0) {
+        var isValid = true;
+        var allZeros = true;
+        for (var i = 0, len = dashArray.length; i < len; i++) {
+          var element = dashArray[i];
+          var validNumber = +element >= 0;
+          if (!validNumber) {
+            isValid = false;
+            break;
+          } else if (element > 0) {
+            allZeros = false;
+          }
+        }
+        if (isValid && !allZeros) {
+          this.dashArray = dashArray;
+        } else {
+          this.width = 0;
+        }
+      } else if (dashArray) {
+        this.width = 0;
+      }
+    },
+    setHorizontalCornerRadius: function AnnotationBorderStyle_setHorizontalCornerRadius(radius) {
+      if (radius === (radius | 0)) {
+        this.horizontalCornerRadius = radius;
+      }
+    },
+    setVerticalCornerRadius: function AnnotationBorderStyle_setVerticalCornerRadius(radius) {
+      if (radius === (radius | 0)) {
+        this.verticalCornerRadius = radius;
+      }
+    }
+  };
+  return AnnotationBorderStyle;
 }();
 var WidgetAnnotation = function WidgetAnnotationClosure() {
- function WidgetAnnotation(params) {
-  Annotation.call(this, params);
-  var dict = params.dict;
-  var data = this.data;
-  data.annotationType = AnnotationType.WIDGET;
-  data.fieldName = this._constructFieldName(dict);
-  data.fieldValue = Util.getInheritableProperty(dict, 'V', true);
-  data.alternativeText = stringToPDFString(dict.get('TU') || '');
-  data.defaultAppearance = Util.getInheritableProperty(dict, 'DA') || '';
-  var fieldType = Util.getInheritableProperty(dict, 'FT');
-  data.fieldType = isName(fieldType) ? fieldType.name : null;
-  this.fieldResources = Util.getInheritableProperty(dict, 'DR') || Dict.empty;
-  data.fieldFlags = Util.getInheritableProperty(dict, 'Ff');
-  if (!isInt(data.fieldFlags) || data.fieldFlags < 0) {
-   data.fieldFlags = 0;
-  }
-  data.readOnly = this.hasFieldFlag(AnnotationFieldFlag.READONLY);
-  if (data.fieldType === 'Sig') {
-   this.setFlags(AnnotationFlag.HIDDEN);
-  }
- }
- Util.inherit(WidgetAnnotation, Annotation, {
-  _constructFieldName: function WidgetAnnotation_constructFieldName(dict) {
-   if (!dict.has('T') && !dict.has('Parent')) {
-    warn('Unknown field name, falling back to empty field name.');
-    return '';
-   }
-   if (!dict.has('Parent')) {
-    return stringToPDFString(dict.get('T'));
-   }
-   var fieldName = [];
-   if (dict.has('T')) {
-    fieldName.unshift(stringToPDFString(dict.get('T')));
-   }
-   var loopDict = dict;
-   while (loopDict.has('Parent')) {
-    loopDict = loopDict.get('Parent');
-    if (!isDict(loopDict)) {
-     break;
+  function WidgetAnnotation(params) {
+    Annotation.call(this, params);
+    var dict = params.dict;
+    var data = this.data;
+    data.annotationType = AnnotationType.WIDGET;
+    data.fieldName = this._constructFieldName(dict);
+    data.fieldValue = Util.getInheritableProperty(dict, 'V', true);
+    data.alternativeText = stringToPDFString(dict.get('TU') || '');
+    data.defaultAppearance = Util.getInheritableProperty(dict, 'DA') || '';
+    var fieldType = Util.getInheritableProperty(dict, 'FT');
+    data.fieldType = isName(fieldType) ? fieldType.name : null;
+    this.fieldResources = Util.getInheritableProperty(dict, 'DR') || Dict.empty;
+    data.fieldFlags = Util.getInheritableProperty(dict, 'Ff');
+    if (!isInt(data.fieldFlags) || data.fieldFlags < 0) {
+      data.fieldFlags = 0;
     }
-    if (loopDict.has('T')) {
-     fieldName.unshift(stringToPDFString(loopDict.get('T')));
+    data.readOnly = this.hasFieldFlag(AnnotationFieldFlag.READONLY);
+    if (data.fieldType === 'Sig') {
+      this.setFlags(AnnotationFlag.HIDDEN);
     }
-   }
-   return fieldName.join('.');
-  },
-  hasFieldFlag: function WidgetAnnotation_hasFieldFlag(flag) {
-   return !!(this.data.fieldFlags & flag);
   }
- });
- return WidgetAnnotation;
+  Util.inherit(WidgetAnnotation, Annotation, {
+    _constructFieldName: function WidgetAnnotation_constructFieldName(dict) {
+      if (!dict.has('T') && !dict.has('Parent')) {
+        warn('Unknown field name, falling back to empty field name.');
+        return '';
+      }
+      if (!dict.has('Parent')) {
+        return stringToPDFString(dict.get('T'));
+      }
+      var fieldName = [];
+      if (dict.has('T')) {
+        fieldName.unshift(stringToPDFString(dict.get('T')));
+      }
+      var loopDict = dict;
+      while (loopDict.has('Parent')) {
+        loopDict = loopDict.get('Parent');
+        if (!isDict(loopDict)) {
+          break;
+        }
+        if (loopDict.has('T')) {
+          fieldName.unshift(stringToPDFString(loopDict.get('T')));
+        }
+      }
+      return fieldName.join('.');
+    },
+    hasFieldFlag: function WidgetAnnotation_hasFieldFlag(flag) {
+      return !!(this.data.fieldFlags & flag);
+    }
+  });
+  return WidgetAnnotation;
 }();
 var TextWidgetAnnotation = function TextWidgetAnnotationClosure() {
- function TextWidgetAnnotation(params) {
-  WidgetAnnotation.call(this, params);
-  this.data.fieldValue = stringToPDFString(this.data.fieldValue || '');
-  var alignment = Util.getInheritableProperty(params.dict, 'Q');
-  if (!isInt(alignment) || alignment < 0 || alignment > 2) {
-   alignment = null;
-  }
-  this.data.textAlignment = alignment;
-  var maximumLength = Util.getInheritableProperty(params.dict, 'MaxLen');
-  if (!isInt(maximumLength) || maximumLength < 0) {
-   maximumLength = null;
-  }
-  this.data.maxLen = maximumLength;
-  this.data.multiLine = this.hasFieldFlag(AnnotationFieldFlag.MULTILINE);
-  this.data.comb = this.hasFieldFlag(AnnotationFieldFlag.COMB) && !this.hasFieldFlag(AnnotationFieldFlag.MULTILINE) && !this.hasFieldFlag(AnnotationFieldFlag.PASSWORD) && !this.hasFieldFlag(AnnotationFieldFlag.FILESELECT) && this.data.maxLen !== null;
- }
- Util.inherit(TextWidgetAnnotation, WidgetAnnotation, {
-  getOperatorList: function TextWidgetAnnotation_getOperatorList(evaluator, task, renderForms) {
-   var operatorList = new OperatorList();
-   if (renderForms) {
-    return Promise.resolve(operatorList);
-   }
-   if (this.appearance) {
-    return Annotation.prototype.getOperatorList.call(this, evaluator, task, renderForms);
-   }
-   if (!this.data.defaultAppearance) {
-    return Promise.resolve(operatorList);
-   }
-   var stream = new Stream(stringToBytes(this.data.defaultAppearance));
-   return evaluator.getOperatorList(stream, task, this.fieldResources, operatorList).then(function () {
-    return operatorList;
-   });
+  function TextWidgetAnnotation(params) {
+    WidgetAnnotation.call(this, params);
+    this.data.fieldValue = stringToPDFString(this.data.fieldValue || '');
+    var alignment = Util.getInheritableProperty(params.dict, 'Q');
+    if (!isInt(alignment) || alignment < 0 || alignment > 2) {
+      alignment = null;
+    }
+    this.data.textAlignment = alignment;
+    var maximumLength = Util.getInheritableProperty(params.dict, 'MaxLen');
+    if (!isInt(maximumLength) || maximumLength < 0) {
+      maximumLength = null;
+    }
+    this.data.maxLen = maximumLength;
+    this.data.multiLine = this.hasFieldFlag(AnnotationFieldFlag.MULTILINE);
+    this.data.comb = this.hasFieldFlag(AnnotationFieldFlag.COMB) && !this.hasFieldFlag(AnnotationFieldFlag.MULTILINE) && !this.hasFieldFlag(AnnotationFieldFlag.PASSWORD) && !this.hasFieldFlag(AnnotationFieldFlag.FILESELECT) && this.data.maxLen !== null;
   }
- });
- return TextWidgetAnnotation;
+  Util.inherit(TextWidgetAnnotation, WidgetAnnotation, {
+    getOperatorList: function TextWidgetAnnotation_getOperatorList(evaluator, task, renderForms) {
+      var operatorList = new OperatorList();
+      if (renderForms) {
+        return Promise.resolve(operatorList);
+      }
+      if (this.appearance) {
+        return Annotation.prototype.getOperatorList.call(this, evaluator, task, renderForms);
+      }
+      if (!this.data.defaultAppearance) {
+        return Promise.resolve(operatorList);
+      }
+      var stream = new Stream(stringToBytes(this.data.defaultAppearance));
+      return evaluator.getOperatorList(stream, task, this.fieldResources, operatorList).then(function () {
+        return operatorList;
+      });
+    }
+  });
+  return TextWidgetAnnotation;
 }();
 var ButtonWidgetAnnotation = function ButtonWidgetAnnotationClosure() {
- function ButtonWidgetAnnotation(params) {
-  WidgetAnnotation.call(this, params);
-  this.data.checkBox = !this.hasFieldFlag(AnnotationFieldFlag.RADIO) && !this.hasFieldFlag(AnnotationFieldFlag.PUSHBUTTON);
-  if (this.data.checkBox) {
-   if (!isName(this.data.fieldValue)) {
-    return;
-   }
-   this.data.fieldValue = this.data.fieldValue.name;
-  }
-  this.data.radioButton = this.hasFieldFlag(AnnotationFieldFlag.RADIO) && !this.hasFieldFlag(AnnotationFieldFlag.PUSHBUTTON);
-  if (this.data.radioButton) {
-   this.data.fieldValue = this.data.buttonValue = null;
-   var fieldParent = params.dict.get('Parent');
-   if (isDict(fieldParent) && fieldParent.has('V')) {
-    var fieldParentValue = fieldParent.get('V');
-    if (isName(fieldParentValue)) {
-     this.data.fieldValue = fieldParentValue.name;
+  function ButtonWidgetAnnotation(params) {
+    WidgetAnnotation.call(this, params);
+    this.data.checkBox = !this.hasFieldFlag(AnnotationFieldFlag.RADIO) && !this.hasFieldFlag(AnnotationFieldFlag.PUSHBUTTON);
+    if (this.data.checkBox) {
+      if (!isName(this.data.fieldValue)) {
+        return;
+      }
+      this.data.fieldValue = this.data.fieldValue.name;
     }
-   }
-   var appearanceStates = params.dict.get('AP');
-   if (!isDict(appearanceStates)) {
-    return;
-   }
-   var normalAppearanceState = appearanceStates.get('N');
-   if (!isDict(normalAppearanceState)) {
-    return;
-   }
-   var keys = normalAppearanceState.getKeys();
-   for (var i = 0, ii = keys.length; i < ii; i++) {
-    if (keys[i] !== 'Off') {
-     this.data.buttonValue = keys[i];
-     break;
+    this.data.radioButton = this.hasFieldFlag(AnnotationFieldFlag.RADIO) && !this.hasFieldFlag(AnnotationFieldFlag.PUSHBUTTON);
+    if (this.data.radioButton) {
+      this.data.fieldValue = this.data.buttonValue = null;
+      var fieldParent = params.dict.get('Parent');
+      if (isDict(fieldParent) && fieldParent.has('V')) {
+        var fieldParentValue = fieldParent.get('V');
+        if (isName(fieldParentValue)) {
+          this.data.fieldValue = fieldParentValue.name;
+        }
+      }
+      var appearanceStates = params.dict.get('AP');
+      if (!isDict(appearanceStates)) {
+        return;
+      }
+      var normalAppearanceState = appearanceStates.get('N');
+      if (!isDict(normalAppearanceState)) {
+        return;
+      }
+      var keys = normalAppearanceState.getKeys();
+      for (var i = 0, ii = keys.length; i < ii; i++) {
+        if (keys[i] !== 'Off') {
+          this.data.buttonValue = keys[i];
+          break;
+        }
+      }
     }
-   }
-  }
- }
- Util.inherit(ButtonWidgetAnnotation, WidgetAnnotation, {
-  getOperatorList: function ButtonWidgetAnnotation_getOperatorList(evaluator, task, renderForms) {
-   var operatorList = new OperatorList();
-   if (renderForms) {
-    return Promise.resolve(operatorList);
-   }
-   if (this.appearance) {
-    return Annotation.prototype.getOperatorList.call(this, evaluator, task, renderForms);
-   }
-   return Promise.resolve(operatorList);
   }
- });
- return ButtonWidgetAnnotation;
+  Util.inherit(ButtonWidgetAnnotation, WidgetAnnotation, {
+    getOperatorList: function ButtonWidgetAnnotation_getOperatorList(evaluator, task, renderForms) {
+      var operatorList = new OperatorList();
+      if (renderForms) {
+        return Promise.resolve(operatorList);
+      }
+      if (this.appearance) {
+        return Annotation.prototype.getOperatorList.call(this, evaluator, task, renderForms);
+      }
+      return Promise.resolve(operatorList);
+    }
+  });
+  return ButtonWidgetAnnotation;
 }();
 var ChoiceWidgetAnnotation = function ChoiceWidgetAnnotationClosure() {
- function ChoiceWidgetAnnotation(params) {
-  WidgetAnnotation.call(this, params);
-  this.data.options = [];
-  var options = Util.getInheritableProperty(params.dict, 'Opt');
-  if (isArray(options)) {
-   var xref = params.xref;
-   for (var i = 0, ii = options.length; i < ii; i++) {
-    var option = xref.fetchIfRef(options[i]);
-    var isOptionArray = isArray(option);
-    this.data.options[i] = {
-     exportValue: isOptionArray ? xref.fetchIfRef(option[0]) : option,
-     displayValue: isOptionArray ? xref.fetchIfRef(option[1]) : option
-    };
-   }
-  }
-  if (!isArray(this.data.fieldValue)) {
-   this.data.fieldValue = [this.data.fieldValue];
-  }
-  this.data.combo = this.hasFieldFlag(AnnotationFieldFlag.COMBO);
-  this.data.multiSelect = this.hasFieldFlag(AnnotationFieldFlag.MULTISELECT);
- }
- Util.inherit(ChoiceWidgetAnnotation, WidgetAnnotation, {
-  getOperatorList: function ChoiceWidgetAnnotation_getOperatorList(evaluator, task, renderForms) {
-   var operatorList = new OperatorList();
-   if (renderForms) {
-    return Promise.resolve(operatorList);
-   }
-   return Annotation.prototype.getOperatorList.call(this, evaluator, task, renderForms);
+  function ChoiceWidgetAnnotation(params) {
+    WidgetAnnotation.call(this, params);
+    this.data.options = [];
+    var options = Util.getInheritableProperty(params.dict, 'Opt');
+    if (isArray(options)) {
+      var xref = params.xref;
+      for (var i = 0, ii = options.length; i < ii; i++) {
+        var option = xref.fetchIfRef(options[i]);
+        var isOptionArray = isArray(option);
+        this.data.options[i] = {
+          exportValue: isOptionArray ? xref.fetchIfRef(option[0]) : option,
+          displayValue: isOptionArray ? xref.fetchIfRef(option[1]) : option
+        };
+      }
+    }
+    if (!isArray(this.data.fieldValue)) {
+      this.data.fieldValue = [this.data.fieldValue];
+    }
+    this.data.combo = this.hasFieldFlag(AnnotationFieldFlag.COMBO);
+    this.data.multiSelect = this.hasFieldFlag(AnnotationFieldFlag.MULTISELECT);
   }
- });
- return ChoiceWidgetAnnotation;
+  Util.inherit(ChoiceWidgetAnnotation, WidgetAnnotation, {
+    getOperatorList: function ChoiceWidgetAnnotation_getOperatorList(evaluator, task, renderForms) {
+      var operatorList = new OperatorList();
+      if (renderForms) {
+        return Promise.resolve(operatorList);
+      }
+      return Annotation.prototype.getOperatorList.call(this, evaluator, task, renderForms);
+    }
+  });
+  return ChoiceWidgetAnnotation;
 }();
 var TextAnnotation = function TextAnnotationClosure() {
- var DEFAULT_ICON_SIZE = 22;
- function TextAnnotation(parameters) {
-  Annotation.call(this, parameters);
-  this.data.annotationType = AnnotationType.TEXT;
-  if (this.data.hasAppearance) {
-   this.data.name = 'NoIcon';
-  } else {
-   this.data.rect[1] = this.data.rect[3] - DEFAULT_ICON_SIZE;
-   this.data.rect[2] = this.data.rect[0] + DEFAULT_ICON_SIZE;
-   this.data.name = parameters.dict.has('Name') ? parameters.dict.get('Name').name : 'Note';
+  var DEFAULT_ICON_SIZE = 22;
+  function TextAnnotation(parameters) {
+    Annotation.call(this, parameters);
+    this.data.annotationType = AnnotationType.TEXT;
+    if (this.data.hasAppearance) {
+      this.data.name = 'NoIcon';
+    } else {
+      this.data.rect[1] = this.data.rect[3] - DEFAULT_ICON_SIZE;
+      this.data.rect[2] = this.data.rect[0] + DEFAULT_ICON_SIZE;
+      this.data.name = parameters.dict.has('Name') ? parameters.dict.get('Name').name : 'Note';
+    }
+    this._preparePopup(parameters.dict);
   }
-  this._preparePopup(parameters.dict);
- }
- Util.inherit(TextAnnotation, Annotation, {});
- return TextAnnotation;
+  Util.inherit(TextAnnotation, Annotation, {});
+  return TextAnnotation;
 }();
 var LinkAnnotation = function LinkAnnotationClosure() {
- function LinkAnnotation(params) {
-  Annotation.call(this, params);
-  var data = this.data;
-  data.annotationType = AnnotationType.LINK;
-  Catalog.parseDestDictionary({
-   destDict: params.dict,
-   resultObj: data,
-   docBaseUrl: params.pdfManager.docBaseUrl
-  });
- }
- Util.inherit(LinkAnnotation, Annotation, {});
- return LinkAnnotation;
+  function LinkAnnotation(params) {
+    Annotation.call(this, params);
+    var data = this.data;
+    data.annotationType = AnnotationType.LINK;
+    Catalog.parseDestDictionary({
+      destDict: params.dict,
+      resultObj: data,
+      docBaseUrl: params.pdfManager.docBaseUrl
+    });
+  }
+  Util.inherit(LinkAnnotation, Annotation, {});
+  return LinkAnnotation;
 }();
 var PopupAnnotation = function PopupAnnotationClosure() {
- function PopupAnnotation(parameters) {
-  Annotation.call(this, parameters);
-  this.data.annotationType = AnnotationType.POPUP;
-  var dict = parameters.dict;
-  var parentItem = dict.get('Parent');
-  if (!parentItem) {
-   warn('Popup annotation has a missing or invalid parent annotation.');
-   return;
-  }
-  this.data.parentId = dict.getRaw('Parent').toString();
-  this.data.title = stringToPDFString(parentItem.get('T') || '');
-  this.data.contents = stringToPDFString(parentItem.get('Contents') || '');
-  if (!parentItem.has('C')) {
-   this.data.color = null;
-  } else {
-   this.setColor(parentItem.getArray('C'));
-   this.data.color = this.color;
-  }
-  if (!this.viewable) {
-   var parentFlags = parentItem.get('F');
-   if (this._isViewable(parentFlags)) {
-    this.setFlags(parentFlags);
-   }
+  function PopupAnnotation(parameters) {
+    Annotation.call(this, parameters);
+    this.data.annotationType = AnnotationType.POPUP;
+    var dict = parameters.dict;
+    var parentItem = dict.get('Parent');
+    if (!parentItem) {
+      warn('Popup annotation has a missing or invalid parent annotation.');
+      return;
+    }
+    this.data.parentId = dict.getRaw('Parent').toString();
+    this.data.title = stringToPDFString(parentItem.get('T') || '');
+    this.data.contents = stringToPDFString(parentItem.get('Contents') || '');
+    if (!parentItem.has('C')) {
+      this.data.color = null;
+    } else {
+      this.setColor(parentItem.getArray('C'));
+      this.data.color = this.color;
+    }
+    if (!this.viewable) {
+      var parentFlags = parentItem.get('F');
+      if (this._isViewable(parentFlags)) {
+        this.setFlags(parentFlags);
+      }
+    }
   }
- }
- Util.inherit(PopupAnnotation, Annotation, {});
- return PopupAnnotation;
+  Util.inherit(PopupAnnotation, Annotation, {});
+  return PopupAnnotation;
 }();
 var HighlightAnnotation = function HighlightAnnotationClosure() {
- function HighlightAnnotation(parameters) {
-  Annotation.call(this, parameters);
-  this.data.annotationType = AnnotationType.HIGHLIGHT;
-  this._preparePopup(parameters.dict);
-  this.data.borderStyle.setWidth(0);
- }
- Util.inherit(HighlightAnnotation, Annotation, {});
- return HighlightAnnotation;
+  function HighlightAnnotation(parameters) {
+    Annotation.call(this, parameters);
+    this.data.annotationType = AnnotationType.HIGHLIGHT;
+    this._preparePopup(parameters.dict);
+    this.data.borderStyle.setWidth(0);
+  }
+  Util.inherit(HighlightAnnotation, Annotation, {});
+  return HighlightAnnotation;
 }();
 var UnderlineAnnotation = function UnderlineAnnotationClosure() {
- function UnderlineAnnotation(parameters) {
-  Annotation.call(this, parameters);
-  this.data.annotationType = AnnotationType.UNDERLINE;
-  this._preparePopup(parameters.dict);
-  this.data.borderStyle.setWidth(0);
- }
- Util.inherit(UnderlineAnnotation, Annotation, {});
- return UnderlineAnnotation;
+  function UnderlineAnnotation(parameters) {
+    Annotation.call(this, parameters);
+    this.data.annotationType = AnnotationType.UNDERLINE;
+    this._preparePopup(parameters.dict);
+    this.data.borderStyle.setWidth(0);
+  }
+  Util.inherit(UnderlineAnnotation, Annotation, {});
+  return UnderlineAnnotation;
 }();
 var SquigglyAnnotation = function SquigglyAnnotationClosure() {
- function SquigglyAnnotation(parameters) {
-  Annotation.call(this, parameters);
-  this.data.annotationType = AnnotationType.SQUIGGLY;
-  this._preparePopup(parameters.dict);
-  this.data.borderStyle.setWidth(0);
- }
- Util.inherit(SquigglyAnnotation, Annotation, {});
- return SquigglyAnnotation;
+  function SquigglyAnnotation(parameters) {
+    Annotation.call(this, parameters);
+    this.data.annotationType = AnnotationType.SQUIGGLY;
+    this._preparePopup(parameters.dict);
+    this.data.borderStyle.setWidth(0);
+  }
+  Util.inherit(SquigglyAnnotation, Annotation, {});
+  return SquigglyAnnotation;
 }();
 var StrikeOutAnnotation = function StrikeOutAnnotationClosure() {
- function StrikeOutAnnotation(parameters) {
-  Annotation.call(this, parameters);
-  this.data.annotationType = AnnotationType.STRIKEOUT;
-  this._preparePopup(parameters.dict);
-  this.data.borderStyle.setWidth(0);
- }
- Util.inherit(StrikeOutAnnotation, Annotation, {});
- return StrikeOutAnnotation;
+  function StrikeOutAnnotation(parameters) {
+    Annotation.call(this, parameters);
+    this.data.annotationType = AnnotationType.STRIKEOUT;
+    this._preparePopup(parameters.dict);
+    this.data.borderStyle.setWidth(0);
+  }
+  Util.inherit(StrikeOutAnnotation, Annotation, {});
+  return StrikeOutAnnotation;
 }();
 var FileAttachmentAnnotation = function FileAttachmentAnnotationClosure() {
- function FileAttachmentAnnotation(parameters) {
-  Annotation.call(this, parameters);
-  var file = new FileSpec(parameters.dict.get('FS'), parameters.xref);
-  this.data.annotationType = AnnotationType.FILEATTACHMENT;
-  this.data.file = file.serializable;
-  this._preparePopup(parameters.dict);
- }
- Util.inherit(FileAttachmentAnnotation, Annotation, {});
- return FileAttachmentAnnotation;
+  function FileAttachmentAnnotation(parameters) {
+    Annotation.call(this, parameters);
+    var file = new FileSpec(parameters.dict.get('FS'), parameters.xref);
+    this.data.annotationType = AnnotationType.FILEATTACHMENT;
+    this.data.file = file.serializable;
+    this._preparePopup(parameters.dict);
+  }
+  Util.inherit(FileAttachmentAnnotation, Annotation, {});
+  return FileAttachmentAnnotation;
 }();
 exports.Annotation = Annotation;
 exports.AnnotationBorderStyle = AnnotationBorderStyle;

+ 325 - 371
lib/core/arithmetic_decoder.js

@@ -13,379 +13,333 @@
  * limitations under the License.
  */
 'use strict';
+
 var ArithmeticDecoder = function ArithmeticDecoderClosure() {
- var QeTable = [
-  {
-   qe: 0x5601,
-   nmps: 1,
-   nlps: 1,
-   switchFlag: 1
-  },
-  {
-   qe: 0x3401,
-   nmps: 2,
-   nlps: 6,
-   switchFlag: 0
-  },
-  {
-   qe: 0x1801,
-   nmps: 3,
-   nlps: 9,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0AC1,
-   nmps: 4,
-   nlps: 12,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0521,
-   nmps: 5,
-   nlps: 29,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0221,
-   nmps: 38,
-   nlps: 33,
-   switchFlag: 0
-  },
-  {
-   qe: 0x5601,
-   nmps: 7,
-   nlps: 6,
-   switchFlag: 1
-  },
-  {
-   qe: 0x5401,
-   nmps: 8,
-   nlps: 14,
-   switchFlag: 0
-  },
-  {
-   qe: 0x4801,
-   nmps: 9,
-   nlps: 14,
-   switchFlag: 0
-  },
-  {
-   qe: 0x3801,
-   nmps: 10,
-   nlps: 14,
-   switchFlag: 0
-  },
-  {
-   qe: 0x3001,
-   nmps: 11,
-   nlps: 17,
-   switchFlag: 0
-  },
-  {
-   qe: 0x2401,
-   nmps: 12,
-   nlps: 18,
-   switchFlag: 0
-  },
-  {
-   qe: 0x1C01,
-   nmps: 13,
-   nlps: 20,
-   switchFlag: 0
-  },
-  {
-   qe: 0x1601,
-   nmps: 29,
-   nlps: 21,
-   switchFlag: 0
-  },
-  {
-   qe: 0x5601,
-   nmps: 15,
-   nlps: 14,
-   switchFlag: 1
-  },
-  {
-   qe: 0x5401,
-   nmps: 16,
-   nlps: 14,
-   switchFlag: 0
-  },
-  {
-   qe: 0x5101,
-   nmps: 17,
-   nlps: 15,
-   switchFlag: 0
-  },
-  {
-   qe: 0x4801,
-   nmps: 18,
-   nlps: 16,
-   switchFlag: 0
-  },
-  {
-   qe: 0x3801,
-   nmps: 19,
-   nlps: 17,
-   switchFlag: 0
-  },
-  {
-   qe: 0x3401,
-   nmps: 20,
-   nlps: 18,
-   switchFlag: 0
-  },
-  {
-   qe: 0x3001,
-   nmps: 21,
-   nlps: 19,
-   switchFlag: 0
-  },
-  {
-   qe: 0x2801,
-   nmps: 22,
-   nlps: 19,
-   switchFlag: 0
-  },
-  {
-   qe: 0x2401,
-   nmps: 23,
-   nlps: 20,
-   switchFlag: 0
-  },
-  {
-   qe: 0x2201,
-   nmps: 24,
-   nlps: 21,
-   switchFlag: 0
-  },
-  {
-   qe: 0x1C01,
-   nmps: 25,
-   nlps: 22,
-   switchFlag: 0
-  },
-  {
-   qe: 0x1801,
-   nmps: 26,
-   nlps: 23,
-   switchFlag: 0
-  },
-  {
-   qe: 0x1601,
-   nmps: 27,
-   nlps: 24,
-   switchFlag: 0
-  },
-  {
-   qe: 0x1401,
-   nmps: 28,
-   nlps: 25,
-   switchFlag: 0
-  },
-  {
-   qe: 0x1201,
-   nmps: 29,
-   nlps: 26,
-   switchFlag: 0
-  },
-  {
-   qe: 0x1101,
-   nmps: 30,
-   nlps: 27,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0AC1,
-   nmps: 31,
-   nlps: 28,
-   switchFlag: 0
-  },
-  {
-   qe: 0x09C1,
-   nmps: 32,
-   nlps: 29,
-   switchFlag: 0
-  },
-  {
-   qe: 0x08A1,
-   nmps: 33,
-   nlps: 30,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0521,
-   nmps: 34,
-   nlps: 31,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0441,
-   nmps: 35,
-   nlps: 32,
-   switchFlag: 0
-  },
-  {
-   qe: 0x02A1,
-   nmps: 36,
-   nlps: 33,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0221,
-   nmps: 37,
-   nlps: 34,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0141,
-   nmps: 38,
-   nlps: 35,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0111,
-   nmps: 39,
-   nlps: 36,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0085,
-   nmps: 40,
-   nlps: 37,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0049,
-   nmps: 41,
-   nlps: 38,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0025,
-   nmps: 42,
-   nlps: 39,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0015,
-   nmps: 43,
-   nlps: 40,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0009,
-   nmps: 44,
-   nlps: 41,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0005,
-   nmps: 45,
-   nlps: 42,
-   switchFlag: 0
-  },
-  {
-   qe: 0x0001,
-   nmps: 45,
-   nlps: 43,
-   switchFlag: 0
-  },
-  {
-   qe: 0x5601,
-   nmps: 46,
-   nlps: 46,
-   switchFlag: 0
+  var QeTable = [{
+    qe: 0x5601,
+    nmps: 1,
+    nlps: 1,
+    switchFlag: 1
+  }, {
+    qe: 0x3401,
+    nmps: 2,
+    nlps: 6,
+    switchFlag: 0
+  }, {
+    qe: 0x1801,
+    nmps: 3,
+    nlps: 9,
+    switchFlag: 0
+  }, {
+    qe: 0x0AC1,
+    nmps: 4,
+    nlps: 12,
+    switchFlag: 0
+  }, {
+    qe: 0x0521,
+    nmps: 5,
+    nlps: 29,
+    switchFlag: 0
+  }, {
+    qe: 0x0221,
+    nmps: 38,
+    nlps: 33,
+    switchFlag: 0
+  }, {
+    qe: 0x5601,
+    nmps: 7,
+    nlps: 6,
+    switchFlag: 1
+  }, {
+    qe: 0x5401,
+    nmps: 8,
+    nlps: 14,
+    switchFlag: 0
+  }, {
+    qe: 0x4801,
+    nmps: 9,
+    nlps: 14,
+    switchFlag: 0
+  }, {
+    qe: 0x3801,
+    nmps: 10,
+    nlps: 14,
+    switchFlag: 0
+  }, {
+    qe: 0x3001,
+    nmps: 11,
+    nlps: 17,
+    switchFlag: 0
+  }, {
+    qe: 0x2401,
+    nmps: 12,
+    nlps: 18,
+    switchFlag: 0
+  }, {
+    qe: 0x1C01,
+    nmps: 13,
+    nlps: 20,
+    switchFlag: 0
+  }, {
+    qe: 0x1601,
+    nmps: 29,
+    nlps: 21,
+    switchFlag: 0
+  }, {
+    qe: 0x5601,
+    nmps: 15,
+    nlps: 14,
+    switchFlag: 1
+  }, {
+    qe: 0x5401,
+    nmps: 16,
+    nlps: 14,
+    switchFlag: 0
+  }, {
+    qe: 0x5101,
+    nmps: 17,
+    nlps: 15,
+    switchFlag: 0
+  }, {
+    qe: 0x4801,
+    nmps: 18,
+    nlps: 16,
+    switchFlag: 0
+  }, {
+    qe: 0x3801,
+    nmps: 19,
+    nlps: 17,
+    switchFlag: 0
+  }, {
+    qe: 0x3401,
+    nmps: 20,
+    nlps: 18,
+    switchFlag: 0
+  }, {
+    qe: 0x3001,
+    nmps: 21,
+    nlps: 19,
+    switchFlag: 0
+  }, {
+    qe: 0x2801,
+    nmps: 22,
+    nlps: 19,
+    switchFlag: 0
+  }, {
+    qe: 0x2401,
+    nmps: 23,
+    nlps: 20,
+    switchFlag: 0
+  }, {
+    qe: 0x2201,
+    nmps: 24,
+    nlps: 21,
+    switchFlag: 0
+  }, {
+    qe: 0x1C01,
+    nmps: 25,
+    nlps: 22,
+    switchFlag: 0
+  }, {
+    qe: 0x1801,
+    nmps: 26,
+    nlps: 23,
+    switchFlag: 0
+  }, {
+    qe: 0x1601,
+    nmps: 27,
+    nlps: 24,
+    switchFlag: 0
+  }, {
+    qe: 0x1401,
+    nmps: 28,
+    nlps: 25,
+    switchFlag: 0
+  }, {
+    qe: 0x1201,
+    nmps: 29,
+    nlps: 26,
+    switchFlag: 0
+  }, {
+    qe: 0x1101,
+    nmps: 30,
+    nlps: 27,
+    switchFlag: 0
+  }, {
+    qe: 0x0AC1,
+    nmps: 31,
+    nlps: 28,
+    switchFlag: 0
+  }, {
+    qe: 0x09C1,
+    nmps: 32,
+    nlps: 29,
+    switchFlag: 0
+  }, {
+    qe: 0x08A1,
+    nmps: 33,
+    nlps: 30,
+    switchFlag: 0
+  }, {
+    qe: 0x0521,
+    nmps: 34,
+    nlps: 31,
+    switchFlag: 0
+  }, {
+    qe: 0x0441,
+    nmps: 35,
+    nlps: 32,
+    switchFlag: 0
+  }, {
+    qe: 0x02A1,
+    nmps: 36,
+    nlps: 33,
+    switchFlag: 0
+  }, {
+    qe: 0x0221,
+    nmps: 37,
+    nlps: 34,
+    switchFlag: 0
+  }, {
+    qe: 0x0141,
+    nmps: 38,
+    nlps: 35,
+    switchFlag: 0
+  }, {
+    qe: 0x0111,
+    nmps: 39,
+    nlps: 36,
+    switchFlag: 0
+  }, {
+    qe: 0x0085,
+    nmps: 40,
+    nlps: 37,
+    switchFlag: 0
+  }, {
+    qe: 0x0049,
+    nmps: 41,
+    nlps: 38,
+    switchFlag: 0
+  }, {
+    qe: 0x0025,
+    nmps: 42,
+    nlps: 39,
+    switchFlag: 0
+  }, {
+    qe: 0x0015,
+    nmps: 43,
+    nlps: 40,
+    switchFlag: 0
+  }, {
+    qe: 0x0009,
+    nmps: 44,
+    nlps: 41,
+    switchFlag: 0
+  }, {
+    qe: 0x0005,
+    nmps: 45,
+    nlps: 42,
+    switchFlag: 0
+  }, {
+    qe: 0x0001,
+    nmps: 45,
+    nlps: 43,
+    switchFlag: 0
+  }, {
+    qe: 0x5601,
+    nmps: 46,
+    nlps: 46,
+    switchFlag: 0
+  }];
+  function ArithmeticDecoder(data, start, end) {
+    this.data = data;
+    this.bp = start;
+    this.dataEnd = end;
+    this.chigh = data[start];
+    this.clow = 0;
+    this.byteIn();
+    this.chigh = this.chigh << 7 & 0xFFFF | this.clow >> 9 & 0x7F;
+    this.clow = this.clow << 7 & 0xFFFF;
+    this.ct -= 7;
+    this.a = 0x8000;
   }
- ];
- function ArithmeticDecoder(data, start, end) {
-  this.data = data;
-  this.bp = start;
-  this.dataEnd = end;
-  this.chigh = data[start];
-  this.clow = 0;
-  this.byteIn();
-  this.chigh = this.chigh << 7 & 0xFFFF | this.clow >> 9 & 0x7F;
-  this.clow = this.clow << 7 & 0xFFFF;
-  this.ct -= 7;
-  this.a = 0x8000;
- }
- ArithmeticDecoder.prototype = {
-  byteIn: function ArithmeticDecoder_byteIn() {
-   var data = this.data;
-   var bp = this.bp;
-   if (data[bp] === 0xFF) {
-    var b1 = data[bp + 1];
-    if (b1 > 0x8F) {
-     this.clow += 0xFF00;
-     this.ct = 8;
-    } else {
-     bp++;
-     this.clow += data[bp] << 9;
-     this.ct = 7;
-     this.bp = bp;
+  ArithmeticDecoder.prototype = {
+    byteIn: function ArithmeticDecoder_byteIn() {
+      var data = this.data;
+      var bp = this.bp;
+      if (data[bp] === 0xFF) {
+        var b1 = data[bp + 1];
+        if (b1 > 0x8F) {
+          this.clow += 0xFF00;
+          this.ct = 8;
+        } else {
+          bp++;
+          this.clow += data[bp] << 9;
+          this.ct = 7;
+          this.bp = bp;
+        }
+      } else {
+        bp++;
+        this.clow += bp < this.dataEnd ? data[bp] << 8 : 0xFF00;
+        this.ct = 8;
+        this.bp = bp;
+      }
+      if (this.clow > 0xFFFF) {
+        this.chigh += this.clow >> 16;
+        this.clow &= 0xFFFF;
+      }
+    },
+    readBit: function ArithmeticDecoder_readBit(contexts, pos) {
+      var cx_index = contexts[pos] >> 1,
+          cx_mps = contexts[pos] & 1;
+      var qeTableIcx = QeTable[cx_index];
+      var qeIcx = qeTableIcx.qe;
+      var d;
+      var a = this.a - qeIcx;
+      if (this.chigh < qeIcx) {
+        if (a < qeIcx) {
+          a = qeIcx;
+          d = cx_mps;
+          cx_index = qeTableIcx.nmps;
+        } else {
+          a = qeIcx;
+          d = 1 ^ cx_mps;
+          if (qeTableIcx.switchFlag === 1) {
+            cx_mps = d;
+          }
+          cx_index = qeTableIcx.nlps;
+        }
+      } else {
+        this.chigh -= qeIcx;
+        if ((a & 0x8000) !== 0) {
+          this.a = a;
+          return cx_mps;
+        }
+        if (a < qeIcx) {
+          d = 1 ^ cx_mps;
+          if (qeTableIcx.switchFlag === 1) {
+            cx_mps = d;
+          }
+          cx_index = qeTableIcx.nlps;
+        } else {
+          d = cx_mps;
+          cx_index = qeTableIcx.nmps;
+        }
+      }
+      do {
+        if (this.ct === 0) {
+          this.byteIn();
+        }
+        a <<= 1;
+        this.chigh = this.chigh << 1 & 0xFFFF | this.clow >> 15 & 1;
+        this.clow = this.clow << 1 & 0xFFFF;
+        this.ct--;
+      } while ((a & 0x8000) === 0);
+      this.a = a;
+      contexts[pos] = cx_index << 1 | cx_mps;
+      return d;
     }
-   } else {
-    bp++;
-    this.clow += bp < this.dataEnd ? data[bp] << 8 : 0xFF00;
-    this.ct = 8;
-    this.bp = bp;
-   }
-   if (this.clow > 0xFFFF) {
-    this.chigh += this.clow >> 16;
-    this.clow &= 0xFFFF;
-   }
-  },
-  readBit: function ArithmeticDecoder_readBit(contexts, pos) {
-   var cx_index = contexts[pos] >> 1, cx_mps = contexts[pos] & 1;
-   var qeTableIcx = QeTable[cx_index];
-   var qeIcx = qeTableIcx.qe;
-   var d;
-   var a = this.a - qeIcx;
-   if (this.chigh < qeIcx) {
-    if (a < qeIcx) {
-     a = qeIcx;
-     d = cx_mps;
-     cx_index = qeTableIcx.nmps;
-    } else {
-     a = qeIcx;
-     d = 1 ^ cx_mps;
-     if (qeTableIcx.switchFlag === 1) {
-      cx_mps = d;
-     }
-     cx_index = qeTableIcx.nlps;
-    }
-   } else {
-    this.chigh -= qeIcx;
-    if ((a & 0x8000) !== 0) {
-     this.a = a;
-     return cx_mps;
-    }
-    if (a < qeIcx) {
-     d = 1 ^ cx_mps;
-     if (qeTableIcx.switchFlag === 1) {
-      cx_mps = d;
-     }
-     cx_index = qeTableIcx.nlps;
-    } else {
-     d = cx_mps;
-     cx_index = qeTableIcx.nmps;
-    }
-   }
-   do {
-    if (this.ct === 0) {
-     this.byteIn();
-    }
-    a <<= 1;
-    this.chigh = this.chigh << 1 & 0xFFFF | this.clow >> 15 & 1;
-    this.clow = this.clow << 1 & 0xFFFF;
-    this.ct--;
-   } while ((a & 0x8000) === 0);
-   this.a = a;
-   contexts[pos] = cx_index << 1 | cx_mps;
-   return d;
-  }
- };
- return ArithmeticDecoder;
+  };
+  return ArithmeticDecoder;
 }();
 exports.ArithmeticDecoder = ArithmeticDecoder;

+ 195 - 708
lib/core/bidi.js

@@ -13,749 +13,236 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var warn = sharedUtil.warn;
-var baseTypes = [
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'S',
- 'B',
- 'S',
- 'WS',
- 'B',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'B',
- 'B',
- 'B',
- 'S',
- 'WS',
- 'ON',
- 'ON',
- 'ET',
- 'ET',
- 'ET',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'ES',
- 'CS',
- 'ES',
- 'CS',
- 'CS',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'CS',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'B',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'BN',
- 'CS',
- 'ON',
- 'ET',
- 'ET',
- 'ET',
- 'ET',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'L',
- 'ON',
- 'ON',
- 'BN',
- 'ON',
- 'ON',
- 'ET',
- 'ET',
- 'EN',
- 'EN',
- 'ON',
- 'L',
- 'ON',
- 'ON',
- 'ON',
- 'EN',
- 'L',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'ON',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'ON',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'ON',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L',
- 'L'
-];
-var arabicTypes = [
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'ON',
- 'ON',
- 'AL',
- 'ET',
- 'ET',
- 'AL',
- 'CS',
- 'AL',
- 'ON',
- 'ON',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'AL',
- 'AL',
- '',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'AN',
- 'ET',
- 'AN',
- 'AN',
- 'AL',
- 'AL',
- 'AL',
- 'NSM',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'AN',
- 'ON',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'AL',
- 'AL',
- 'NSM',
- 'NSM',
- 'ON',
- 'NSM',
- 'NSM',
- 'NSM',
- 'NSM',
- 'AL',
- 'AL',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'EN',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL',
- 'AL'
-];
+var baseTypes = ['BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'S', 'B', 'S', 'WS', 'B', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'B', 'B', 'B', 'S', 'WS', 'ON', 'ON', 'ET', 'ET', 'ET', 'ON', 'ON', 'ON', 'ON', 'ON', 'ES', 'CS', 'ES', 'CS', 'CS', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'CS', 'ON', 'ON', 'ON', 'ON', 'ON', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'ON', 'ON', 'ON', 'ON', 'ON', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'ON', 'ON', 'ON', 'ON', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'B', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'CS', 'ON', 'ET', 'ET', 'ET', 'ET', 'ON', 'ON', 'ON', 'ON', 'L', 'ON', 'ON', 'BN', 'ON', 'ON', 'ET', 'ET', 'EN', 'EN', 'ON', 'L', 'ON', 'ON', 'ON', 'EN', 'L', 'ON', 'ON', 'ON', 'ON', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L'];
+var arabicTypes = ['AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'ON', 'ON', 'AL', 'ET', 'ET', 'AL', 'CS', 'AL', 'ON', 'ON', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'AL', 'AL', '', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'ET', 'AN', 'AN', 'AL', 'AL', 'AL', 'NSM', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'AN', 'ON', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'AL', 'AL', 'NSM', 'NSM', 'ON', 'NSM', 'NSM', 'NSM', 'NSM', 'AL', 'AL', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL'];
 function isOdd(i) {
- return (i & 1) !== 0;
+  return (i & 1) !== 0;
 }
 function isEven(i) {
- return (i & 1) === 0;
+  return (i & 1) === 0;
 }
 function findUnequal(arr, start, value) {
- for (var j = start, jj = arr.length; j < jj; ++j) {
-  if (arr[j] !== value) {
-   return j;
+  for (var j = start, jj = arr.length; j < jj; ++j) {
+    if (arr[j] !== value) {
+      return j;
+    }
   }
- }
- return j;
+  return j;
 }
 function setValues(arr, start, end, value) {
- for (var j = start; j < end; ++j) {
-  arr[j] = value;
- }
+  for (var j = start; j < end; ++j) {
+    arr[j] = value;
+  }
 }
 function reverseValues(arr, start, end) {
- for (var i = start, j = end - 1; i < j; ++i, --j) {
-  var temp = arr[i];
-  arr[i] = arr[j];
-  arr[j] = temp;
- }
+  for (var i = start, j = end - 1; i < j; ++i, --j) {
+    var temp = arr[i];
+    arr[i] = arr[j];
+    arr[j] = temp;
+  }
 }
 function createBidiText(str, isLTR, vertical) {
- return {
-  str: str,
-  dir: vertical ? 'ttb' : isLTR ? 'ltr' : 'rtl'
- };
+  return {
+    str: str,
+    dir: vertical ? 'ttb' : isLTR ? 'ltr' : 'rtl'
+  };
 }
 var chars = [];
 var types = [];
 function bidi(str, startLevel, vertical) {
- var isLTR = true;
- var strLength = str.length;
- if (strLength === 0 || vertical) {
-  return createBidiText(str, isLTR, vertical);
- }
- chars.length = strLength;
- types.length = strLength;
- var numBidi = 0;
- var i, ii;
- for (i = 0; i < strLength; ++i) {
-  chars[i] = str.charAt(i);
-  var charCode = str.charCodeAt(i);
-  var charType = 'L';
-  if (charCode <= 0x00ff) {
-   charType = baseTypes[charCode];
-  } else if (0x0590 <= charCode && charCode <= 0x05f4) {
-   charType = 'R';
-  } else if (0x0600 <= charCode && charCode <= 0x06ff) {
-   charType = arabicTypes[charCode & 0xff];
-   if (!charType) {
-    warn('Bidi: invalid Unicode character ' + charCode.toString(16));
-   }
-  } else if (0x0700 <= charCode && charCode <= 0x08AC) {
-   charType = 'AL';
-  }
-  if (charType === 'R' || charType === 'AL' || charType === 'AN') {
-   numBidi++;
-  }
-  types[i] = charType;
- }
- if (numBidi === 0) {
-  isLTR = true;
-  return createBidiText(str, isLTR);
- }
- if (startLevel === -1) {
-  if (numBidi / strLength < 0.3) {
-   isLTR = true;
-   startLevel = 0;
-  } else {
-   isLTR = false;
-   startLevel = 1;
-  }
- }
- var levels = [];
- for (i = 0; i < strLength; ++i) {
-  levels[i] = startLevel;
- }
- var e = isOdd(startLevel) ? 'R' : 'L';
- var sor = e;
- var eor = sor;
- var lastType = sor;
- for (i = 0; i < strLength; ++i) {
-  if (types[i] === 'NSM') {
-   types[i] = lastType;
-  } else {
-   lastType = types[i];
-  }
- }
- lastType = sor;
- var t;
- for (i = 0; i < strLength; ++i) {
-  t = types[i];
-  if (t === 'EN') {
-   types[i] = lastType === 'AL' ? 'AN' : 'EN';
-  } else if (t === 'R' || t === 'L' || t === 'AL') {
-   lastType = t;
+  var isLTR = true;
+  var strLength = str.length;
+  if (strLength === 0 || vertical) {
+    return createBidiText(str, isLTR, vertical);
+  }
+  chars.length = strLength;
+  types.length = strLength;
+  var numBidi = 0;
+  var i, ii;
+  for (i = 0; i < strLength; ++i) {
+    chars[i] = str.charAt(i);
+    var charCode = str.charCodeAt(i);
+    var charType = 'L';
+    if (charCode <= 0x00ff) {
+      charType = baseTypes[charCode];
+    } else if (0x0590 <= charCode && charCode <= 0x05f4) {
+      charType = 'R';
+    } else if (0x0600 <= charCode && charCode <= 0x06ff) {
+      charType = arabicTypes[charCode & 0xff];
+      if (!charType) {
+        warn('Bidi: invalid Unicode character ' + charCode.toString(16));
+      }
+    } else if (0x0700 <= charCode && charCode <= 0x08AC) {
+      charType = 'AL';
+    }
+    if (charType === 'R' || charType === 'AL' || charType === 'AN') {
+      numBidi++;
+    }
+    types[i] = charType;
+  }
+  if (numBidi === 0) {
+    isLTR = true;
+    return createBidiText(str, isLTR);
+  }
+  if (startLevel === -1) {
+    if (numBidi / strLength < 0.3) {
+      isLTR = true;
+      startLevel = 0;
+    } else {
+      isLTR = false;
+      startLevel = 1;
+    }
   }
- }
- for (i = 0; i < strLength; ++i) {
-  t = types[i];
-  if (t === 'AL') {
-   types[i] = 'R';
+  var levels = [];
+  for (i = 0; i < strLength; ++i) {
+    levels[i] = startLevel;
+  }
+  var e = isOdd(startLevel) ? 'R' : 'L';
+  var sor = e;
+  var eor = sor;
+  var lastType = sor;
+  for (i = 0; i < strLength; ++i) {
+    if (types[i] === 'NSM') {
+      types[i] = lastType;
+    } else {
+      lastType = types[i];
+    }
   }
- }
- for (i = 1; i < strLength - 1; ++i) {
-  if (types[i] === 'ES' && types[i - 1] === 'EN' && types[i + 1] === 'EN') {
-   types[i] = 'EN';
+  lastType = sor;
+  var t;
+  for (i = 0; i < strLength; ++i) {
+    t = types[i];
+    if (t === 'EN') {
+      types[i] = lastType === 'AL' ? 'AN' : 'EN';
+    } else if (t === 'R' || t === 'L' || t === 'AL') {
+      lastType = t;
+    }
   }
-  if (types[i] === 'CS' && (types[i - 1] === 'EN' || types[i - 1] === 'AN') && types[i + 1] === types[i - 1]) {
-   types[i] = types[i - 1];
+  for (i = 0; i < strLength; ++i) {
+    t = types[i];
+    if (t === 'AL') {
+      types[i] = 'R';
+    }
   }
- }
- for (i = 0; i < strLength; ++i) {
-  if (types[i] === 'EN') {
-   var j;
-   for (j = i - 1; j >= 0; --j) {
-    if (types[j] !== 'ET') {
-     break;
+  for (i = 1; i < strLength - 1; ++i) {
+    if (types[i] === 'ES' && types[i - 1] === 'EN' && types[i + 1] === 'EN') {
+      types[i] = 'EN';
     }
-    types[j] = 'EN';
-   }
-   for (j = i + 1; j < strLength; ++j) {
-    if (types[j] !== 'ET') {
-     break;
+    if (types[i] === 'CS' && (types[i - 1] === 'EN' || types[i - 1] === 'AN') && types[i + 1] === types[i - 1]) {
+      types[i] = types[i - 1];
     }
-    types[j] = 'EN';
-   }
   }
- }
- for (i = 0; i < strLength; ++i) {
-  t = types[i];
-  if (t === 'WS' || t === 'ES' || t === 'ET' || t === 'CS') {
-   types[i] = 'ON';
-  }
- }
- lastType = sor;
- for (i = 0; i < strLength; ++i) {
-  t = types[i];
-  if (t === 'EN') {
-   types[i] = lastType === 'L' ? 'L' : 'EN';
-  } else if (t === 'R' || t === 'L') {
-   lastType = t;
+  for (i = 0; i < strLength; ++i) {
+    if (types[i] === 'EN') {
+      var j;
+      for (j = i - 1; j >= 0; --j) {
+        if (types[j] !== 'ET') {
+          break;
+        }
+        types[j] = 'EN';
+      }
+      for (j = i + 1; j < strLength; ++j) {
+        if (types[j] !== 'ET') {
+          break;
+        }
+        types[j] = 'EN';
+      }
+    }
   }
- }
- for (i = 0; i < strLength; ++i) {
-  if (types[i] === 'ON') {
-   var end = findUnequal(types, i + 1, 'ON');
-   var before = sor;
-   if (i > 0) {
-    before = types[i - 1];
-   }
-   var after = eor;
-   if (end + 1 < strLength) {
-    after = types[end + 1];
-   }
-   if (before !== 'L') {
-    before = 'R';
-   }
-   if (after !== 'L') {
-    after = 'R';
-   }
-   if (before === after) {
-    setValues(types, i, end, before);
-   }
-   i = end - 1;
+  for (i = 0; i < strLength; ++i) {
+    t = types[i];
+    if (t === 'WS' || t === 'ES' || t === 'ET' || t === 'CS') {
+      types[i] = 'ON';
+    }
   }
- }
- for (i = 0; i < strLength; ++i) {
-  if (types[i] === 'ON') {
-   types[i] = e;
+  lastType = sor;
+  for (i = 0; i < strLength; ++i) {
+    t = types[i];
+    if (t === 'EN') {
+      types[i] = lastType === 'L' ? 'L' : 'EN';
+    } else if (t === 'R' || t === 'L') {
+      lastType = t;
+    }
   }
- }
- for (i = 0; i < strLength; ++i) {
-  t = types[i];
-  if (isEven(levels[i])) {
-   if (t === 'R') {
-    levels[i] += 1;
-   } else if (t === 'AN' || t === 'EN') {
-    levels[i] += 2;
-   }
-  } else {
-   if (t === 'L' || t === 'AN' || t === 'EN') {
-    levels[i] += 1;
-   }
+  for (i = 0; i < strLength; ++i) {
+    if (types[i] === 'ON') {
+      var end = findUnequal(types, i + 1, 'ON');
+      var before = sor;
+      if (i > 0) {
+        before = types[i - 1];
+      }
+      var after = eor;
+      if (end + 1 < strLength) {
+        after = types[end + 1];
+      }
+      if (before !== 'L') {
+        before = 'R';
+      }
+      if (after !== 'L') {
+        after = 'R';
+      }
+      if (before === after) {
+        setValues(types, i, end, before);
+      }
+      i = end - 1;
+    }
   }
- }
- var highestLevel = -1;
- var lowestOddLevel = 99;
- var level;
- for (i = 0, ii = levels.length; i < ii; ++i) {
-  level = levels[i];
-  if (highestLevel < level) {
-   highestLevel = level;
+  for (i = 0; i < strLength; ++i) {
+    if (types[i] === 'ON') {
+      types[i] = e;
+    }
   }
-  if (lowestOddLevel > level && isOdd(level)) {
-   lowestOddLevel = level;
+  for (i = 0; i < strLength; ++i) {
+    t = types[i];
+    if (isEven(levels[i])) {
+      if (t === 'R') {
+        levels[i] += 1;
+      } else if (t === 'AN' || t === 'EN') {
+        levels[i] += 2;
+      }
+    } else {
+      if (t === 'L' || t === 'AN' || t === 'EN') {
+        levels[i] += 1;
+      }
+    }
   }
- }
- for (level = highestLevel; level >= lowestOddLevel; --level) {
-  var start = -1;
+  var highestLevel = -1;
+  var lowestOddLevel = 99;
+  var level;
   for (i = 0, ii = levels.length; i < ii; ++i) {
-   if (levels[i] < level) {
-    if (start >= 0) {
-     reverseValues(chars, start, i);
-     start = -1;
+    level = levels[i];
+    if (highestLevel < level) {
+      highestLevel = level;
+    }
+    if (lowestOddLevel > level && isOdd(level)) {
+      lowestOddLevel = level;
     }
-   } else if (start < 0) {
-    start = i;
-   }
   }
-  if (start >= 0) {
-   reverseValues(chars, start, levels.length);
+  for (level = highestLevel; level >= lowestOddLevel; --level) {
+    var start = -1;
+    for (i = 0, ii = levels.length; i < ii; ++i) {
+      if (levels[i] < level) {
+        if (start >= 0) {
+          reverseValues(chars, start, i);
+          start = -1;
+        }
+      } else if (start < 0) {
+        start = i;
+      }
+    }
+    if (start >= 0) {
+      reverseValues(chars, start, levels.length);
+    }
   }
- }
- for (i = 0, ii = chars.length; i < ii; ++i) {
-  var ch = chars[i];
-  if (ch === '<' || ch === '>') {
-   chars[i] = '';
+  for (i = 0, ii = chars.length; i < ii; ++i) {
+    var ch = chars[i];
+    if (ch === '<' || ch === '>') {
+      chars[i] = '';
+    }
   }
- }
- return createBidiText(chars.join(''), isLTR);
+  return createBidiText(chars.join(''), isLTR);
 }
 exports.bidi = bidi;

File diff suppressed because it is too large
+ 1 - 393
lib/core/cff_parser.js


File diff suppressed because it is too large
+ 1 - 488
lib/core/charsets.js


+ 441 - 439
lib/core/chunked_stream.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var MissingDataException = sharedUtil.MissingDataException;
 var arrayByteLength = sharedUtil.arrayByteLength;
@@ -22,450 +23,451 @@ var createPromiseCapability = sharedUtil.createPromiseCapability;
 var isInt = sharedUtil.isInt;
 var isEmptyObj = sharedUtil.isEmptyObj;
 var ChunkedStream = function ChunkedStreamClosure() {
- function ChunkedStream(length, chunkSize, manager) {
-  this.bytes = new Uint8Array(length);
-  this.start = 0;
-  this.pos = 0;
-  this.end = length;
-  this.chunkSize = chunkSize;
-  this.loadedChunks = [];
-  this.numChunksLoaded = 0;
-  this.numChunks = Math.ceil(length / chunkSize);
-  this.manager = manager;
-  this.progressiveDataLength = 0;
-  this.lastSuccessfulEnsureByteChunk = -1;
- }
- ChunkedStream.prototype = {
-  getMissingChunks: function ChunkedStream_getMissingChunks() {
-   var chunks = [];
-   for (var chunk = 0, n = this.numChunks; chunk < n; ++chunk) {
-    if (!this.loadedChunks[chunk]) {
-     chunks.push(chunk);
-    }
-   }
-   return chunks;
-  },
-  getBaseStreams: function ChunkedStream_getBaseStreams() {
-   return [this];
-  },
-  allChunksLoaded: function ChunkedStream_allChunksLoaded() {
-   return this.numChunksLoaded === this.numChunks;
-  },
-  onReceiveData: function ChunkedStream_onReceiveData(begin, chunk) {
-   var end = begin + chunk.byteLength;
-   assert(begin % this.chunkSize === 0, 'Bad begin offset: ' + begin);
-   var length = this.bytes.length;
-   assert(end % this.chunkSize === 0 || end === length, 'Bad end offset: ' + end);
-   this.bytes.set(new Uint8Array(chunk), begin);
-   var chunkSize = this.chunkSize;
-   var beginChunk = Math.floor(begin / chunkSize);
-   var endChunk = Math.floor((end - 1) / chunkSize) + 1;
-   var curChunk;
-   for (curChunk = beginChunk; curChunk < endChunk; ++curChunk) {
-    if (!this.loadedChunks[curChunk]) {
-     this.loadedChunks[curChunk] = true;
-     ++this.numChunksLoaded;
-    }
-   }
-  },
-  onReceiveProgressiveData: function ChunkedStream_onReceiveProgressiveData(data) {
-   var position = this.progressiveDataLength;
-   var beginChunk = Math.floor(position / this.chunkSize);
-   this.bytes.set(new Uint8Array(data), position);
-   position += data.byteLength;
-   this.progressiveDataLength = position;
-   var endChunk = position >= this.end ? this.numChunks : Math.floor(position / this.chunkSize);
-   var curChunk;
-   for (curChunk = beginChunk; curChunk < endChunk; ++curChunk) {
-    if (!this.loadedChunks[curChunk]) {
-     this.loadedChunks[curChunk] = true;
-     ++this.numChunksLoaded;
-    }
-   }
-  },
-  ensureByte: function ChunkedStream_ensureByte(pos) {
-   var chunk = Math.floor(pos / this.chunkSize);
-   if (chunk === this.lastSuccessfulEnsureByteChunk) {
-    return;
-   }
-   if (!this.loadedChunks[chunk]) {
-    throw new MissingDataException(pos, pos + 1);
-   }
-   this.lastSuccessfulEnsureByteChunk = chunk;
-  },
-  ensureRange: function ChunkedStream_ensureRange(begin, end) {
-   if (begin >= end) {
-    return;
-   }
-   if (end <= this.progressiveDataLength) {
-    return;
-   }
-   var chunkSize = this.chunkSize;
-   var beginChunk = Math.floor(begin / chunkSize);
-   var endChunk = Math.floor((end - 1) / chunkSize) + 1;
-   for (var chunk = beginChunk; chunk < endChunk; ++chunk) {
-    if (!this.loadedChunks[chunk]) {
-     throw new MissingDataException(begin, end);
-    }
-   }
-  },
-  nextEmptyChunk: function ChunkedStream_nextEmptyChunk(beginChunk) {
-   var chunk, numChunks = this.numChunks;
-   for (var i = 0; i < numChunks; ++i) {
-    chunk = (beginChunk + i) % numChunks;
-    if (!this.loadedChunks[chunk]) {
-     return chunk;
-    }
-   }
-   return null;
-  },
-  hasChunk: function ChunkedStream_hasChunk(chunk) {
-   return !!this.loadedChunks[chunk];
-  },
-  get length() {
-   return this.end - this.start;
-  },
-  get isEmpty() {
-   return this.length === 0;
-  },
-  getByte: function ChunkedStream_getByte() {
-   var pos = this.pos;
-   if (pos >= this.end) {
-    return -1;
-   }
-   this.ensureByte(pos);
-   return this.bytes[this.pos++];
-  },
-  getUint16: function ChunkedStream_getUint16() {
-   var b0 = this.getByte();
-   var b1 = this.getByte();
-   if (b0 === -1 || b1 === -1) {
-    return -1;
-   }
-   return (b0 << 8) + b1;
-  },
-  getInt32: function ChunkedStream_getInt32() {
-   var b0 = this.getByte();
-   var b1 = this.getByte();
-   var b2 = this.getByte();
-   var b3 = this.getByte();
-   return (b0 << 24) + (b1 << 16) + (b2 << 8) + b3;
-  },
-  getBytes: function ChunkedStream_getBytes(length) {
-   var bytes = this.bytes;
-   var pos = this.pos;
-   var strEnd = this.end;
-   if (!length) {
-    this.ensureRange(pos, strEnd);
-    return bytes.subarray(pos, strEnd);
-   }
-   var end = pos + length;
-   if (end > strEnd) {
-    end = strEnd;
-   }
-   this.ensureRange(pos, end);
-   this.pos = end;
-   return bytes.subarray(pos, end);
-  },
-  peekByte: function ChunkedStream_peekByte() {
-   var peekedByte = this.getByte();
-   this.pos--;
-   return peekedByte;
-  },
-  peekBytes: function ChunkedStream_peekBytes(length) {
-   var bytes = this.getBytes(length);
-   this.pos -= bytes.length;
-   return bytes;
-  },
-  getByteRange: function ChunkedStream_getBytes(begin, end) {
-   this.ensureRange(begin, end);
-   return this.bytes.subarray(begin, end);
-  },
-  skip: function ChunkedStream_skip(n) {
-   if (!n) {
-    n = 1;
-   }
-   this.pos += n;
-  },
-  reset: function ChunkedStream_reset() {
-   this.pos = this.start;
-  },
-  moveStart: function ChunkedStream_moveStart() {
-   this.start = this.pos;
-  },
-  makeSubStream: function ChunkedStream_makeSubStream(start, length, dict) {
-   this.ensureRange(start, start + length);
-   function ChunkedStreamSubstream() {
-   }
-   ChunkedStreamSubstream.prototype = Object.create(this);
-   ChunkedStreamSubstream.prototype.getMissingChunks = function () {
-    var chunkSize = this.chunkSize;
-    var beginChunk = Math.floor(this.start / chunkSize);
-    var endChunk = Math.floor((this.end - 1) / chunkSize) + 1;
-    var missingChunks = [];
-    for (var chunk = beginChunk; chunk < endChunk; ++chunk) {
-     if (!this.loadedChunks[chunk]) {
-      missingChunks.push(chunk);
-     }
-    }
-    return missingChunks;
-   };
-   var subStream = new ChunkedStreamSubstream();
-   subStream.pos = subStream.start = start;
-   subStream.end = start + length || this.end;
-   subStream.dict = dict;
-   return subStream;
+  function ChunkedStream(length, chunkSize, manager) {
+    this.bytes = new Uint8Array(length);
+    this.start = 0;
+    this.pos = 0;
+    this.end = length;
+    this.chunkSize = chunkSize;
+    this.loadedChunks = [];
+    this.numChunksLoaded = 0;
+    this.numChunks = Math.ceil(length / chunkSize);
+    this.manager = manager;
+    this.progressiveDataLength = 0;
+    this.lastSuccessfulEnsureByteChunk = -1;
   }
- };
- return ChunkedStream;
+  ChunkedStream.prototype = {
+    getMissingChunks: function ChunkedStream_getMissingChunks() {
+      var chunks = [];
+      for (var chunk = 0, n = this.numChunks; chunk < n; ++chunk) {
+        if (!this.loadedChunks[chunk]) {
+          chunks.push(chunk);
+        }
+      }
+      return chunks;
+    },
+    getBaseStreams: function ChunkedStream_getBaseStreams() {
+      return [this];
+    },
+    allChunksLoaded: function ChunkedStream_allChunksLoaded() {
+      return this.numChunksLoaded === this.numChunks;
+    },
+    onReceiveData: function ChunkedStream_onReceiveData(begin, chunk) {
+      var end = begin + chunk.byteLength;
+      assert(begin % this.chunkSize === 0, 'Bad begin offset: ' + begin);
+      var length = this.bytes.length;
+      assert(end % this.chunkSize === 0 || end === length, 'Bad end offset: ' + end);
+      this.bytes.set(new Uint8Array(chunk), begin);
+      var chunkSize = this.chunkSize;
+      var beginChunk = Math.floor(begin / chunkSize);
+      var endChunk = Math.floor((end - 1) / chunkSize) + 1;
+      var curChunk;
+      for (curChunk = beginChunk; curChunk < endChunk; ++curChunk) {
+        if (!this.loadedChunks[curChunk]) {
+          this.loadedChunks[curChunk] = true;
+          ++this.numChunksLoaded;
+        }
+      }
+    },
+    onReceiveProgressiveData: function ChunkedStream_onReceiveProgressiveData(data) {
+      var position = this.progressiveDataLength;
+      var beginChunk = Math.floor(position / this.chunkSize);
+      this.bytes.set(new Uint8Array(data), position);
+      position += data.byteLength;
+      this.progressiveDataLength = position;
+      var endChunk = position >= this.end ? this.numChunks : Math.floor(position / this.chunkSize);
+      var curChunk;
+      for (curChunk = beginChunk; curChunk < endChunk; ++curChunk) {
+        if (!this.loadedChunks[curChunk]) {
+          this.loadedChunks[curChunk] = true;
+          ++this.numChunksLoaded;
+        }
+      }
+    },
+    ensureByte: function ChunkedStream_ensureByte(pos) {
+      var chunk = Math.floor(pos / this.chunkSize);
+      if (chunk === this.lastSuccessfulEnsureByteChunk) {
+        return;
+      }
+      if (!this.loadedChunks[chunk]) {
+        throw new MissingDataException(pos, pos + 1);
+      }
+      this.lastSuccessfulEnsureByteChunk = chunk;
+    },
+    ensureRange: function ChunkedStream_ensureRange(begin, end) {
+      if (begin >= end) {
+        return;
+      }
+      if (end <= this.progressiveDataLength) {
+        return;
+      }
+      var chunkSize = this.chunkSize;
+      var beginChunk = Math.floor(begin / chunkSize);
+      var endChunk = Math.floor((end - 1) / chunkSize) + 1;
+      for (var chunk = beginChunk; chunk < endChunk; ++chunk) {
+        if (!this.loadedChunks[chunk]) {
+          throw new MissingDataException(begin, end);
+        }
+      }
+    },
+    nextEmptyChunk: function ChunkedStream_nextEmptyChunk(beginChunk) {
+      var chunk,
+          numChunks = this.numChunks;
+      for (var i = 0; i < numChunks; ++i) {
+        chunk = (beginChunk + i) % numChunks;
+        if (!this.loadedChunks[chunk]) {
+          return chunk;
+        }
+      }
+      return null;
+    },
+    hasChunk: function ChunkedStream_hasChunk(chunk) {
+      return !!this.loadedChunks[chunk];
+    },
+    get length() {
+      return this.end - this.start;
+    },
+    get isEmpty() {
+      return this.length === 0;
+    },
+    getByte: function ChunkedStream_getByte() {
+      var pos = this.pos;
+      if (pos >= this.end) {
+        return -1;
+      }
+      this.ensureByte(pos);
+      return this.bytes[this.pos++];
+    },
+    getUint16: function ChunkedStream_getUint16() {
+      var b0 = this.getByte();
+      var b1 = this.getByte();
+      if (b0 === -1 || b1 === -1) {
+        return -1;
+      }
+      return (b0 << 8) + b1;
+    },
+    getInt32: function ChunkedStream_getInt32() {
+      var b0 = this.getByte();
+      var b1 = this.getByte();
+      var b2 = this.getByte();
+      var b3 = this.getByte();
+      return (b0 << 24) + (b1 << 16) + (b2 << 8) + b3;
+    },
+    getBytes: function ChunkedStream_getBytes(length) {
+      var bytes = this.bytes;
+      var pos = this.pos;
+      var strEnd = this.end;
+      if (!length) {
+        this.ensureRange(pos, strEnd);
+        return bytes.subarray(pos, strEnd);
+      }
+      var end = pos + length;
+      if (end > strEnd) {
+        end = strEnd;
+      }
+      this.ensureRange(pos, end);
+      this.pos = end;
+      return bytes.subarray(pos, end);
+    },
+    peekByte: function ChunkedStream_peekByte() {
+      var peekedByte = this.getByte();
+      this.pos--;
+      return peekedByte;
+    },
+    peekBytes: function ChunkedStream_peekBytes(length) {
+      var bytes = this.getBytes(length);
+      this.pos -= bytes.length;
+      return bytes;
+    },
+    getByteRange: function ChunkedStream_getBytes(begin, end) {
+      this.ensureRange(begin, end);
+      return this.bytes.subarray(begin, end);
+    },
+    skip: function ChunkedStream_skip(n) {
+      if (!n) {
+        n = 1;
+      }
+      this.pos += n;
+    },
+    reset: function ChunkedStream_reset() {
+      this.pos = this.start;
+    },
+    moveStart: function ChunkedStream_moveStart() {
+      this.start = this.pos;
+    },
+    makeSubStream: function ChunkedStream_makeSubStream(start, length, dict) {
+      this.ensureRange(start, start + length);
+      function ChunkedStreamSubstream() {}
+      ChunkedStreamSubstream.prototype = Object.create(this);
+      ChunkedStreamSubstream.prototype.getMissingChunks = function () {
+        var chunkSize = this.chunkSize;
+        var beginChunk = Math.floor(this.start / chunkSize);
+        var endChunk = Math.floor((this.end - 1) / chunkSize) + 1;
+        var missingChunks = [];
+        for (var chunk = beginChunk; chunk < endChunk; ++chunk) {
+          if (!this.loadedChunks[chunk]) {
+            missingChunks.push(chunk);
+          }
+        }
+        return missingChunks;
+      };
+      var subStream = new ChunkedStreamSubstream();
+      subStream.pos = subStream.start = start;
+      subStream.end = start + length || this.end;
+      subStream.dict = dict;
+      return subStream;
+    }
+  };
+  return ChunkedStream;
 }();
 var ChunkedStreamManager = function ChunkedStreamManagerClosure() {
- function ChunkedStreamManager(pdfNetworkStream, args) {
-  var chunkSize = args.rangeChunkSize;
-  var length = args.length;
-  this.stream = new ChunkedStream(length, chunkSize, this);
-  this.length = length;
-  this.chunkSize = chunkSize;
-  this.pdfNetworkStream = pdfNetworkStream;
-  this.url = args.url;
-  this.disableAutoFetch = args.disableAutoFetch;
-  this.msgHandler = args.msgHandler;
-  this.currRequestId = 0;
-  this.chunksNeededByRequest = Object.create(null);
-  this.requestsByChunk = Object.create(null);
-  this.promisesByRequest = Object.create(null);
-  this.progressiveDataLength = 0;
-  this.aborted = false;
-  this._loadedStreamCapability = createPromiseCapability();
- }
- ChunkedStreamManager.prototype = {
-  onLoadedStream: function ChunkedStreamManager_getLoadedStream() {
-   return this._loadedStreamCapability.promise;
-  },
-  sendRequest: function ChunkedStreamManager_sendRequest(begin, end) {
-   var rangeReader = this.pdfNetworkStream.getRangeReader(begin, end);
-   if (!rangeReader.isStreamingSupported) {
-    rangeReader.onProgress = this.onProgress.bind(this);
-   }
-   var chunks = [], loaded = 0;
-   var manager = this;
-   var promise = new Promise(function (resolve, reject) {
-    var readChunk = function (chunk) {
-     try {
-      if (!chunk.done) {
-       var data = chunk.value;
-       chunks.push(data);
-       loaded += arrayByteLength(data);
-       if (rangeReader.isStreamingSupported) {
-        manager.onProgress({ loaded: loaded });
-       }
-       rangeReader.read().then(readChunk, reject);
-       return;
+  function ChunkedStreamManager(pdfNetworkStream, args) {
+    var chunkSize = args.rangeChunkSize;
+    var length = args.length;
+    this.stream = new ChunkedStream(length, chunkSize, this);
+    this.length = length;
+    this.chunkSize = chunkSize;
+    this.pdfNetworkStream = pdfNetworkStream;
+    this.url = args.url;
+    this.disableAutoFetch = args.disableAutoFetch;
+    this.msgHandler = args.msgHandler;
+    this.currRequestId = 0;
+    this.chunksNeededByRequest = Object.create(null);
+    this.requestsByChunk = Object.create(null);
+    this.promisesByRequest = Object.create(null);
+    this.progressiveDataLength = 0;
+    this.aborted = false;
+    this._loadedStreamCapability = createPromiseCapability();
+  }
+  ChunkedStreamManager.prototype = {
+    onLoadedStream: function ChunkedStreamManager_getLoadedStream() {
+      return this._loadedStreamCapability.promise;
+    },
+    sendRequest: function ChunkedStreamManager_sendRequest(begin, end) {
+      var rangeReader = this.pdfNetworkStream.getRangeReader(begin, end);
+      if (!rangeReader.isStreamingSupported) {
+        rangeReader.onProgress = this.onProgress.bind(this);
+      }
+      var chunks = [],
+          loaded = 0;
+      var manager = this;
+      var promise = new Promise(function (resolve, reject) {
+        var readChunk = function (chunk) {
+          try {
+            if (!chunk.done) {
+              var data = chunk.value;
+              chunks.push(data);
+              loaded += arrayByteLength(data);
+              if (rangeReader.isStreamingSupported) {
+                manager.onProgress({ loaded: loaded });
+              }
+              rangeReader.read().then(readChunk, reject);
+              return;
+            }
+            var chunkData = arraysToBytes(chunks);
+            chunks = null;
+            resolve(chunkData);
+          } catch (e) {
+            reject(e);
+          }
+        };
+        rangeReader.read().then(readChunk, reject);
+      });
+      promise.then(function (data) {
+        if (this.aborted) {
+          return;
+        }
+        this.onReceiveData({
+          chunk: data,
+          begin: begin
+        });
+      }.bind(this));
+    },
+    requestAllChunks: function ChunkedStreamManager_requestAllChunks() {
+      var missingChunks = this.stream.getMissingChunks();
+      this._requestChunks(missingChunks);
+      return this._loadedStreamCapability.promise;
+    },
+    _requestChunks: function ChunkedStreamManager_requestChunks(chunks) {
+      var requestId = this.currRequestId++;
+      var i, ii;
+      var chunksNeeded = Object.create(null);
+      this.chunksNeededByRequest[requestId] = chunksNeeded;
+      for (i = 0, ii = chunks.length; i < ii; i++) {
+        if (!this.stream.hasChunk(chunks[i])) {
+          chunksNeeded[chunks[i]] = true;
+        }
+      }
+      if (isEmptyObj(chunksNeeded)) {
+        return Promise.resolve();
+      }
+      var capability = createPromiseCapability();
+      this.promisesByRequest[requestId] = capability;
+      var chunksToRequest = [];
+      for (var chunk in chunksNeeded) {
+        chunk = chunk | 0;
+        if (!(chunk in this.requestsByChunk)) {
+          this.requestsByChunk[chunk] = [];
+          chunksToRequest.push(chunk);
+        }
+        this.requestsByChunk[chunk].push(requestId);
+      }
+      if (!chunksToRequest.length) {
+        return capability.promise;
+      }
+      var groupedChunksToRequest = this.groupChunks(chunksToRequest);
+      for (i = 0; i < groupedChunksToRequest.length; ++i) {
+        var groupedChunk = groupedChunksToRequest[i];
+        var begin = groupedChunk.beginChunk * this.chunkSize;
+        var end = Math.min(groupedChunk.endChunk * this.chunkSize, this.length);
+        this.sendRequest(begin, end);
+      }
+      return capability.promise;
+    },
+    getStream: function ChunkedStreamManager_getStream() {
+      return this.stream;
+    },
+    requestRange: function ChunkedStreamManager_requestRange(begin, end) {
+      end = Math.min(end, this.length);
+      var beginChunk = this.getBeginChunk(begin);
+      var endChunk = this.getEndChunk(end);
+      var chunks = [];
+      for (var chunk = beginChunk; chunk < endChunk; ++chunk) {
+        chunks.push(chunk);
+      }
+      return this._requestChunks(chunks);
+    },
+    requestRanges: function ChunkedStreamManager_requestRanges(ranges) {
+      ranges = ranges || [];
+      var chunksToRequest = [];
+      for (var i = 0; i < ranges.length; i++) {
+        var beginChunk = this.getBeginChunk(ranges[i].begin);
+        var endChunk = this.getEndChunk(ranges[i].end);
+        for (var chunk = beginChunk; chunk < endChunk; ++chunk) {
+          if (chunksToRequest.indexOf(chunk) < 0) {
+            chunksToRequest.push(chunk);
+          }
+        }
+      }
+      chunksToRequest.sort(function (a, b) {
+        return a - b;
+      });
+      return this._requestChunks(chunksToRequest);
+    },
+    groupChunks: function ChunkedStreamManager_groupChunks(chunks) {
+      var groupedChunks = [];
+      var beginChunk = -1;
+      var prevChunk = -1;
+      for (var i = 0; i < chunks.length; ++i) {
+        var chunk = chunks[i];
+        if (beginChunk < 0) {
+          beginChunk = chunk;
+        }
+        if (prevChunk >= 0 && prevChunk + 1 !== chunk) {
+          groupedChunks.push({
+            beginChunk: beginChunk,
+            endChunk: prevChunk + 1
+          });
+          beginChunk = chunk;
+        }
+        if (i + 1 === chunks.length) {
+          groupedChunks.push({
+            beginChunk: beginChunk,
+            endChunk: chunk + 1
+          });
+        }
+        prevChunk = chunk;
+      }
+      return groupedChunks;
+    },
+    onProgress: function ChunkedStreamManager_onProgress(args) {
+      var bytesLoaded = this.stream.numChunksLoaded * this.chunkSize + args.loaded;
+      this.msgHandler.send('DocProgress', {
+        loaded: bytesLoaded,
+        total: this.length
+      });
+    },
+    onReceiveData: function ChunkedStreamManager_onReceiveData(args) {
+      var chunk = args.chunk;
+      var isProgressive = args.begin === undefined;
+      var begin = isProgressive ? this.progressiveDataLength : args.begin;
+      var end = begin + chunk.byteLength;
+      var beginChunk = Math.floor(begin / this.chunkSize);
+      var endChunk = end < this.length ? Math.floor(end / this.chunkSize) : Math.ceil(end / this.chunkSize);
+      if (isProgressive) {
+        this.stream.onReceiveProgressiveData(chunk);
+        this.progressiveDataLength = end;
+      } else {
+        this.stream.onReceiveData(begin, chunk);
+      }
+      if (this.stream.allChunksLoaded()) {
+        this._loadedStreamCapability.resolve(this.stream);
+      }
+      var loadedRequests = [];
+      var i, requestId;
+      for (chunk = beginChunk; chunk < endChunk; ++chunk) {
+        var requestIds = this.requestsByChunk[chunk] || [];
+        delete this.requestsByChunk[chunk];
+        for (i = 0; i < requestIds.length; ++i) {
+          requestId = requestIds[i];
+          var chunksNeeded = this.chunksNeededByRequest[requestId];
+          if (chunk in chunksNeeded) {
+            delete chunksNeeded[chunk];
+          }
+          if (!isEmptyObj(chunksNeeded)) {
+            continue;
+          }
+          loadedRequests.push(requestId);
+        }
+      }
+      if (!this.disableAutoFetch && isEmptyObj(this.requestsByChunk)) {
+        var nextEmptyChunk;
+        if (this.stream.numChunksLoaded === 1) {
+          var lastChunk = this.stream.numChunks - 1;
+          if (!this.stream.hasChunk(lastChunk)) {
+            nextEmptyChunk = lastChunk;
+          }
+        } else {
+          nextEmptyChunk = this.stream.nextEmptyChunk(endChunk);
+        }
+        if (isInt(nextEmptyChunk)) {
+          this._requestChunks([nextEmptyChunk]);
+        }
+      }
+      for (i = 0; i < loadedRequests.length; ++i) {
+        requestId = loadedRequests[i];
+        var capability = this.promisesByRequest[requestId];
+        delete this.promisesByRequest[requestId];
+        capability.resolve();
+      }
+      this.msgHandler.send('DocProgress', {
+        loaded: this.stream.numChunksLoaded * this.chunkSize,
+        total: this.length
+      });
+    },
+    onError: function ChunkedStreamManager_onError(err) {
+      this._loadedStreamCapability.reject(err);
+    },
+    getBeginChunk: function ChunkedStreamManager_getBeginChunk(begin) {
+      var chunk = Math.floor(begin / this.chunkSize);
+      return chunk;
+    },
+    getEndChunk: function ChunkedStreamManager_getEndChunk(end) {
+      var chunk = Math.floor((end - 1) / this.chunkSize) + 1;
+      return chunk;
+    },
+    abort: function ChunkedStreamManager_abort() {
+      this.aborted = true;
+      if (this.pdfNetworkStream) {
+        this.pdfNetworkStream.cancelAllRequests('abort');
+      }
+      for (var requestId in this.promisesByRequest) {
+        var capability = this.promisesByRequest[requestId];
+        capability.reject(new Error('Request was aborted'));
       }
-      var chunkData = arraysToBytes(chunks);
-      chunks = null;
-      resolve(chunkData);
-     } catch (e) {
-      reject(e);
-     }
-    };
-    rangeReader.read().then(readChunk, reject);
-   });
-   promise.then(function (data) {
-    if (this.aborted) {
-     return;
-    }
-    this.onReceiveData({
-     chunk: data,
-     begin: begin
-    });
-   }.bind(this));
-  },
-  requestAllChunks: function ChunkedStreamManager_requestAllChunks() {
-   var missingChunks = this.stream.getMissingChunks();
-   this._requestChunks(missingChunks);
-   return this._loadedStreamCapability.promise;
-  },
-  _requestChunks: function ChunkedStreamManager_requestChunks(chunks) {
-   var requestId = this.currRequestId++;
-   var i, ii;
-   var chunksNeeded = Object.create(null);
-   this.chunksNeededByRequest[requestId] = chunksNeeded;
-   for (i = 0, ii = chunks.length; i < ii; i++) {
-    if (!this.stream.hasChunk(chunks[i])) {
-     chunksNeeded[chunks[i]] = true;
-    }
-   }
-   if (isEmptyObj(chunksNeeded)) {
-    return Promise.resolve();
-   }
-   var capability = createPromiseCapability();
-   this.promisesByRequest[requestId] = capability;
-   var chunksToRequest = [];
-   for (var chunk in chunksNeeded) {
-    chunk = chunk | 0;
-    if (!(chunk in this.requestsByChunk)) {
-     this.requestsByChunk[chunk] = [];
-     chunksToRequest.push(chunk);
-    }
-    this.requestsByChunk[chunk].push(requestId);
-   }
-   if (!chunksToRequest.length) {
-    return capability.promise;
-   }
-   var groupedChunksToRequest = this.groupChunks(chunksToRequest);
-   for (i = 0; i < groupedChunksToRequest.length; ++i) {
-    var groupedChunk = groupedChunksToRequest[i];
-    var begin = groupedChunk.beginChunk * this.chunkSize;
-    var end = Math.min(groupedChunk.endChunk * this.chunkSize, this.length);
-    this.sendRequest(begin, end);
-   }
-   return capability.promise;
-  },
-  getStream: function ChunkedStreamManager_getStream() {
-   return this.stream;
-  },
-  requestRange: function ChunkedStreamManager_requestRange(begin, end) {
-   end = Math.min(end, this.length);
-   var beginChunk = this.getBeginChunk(begin);
-   var endChunk = this.getEndChunk(end);
-   var chunks = [];
-   for (var chunk = beginChunk; chunk < endChunk; ++chunk) {
-    chunks.push(chunk);
-   }
-   return this._requestChunks(chunks);
-  },
-  requestRanges: function ChunkedStreamManager_requestRanges(ranges) {
-   ranges = ranges || [];
-   var chunksToRequest = [];
-   for (var i = 0; i < ranges.length; i++) {
-    var beginChunk = this.getBeginChunk(ranges[i].begin);
-    var endChunk = this.getEndChunk(ranges[i].end);
-    for (var chunk = beginChunk; chunk < endChunk; ++chunk) {
-     if (chunksToRequest.indexOf(chunk) < 0) {
-      chunksToRequest.push(chunk);
-     }
-    }
-   }
-   chunksToRequest.sort(function (a, b) {
-    return a - b;
-   });
-   return this._requestChunks(chunksToRequest);
-  },
-  groupChunks: function ChunkedStreamManager_groupChunks(chunks) {
-   var groupedChunks = [];
-   var beginChunk = -1;
-   var prevChunk = -1;
-   for (var i = 0; i < chunks.length; ++i) {
-    var chunk = chunks[i];
-    if (beginChunk < 0) {
-     beginChunk = chunk;
-    }
-    if (prevChunk >= 0 && prevChunk + 1 !== chunk) {
-     groupedChunks.push({
-      beginChunk: beginChunk,
-      endChunk: prevChunk + 1
-     });
-     beginChunk = chunk;
-    }
-    if (i + 1 === chunks.length) {
-     groupedChunks.push({
-      beginChunk: beginChunk,
-      endChunk: chunk + 1
-     });
-    }
-    prevChunk = chunk;
-   }
-   return groupedChunks;
-  },
-  onProgress: function ChunkedStreamManager_onProgress(args) {
-   var bytesLoaded = this.stream.numChunksLoaded * this.chunkSize + args.loaded;
-   this.msgHandler.send('DocProgress', {
-    loaded: bytesLoaded,
-    total: this.length
-   });
-  },
-  onReceiveData: function ChunkedStreamManager_onReceiveData(args) {
-   var chunk = args.chunk;
-   var isProgressive = args.begin === undefined;
-   var begin = isProgressive ? this.progressiveDataLength : args.begin;
-   var end = begin + chunk.byteLength;
-   var beginChunk = Math.floor(begin / this.chunkSize);
-   var endChunk = end < this.length ? Math.floor(end / this.chunkSize) : Math.ceil(end / this.chunkSize);
-   if (isProgressive) {
-    this.stream.onReceiveProgressiveData(chunk);
-    this.progressiveDataLength = end;
-   } else {
-    this.stream.onReceiveData(begin, chunk);
-   }
-   if (this.stream.allChunksLoaded()) {
-    this._loadedStreamCapability.resolve(this.stream);
-   }
-   var loadedRequests = [];
-   var i, requestId;
-   for (chunk = beginChunk; chunk < endChunk; ++chunk) {
-    var requestIds = this.requestsByChunk[chunk] || [];
-    delete this.requestsByChunk[chunk];
-    for (i = 0; i < requestIds.length; ++i) {
-     requestId = requestIds[i];
-     var chunksNeeded = this.chunksNeededByRequest[requestId];
-     if (chunk in chunksNeeded) {
-      delete chunksNeeded[chunk];
-     }
-     if (!isEmptyObj(chunksNeeded)) {
-      continue;
-     }
-     loadedRequests.push(requestId);
-    }
-   }
-   if (!this.disableAutoFetch && isEmptyObj(this.requestsByChunk)) {
-    var nextEmptyChunk;
-    if (this.stream.numChunksLoaded === 1) {
-     var lastChunk = this.stream.numChunks - 1;
-     if (!this.stream.hasChunk(lastChunk)) {
-      nextEmptyChunk = lastChunk;
-     }
-    } else {
-     nextEmptyChunk = this.stream.nextEmptyChunk(endChunk);
-    }
-    if (isInt(nextEmptyChunk)) {
-     this._requestChunks([nextEmptyChunk]);
     }
-   }
-   for (i = 0; i < loadedRequests.length; ++i) {
-    requestId = loadedRequests[i];
-    var capability = this.promisesByRequest[requestId];
-    delete this.promisesByRequest[requestId];
-    capability.resolve();
-   }
-   this.msgHandler.send('DocProgress', {
-    loaded: this.stream.numChunksLoaded * this.chunkSize,
-    total: this.length
-   });
-  },
-  onError: function ChunkedStreamManager_onError(err) {
-   this._loadedStreamCapability.reject(err);
-  },
-  getBeginChunk: function ChunkedStreamManager_getBeginChunk(begin) {
-   var chunk = Math.floor(begin / this.chunkSize);
-   return chunk;
-  },
-  getEndChunk: function ChunkedStreamManager_getEndChunk(end) {
-   var chunk = Math.floor((end - 1) / this.chunkSize) + 1;
-   return chunk;
-  },
-  abort: function ChunkedStreamManager_abort() {
-   this.aborted = true;
-   if (this.pdfNetworkStream) {
-    this.pdfNetworkStream.cancelAllRequests('abort');
-   }
-   for (var requestId in this.promisesByRequest) {
-    var capability = this.promisesByRequest[requestId];
-    capability.reject(new Error('Request was aborted'));
-   }
-  }
- };
- return ChunkedStreamManager;
+  };
+  return ChunkedStreamManager;
 }();
 exports.ChunkedStream = ChunkedStream;
 exports.ChunkedStreamManager = ChunkedStreamManager;

File diff suppressed because it is too large
+ 1 - 170
lib/core/cmap.js


+ 878 - 964
lib/core/colorspace.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var coreFunction = require('./function.js');
@@ -27,1004 +28,917 @@ var isName = corePrimitives.isName;
 var isStream = corePrimitives.isStream;
 var PDFFunction = coreFunction.PDFFunction;
 var ColorSpace = function ColorSpaceClosure() {
- function resizeRgbImage(src, bpc, w1, h1, w2, h2, alpha01, dest) {
-  var COMPONENTS = 3;
-  alpha01 = alpha01 !== 1 ? 0 : alpha01;
-  var xRatio = w1 / w2;
-  var yRatio = h1 / h2;
-  var i, j, py, newIndex = 0, oldIndex;
-  var xScaled = new Uint16Array(w2);
-  var w1Scanline = w1 * COMPONENTS;
-  for (i = 0; i < w2; i++) {
-   xScaled[i] = Math.floor(i * xRatio) * COMPONENTS;
-  }
-  for (i = 0; i < h2; i++) {
-   py = Math.floor(i * yRatio) * w1Scanline;
-   for (j = 0; j < w2; j++) {
-    oldIndex = py + xScaled[j];
-    dest[newIndex++] = src[oldIndex++];
-    dest[newIndex++] = src[oldIndex++];
-    dest[newIndex++] = src[oldIndex++];
-    newIndex += alpha01;
-   }
+  function resizeRgbImage(src, bpc, w1, h1, w2, h2, alpha01, dest) {
+    var COMPONENTS = 3;
+    alpha01 = alpha01 !== 1 ? 0 : alpha01;
+    var xRatio = w1 / w2;
+    var yRatio = h1 / h2;
+    var i,
+        j,
+        py,
+        newIndex = 0,
+        oldIndex;
+    var xScaled = new Uint16Array(w2);
+    var w1Scanline = w1 * COMPONENTS;
+    for (i = 0; i < w2; i++) {
+      xScaled[i] = Math.floor(i * xRatio) * COMPONENTS;
+    }
+    for (i = 0; i < h2; i++) {
+      py = Math.floor(i * yRatio) * w1Scanline;
+      for (j = 0; j < w2; j++) {
+        oldIndex = py + xScaled[j];
+        dest[newIndex++] = src[oldIndex++];
+        dest[newIndex++] = src[oldIndex++];
+        dest[newIndex++] = src[oldIndex++];
+        newIndex += alpha01;
+      }
+    }
   }
- }
- function ColorSpace() {
-  error('should not call ColorSpace constructor');
- }
- ColorSpace.prototype = {
-  getRgb: function ColorSpace_getRgb(src, srcOffset) {
-   var rgb = new Uint8Array(3);
-   this.getRgbItem(src, srcOffset, rgb, 0);
-   return rgb;
-  },
-  getRgbItem: function ColorSpace_getRgbItem(src, srcOffset, dest, destOffset) {
-   error('Should not call ColorSpace.getRgbItem');
-  },
-  getRgbBuffer: function ColorSpace_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
-   error('Should not call ColorSpace.getRgbBuffer');
-  },
-  getOutputLength: function ColorSpace_getOutputLength(inputLength, alpha01) {
-   error('Should not call ColorSpace.getOutputLength');
-  },
-  isPassthrough: function ColorSpace_isPassthrough(bits) {
-   return false;
-  },
-  fillRgb: function ColorSpace_fillRgb(dest, originalWidth, originalHeight, width, height, actualHeight, bpc, comps, alpha01) {
-   var count = originalWidth * originalHeight;
-   var rgbBuf = null;
-   var numComponentColors = 1 << bpc;
-   var needsResizing = originalHeight !== height || originalWidth !== width;
-   var i, ii;
-   if (this.isPassthrough(bpc)) {
-    rgbBuf = comps;
-   } else if (this.numComps === 1 && count > numComponentColors && this.name !== 'DeviceGray' && this.name !== 'DeviceRGB') {
-    var allColors = bpc <= 8 ? new Uint8Array(numComponentColors) : new Uint16Array(numComponentColors);
-    var key;
-    for (i = 0; i < numComponentColors; i++) {
-     allColors[i] = i;
+  function ColorSpace() {
+    error('should not call ColorSpace constructor');
+  }
+  ColorSpace.prototype = {
+    getRgb: function ColorSpace_getRgb(src, srcOffset) {
+      var rgb = new Uint8Array(3);
+      this.getRgbItem(src, srcOffset, rgb, 0);
+      return rgb;
+    },
+    getRgbItem: function ColorSpace_getRgbItem(src, srcOffset, dest, destOffset) {
+      error('Should not call ColorSpace.getRgbItem');
+    },
+    getRgbBuffer: function ColorSpace_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
+      error('Should not call ColorSpace.getRgbBuffer');
+    },
+    getOutputLength: function ColorSpace_getOutputLength(inputLength, alpha01) {
+      error('Should not call ColorSpace.getOutputLength');
+    },
+    isPassthrough: function ColorSpace_isPassthrough(bits) {
+      return false;
+    },
+    fillRgb: function ColorSpace_fillRgb(dest, originalWidth, originalHeight, width, height, actualHeight, bpc, comps, alpha01) {
+      var count = originalWidth * originalHeight;
+      var rgbBuf = null;
+      var numComponentColors = 1 << bpc;
+      var needsResizing = originalHeight !== height || originalWidth !== width;
+      var i, ii;
+      if (this.isPassthrough(bpc)) {
+        rgbBuf = comps;
+      } else if (this.numComps === 1 && count > numComponentColors && this.name !== 'DeviceGray' && this.name !== 'DeviceRGB') {
+        var allColors = bpc <= 8 ? new Uint8Array(numComponentColors) : new Uint16Array(numComponentColors);
+        var key;
+        for (i = 0; i < numComponentColors; i++) {
+          allColors[i] = i;
+        }
+        var colorMap = new Uint8Array(numComponentColors * 3);
+        this.getRgbBuffer(allColors, 0, numComponentColors, colorMap, 0, bpc, 0);
+        var destPos, rgbPos;
+        if (!needsResizing) {
+          destPos = 0;
+          for (i = 0; i < count; ++i) {
+            key = comps[i] * 3;
+            dest[destPos++] = colorMap[key];
+            dest[destPos++] = colorMap[key + 1];
+            dest[destPos++] = colorMap[key + 2];
+            destPos += alpha01;
+          }
+        } else {
+          rgbBuf = new Uint8Array(count * 3);
+          rgbPos = 0;
+          for (i = 0; i < count; ++i) {
+            key = comps[i] * 3;
+            rgbBuf[rgbPos++] = colorMap[key];
+            rgbBuf[rgbPos++] = colorMap[key + 1];
+            rgbBuf[rgbPos++] = colorMap[key + 2];
+          }
+        }
+      } else {
+        if (!needsResizing) {
+          this.getRgbBuffer(comps, 0, width * actualHeight, dest, 0, bpc, alpha01);
+        } else {
+          rgbBuf = new Uint8Array(count * 3);
+          this.getRgbBuffer(comps, 0, count, rgbBuf, 0, bpc, 0);
+        }
+      }
+      if (rgbBuf) {
+        if (needsResizing) {
+          resizeRgbImage(rgbBuf, bpc, originalWidth, originalHeight, width, height, alpha01, dest);
+        } else {
+          rgbPos = 0;
+          destPos = 0;
+          for (i = 0, ii = width * actualHeight; i < ii; i++) {
+            dest[destPos++] = rgbBuf[rgbPos++];
+            dest[destPos++] = rgbBuf[rgbPos++];
+            dest[destPos++] = rgbBuf[rgbPos++];
+            destPos += alpha01;
+          }
+        }
+      }
+    },
+    usesZeroToOneRange: true
+  };
+  ColorSpace.parse = function ColorSpace_parse(cs, xref, res) {
+    var IR = ColorSpace.parseToIR(cs, xref, res);
+    if (IR instanceof AlternateCS) {
+      return IR;
     }
-    var colorMap = new Uint8Array(numComponentColors * 3);
-    this.getRgbBuffer(allColors, 0, numComponentColors, colorMap, 0, bpc, 0);
-    var destPos, rgbPos;
-    if (!needsResizing) {
-     destPos = 0;
-     for (i = 0; i < count; ++i) {
-      key = comps[i] * 3;
-      dest[destPos++] = colorMap[key];
-      dest[destPos++] = colorMap[key + 1];
-      dest[destPos++] = colorMap[key + 2];
-      destPos += alpha01;
-     }
-    } else {
-     rgbBuf = new Uint8Array(count * 3);
-     rgbPos = 0;
-     for (i = 0; i < count; ++i) {
-      key = comps[i] * 3;
-      rgbBuf[rgbPos++] = colorMap[key];
-      rgbBuf[rgbPos++] = colorMap[key + 1];
-      rgbBuf[rgbPos++] = colorMap[key + 2];
-     }
+    return ColorSpace.fromIR(IR);
+  };
+  ColorSpace.fromIR = function ColorSpace_fromIR(IR) {
+    var name = isArray(IR) ? IR[0] : IR;
+    var whitePoint, blackPoint, gamma;
+    switch (name) {
+      case 'DeviceGrayCS':
+        return this.singletons.gray;
+      case 'DeviceRgbCS':
+        return this.singletons.rgb;
+      case 'DeviceCmykCS':
+        return this.singletons.cmyk;
+      case 'CalGrayCS':
+        whitePoint = IR[1];
+        blackPoint = IR[2];
+        gamma = IR[3];
+        return new CalGrayCS(whitePoint, blackPoint, gamma);
+      case 'CalRGBCS':
+        whitePoint = IR[1];
+        blackPoint = IR[2];
+        gamma = IR[3];
+        var matrix = IR[4];
+        return new CalRGBCS(whitePoint, blackPoint, gamma, matrix);
+      case 'PatternCS':
+        var basePatternCS = IR[1];
+        if (basePatternCS) {
+          basePatternCS = ColorSpace.fromIR(basePatternCS);
+        }
+        return new PatternCS(basePatternCS);
+      case 'IndexedCS':
+        var baseIndexedCS = IR[1];
+        var hiVal = IR[2];
+        var lookup = IR[3];
+        return new IndexedCS(ColorSpace.fromIR(baseIndexedCS), hiVal, lookup);
+      case 'AlternateCS':
+        var numComps = IR[1];
+        var alt = IR[2];
+        var tintFnIR = IR[3];
+        return new AlternateCS(numComps, ColorSpace.fromIR(alt), PDFFunction.fromIR(tintFnIR));
+      case 'LabCS':
+        whitePoint = IR[1];
+        blackPoint = IR[2];
+        var range = IR[3];
+        return new LabCS(whitePoint, blackPoint, range);
+      default:
+        error('Unknown name ' + name);
     }
-   } else {
-    if (!needsResizing) {
-     this.getRgbBuffer(comps, 0, width * actualHeight, dest, 0, bpc, alpha01);
-    } else {
-     rgbBuf = new Uint8Array(count * 3);
-     this.getRgbBuffer(comps, 0, count, rgbBuf, 0, bpc, 0);
+    return null;
+  };
+  ColorSpace.parseToIR = function ColorSpace_parseToIR(cs, xref, res) {
+    if (isName(cs)) {
+      var colorSpaces = res.get('ColorSpace');
+      if (isDict(colorSpaces)) {
+        var refcs = colorSpaces.get(cs.name);
+        if (refcs) {
+          cs = refcs;
+        }
+      }
     }
-   }
-   if (rgbBuf) {
-    if (needsResizing) {
-     resizeRgbImage(rgbBuf, bpc, originalWidth, originalHeight, width, height, alpha01, dest);
+    cs = xref.fetchIfRef(cs);
+    if (isName(cs)) {
+      switch (cs.name) {
+        case 'DeviceGray':
+        case 'G':
+          return 'DeviceGrayCS';
+        case 'DeviceRGB':
+        case 'RGB':
+          return 'DeviceRgbCS';
+        case 'DeviceCMYK':
+        case 'CMYK':
+          return 'DeviceCmykCS';
+        case 'Pattern':
+          return ['PatternCS', null];
+        default:
+          error('unrecognized colorspace ' + cs.name);
+      }
+    } else if (isArray(cs)) {
+      var mode = xref.fetchIfRef(cs[0]).name;
+      var numComps, params, alt, whitePoint, blackPoint, gamma;
+      switch (mode) {
+        case 'DeviceGray':
+        case 'G':
+          return 'DeviceGrayCS';
+        case 'DeviceRGB':
+        case 'RGB':
+          return 'DeviceRgbCS';
+        case 'DeviceCMYK':
+        case 'CMYK':
+          return 'DeviceCmykCS';
+        case 'CalGray':
+          params = xref.fetchIfRef(cs[1]);
+          whitePoint = params.getArray('WhitePoint');
+          blackPoint = params.getArray('BlackPoint');
+          gamma = params.get('Gamma');
+          return ['CalGrayCS', whitePoint, blackPoint, gamma];
+        case 'CalRGB':
+          params = xref.fetchIfRef(cs[1]);
+          whitePoint = params.getArray('WhitePoint');
+          blackPoint = params.getArray('BlackPoint');
+          gamma = params.getArray('Gamma');
+          var matrix = params.getArray('Matrix');
+          return ['CalRGBCS', whitePoint, blackPoint, gamma, matrix];
+        case 'ICCBased':
+          var stream = xref.fetchIfRef(cs[1]);
+          var dict = stream.dict;
+          numComps = dict.get('N');
+          alt = dict.get('Alternate');
+          if (alt) {
+            var altIR = ColorSpace.parseToIR(alt, xref, res);
+            var altCS = ColorSpace.fromIR(altIR);
+            if (altCS.numComps === numComps) {
+              return altIR;
+            }
+            warn('ICCBased color space: Ignoring incorrect /Alternate entry.');
+          }
+          if (numComps === 1) {
+            return 'DeviceGrayCS';
+          } else if (numComps === 3) {
+            return 'DeviceRgbCS';
+          } else if (numComps === 4) {
+            return 'DeviceCmykCS';
+          }
+          break;
+        case 'Pattern':
+          var basePatternCS = cs[1] || null;
+          if (basePatternCS) {
+            basePatternCS = ColorSpace.parseToIR(basePatternCS, xref, res);
+          }
+          return ['PatternCS', basePatternCS];
+        case 'Indexed':
+        case 'I':
+          var baseIndexedCS = ColorSpace.parseToIR(cs[1], xref, res);
+          var hiVal = xref.fetchIfRef(cs[2]) + 1;
+          var lookup = xref.fetchIfRef(cs[3]);
+          if (isStream(lookup)) {
+            lookup = lookup.getBytes();
+          }
+          return ['IndexedCS', baseIndexedCS, hiVal, lookup];
+        case 'Separation':
+        case 'DeviceN':
+          var name = xref.fetchIfRef(cs[1]);
+          numComps = isArray(name) ? name.length : 1;
+          alt = ColorSpace.parseToIR(cs[2], xref, res);
+          var tintFnIR = PDFFunction.getIR(xref, xref.fetchIfRef(cs[3]));
+          return ['AlternateCS', numComps, alt, tintFnIR];
+        case 'Lab':
+          params = xref.fetchIfRef(cs[1]);
+          whitePoint = params.getArray('WhitePoint');
+          blackPoint = params.getArray('BlackPoint');
+          var range = params.getArray('Range');
+          return ['LabCS', whitePoint, blackPoint, range];
+        default:
+          error('unimplemented color space object "' + mode + '"');
+      }
     } else {
-     rgbPos = 0;
-     destPos = 0;
-     for (i = 0, ii = width * actualHeight; i < ii; i++) {
-      dest[destPos++] = rgbBuf[rgbPos++];
-      dest[destPos++] = rgbBuf[rgbPos++];
-      dest[destPos++] = rgbBuf[rgbPos++];
-      destPos += alpha01;
-     }
-    }
-   }
-  },
-  usesZeroToOneRange: true
- };
- ColorSpace.parse = function ColorSpace_parse(cs, xref, res) {
-  var IR = ColorSpace.parseToIR(cs, xref, res);
-  if (IR instanceof AlternateCS) {
-   return IR;
-  }
-  return ColorSpace.fromIR(IR);
- };
- ColorSpace.fromIR = function ColorSpace_fromIR(IR) {
-  var name = isArray(IR) ? IR[0] : IR;
-  var whitePoint, blackPoint, gamma;
-  switch (name) {
-  case 'DeviceGrayCS':
-   return this.singletons.gray;
-  case 'DeviceRgbCS':
-   return this.singletons.rgb;
-  case 'DeviceCmykCS':
-   return this.singletons.cmyk;
-  case 'CalGrayCS':
-   whitePoint = IR[1];
-   blackPoint = IR[2];
-   gamma = IR[3];
-   return new CalGrayCS(whitePoint, blackPoint, gamma);
-  case 'CalRGBCS':
-   whitePoint = IR[1];
-   blackPoint = IR[2];
-   gamma = IR[3];
-   var matrix = IR[4];
-   return new CalRGBCS(whitePoint, blackPoint, gamma, matrix);
-  case 'PatternCS':
-   var basePatternCS = IR[1];
-   if (basePatternCS) {
-    basePatternCS = ColorSpace.fromIR(basePatternCS);
-   }
-   return new PatternCS(basePatternCS);
-  case 'IndexedCS':
-   var baseIndexedCS = IR[1];
-   var hiVal = IR[2];
-   var lookup = IR[3];
-   return new IndexedCS(ColorSpace.fromIR(baseIndexedCS), hiVal, lookup);
-  case 'AlternateCS':
-   var numComps = IR[1];
-   var alt = IR[2];
-   var tintFnIR = IR[3];
-   return new AlternateCS(numComps, ColorSpace.fromIR(alt), PDFFunction.fromIR(tintFnIR));
-  case 'LabCS':
-   whitePoint = IR[1];
-   blackPoint = IR[2];
-   var range = IR[3];
-   return new LabCS(whitePoint, blackPoint, range);
-  default:
-   error('Unknown name ' + name);
-  }
-  return null;
- };
- ColorSpace.parseToIR = function ColorSpace_parseToIR(cs, xref, res) {
-  if (isName(cs)) {
-   var colorSpaces = res.get('ColorSpace');
-   if (isDict(colorSpaces)) {
-    var refcs = colorSpaces.get(cs.name);
-    if (refcs) {
-     cs = refcs;
+      error('unrecognized color space object: "' + cs + '"');
     }
-   }
-  }
-  cs = xref.fetchIfRef(cs);
-  if (isName(cs)) {
-   switch (cs.name) {
-   case 'DeviceGray':
-   case 'G':
-    return 'DeviceGrayCS';
-   case 'DeviceRGB':
-   case 'RGB':
-    return 'DeviceRgbCS';
-   case 'DeviceCMYK':
-   case 'CMYK':
-    return 'DeviceCmykCS';
-   case 'Pattern':
-    return [
-     'PatternCS',
-     null
-    ];
-   default:
-    error('unrecognized colorspace ' + cs.name);
-   }
-  } else if (isArray(cs)) {
-   var mode = xref.fetchIfRef(cs[0]).name;
-   var numComps, params, alt, whitePoint, blackPoint, gamma;
-   switch (mode) {
-   case 'DeviceGray':
-   case 'G':
-    return 'DeviceGrayCS';
-   case 'DeviceRGB':
-   case 'RGB':
-    return 'DeviceRgbCS';
-   case 'DeviceCMYK':
-   case 'CMYK':
-    return 'DeviceCmykCS';
-   case 'CalGray':
-    params = xref.fetchIfRef(cs[1]);
-    whitePoint = params.getArray('WhitePoint');
-    blackPoint = params.getArray('BlackPoint');
-    gamma = params.get('Gamma');
-    return [
-     'CalGrayCS',
-     whitePoint,
-     blackPoint,
-     gamma
-    ];
-   case 'CalRGB':
-    params = xref.fetchIfRef(cs[1]);
-    whitePoint = params.getArray('WhitePoint');
-    blackPoint = params.getArray('BlackPoint');
-    gamma = params.getArray('Gamma');
-    var matrix = params.getArray('Matrix');
-    return [
-     'CalRGBCS',
-     whitePoint,
-     blackPoint,
-     gamma,
-     matrix
-    ];
-   case 'ICCBased':
-    var stream = xref.fetchIfRef(cs[1]);
-    var dict = stream.dict;
-    numComps = dict.get('N');
-    alt = dict.get('Alternate');
-    if (alt) {
-     var altIR = ColorSpace.parseToIR(alt, xref, res);
-     var altCS = ColorSpace.fromIR(altIR);
-     if (altCS.numComps === numComps) {
-      return altIR;
-     }
-     warn('ICCBased color space: Ignoring incorrect /Alternate entry.');
+    return null;
+  };
+  ColorSpace.isDefaultDecode = function ColorSpace_isDefaultDecode(decode, n) {
+    if (!isArray(decode)) {
+      return true;
     }
-    if (numComps === 1) {
-     return 'DeviceGrayCS';
-    } else if (numComps === 3) {
-     return 'DeviceRgbCS';
-    } else if (numComps === 4) {
-     return 'DeviceCmykCS';
+    if (n * 2 !== decode.length) {
+      warn('The decode map is not the correct length');
+      return true;
     }
-    break;
-   case 'Pattern':
-    var basePatternCS = cs[1] || null;
-    if (basePatternCS) {
-     basePatternCS = ColorSpace.parseToIR(basePatternCS, xref, res);
+    for (var i = 0, ii = decode.length; i < ii; i += 2) {
+      if (decode[i] !== 0 || decode[i + 1] !== 1) {
+        return false;
+      }
     }
-    return [
-     'PatternCS',
-     basePatternCS
-    ];
-   case 'Indexed':
-   case 'I':
-    var baseIndexedCS = ColorSpace.parseToIR(cs[1], xref, res);
-    var hiVal = xref.fetchIfRef(cs[2]) + 1;
-    var lookup = xref.fetchIfRef(cs[3]);
-    if (isStream(lookup)) {
-     lookup = lookup.getBytes();
+    return true;
+  };
+  ColorSpace.singletons = {
+    get gray() {
+      return shadow(this, 'gray', new DeviceGrayCS());
+    },
+    get rgb() {
+      return shadow(this, 'rgb', new DeviceRgbCS());
+    },
+    get cmyk() {
+      return shadow(this, 'cmyk', new DeviceCmykCS());
     }
-    return [
-     'IndexedCS',
-     baseIndexedCS,
-     hiVal,
-     lookup
-    ];
-   case 'Separation':
-   case 'DeviceN':
-    var name = xref.fetchIfRef(cs[1]);
-    numComps = isArray(name) ? name.length : 1;
-    alt = ColorSpace.parseToIR(cs[2], xref, res);
-    var tintFnIR = PDFFunction.getIR(xref, xref.fetchIfRef(cs[3]));
-    return [
-     'AlternateCS',
-     numComps,
-     alt,
-     tintFnIR
-    ];
-   case 'Lab':
-    params = xref.fetchIfRef(cs[1]);
-    whitePoint = params.getArray('WhitePoint');
-    blackPoint = params.getArray('BlackPoint');
-    var range = params.getArray('Range');
-    return [
-     'LabCS',
-     whitePoint,
-     blackPoint,
-     range
-    ];
-   default:
-    error('unimplemented color space object "' + mode + '"');
-   }
-  } else {
-   error('unrecognized color space object: "' + cs + '"');
-  }
-  return null;
- };
- ColorSpace.isDefaultDecode = function ColorSpace_isDefaultDecode(decode, n) {
-  if (!isArray(decode)) {
-   return true;
-  }
-  if (n * 2 !== decode.length) {
-   warn('The decode map is not the correct length');
-   return true;
-  }
-  for (var i = 0, ii = decode.length; i < ii; i += 2) {
-   if (decode[i] !== 0 || decode[i + 1] !== 1) {
-    return false;
-   }
-  }
-  return true;
- };
- ColorSpace.singletons = {
-  get gray() {
-   return shadow(this, 'gray', new DeviceGrayCS());
-  },
-  get rgb() {
-   return shadow(this, 'rgb', new DeviceRgbCS());
-  },
-  get cmyk() {
-   return shadow(this, 'cmyk', new DeviceCmykCS());
-  }
- };
- return ColorSpace;
+  };
+  return ColorSpace;
 }();
 var AlternateCS = function AlternateCSClosure() {
- function AlternateCS(numComps, base, tintFn) {
-  this.name = 'Alternate';
-  this.numComps = numComps;
-  this.defaultColor = new Float32Array(numComps);
-  for (var i = 0; i < numComps; ++i) {
-   this.defaultColor[i] = 1;
-  }
-  this.base = base;
-  this.tintFn = tintFn;
-  this.tmpBuf = new Float32Array(base.numComps);
- }
- AlternateCS.prototype = {
-  getRgb: ColorSpace.prototype.getRgb,
-  getRgbItem: function AlternateCS_getRgbItem(src, srcOffset, dest, destOffset) {
-   var tmpBuf = this.tmpBuf;
-   this.tintFn(src, srcOffset, tmpBuf, 0);
-   this.base.getRgbItem(tmpBuf, 0, dest, destOffset);
-  },
-  getRgbBuffer: function AlternateCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
-   var tintFn = this.tintFn;
-   var base = this.base;
-   var scale = 1 / ((1 << bits) - 1);
-   var baseNumComps = base.numComps;
-   var usesZeroToOneRange = base.usesZeroToOneRange;
-   var isPassthrough = (base.isPassthrough(8) || !usesZeroToOneRange) && alpha01 === 0;
-   var pos = isPassthrough ? destOffset : 0;
-   var baseBuf = isPassthrough ? dest : new Uint8Array(baseNumComps * count);
-   var numComps = this.numComps;
-   var scaled = new Float32Array(numComps);
-   var tinted = new Float32Array(baseNumComps);
-   var i, j;
-   for (i = 0; i < count; i++) {
-    for (j = 0; j < numComps; j++) {
-     scaled[j] = src[srcOffset++] * scale;
+  function AlternateCS(numComps, base, tintFn) {
+    this.name = 'Alternate';
+    this.numComps = numComps;
+    this.defaultColor = new Float32Array(numComps);
+    for (var i = 0; i < numComps; ++i) {
+      this.defaultColor[i] = 1;
     }
-    tintFn(scaled, 0, tinted, 0);
-    if (usesZeroToOneRange) {
-     for (j = 0; j < baseNumComps; j++) {
-      baseBuf[pos++] = tinted[j] * 255;
-     }
-    } else {
-     base.getRgbItem(tinted, 0, baseBuf, pos);
-     pos += baseNumComps;
-    }
-   }
-   if (!isPassthrough) {
-    base.getRgbBuffer(baseBuf, 0, count, dest, destOffset, 8, alpha01);
-   }
-  },
-  getOutputLength: function AlternateCS_getOutputLength(inputLength, alpha01) {
-   return this.base.getOutputLength(inputLength * this.base.numComps / this.numComps, alpha01);
-  },
-  isPassthrough: ColorSpace.prototype.isPassthrough,
-  fillRgb: ColorSpace.prototype.fillRgb,
-  isDefaultDecode: function AlternateCS_isDefaultDecode(decodeMap) {
-   return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
-  },
-  usesZeroToOneRange: true
- };
- return AlternateCS;
+    this.base = base;
+    this.tintFn = tintFn;
+    this.tmpBuf = new Float32Array(base.numComps);
+  }
+  AlternateCS.prototype = {
+    getRgb: ColorSpace.prototype.getRgb,
+    getRgbItem: function AlternateCS_getRgbItem(src, srcOffset, dest, destOffset) {
+      var tmpBuf = this.tmpBuf;
+      this.tintFn(src, srcOffset, tmpBuf, 0);
+      this.base.getRgbItem(tmpBuf, 0, dest, destOffset);
+    },
+    getRgbBuffer: function AlternateCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
+      var tintFn = this.tintFn;
+      var base = this.base;
+      var scale = 1 / ((1 << bits) - 1);
+      var baseNumComps = base.numComps;
+      var usesZeroToOneRange = base.usesZeroToOneRange;
+      var isPassthrough = (base.isPassthrough(8) || !usesZeroToOneRange) && alpha01 === 0;
+      var pos = isPassthrough ? destOffset : 0;
+      var baseBuf = isPassthrough ? dest : new Uint8Array(baseNumComps * count);
+      var numComps = this.numComps;
+      var scaled = new Float32Array(numComps);
+      var tinted = new Float32Array(baseNumComps);
+      var i, j;
+      for (i = 0; i < count; i++) {
+        for (j = 0; j < numComps; j++) {
+          scaled[j] = src[srcOffset++] * scale;
+        }
+        tintFn(scaled, 0, tinted, 0);
+        if (usesZeroToOneRange) {
+          for (j = 0; j < baseNumComps; j++) {
+            baseBuf[pos++] = tinted[j] * 255;
+          }
+        } else {
+          base.getRgbItem(tinted, 0, baseBuf, pos);
+          pos += baseNumComps;
+        }
+      }
+      if (!isPassthrough) {
+        base.getRgbBuffer(baseBuf, 0, count, dest, destOffset, 8, alpha01);
+      }
+    },
+    getOutputLength: function AlternateCS_getOutputLength(inputLength, alpha01) {
+      return this.base.getOutputLength(inputLength * this.base.numComps / this.numComps, alpha01);
+    },
+    isPassthrough: ColorSpace.prototype.isPassthrough,
+    fillRgb: ColorSpace.prototype.fillRgb,
+    isDefaultDecode: function AlternateCS_isDefaultDecode(decodeMap) {
+      return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
+    },
+    usesZeroToOneRange: true
+  };
+  return AlternateCS;
 }();
 var PatternCS = function PatternCSClosure() {
- function PatternCS(baseCS) {
-  this.name = 'Pattern';
-  this.base = baseCS;
- }
- PatternCS.prototype = {};
- return PatternCS;
+  function PatternCS(baseCS) {
+    this.name = 'Pattern';
+    this.base = baseCS;
+  }
+  PatternCS.prototype = {};
+  return PatternCS;
 }();
 var IndexedCS = function IndexedCSClosure() {
- function IndexedCS(base, highVal, lookup) {
-  this.name = 'Indexed';
-  this.numComps = 1;
-  this.defaultColor = new Uint8Array(this.numComps);
-  this.base = base;
-  this.highVal = highVal;
-  var baseNumComps = base.numComps;
-  var length = baseNumComps * highVal;
-  if (isStream(lookup)) {
-   this.lookup = new Uint8Array(length);
-   var bytes = lookup.getBytes(length);
-   this.lookup.set(bytes);
-  } else if (isString(lookup)) {
-   this.lookup = new Uint8Array(length);
-   for (var i = 0; i < length; ++i) {
-    this.lookup[i] = lookup.charCodeAt(i);
-   }
-  } else if (lookup instanceof Uint8Array || lookup instanceof Array) {
-   this.lookup = lookup;
-  } else {
-   error('Unrecognized lookup table: ' + lookup);
+  function IndexedCS(base, highVal, lookup) {
+    this.name = 'Indexed';
+    this.numComps = 1;
+    this.defaultColor = new Uint8Array(this.numComps);
+    this.base = base;
+    this.highVal = highVal;
+    var baseNumComps = base.numComps;
+    var length = baseNumComps * highVal;
+    if (isStream(lookup)) {
+      this.lookup = new Uint8Array(length);
+      var bytes = lookup.getBytes(length);
+      this.lookup.set(bytes);
+    } else if (isString(lookup)) {
+      this.lookup = new Uint8Array(length);
+      for (var i = 0; i < length; ++i) {
+        this.lookup[i] = lookup.charCodeAt(i);
+      }
+    } else if (lookup instanceof Uint8Array || lookup instanceof Array) {
+      this.lookup = lookup;
+    } else {
+      error('Unrecognized lookup table: ' + lookup);
+    }
   }
- }
- IndexedCS.prototype = {
-  getRgb: ColorSpace.prototype.getRgb,
-  getRgbItem: function IndexedCS_getRgbItem(src, srcOffset, dest, destOffset) {
-   var numComps = this.base.numComps;
-   var start = src[srcOffset] * numComps;
-   this.base.getRgbItem(this.lookup, start, dest, destOffset);
-  },
-  getRgbBuffer: function IndexedCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
-   var base = this.base;
-   var numComps = base.numComps;
-   var outputDelta = base.getOutputLength(numComps, alpha01);
-   var lookup = this.lookup;
-   for (var i = 0; i < count; ++i) {
-    var lookupPos = src[srcOffset++] * numComps;
-    base.getRgbBuffer(lookup, lookupPos, 1, dest, destOffset, 8, alpha01);
-    destOffset += outputDelta;
-   }
-  },
-  getOutputLength: function IndexedCS_getOutputLength(inputLength, alpha01) {
-   return this.base.getOutputLength(inputLength * this.base.numComps, alpha01);
-  },
-  isPassthrough: ColorSpace.prototype.isPassthrough,
-  fillRgb: ColorSpace.prototype.fillRgb,
-  isDefaultDecode: function IndexedCS_isDefaultDecode(decodeMap) {
-   return true;
-  },
-  usesZeroToOneRange: true
- };
- return IndexedCS;
+  IndexedCS.prototype = {
+    getRgb: ColorSpace.prototype.getRgb,
+    getRgbItem: function IndexedCS_getRgbItem(src, srcOffset, dest, destOffset) {
+      var numComps = this.base.numComps;
+      var start = src[srcOffset] * numComps;
+      this.base.getRgbItem(this.lookup, start, dest, destOffset);
+    },
+    getRgbBuffer: function IndexedCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
+      var base = this.base;
+      var numComps = base.numComps;
+      var outputDelta = base.getOutputLength(numComps, alpha01);
+      var lookup = this.lookup;
+      for (var i = 0; i < count; ++i) {
+        var lookupPos = src[srcOffset++] * numComps;
+        base.getRgbBuffer(lookup, lookupPos, 1, dest, destOffset, 8, alpha01);
+        destOffset += outputDelta;
+      }
+    },
+    getOutputLength: function IndexedCS_getOutputLength(inputLength, alpha01) {
+      return this.base.getOutputLength(inputLength * this.base.numComps, alpha01);
+    },
+    isPassthrough: ColorSpace.prototype.isPassthrough,
+    fillRgb: ColorSpace.prototype.fillRgb,
+    isDefaultDecode: function IndexedCS_isDefaultDecode(decodeMap) {
+      return true;
+    },
+    usesZeroToOneRange: true
+  };
+  return IndexedCS;
 }();
 var DeviceGrayCS = function DeviceGrayCSClosure() {
- function DeviceGrayCS() {
-  this.name = 'DeviceGray';
-  this.numComps = 1;
-  this.defaultColor = new Float32Array(this.numComps);
- }
- DeviceGrayCS.prototype = {
-  getRgb: ColorSpace.prototype.getRgb,
-  getRgbItem: function DeviceGrayCS_getRgbItem(src, srcOffset, dest, destOffset) {
-   var c = src[srcOffset] * 255 | 0;
-   c = c < 0 ? 0 : c > 255 ? 255 : c;
-   dest[destOffset] = dest[destOffset + 1] = dest[destOffset + 2] = c;
-  },
-  getRgbBuffer: function DeviceGrayCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
-   var scale = 255 / ((1 << bits) - 1);
-   var j = srcOffset, q = destOffset;
-   for (var i = 0; i < count; ++i) {
-    var c = scale * src[j++] | 0;
-    dest[q++] = c;
-    dest[q++] = c;
-    dest[q++] = c;
-    q += alpha01;
-   }
-  },
-  getOutputLength: function DeviceGrayCS_getOutputLength(inputLength, alpha01) {
-   return inputLength * (3 + alpha01);
-  },
-  isPassthrough: ColorSpace.prototype.isPassthrough,
-  fillRgb: ColorSpace.prototype.fillRgb,
-  isDefaultDecode: function DeviceGrayCS_isDefaultDecode(decodeMap) {
-   return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
-  },
-  usesZeroToOneRange: true
- };
- return DeviceGrayCS;
+  function DeviceGrayCS() {
+    this.name = 'DeviceGray';
+    this.numComps = 1;
+    this.defaultColor = new Float32Array(this.numComps);
+  }
+  DeviceGrayCS.prototype = {
+    getRgb: ColorSpace.prototype.getRgb,
+    getRgbItem: function DeviceGrayCS_getRgbItem(src, srcOffset, dest, destOffset) {
+      var c = src[srcOffset] * 255 | 0;
+      c = c < 0 ? 0 : c > 255 ? 255 : c;
+      dest[destOffset] = dest[destOffset + 1] = dest[destOffset + 2] = c;
+    },
+    getRgbBuffer: function DeviceGrayCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
+      var scale = 255 / ((1 << bits) - 1);
+      var j = srcOffset,
+          q = destOffset;
+      for (var i = 0; i < count; ++i) {
+        var c = scale * src[j++] | 0;
+        dest[q++] = c;
+        dest[q++] = c;
+        dest[q++] = c;
+        q += alpha01;
+      }
+    },
+    getOutputLength: function DeviceGrayCS_getOutputLength(inputLength, alpha01) {
+      return inputLength * (3 + alpha01);
+    },
+    isPassthrough: ColorSpace.prototype.isPassthrough,
+    fillRgb: ColorSpace.prototype.fillRgb,
+    isDefaultDecode: function DeviceGrayCS_isDefaultDecode(decodeMap) {
+      return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
+    },
+    usesZeroToOneRange: true
+  };
+  return DeviceGrayCS;
 }();
 var DeviceRgbCS = function DeviceRgbCSClosure() {
- function DeviceRgbCS() {
-  this.name = 'DeviceRGB';
-  this.numComps = 3;
-  this.defaultColor = new Float32Array(this.numComps);
- }
- DeviceRgbCS.prototype = {
-  getRgb: ColorSpace.prototype.getRgb,
-  getRgbItem: function DeviceRgbCS_getRgbItem(src, srcOffset, dest, destOffset) {
-   var r = src[srcOffset] * 255 | 0;
-   var g = src[srcOffset + 1] * 255 | 0;
-   var b = src[srcOffset + 2] * 255 | 0;
-   dest[destOffset] = r < 0 ? 0 : r > 255 ? 255 : r;
-   dest[destOffset + 1] = g < 0 ? 0 : g > 255 ? 255 : g;
-   dest[destOffset + 2] = b < 0 ? 0 : b > 255 ? 255 : b;
-  },
-  getRgbBuffer: function DeviceRgbCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
-   if (bits === 8 && alpha01 === 0) {
-    dest.set(src.subarray(srcOffset, srcOffset + count * 3), destOffset);
-    return;
-   }
-   var scale = 255 / ((1 << bits) - 1);
-   var j = srcOffset, q = destOffset;
-   for (var i = 0; i < count; ++i) {
-    dest[q++] = scale * src[j++] | 0;
-    dest[q++] = scale * src[j++] | 0;
-    dest[q++] = scale * src[j++] | 0;
-    q += alpha01;
-   }
-  },
-  getOutputLength: function DeviceRgbCS_getOutputLength(inputLength, alpha01) {
-   return inputLength * (3 + alpha01) / 3 | 0;
-  },
-  isPassthrough: function DeviceRgbCS_isPassthrough(bits) {
-   return bits === 8;
-  },
-  fillRgb: ColorSpace.prototype.fillRgb,
-  isDefaultDecode: function DeviceRgbCS_isDefaultDecode(decodeMap) {
-   return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
-  },
-  usesZeroToOneRange: true
- };
- return DeviceRgbCS;
+  function DeviceRgbCS() {
+    this.name = 'DeviceRGB';
+    this.numComps = 3;
+    this.defaultColor = new Float32Array(this.numComps);
+  }
+  DeviceRgbCS.prototype = {
+    getRgb: ColorSpace.prototype.getRgb,
+    getRgbItem: function DeviceRgbCS_getRgbItem(src, srcOffset, dest, destOffset) {
+      var r = src[srcOffset] * 255 | 0;
+      var g = src[srcOffset + 1] * 255 | 0;
+      var b = src[srcOffset + 2] * 255 | 0;
+      dest[destOffset] = r < 0 ? 0 : r > 255 ? 255 : r;
+      dest[destOffset + 1] = g < 0 ? 0 : g > 255 ? 255 : g;
+      dest[destOffset + 2] = b < 0 ? 0 : b > 255 ? 255 : b;
+    },
+    getRgbBuffer: function DeviceRgbCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
+      if (bits === 8 && alpha01 === 0) {
+        dest.set(src.subarray(srcOffset, srcOffset + count * 3), destOffset);
+        return;
+      }
+      var scale = 255 / ((1 << bits) - 1);
+      var j = srcOffset,
+          q = destOffset;
+      for (var i = 0; i < count; ++i) {
+        dest[q++] = scale * src[j++] | 0;
+        dest[q++] = scale * src[j++] | 0;
+        dest[q++] = scale * src[j++] | 0;
+        q += alpha01;
+      }
+    },
+    getOutputLength: function DeviceRgbCS_getOutputLength(inputLength, alpha01) {
+      return inputLength * (3 + alpha01) / 3 | 0;
+    },
+    isPassthrough: function DeviceRgbCS_isPassthrough(bits) {
+      return bits === 8;
+    },
+    fillRgb: ColorSpace.prototype.fillRgb,
+    isDefaultDecode: function DeviceRgbCS_isDefaultDecode(decodeMap) {
+      return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
+    },
+    usesZeroToOneRange: true
+  };
+  return DeviceRgbCS;
 }();
 var DeviceCmykCS = function DeviceCmykCSClosure() {
- function convertToRgb(src, srcOffset, srcScale, dest, destOffset) {
-  var c = src[srcOffset + 0] * srcScale;
-  var m = src[srcOffset + 1] * srcScale;
-  var y = src[srcOffset + 2] * srcScale;
-  var k = src[srcOffset + 3] * srcScale;
-  var r = c * (-4.387332384609988 * c + 54.48615194189176 * m + 18.82290502165302 * y + 212.25662451639585 * k + -285.2331026137004) + m * (1.7149763477362134 * m - 5.6096736904047315 * y + -17.873870861415444 * k - 5.497006427196366) + y * (-2.5217340131683033 * y - 21.248923337353073 * k + 17.5119270841813) + k * (-21.86122147463605 * k - 189.48180835922747) + 255 | 0;
-  var g = c * (8.841041422036149 * c + 60.118027045597366 * m + 6.871425592049007 * y + 31.159100130055922 * k + -79.2970844816548) + m * (-15.310361306967817 * m + 17.575251261109482 * y + 131.35250912493976 * k - 190.9453302588951) + y * (4.444339102852739 * y + 9.8632861493405 * k - 24.86741582555878) + k * (-20.737325471181034 * k - 187.80453709719578) + 255 | 0;
-  var b = c * (0.8842522430003296 * c + 8.078677503112928 * m + 30.89978309703729 * y - 0.23883238689178934 * k + -14.183576799673286) + m * (10.49593273432072 * m + 63.02378494754052 * y + 50.606957656360734 * k - 112.23884253719248) + y * (0.03296041114873217 * y + 115.60384449646641 * k + -193.58209356861505) + k * (-22.33816807309886 * k - 180.12613974708367) + 255 | 0;
-  dest[destOffset] = r > 255 ? 255 : r < 0 ? 0 : r;
-  dest[destOffset + 1] = g > 255 ? 255 : g < 0 ? 0 : g;
-  dest[destOffset + 2] = b > 255 ? 255 : b < 0 ? 0 : b;
- }
- function DeviceCmykCS() {
-  this.name = 'DeviceCMYK';
-  this.numComps = 4;
-  this.defaultColor = new Float32Array(this.numComps);
-  this.defaultColor[3] = 1;
- }
- DeviceCmykCS.prototype = {
-  getRgb: ColorSpace.prototype.getRgb,
-  getRgbItem: function DeviceCmykCS_getRgbItem(src, srcOffset, dest, destOffset) {
-   convertToRgb(src, srcOffset, 1, dest, destOffset);
-  },
-  getRgbBuffer: function DeviceCmykCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
-   var scale = 1 / ((1 << bits) - 1);
-   for (var i = 0; i < count; i++) {
-    convertToRgb(src, srcOffset, scale, dest, destOffset);
-    srcOffset += 4;
-    destOffset += 3 + alpha01;
-   }
-  },
-  getOutputLength: function DeviceCmykCS_getOutputLength(inputLength, alpha01) {
-   return inputLength / 4 * (3 + alpha01) | 0;
-  },
-  isPassthrough: ColorSpace.prototype.isPassthrough,
-  fillRgb: ColorSpace.prototype.fillRgb,
-  isDefaultDecode: function DeviceCmykCS_isDefaultDecode(decodeMap) {
-   return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
-  },
-  usesZeroToOneRange: true
- };
- return DeviceCmykCS;
+  function convertToRgb(src, srcOffset, srcScale, dest, destOffset) {
+    var c = src[srcOffset + 0] * srcScale;
+    var m = src[srcOffset + 1] * srcScale;
+    var y = src[srcOffset + 2] * srcScale;
+    var k = src[srcOffset + 3] * srcScale;
+    var r = c * (-4.387332384609988 * c + 54.48615194189176 * m + 18.82290502165302 * y + 212.25662451639585 * k + -285.2331026137004) + m * (1.7149763477362134 * m - 5.6096736904047315 * y + -17.873870861415444 * k - 5.497006427196366) + y * (-2.5217340131683033 * y - 21.248923337353073 * k + 17.5119270841813) + k * (-21.86122147463605 * k - 189.48180835922747) + 255 | 0;
+    var g = c * (8.841041422036149 * c + 60.118027045597366 * m + 6.871425592049007 * y + 31.159100130055922 * k + -79.2970844816548) + m * (-15.310361306967817 * m + 17.575251261109482 * y + 131.35250912493976 * k - 190.9453302588951) + y * (4.444339102852739 * y + 9.8632861493405 * k - 24.86741582555878) + k * (-20.737325471181034 * k - 187.80453709719578) + 255 | 0;
+    var b = c * (0.8842522430003296 * c + 8.078677503112928 * m + 30.89978309703729 * y - 0.23883238689178934 * k + -14.183576799673286) + m * (10.49593273432072 * m + 63.02378494754052 * y + 50.606957656360734 * k - 112.23884253719248) + y * (0.03296041114873217 * y + 115.60384449646641 * k + -193.58209356861505) + k * (-22.33816807309886 * k - 180.12613974708367) + 255 | 0;
+    dest[destOffset] = r > 255 ? 255 : r < 0 ? 0 : r;
+    dest[destOffset + 1] = g > 255 ? 255 : g < 0 ? 0 : g;
+    dest[destOffset + 2] = b > 255 ? 255 : b < 0 ? 0 : b;
+  }
+  function DeviceCmykCS() {
+    this.name = 'DeviceCMYK';
+    this.numComps = 4;
+    this.defaultColor = new Float32Array(this.numComps);
+    this.defaultColor[3] = 1;
+  }
+  DeviceCmykCS.prototype = {
+    getRgb: ColorSpace.prototype.getRgb,
+    getRgbItem: function DeviceCmykCS_getRgbItem(src, srcOffset, dest, destOffset) {
+      convertToRgb(src, srcOffset, 1, dest, destOffset);
+    },
+    getRgbBuffer: function DeviceCmykCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
+      var scale = 1 / ((1 << bits) - 1);
+      for (var i = 0; i < count; i++) {
+        convertToRgb(src, srcOffset, scale, dest, destOffset);
+        srcOffset += 4;
+        destOffset += 3 + alpha01;
+      }
+    },
+    getOutputLength: function DeviceCmykCS_getOutputLength(inputLength, alpha01) {
+      return inputLength / 4 * (3 + alpha01) | 0;
+    },
+    isPassthrough: ColorSpace.prototype.isPassthrough,
+    fillRgb: ColorSpace.prototype.fillRgb,
+    isDefaultDecode: function DeviceCmykCS_isDefaultDecode(decodeMap) {
+      return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
+    },
+    usesZeroToOneRange: true
+  };
+  return DeviceCmykCS;
 }();
 var CalGrayCS = function CalGrayCSClosure() {
- function CalGrayCS(whitePoint, blackPoint, gamma) {
-  this.name = 'CalGray';
-  this.numComps = 1;
-  this.defaultColor = new Float32Array(this.numComps);
-  if (!whitePoint) {
-   error('WhitePoint missing - required for color space CalGray');
-  }
-  blackPoint = blackPoint || [
-   0,
-   0,
-   0
-  ];
-  gamma = gamma || 1;
-  this.XW = whitePoint[0];
-  this.YW = whitePoint[1];
-  this.ZW = whitePoint[2];
-  this.XB = blackPoint[0];
-  this.YB = blackPoint[1];
-  this.ZB = blackPoint[2];
-  this.G = gamma;
-  if (this.XW < 0 || this.ZW < 0 || this.YW !== 1) {
-   error('Invalid WhitePoint components for ' + this.name + ', no fallback available');
-  }
-  if (this.XB < 0 || this.YB < 0 || this.ZB < 0) {
-   info('Invalid BlackPoint for ' + this.name + ', falling back to default');
-   this.XB = this.YB = this.ZB = 0;
-  }
-  if (this.XB !== 0 || this.YB !== 0 || this.ZB !== 0) {
-   warn(this.name + ', BlackPoint: XB: ' + this.XB + ', YB: ' + this.YB + ', ZB: ' + this.ZB + ', only default values are supported.');
-  }
-  if (this.G < 1) {
-   info('Invalid Gamma: ' + this.G + ' for ' + this.name + ', falling back to default');
-   this.G = 1;
+  function CalGrayCS(whitePoint, blackPoint, gamma) {
+    this.name = 'CalGray';
+    this.numComps = 1;
+    this.defaultColor = new Float32Array(this.numComps);
+    if (!whitePoint) {
+      error('WhitePoint missing - required for color space CalGray');
+    }
+    blackPoint = blackPoint || [0, 0, 0];
+    gamma = gamma || 1;
+    this.XW = whitePoint[0];
+    this.YW = whitePoint[1];
+    this.ZW = whitePoint[2];
+    this.XB = blackPoint[0];
+    this.YB = blackPoint[1];
+    this.ZB = blackPoint[2];
+    this.G = gamma;
+    if (this.XW < 0 || this.ZW < 0 || this.YW !== 1) {
+      error('Invalid WhitePoint components for ' + this.name + ', no fallback available');
+    }
+    if (this.XB < 0 || this.YB < 0 || this.ZB < 0) {
+      info('Invalid BlackPoint for ' + this.name + ', falling back to default');
+      this.XB = this.YB = this.ZB = 0;
+    }
+    if (this.XB !== 0 || this.YB !== 0 || this.ZB !== 0) {
+      warn(this.name + ', BlackPoint: XB: ' + this.XB + ', YB: ' + this.YB + ', ZB: ' + this.ZB + ', only default values are supported.');
+    }
+    if (this.G < 1) {
+      info('Invalid Gamma: ' + this.G + ' for ' + this.name + ', falling back to default');
+      this.G = 1;
+    }
   }
- }
- function convertToRgb(cs, src, srcOffset, dest, destOffset, scale) {
-  var A = src[srcOffset] * scale;
-  var AG = Math.pow(A, cs.G);
-  var L = cs.YW * AG;
-  var val = Math.max(295.8 * Math.pow(L, 0.333333333333333333) - 40.8, 0) | 0;
-  dest[destOffset] = val;
-  dest[destOffset + 1] = val;
-  dest[destOffset + 2] = val;
- }
- CalGrayCS.prototype = {
-  getRgb: ColorSpace.prototype.getRgb,
-  getRgbItem: function CalGrayCS_getRgbItem(src, srcOffset, dest, destOffset) {
-   convertToRgb(this, src, srcOffset, dest, destOffset, 1);
-  },
-  getRgbBuffer: function CalGrayCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
-   var scale = 1 / ((1 << bits) - 1);
-   for (var i = 0; i < count; ++i) {
-    convertToRgb(this, src, srcOffset, dest, destOffset, scale);
-    srcOffset += 1;
-    destOffset += 3 + alpha01;
-   }
-  },
-  getOutputLength: function CalGrayCS_getOutputLength(inputLength, alpha01) {
-   return inputLength * (3 + alpha01);
-  },
-  isPassthrough: ColorSpace.prototype.isPassthrough,
-  fillRgb: ColorSpace.prototype.fillRgb,
-  isDefaultDecode: function CalGrayCS_isDefaultDecode(decodeMap) {
-   return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
-  },
-  usesZeroToOneRange: true
- };
- return CalGrayCS;
+  function convertToRgb(cs, src, srcOffset, dest, destOffset, scale) {
+    var A = src[srcOffset] * scale;
+    var AG = Math.pow(A, cs.G);
+    var L = cs.YW * AG;
+    var val = Math.max(295.8 * Math.pow(L, 0.333333333333333333) - 40.8, 0) | 0;
+    dest[destOffset] = val;
+    dest[destOffset + 1] = val;
+    dest[destOffset + 2] = val;
+  }
+  CalGrayCS.prototype = {
+    getRgb: ColorSpace.prototype.getRgb,
+    getRgbItem: function CalGrayCS_getRgbItem(src, srcOffset, dest, destOffset) {
+      convertToRgb(this, src, srcOffset, dest, destOffset, 1);
+    },
+    getRgbBuffer: function CalGrayCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
+      var scale = 1 / ((1 << bits) - 1);
+      for (var i = 0; i < count; ++i) {
+        convertToRgb(this, src, srcOffset, dest, destOffset, scale);
+        srcOffset += 1;
+        destOffset += 3 + alpha01;
+      }
+    },
+    getOutputLength: function CalGrayCS_getOutputLength(inputLength, alpha01) {
+      return inputLength * (3 + alpha01);
+    },
+    isPassthrough: ColorSpace.prototype.isPassthrough,
+    fillRgb: ColorSpace.prototype.fillRgb,
+    isDefaultDecode: function CalGrayCS_isDefaultDecode(decodeMap) {
+      return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
+    },
+    usesZeroToOneRange: true
+  };
+  return CalGrayCS;
 }();
 var CalRGBCS = function CalRGBCSClosure() {
- var BRADFORD_SCALE_MATRIX = new Float32Array([
-  0.8951,
-  0.2664,
-  -0.1614,
-  -0.7502,
-  1.7135,
-  0.0367,
-  0.0389,
-  -0.0685,
-  1.0296
- ]);
- var BRADFORD_SCALE_INVERSE_MATRIX = new Float32Array([
-  0.9869929,
-  -0.1470543,
-  0.1599627,
-  0.4323053,
-  0.5183603,
-  0.0492912,
-  -0.0085287,
-  0.0400428,
-  0.9684867
- ]);
- var SRGB_D65_XYZ_TO_RGB_MATRIX = new Float32Array([
-  3.2404542,
-  -1.5371385,
-  -0.4985314,
-  -0.9692660,
-  1.8760108,
-  0.0415560,
-  0.0556434,
-  -0.2040259,
-  1.0572252
- ]);
- var FLAT_WHITEPOINT_MATRIX = new Float32Array([
-  1,
-  1,
-  1
- ]);
- var tempNormalizeMatrix = new Float32Array(3);
- var tempConvertMatrix1 = new Float32Array(3);
- var tempConvertMatrix2 = new Float32Array(3);
- var DECODE_L_CONSTANT = Math.pow((8 + 16) / 116, 3) / 8.0;
- function CalRGBCS(whitePoint, blackPoint, gamma, matrix) {
-  this.name = 'CalRGB';
-  this.numComps = 3;
-  this.defaultColor = new Float32Array(this.numComps);
-  if (!whitePoint) {
-   error('WhitePoint missing - required for color space CalRGB');
-  }
-  blackPoint = blackPoint || new Float32Array(3);
-  gamma = gamma || new Float32Array([
-   1,
-   1,
-   1
-  ]);
-  matrix = matrix || new Float32Array([
-   1,
-   0,
-   0,
-   0,
-   1,
-   0,
-   0,
-   0,
-   1
-  ]);
-  var XW = whitePoint[0];
-  var YW = whitePoint[1];
-  var ZW = whitePoint[2];
-  this.whitePoint = whitePoint;
-  var XB = blackPoint[0];
-  var YB = blackPoint[1];
-  var ZB = blackPoint[2];
-  this.blackPoint = blackPoint;
-  this.GR = gamma[0];
-  this.GG = gamma[1];
-  this.GB = gamma[2];
-  this.MXA = matrix[0];
-  this.MYA = matrix[1];
-  this.MZA = matrix[2];
-  this.MXB = matrix[3];
-  this.MYB = matrix[4];
-  this.MZB = matrix[5];
-  this.MXC = matrix[6];
-  this.MYC = matrix[7];
-  this.MZC = matrix[8];
-  if (XW < 0 || ZW < 0 || YW !== 1) {
-   error('Invalid WhitePoint components for ' + this.name + ', no fallback available');
-  }
-  if (XB < 0 || YB < 0 || ZB < 0) {
-   info('Invalid BlackPoint for ' + this.name + ' [' + XB + ', ' + YB + ', ' + ZB + '], falling back to default');
-   this.blackPoint = new Float32Array(3);
-  }
-  if (this.GR < 0 || this.GG < 0 || this.GB < 0) {
-   info('Invalid Gamma [' + this.GR + ', ' + this.GG + ', ' + this.GB + '] for ' + this.name + ', falling back to default');
-   this.GR = this.GG = this.GB = 1;
-  }
-  if (this.MXA < 0 || this.MYA < 0 || this.MZA < 0 || this.MXB < 0 || this.MYB < 0 || this.MZB < 0 || this.MXC < 0 || this.MYC < 0 || this.MZC < 0) {
-   info('Invalid Matrix for ' + this.name + ' [' + this.MXA + ', ' + this.MYA + ', ' + this.MZA + this.MXB + ', ' + this.MYB + ', ' + this.MZB + this.MXC + ', ' + this.MYC + ', ' + this.MZC + '], falling back to default');
-   this.MXA = this.MYB = this.MZC = 1;
-   this.MXB = this.MYA = this.MZA = this.MXC = this.MYC = this.MZB = 0;
-  }
- }
- function matrixProduct(a, b, result) {
-  result[0] = a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
-  result[1] = a[3] * b[0] + a[4] * b[1] + a[5] * b[2];
-  result[2] = a[6] * b[0] + a[7] * b[1] + a[8] * b[2];
- }
- function convertToFlat(sourceWhitePoint, LMS, result) {
-  result[0] = LMS[0] * 1 / sourceWhitePoint[0];
-  result[1] = LMS[1] * 1 / sourceWhitePoint[1];
-  result[2] = LMS[2] * 1 / sourceWhitePoint[2];
- }
- function convertToD65(sourceWhitePoint, LMS, result) {
-  var D65X = 0.95047;
-  var D65Y = 1;
-  var D65Z = 1.08883;
-  result[0] = LMS[0] * D65X / sourceWhitePoint[0];
-  result[1] = LMS[1] * D65Y / sourceWhitePoint[1];
-  result[2] = LMS[2] * D65Z / sourceWhitePoint[2];
- }
- function sRGBTransferFunction(color) {
-  if (color <= 0.0031308) {
-   return adjustToRange(0, 1, 12.92 * color);
-  }
-  return adjustToRange(0, 1, (1 + 0.055) * Math.pow(color, 1 / 2.4) - 0.055);
- }
- function adjustToRange(min, max, value) {
-  return Math.max(min, Math.min(max, value));
- }
- function decodeL(L) {
-  if (L < 0) {
-   return -decodeL(-L);
-  }
-  if (L > 8.0) {
-   return Math.pow((L + 16) / 116, 3);
+  var BRADFORD_SCALE_MATRIX = new Float32Array([0.8951, 0.2664, -0.1614, -0.7502, 1.7135, 0.0367, 0.0389, -0.0685, 1.0296]);
+  var BRADFORD_SCALE_INVERSE_MATRIX = new Float32Array([0.9869929, -0.1470543, 0.1599627, 0.4323053, 0.5183603, 0.0492912, -0.0085287, 0.0400428, 0.9684867]);
+  var SRGB_D65_XYZ_TO_RGB_MATRIX = new Float32Array([3.2404542, -1.5371385, -0.4985314, -0.9692660, 1.8760108, 0.0415560, 0.0556434, -0.2040259, 1.0572252]);
+  var FLAT_WHITEPOINT_MATRIX = new Float32Array([1, 1, 1]);
+  var tempNormalizeMatrix = new Float32Array(3);
+  var tempConvertMatrix1 = new Float32Array(3);
+  var tempConvertMatrix2 = new Float32Array(3);
+  var DECODE_L_CONSTANT = Math.pow((8 + 16) / 116, 3) / 8.0;
+  function CalRGBCS(whitePoint, blackPoint, gamma, matrix) {
+    this.name = 'CalRGB';
+    this.numComps = 3;
+    this.defaultColor = new Float32Array(this.numComps);
+    if (!whitePoint) {
+      error('WhitePoint missing - required for color space CalRGB');
+    }
+    blackPoint = blackPoint || new Float32Array(3);
+    gamma = gamma || new Float32Array([1, 1, 1]);
+    matrix = matrix || new Float32Array([1, 0, 0, 0, 1, 0, 0, 0, 1]);
+    var XW = whitePoint[0];
+    var YW = whitePoint[1];
+    var ZW = whitePoint[2];
+    this.whitePoint = whitePoint;
+    var XB = blackPoint[0];
+    var YB = blackPoint[1];
+    var ZB = blackPoint[2];
+    this.blackPoint = blackPoint;
+    this.GR = gamma[0];
+    this.GG = gamma[1];
+    this.GB = gamma[2];
+    this.MXA = matrix[0];
+    this.MYA = matrix[1];
+    this.MZA = matrix[2];
+    this.MXB = matrix[3];
+    this.MYB = matrix[4];
+    this.MZB = matrix[5];
+    this.MXC = matrix[6];
+    this.MYC = matrix[7];
+    this.MZC = matrix[8];
+    if (XW < 0 || ZW < 0 || YW !== 1) {
+      error('Invalid WhitePoint components for ' + this.name + ', no fallback available');
+    }
+    if (XB < 0 || YB < 0 || ZB < 0) {
+      info('Invalid BlackPoint for ' + this.name + ' [' + XB + ', ' + YB + ', ' + ZB + '], falling back to default');
+      this.blackPoint = new Float32Array(3);
+    }
+    if (this.GR < 0 || this.GG < 0 || this.GB < 0) {
+      info('Invalid Gamma [' + this.GR + ', ' + this.GG + ', ' + this.GB + '] for ' + this.name + ', falling back to default');
+      this.GR = this.GG = this.GB = 1;
+    }
+    if (this.MXA < 0 || this.MYA < 0 || this.MZA < 0 || this.MXB < 0 || this.MYB < 0 || this.MZB < 0 || this.MXC < 0 || this.MYC < 0 || this.MZC < 0) {
+      info('Invalid Matrix for ' + this.name + ' [' + this.MXA + ', ' + this.MYA + ', ' + this.MZA + this.MXB + ', ' + this.MYB + ', ' + this.MZB + this.MXC + ', ' + this.MYC + ', ' + this.MZC + '], falling back to default');
+      this.MXA = this.MYB = this.MZC = 1;
+      this.MXB = this.MYA = this.MZA = this.MXC = this.MYC = this.MZB = 0;
+    }
   }
-  return L * DECODE_L_CONSTANT;
- }
- function compensateBlackPoint(sourceBlackPoint, XYZ_Flat, result) {
-  if (sourceBlackPoint[0] === 0 && sourceBlackPoint[1] === 0 && sourceBlackPoint[2] === 0) {
-   result[0] = XYZ_Flat[0];
-   result[1] = XYZ_Flat[1];
-   result[2] = XYZ_Flat[2];
-   return;
+  function matrixProduct(a, b, result) {
+    result[0] = a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
+    result[1] = a[3] * b[0] + a[4] * b[1] + a[5] * b[2];
+    result[2] = a[6] * b[0] + a[7] * b[1] + a[8] * b[2];
+  }
+  function convertToFlat(sourceWhitePoint, LMS, result) {
+    result[0] = LMS[0] * 1 / sourceWhitePoint[0];
+    result[1] = LMS[1] * 1 / sourceWhitePoint[1];
+    result[2] = LMS[2] * 1 / sourceWhitePoint[2];
+  }
+  function convertToD65(sourceWhitePoint, LMS, result) {
+    var D65X = 0.95047;
+    var D65Y = 1;
+    var D65Z = 1.08883;
+    result[0] = LMS[0] * D65X / sourceWhitePoint[0];
+    result[1] = LMS[1] * D65Y / sourceWhitePoint[1];
+    result[2] = LMS[2] * D65Z / sourceWhitePoint[2];
+  }
+  function sRGBTransferFunction(color) {
+    if (color <= 0.0031308) {
+      return adjustToRange(0, 1, 12.92 * color);
+    }
+    return adjustToRange(0, 1, (1 + 0.055) * Math.pow(color, 1 / 2.4) - 0.055);
   }
-  var zeroDecodeL = decodeL(0);
-  var X_DST = zeroDecodeL;
-  var X_SRC = decodeL(sourceBlackPoint[0]);
-  var Y_DST = zeroDecodeL;
-  var Y_SRC = decodeL(sourceBlackPoint[1]);
-  var Z_DST = zeroDecodeL;
-  var Z_SRC = decodeL(sourceBlackPoint[2]);
-  var X_Scale = (1 - X_DST) / (1 - X_SRC);
-  var X_Offset = 1 - X_Scale;
-  var Y_Scale = (1 - Y_DST) / (1 - Y_SRC);
-  var Y_Offset = 1 - Y_Scale;
-  var Z_Scale = (1 - Z_DST) / (1 - Z_SRC);
-  var Z_Offset = 1 - Z_Scale;
-  result[0] = XYZ_Flat[0] * X_Scale + X_Offset;
-  result[1] = XYZ_Flat[1] * Y_Scale + Y_Offset;
-  result[2] = XYZ_Flat[2] * Z_Scale + Z_Offset;
- }
- function normalizeWhitePointToFlat(sourceWhitePoint, XYZ_In, result) {
-  if (sourceWhitePoint[0] === 1 && sourceWhitePoint[2] === 1) {
-   result[0] = XYZ_In[0];
-   result[1] = XYZ_In[1];
-   result[2] = XYZ_In[2];
-   return;
+  function adjustToRange(min, max, value) {
+    return Math.max(min, Math.min(max, value));
   }
-  var LMS = result;
-  matrixProduct(BRADFORD_SCALE_MATRIX, XYZ_In, LMS);
-  var LMS_Flat = tempNormalizeMatrix;
-  convertToFlat(sourceWhitePoint, LMS, LMS_Flat);
-  matrixProduct(BRADFORD_SCALE_INVERSE_MATRIX, LMS_Flat, result);
- }
- function normalizeWhitePointToD65(sourceWhitePoint, XYZ_In, result) {
-  var LMS = result;
-  matrixProduct(BRADFORD_SCALE_MATRIX, XYZ_In, LMS);
-  var LMS_D65 = tempNormalizeMatrix;
-  convertToD65(sourceWhitePoint, LMS, LMS_D65);
-  matrixProduct(BRADFORD_SCALE_INVERSE_MATRIX, LMS_D65, result);
- }
- function convertToRgb(cs, src, srcOffset, dest, destOffset, scale) {
-  var A = adjustToRange(0, 1, src[srcOffset] * scale);
-  var B = adjustToRange(0, 1, src[srcOffset + 1] * scale);
-  var C = adjustToRange(0, 1, src[srcOffset + 2] * scale);
-  var AGR = Math.pow(A, cs.GR);
-  var BGG = Math.pow(B, cs.GG);
-  var CGB = Math.pow(C, cs.GB);
-  var X = cs.MXA * AGR + cs.MXB * BGG + cs.MXC * CGB;
-  var Y = cs.MYA * AGR + cs.MYB * BGG + cs.MYC * CGB;
-  var Z = cs.MZA * AGR + cs.MZB * BGG + cs.MZC * CGB;
-  var XYZ = tempConvertMatrix1;
-  XYZ[0] = X;
-  XYZ[1] = Y;
-  XYZ[2] = Z;
-  var XYZ_Flat = tempConvertMatrix2;
-  normalizeWhitePointToFlat(cs.whitePoint, XYZ, XYZ_Flat);
-  var XYZ_Black = tempConvertMatrix1;
-  compensateBlackPoint(cs.blackPoint, XYZ_Flat, XYZ_Black);
-  var XYZ_D65 = tempConvertMatrix2;
-  normalizeWhitePointToD65(FLAT_WHITEPOINT_MATRIX, XYZ_Black, XYZ_D65);
-  var SRGB = tempConvertMatrix1;
-  matrixProduct(SRGB_D65_XYZ_TO_RGB_MATRIX, XYZ_D65, SRGB);
-  var sR = sRGBTransferFunction(SRGB[0]);
-  var sG = sRGBTransferFunction(SRGB[1]);
-  var sB = sRGBTransferFunction(SRGB[2]);
-  dest[destOffset] = Math.round(sR * 255);
-  dest[destOffset + 1] = Math.round(sG * 255);
-  dest[destOffset + 2] = Math.round(sB * 255);
- }
- CalRGBCS.prototype = {
-  getRgb: function CalRGBCS_getRgb(src, srcOffset) {
-   var rgb = new Uint8Array(3);
-   this.getRgbItem(src, srcOffset, rgb, 0);
-   return rgb;
-  },
-  getRgbItem: function CalRGBCS_getRgbItem(src, srcOffset, dest, destOffset) {
-   convertToRgb(this, src, srcOffset, dest, destOffset, 1);
-  },
-  getRgbBuffer: function CalRGBCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
-   var scale = 1 / ((1 << bits) - 1);
-   for (var i = 0; i < count; ++i) {
-    convertToRgb(this, src, srcOffset, dest, destOffset, scale);
-    srcOffset += 3;
-    destOffset += 3 + alpha01;
-   }
-  },
-  getOutputLength: function CalRGBCS_getOutputLength(inputLength, alpha01) {
-   return inputLength * (3 + alpha01) / 3 | 0;
-  },
-  isPassthrough: ColorSpace.prototype.isPassthrough,
-  fillRgb: ColorSpace.prototype.fillRgb,
-  isDefaultDecode: function CalRGBCS_isDefaultDecode(decodeMap) {
-   return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
-  },
-  usesZeroToOneRange: true
- };
- return CalRGBCS;
+  function decodeL(L) {
+    if (L < 0) {
+      return -decodeL(-L);
+    }
+    if (L > 8.0) {
+      return Math.pow((L + 16) / 116, 3);
+    }
+    return L * DECODE_L_CONSTANT;
+  }
+  function compensateBlackPoint(sourceBlackPoint, XYZ_Flat, result) {
+    if (sourceBlackPoint[0] === 0 && sourceBlackPoint[1] === 0 && sourceBlackPoint[2] === 0) {
+      result[0] = XYZ_Flat[0];
+      result[1] = XYZ_Flat[1];
+      result[2] = XYZ_Flat[2];
+      return;
+    }
+    var zeroDecodeL = decodeL(0);
+    var X_DST = zeroDecodeL;
+    var X_SRC = decodeL(sourceBlackPoint[0]);
+    var Y_DST = zeroDecodeL;
+    var Y_SRC = decodeL(sourceBlackPoint[1]);
+    var Z_DST = zeroDecodeL;
+    var Z_SRC = decodeL(sourceBlackPoint[2]);
+    var X_Scale = (1 - X_DST) / (1 - X_SRC);
+    var X_Offset = 1 - X_Scale;
+    var Y_Scale = (1 - Y_DST) / (1 - Y_SRC);
+    var Y_Offset = 1 - Y_Scale;
+    var Z_Scale = (1 - Z_DST) / (1 - Z_SRC);
+    var Z_Offset = 1 - Z_Scale;
+    result[0] = XYZ_Flat[0] * X_Scale + X_Offset;
+    result[1] = XYZ_Flat[1] * Y_Scale + Y_Offset;
+    result[2] = XYZ_Flat[2] * Z_Scale + Z_Offset;
+  }
+  function normalizeWhitePointToFlat(sourceWhitePoint, XYZ_In, result) {
+    if (sourceWhitePoint[0] === 1 && sourceWhitePoint[2] === 1) {
+      result[0] = XYZ_In[0];
+      result[1] = XYZ_In[1];
+      result[2] = XYZ_In[2];
+      return;
+    }
+    var LMS = result;
+    matrixProduct(BRADFORD_SCALE_MATRIX, XYZ_In, LMS);
+    var LMS_Flat = tempNormalizeMatrix;
+    convertToFlat(sourceWhitePoint, LMS, LMS_Flat);
+    matrixProduct(BRADFORD_SCALE_INVERSE_MATRIX, LMS_Flat, result);
+  }
+  function normalizeWhitePointToD65(sourceWhitePoint, XYZ_In, result) {
+    var LMS = result;
+    matrixProduct(BRADFORD_SCALE_MATRIX, XYZ_In, LMS);
+    var LMS_D65 = tempNormalizeMatrix;
+    convertToD65(sourceWhitePoint, LMS, LMS_D65);
+    matrixProduct(BRADFORD_SCALE_INVERSE_MATRIX, LMS_D65, result);
+  }
+  function convertToRgb(cs, src, srcOffset, dest, destOffset, scale) {
+    var A = adjustToRange(0, 1, src[srcOffset] * scale);
+    var B = adjustToRange(0, 1, src[srcOffset + 1] * scale);
+    var C = adjustToRange(0, 1, src[srcOffset + 2] * scale);
+    var AGR = Math.pow(A, cs.GR);
+    var BGG = Math.pow(B, cs.GG);
+    var CGB = Math.pow(C, cs.GB);
+    var X = cs.MXA * AGR + cs.MXB * BGG + cs.MXC * CGB;
+    var Y = cs.MYA * AGR + cs.MYB * BGG + cs.MYC * CGB;
+    var Z = cs.MZA * AGR + cs.MZB * BGG + cs.MZC * CGB;
+    var XYZ = tempConvertMatrix1;
+    XYZ[0] = X;
+    XYZ[1] = Y;
+    XYZ[2] = Z;
+    var XYZ_Flat = tempConvertMatrix2;
+    normalizeWhitePointToFlat(cs.whitePoint, XYZ, XYZ_Flat);
+    var XYZ_Black = tempConvertMatrix1;
+    compensateBlackPoint(cs.blackPoint, XYZ_Flat, XYZ_Black);
+    var XYZ_D65 = tempConvertMatrix2;
+    normalizeWhitePointToD65(FLAT_WHITEPOINT_MATRIX, XYZ_Black, XYZ_D65);
+    var SRGB = tempConvertMatrix1;
+    matrixProduct(SRGB_D65_XYZ_TO_RGB_MATRIX, XYZ_D65, SRGB);
+    var sR = sRGBTransferFunction(SRGB[0]);
+    var sG = sRGBTransferFunction(SRGB[1]);
+    var sB = sRGBTransferFunction(SRGB[2]);
+    dest[destOffset] = Math.round(sR * 255);
+    dest[destOffset + 1] = Math.round(sG * 255);
+    dest[destOffset + 2] = Math.round(sB * 255);
+  }
+  CalRGBCS.prototype = {
+    getRgb: function CalRGBCS_getRgb(src, srcOffset) {
+      var rgb = new Uint8Array(3);
+      this.getRgbItem(src, srcOffset, rgb, 0);
+      return rgb;
+    },
+    getRgbItem: function CalRGBCS_getRgbItem(src, srcOffset, dest, destOffset) {
+      convertToRgb(this, src, srcOffset, dest, destOffset, 1);
+    },
+    getRgbBuffer: function CalRGBCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
+      var scale = 1 / ((1 << bits) - 1);
+      for (var i = 0; i < count; ++i) {
+        convertToRgb(this, src, srcOffset, dest, destOffset, scale);
+        srcOffset += 3;
+        destOffset += 3 + alpha01;
+      }
+    },
+    getOutputLength: function CalRGBCS_getOutputLength(inputLength, alpha01) {
+      return inputLength * (3 + alpha01) / 3 | 0;
+    },
+    isPassthrough: ColorSpace.prototype.isPassthrough,
+    fillRgb: ColorSpace.prototype.fillRgb,
+    isDefaultDecode: function CalRGBCS_isDefaultDecode(decodeMap) {
+      return ColorSpace.isDefaultDecode(decodeMap, this.numComps);
+    },
+    usesZeroToOneRange: true
+  };
+  return CalRGBCS;
 }();
 var LabCS = function LabCSClosure() {
- function LabCS(whitePoint, blackPoint, range) {
-  this.name = 'Lab';
-  this.numComps = 3;
-  this.defaultColor = new Float32Array(this.numComps);
-  if (!whitePoint) {
-   error('WhitePoint missing - required for color space Lab');
-  }
-  blackPoint = blackPoint || [
-   0,
-   0,
-   0
-  ];
-  range = range || [
-   -100,
-   100,
-   -100,
-   100
-  ];
-  this.XW = whitePoint[0];
-  this.YW = whitePoint[1];
-  this.ZW = whitePoint[2];
-  this.amin = range[0];
-  this.amax = range[1];
-  this.bmin = range[2];
-  this.bmax = range[3];
-  this.XB = blackPoint[0];
-  this.YB = blackPoint[1];
-  this.ZB = blackPoint[2];
-  if (this.XW < 0 || this.ZW < 0 || this.YW !== 1) {
-   error('Invalid WhitePoint components, no fallback available');
-  }
-  if (this.XB < 0 || this.YB < 0 || this.ZB < 0) {
-   info('Invalid BlackPoint, falling back to default');
-   this.XB = this.YB = this.ZB = 0;
-  }
-  if (this.amin > this.amax || this.bmin > this.bmax) {
-   info('Invalid Range, falling back to defaults');
-   this.amin = -100;
-   this.amax = 100;
-   this.bmin = -100;
-   this.bmax = 100;
-  }
- }
- function fn_g(x) {
-  var result;
-  if (x >= 6 / 29) {
-   result = x * x * x;
-  } else {
-   result = 108 / 841 * (x - 4 / 29);
-  }
-  return result;
- }
- function decode(value, high1, low2, high2) {
-  return low2 + value * (high2 - low2) / high1;
- }
- function convertToRgb(cs, src, srcOffset, maxVal, dest, destOffset) {
-  var Ls = src[srcOffset];
-  var as = src[srcOffset + 1];
-  var bs = src[srcOffset + 2];
-  if (maxVal !== false) {
-   Ls = decode(Ls, maxVal, 0, 100);
-   as = decode(as, maxVal, cs.amin, cs.amax);
-   bs = decode(bs, maxVal, cs.bmin, cs.bmax);
-  }
-  as = as > cs.amax ? cs.amax : as < cs.amin ? cs.amin : as;
-  bs = bs > cs.bmax ? cs.bmax : bs < cs.bmin ? cs.bmin : bs;
-  var M = (Ls + 16) / 116;
-  var L = M + as / 500;
-  var N = M - bs / 200;
-  var X = cs.XW * fn_g(L);
-  var Y = cs.YW * fn_g(M);
-  var Z = cs.ZW * fn_g(N);
-  var r, g, b;
-  if (cs.ZW < 1) {
-   r = X * 3.1339 + Y * -1.6170 + Z * -0.4906;
-   g = X * -0.9785 + Y * 1.9160 + Z * 0.0333;
-   b = X * 0.0720 + Y * -0.2290 + Z * 1.4057;
-  } else {
-   r = X * 3.2406 + Y * -1.5372 + Z * -0.4986;
-   g = X * -0.9689 + Y * 1.8758 + Z * 0.0415;
-   b = X * 0.0557 + Y * -0.2040 + Z * 1.0570;
+  function LabCS(whitePoint, blackPoint, range) {
+    this.name = 'Lab';
+    this.numComps = 3;
+    this.defaultColor = new Float32Array(this.numComps);
+    if (!whitePoint) {
+      error('WhitePoint missing - required for color space Lab');
+    }
+    blackPoint = blackPoint || [0, 0, 0];
+    range = range || [-100, 100, -100, 100];
+    this.XW = whitePoint[0];
+    this.YW = whitePoint[1];
+    this.ZW = whitePoint[2];
+    this.amin = range[0];
+    this.amax = range[1];
+    this.bmin = range[2];
+    this.bmax = range[3];
+    this.XB = blackPoint[0];
+    this.YB = blackPoint[1];
+    this.ZB = blackPoint[2];
+    if (this.XW < 0 || this.ZW < 0 || this.YW !== 1) {
+      error('Invalid WhitePoint components, no fallback available');
+    }
+    if (this.XB < 0 || this.YB < 0 || this.ZB < 0) {
+      info('Invalid BlackPoint, falling back to default');
+      this.XB = this.YB = this.ZB = 0;
+    }
+    if (this.amin > this.amax || this.bmin > this.bmax) {
+      info('Invalid Range, falling back to defaults');
+      this.amin = -100;
+      this.amax = 100;
+      this.bmin = -100;
+      this.bmax = 100;
+    }
   }
-  dest[destOffset] = r <= 0 ? 0 : r >= 1 ? 255 : Math.sqrt(r) * 255 | 0;
-  dest[destOffset + 1] = g <= 0 ? 0 : g >= 1 ? 255 : Math.sqrt(g) * 255 | 0;
-  dest[destOffset + 2] = b <= 0 ? 0 : b >= 1 ? 255 : Math.sqrt(b) * 255 | 0;
- }
- LabCS.prototype = {
-  getRgb: ColorSpace.prototype.getRgb,
-  getRgbItem: function LabCS_getRgbItem(src, srcOffset, dest, destOffset) {
-   convertToRgb(this, src, srcOffset, false, dest, destOffset);
-  },
-  getRgbBuffer: function LabCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
-   var maxVal = (1 << bits) - 1;
-   for (var i = 0; i < count; i++) {
-    convertToRgb(this, src, srcOffset, maxVal, dest, destOffset);
-    srcOffset += 3;
-    destOffset += 3 + alpha01;
-   }
-  },
-  getOutputLength: function LabCS_getOutputLength(inputLength, alpha01) {
-   return inputLength * (3 + alpha01) / 3 | 0;
-  },
-  isPassthrough: ColorSpace.prototype.isPassthrough,
-  fillRgb: ColorSpace.prototype.fillRgb,
-  isDefaultDecode: function LabCS_isDefaultDecode(decodeMap) {
-   return true;
-  },
-  usesZeroToOneRange: false
- };
- return LabCS;
+  function fn_g(x) {
+    var result;
+    if (x >= 6 / 29) {
+      result = x * x * x;
+    } else {
+      result = 108 / 841 * (x - 4 / 29);
+    }
+    return result;
+  }
+  function decode(value, high1, low2, high2) {
+    return low2 + value * (high2 - low2) / high1;
+  }
+  function convertToRgb(cs, src, srcOffset, maxVal, dest, destOffset) {
+    var Ls = src[srcOffset];
+    var as = src[srcOffset + 1];
+    var bs = src[srcOffset + 2];
+    if (maxVal !== false) {
+      Ls = decode(Ls, maxVal, 0, 100);
+      as = decode(as, maxVal, cs.amin, cs.amax);
+      bs = decode(bs, maxVal, cs.bmin, cs.bmax);
+    }
+    as = as > cs.amax ? cs.amax : as < cs.amin ? cs.amin : as;
+    bs = bs > cs.bmax ? cs.bmax : bs < cs.bmin ? cs.bmin : bs;
+    var M = (Ls + 16) / 116;
+    var L = M + as / 500;
+    var N = M - bs / 200;
+    var X = cs.XW * fn_g(L);
+    var Y = cs.YW * fn_g(M);
+    var Z = cs.ZW * fn_g(N);
+    var r, g, b;
+    if (cs.ZW < 1) {
+      r = X * 3.1339 + Y * -1.6170 + Z * -0.4906;
+      g = X * -0.9785 + Y * 1.9160 + Z * 0.0333;
+      b = X * 0.0720 + Y * -0.2290 + Z * 1.4057;
+    } else {
+      r = X * 3.2406 + Y * -1.5372 + Z * -0.4986;
+      g = X * -0.9689 + Y * 1.8758 + Z * 0.0415;
+      b = X * 0.0557 + Y * -0.2040 + Z * 1.0570;
+    }
+    dest[destOffset] = r <= 0 ? 0 : r >= 1 ? 255 : Math.sqrt(r) * 255 | 0;
+    dest[destOffset + 1] = g <= 0 ? 0 : g >= 1 ? 255 : Math.sqrt(g) * 255 | 0;
+    dest[destOffset + 2] = b <= 0 ? 0 : b >= 1 ? 255 : Math.sqrt(b) * 255 | 0;
+  }
+  LabCS.prototype = {
+    getRgb: ColorSpace.prototype.getRgb,
+    getRgbItem: function LabCS_getRgbItem(src, srcOffset, dest, destOffset) {
+      convertToRgb(this, src, srcOffset, false, dest, destOffset);
+    },
+    getRgbBuffer: function LabCS_getRgbBuffer(src, srcOffset, count, dest, destOffset, bits, alpha01) {
+      var maxVal = (1 << bits) - 1;
+      for (var i = 0; i < count; i++) {
+        convertToRgb(this, src, srcOffset, maxVal, dest, destOffset);
+        srcOffset += 3;
+        destOffset += 3 + alpha01;
+      }
+    },
+    getOutputLength: function LabCS_getOutputLength(inputLength, alpha01) {
+      return inputLength * (3 + alpha01) / 3 | 0;
+    },
+    isPassthrough: ColorSpace.prototype.isPassthrough,
+    fillRgb: ColorSpace.prototype.fillRgb,
+    isDefaultDecode: function LabCS_isDefaultDecode(decodeMap) {
+      return true;
+    },
+    usesZeroToOneRange: false
+  };
+  return LabCS;
 }();
 exports.ColorSpace = ColorSpace;

File diff suppressed because it is too large
+ 339 - 763
lib/core/crypto.js


+ 437 - 451
lib/core/document.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var coreStream = require('./stream.js');
@@ -52,468 +53,453 @@ var OperatorList = coreEvaluator.OperatorList;
 var PartialEvaluator = coreEvaluator.PartialEvaluator;
 var AnnotationFactory = coreAnnotation.AnnotationFactory;
 var Page = function PageClosure() {
- var DEFAULT_USER_UNIT = 1.0;
- var LETTER_SIZE_MEDIABOX = [
-  0,
-  0,
-  612,
-  792
- ];
- function isAnnotationRenderable(annotation, intent) {
-  return intent === 'display' && annotation.viewable || intent === 'print' && annotation.printable;
- }
- function Page(pdfManager, xref, pageIndex, pageDict, ref, fontCache, builtInCMapCache) {
-  this.pdfManager = pdfManager;
-  this.pageIndex = pageIndex;
-  this.pageDict = pageDict;
-  this.xref = xref;
-  this.ref = ref;
-  this.fontCache = fontCache;
-  this.builtInCMapCache = builtInCMapCache;
-  this.evaluatorOptions = pdfManager.evaluatorOptions;
-  this.resourcesPromise = null;
-  var uniquePrefix = 'p' + this.pageIndex + '_';
-  var idCounters = { obj: 0 };
-  this.idFactory = {
-   createObjId: function () {
-    return uniquePrefix + ++idCounters.obj;
-   }
-  };
- }
- Page.prototype = {
-  getPageProp: function Page_getPageProp(key) {
-   return this.pageDict.get(key);
-  },
-  getInheritedPageProp: function Page_getInheritedPageProp(key, getArray) {
-   var dict = this.pageDict, valueArray = null, loopCount = 0;
-   var MAX_LOOP_COUNT = 100;
-   getArray = getArray || false;
-   while (dict) {
-    var value = getArray ? dict.getArray(key) : dict.get(key);
-    if (value !== undefined) {
-     if (!valueArray) {
-      valueArray = [];
-     }
-     valueArray.push(value);
-    }
-    if (++loopCount > MAX_LOOP_COUNT) {
-     warn('getInheritedPageProp: maximum loop count exceeded for ' + key);
-     return valueArray ? valueArray[0] : undefined;
-    }
-    dict = dict.get('Parent');
-   }
-   if (!valueArray) {
-    return undefined;
-   }
-   if (valueArray.length === 1 || !isDict(valueArray[0])) {
-    return valueArray[0];
-   }
-   return Dict.merge(this.xref, valueArray);
-  },
-  get content() {
-   return this.getPageProp('Contents');
-  },
-  get resources() {
-   return shadow(this, 'resources', this.getInheritedPageProp('Resources') || Dict.empty);
-  },
-  get mediaBox() {
-   var mediaBox = this.getInheritedPageProp('MediaBox', true);
-   if (!isArray(mediaBox) || mediaBox.length !== 4) {
-    return shadow(this, 'mediaBox', LETTER_SIZE_MEDIABOX);
-   }
-   return shadow(this, 'mediaBox', mediaBox);
-  },
-  get cropBox() {
-   var cropBox = this.getInheritedPageProp('CropBox', true);
-   if (!isArray(cropBox) || cropBox.length !== 4) {
-    return shadow(this, 'cropBox', this.mediaBox);
-   }
-   return shadow(this, 'cropBox', cropBox);
-  },
-  get userUnit() {
-   var obj = this.getPageProp('UserUnit');
-   if (!isNum(obj) || obj <= 0) {
-    obj = DEFAULT_USER_UNIT;
-   }
-   return shadow(this, 'userUnit', obj);
-  },
-  get view() {
-   var mediaBox = this.mediaBox, cropBox = this.cropBox;
-   if (mediaBox === cropBox) {
-    return shadow(this, 'view', mediaBox);
-   }
-   var intersection = Util.intersect(cropBox, mediaBox);
-   return shadow(this, 'view', intersection || mediaBox);
-  },
-  get rotate() {
-   var rotate = this.getInheritedPageProp('Rotate') || 0;
-   if (rotate % 90 !== 0) {
-    rotate = 0;
-   } else if (rotate >= 360) {
-    rotate = rotate % 360;
-   } else if (rotate < 0) {
-    rotate = (rotate % 360 + 360) % 360;
-   }
-   return shadow(this, 'rotate', rotate);
-  },
-  getContentStream: function Page_getContentStream() {
-   var content = this.content;
-   var stream;
-   if (isArray(content)) {
-    var xref = this.xref;
-    var i, n = content.length;
-    var streams = [];
-    for (i = 0; i < n; ++i) {
-     streams.push(xref.fetchIfRef(content[i]));
-    }
-    stream = new StreamsSequenceStream(streams);
-   } else if (isStream(content)) {
-    stream = content;
-   } else {
-    stream = new NullStream();
-   }
-   return stream;
-  },
-  loadResources: function Page_loadResources(keys) {
-   if (!this.resourcesPromise) {
-    this.resourcesPromise = this.pdfManager.ensure(this, 'resources');
-   }
-   return this.resourcesPromise.then(function resourceSuccess() {
-    var objectLoader = new ObjectLoader(this.resources.map, keys, this.xref);
-    return objectLoader.load();
-   }.bind(this));
-  },
-  getOperatorList: function Page_getOperatorList(handler, task, intent, renderInteractiveForms) {
-   var self = this;
-   var pdfManager = this.pdfManager;
-   var contentStreamPromise = pdfManager.ensure(this, 'getContentStream', []);
-   var resourcesPromise = this.loadResources([
-    'ExtGState',
-    'ColorSpace',
-    'Pattern',
-    'Shading',
-    'XObject',
-    'Font'
-   ]);
-   var partialEvaluator = new PartialEvaluator(pdfManager, this.xref, handler, this.pageIndex, this.idFactory, this.fontCache, this.builtInCMapCache, this.evaluatorOptions);
-   var dataPromises = Promise.all([
-    contentStreamPromise,
-    resourcesPromise
-   ]);
-   var pageListPromise = dataPromises.then(function (data) {
-    var contentStream = data[0];
-    var opList = new OperatorList(intent, handler, self.pageIndex);
-    handler.send('StartRenderPage', {
-     transparency: partialEvaluator.hasBlendModes(self.resources),
-     pageIndex: self.pageIndex,
-     intent: intent
-    });
-    return partialEvaluator.getOperatorList(contentStream, task, self.resources, opList).then(function () {
-     return opList;
-    });
-   });
-   var annotationsPromise = pdfManager.ensure(this, 'annotations');
-   return Promise.all([
-    pageListPromise,
-    annotationsPromise
-   ]).then(function (datas) {
-    var pageOpList = datas[0];
-    var annotations = datas[1];
-    if (annotations.length === 0) {
-     pageOpList.flush(true);
-     return pageOpList;
-    }
-    var i, ii, opListPromises = [];
-    for (i = 0, ii = annotations.length; i < ii; i++) {
-     if (isAnnotationRenderable(annotations[i], intent)) {
-      opListPromises.push(annotations[i].getOperatorList(partialEvaluator, task, renderInteractiveForms));
-     }
-    }
-    return Promise.all(opListPromises).then(function (opLists) {
-     pageOpList.addOp(OPS.beginAnnotations, []);
-     for (i = 0, ii = opLists.length; i < ii; i++) {
-      pageOpList.addOpList(opLists[i]);
-     }
-     pageOpList.addOp(OPS.endAnnotations, []);
-     pageOpList.flush(true);
-     return pageOpList;
-    });
-   });
-  },
-  extractTextContent: function Page_extractTextContent(handler, task, normalizeWhitespace, combineTextItems) {
-   var self = this;
-   var pdfManager = this.pdfManager;
-   var contentStreamPromise = pdfManager.ensure(this, 'getContentStream', []);
-   var resourcesPromise = this.loadResources([
-    'ExtGState',
-    'XObject',
-    'Font'
-   ]);
-   var dataPromises = Promise.all([
-    contentStreamPromise,
-    resourcesPromise
-   ]);
-   return dataPromises.then(function (data) {
-    var contentStream = data[0];
-    var partialEvaluator = new PartialEvaluator(pdfManager, self.xref, handler, self.pageIndex, self.idFactory, self.fontCache, self.builtInCMapCache, self.evaluatorOptions);
-    return partialEvaluator.getTextContent(contentStream, task, self.resources, null, normalizeWhitespace, combineTextItems);
-   });
-  },
-  getAnnotationsData: function Page_getAnnotationsData(intent) {
-   var annotations = this.annotations;
-   var annotationsData = [];
-   for (var i = 0, n = annotations.length; i < n; ++i) {
-    if (!intent || isAnnotationRenderable(annotations[i], intent)) {
-     annotationsData.push(annotations[i].data);
-    }
-   }
-   return annotationsData;
-  },
-  get annotations() {
-   var annotations = [];
-   var annotationRefs = this.getInheritedPageProp('Annots') || [];
-   var annotationFactory = new AnnotationFactory();
-   for (var i = 0, n = annotationRefs.length; i < n; ++i) {
-    var annotationRef = annotationRefs[i];
-    var annotation = annotationFactory.create(this.xref, annotationRef, this.pdfManager, this.idFactory);
-    if (annotation) {
-     annotations.push(annotation);
-    }
-   }
-   return shadow(this, 'annotations', annotations);
-  }
- };
- return Page;
-}();
-var PDFDocument = function PDFDocumentClosure() {
- var FINGERPRINT_FIRST_BYTES = 1024;
- var EMPTY_FINGERPRINT = '\x00\x00\x00\x00\x00\x00\x00' + '\x00\x00\x00\x00\x00\x00\x00\x00\x00';
- function PDFDocument(pdfManager, arg) {
-  var stream;
-  if (isStream(arg)) {
-   stream = arg;
-  } else if (isArrayBuffer(arg)) {
-   stream = new Stream(arg);
-  } else {
-   error('PDFDocument: Unknown argument type');
-  }
-  assert(stream.length > 0, 'stream must have data');
-  this.pdfManager = pdfManager;
-  this.stream = stream;
-  this.xref = new XRef(stream, pdfManager);
- }
- function find(stream, needle, limit, backwards) {
-  var pos = stream.pos;
-  var end = stream.end;
-  var strBuf = [];
-  if (pos + limit > end) {
-   limit = end - pos;
-  }
-  for (var n = 0; n < limit; ++n) {
-   strBuf.push(String.fromCharCode(stream.getByte()));
+  var DEFAULT_USER_UNIT = 1.0;
+  var LETTER_SIZE_MEDIABOX = [0, 0, 612, 792];
+  function isAnnotationRenderable(annotation, intent) {
+    return intent === 'display' && annotation.viewable || intent === 'print' && annotation.printable;
   }
-  var str = strBuf.join('');
-  stream.pos = pos;
-  var index = backwards ? str.lastIndexOf(needle) : str.indexOf(needle);
-  if (index === -1) {
-   return false;
-  }
-  stream.pos += index;
-  return true;
- }
- var DocumentInfoValidators = {
-  get entries() {
-   return shadow(this, 'entries', {
-    Title: isString,
-    Author: isString,
-    Subject: isString,
-    Keywords: isString,
-    Creator: isString,
-    Producer: isString,
-    CreationDate: isString,
-    ModDate: isString,
-    Trapped: isName
-   });
+  function Page(pdfManager, xref, pageIndex, pageDict, ref, fontCache, builtInCMapCache) {
+    this.pdfManager = pdfManager;
+    this.pageIndex = pageIndex;
+    this.pageDict = pageDict;
+    this.xref = xref;
+    this.ref = ref;
+    this.fontCache = fontCache;
+    this.builtInCMapCache = builtInCMapCache;
+    this.evaluatorOptions = pdfManager.evaluatorOptions;
+    this.resourcesPromise = null;
+    var uniquePrefix = 'p' + this.pageIndex + '_';
+    var idCounters = { obj: 0 };
+    this.idFactory = {
+      createObjId: function () {
+        return uniquePrefix + ++idCounters.obj;
+      }
+    };
   }
- };
- PDFDocument.prototype = {
-  parse: function PDFDocument_parse(recoveryMode) {
-   this.setup(recoveryMode);
-   var version = this.catalog.catDict.get('Version');
-   if (isName(version)) {
-    this.pdfFormatVersion = version.name;
-   }
-   try {
-    this.acroForm = this.catalog.catDict.get('AcroForm');
-    if (this.acroForm) {
-     this.xfa = this.acroForm.get('XFA');
-     var fields = this.acroForm.get('Fields');
-     if ((!fields || !isArray(fields) || fields.length === 0) && !this.xfa) {
-      this.acroForm = null;
-     }
-    }
-   } catch (ex) {
-    if (ex instanceof MissingDataException) {
-     throw ex;
-    }
-    info('Something wrong with AcroForm entry');
-    this.acroForm = null;
-   }
-  },
-  get linearization() {
-   var linearization = null;
-   if (this.stream.length) {
-    try {
-     linearization = Linearization.create(this.stream);
-    } catch (err) {
-     if (err instanceof MissingDataException) {
-      throw err;
-     }
-     info(err);
-    }
-   }
-   return shadow(this, 'linearization', linearization);
-  },
-  get startXRef() {
-   var stream = this.stream;
-   var startXRef = 0;
-   var linearization = this.linearization;
-   if (linearization) {
-    stream.reset();
-    if (find(stream, 'endobj', 1024)) {
-     startXRef = stream.pos + 6;
-    }
-   } else {
-    var step = 1024;
-    var found = false, pos = stream.end;
-    while (!found && pos > 0) {
-     pos -= step - 'startxref'.length;
-     if (pos < 0) {
-      pos = 0;
-     }
-     stream.pos = pos;
-     found = find(stream, 'startxref', step, true);
+  Page.prototype = {
+    getPageProp: function Page_getPageProp(key) {
+      return this.pageDict.get(key);
+    },
+    getInheritedPageProp: function Page_getInheritedPageProp(key, getArray) {
+      var dict = this.pageDict,
+          valueArray = null,
+          loopCount = 0;
+      var MAX_LOOP_COUNT = 100;
+      getArray = getArray || false;
+      while (dict) {
+        var value = getArray ? dict.getArray(key) : dict.get(key);
+        if (value !== undefined) {
+          if (!valueArray) {
+            valueArray = [];
+          }
+          valueArray.push(value);
+        }
+        if (++loopCount > MAX_LOOP_COUNT) {
+          warn('getInheritedPageProp: maximum loop count exceeded for ' + key);
+          return valueArray ? valueArray[0] : undefined;
+        }
+        dict = dict.get('Parent');
+      }
+      if (!valueArray) {
+        return undefined;
+      }
+      if (valueArray.length === 1 || !isDict(valueArray[0])) {
+        return valueArray[0];
+      }
+      return Dict.merge(this.xref, valueArray);
+    },
+    get content() {
+      return this.getPageProp('Contents');
+    },
+    get resources() {
+      return shadow(this, 'resources', this.getInheritedPageProp('Resources') || Dict.empty);
+    },
+    get mediaBox() {
+      var mediaBox = this.getInheritedPageProp('MediaBox', true);
+      if (!isArray(mediaBox) || mediaBox.length !== 4) {
+        return shadow(this, 'mediaBox', LETTER_SIZE_MEDIABOX);
+      }
+      return shadow(this, 'mediaBox', mediaBox);
+    },
+    get cropBox() {
+      var cropBox = this.getInheritedPageProp('CropBox', true);
+      if (!isArray(cropBox) || cropBox.length !== 4) {
+        return shadow(this, 'cropBox', this.mediaBox);
+      }
+      return shadow(this, 'cropBox', cropBox);
+    },
+    get userUnit() {
+      var obj = this.getPageProp('UserUnit');
+      if (!isNum(obj) || obj <= 0) {
+        obj = DEFAULT_USER_UNIT;
+      }
+      return shadow(this, 'userUnit', obj);
+    },
+    get view() {
+      var mediaBox = this.mediaBox,
+          cropBox = this.cropBox;
+      if (mediaBox === cropBox) {
+        return shadow(this, 'view', mediaBox);
+      }
+      var intersection = Util.intersect(cropBox, mediaBox);
+      return shadow(this, 'view', intersection || mediaBox);
+    },
+    get rotate() {
+      var rotate = this.getInheritedPageProp('Rotate') || 0;
+      if (rotate % 90 !== 0) {
+        rotate = 0;
+      } else if (rotate >= 360) {
+        rotate = rotate % 360;
+      } else if (rotate < 0) {
+        rotate = (rotate % 360 + 360) % 360;
+      }
+      return shadow(this, 'rotate', rotate);
+    },
+    getContentStream: function Page_getContentStream() {
+      var content = this.content;
+      var stream;
+      if (isArray(content)) {
+        var xref = this.xref;
+        var i,
+            n = content.length;
+        var streams = [];
+        for (i = 0; i < n; ++i) {
+          streams.push(xref.fetchIfRef(content[i]));
+        }
+        stream = new StreamsSequenceStream(streams);
+      } else if (isStream(content)) {
+        stream = content;
+      } else {
+        stream = new NullStream();
+      }
+      return stream;
+    },
+    loadResources: function Page_loadResources(keys) {
+      if (!this.resourcesPromise) {
+        this.resourcesPromise = this.pdfManager.ensure(this, 'resources');
+      }
+      return this.resourcesPromise.then(function resourceSuccess() {
+        var objectLoader = new ObjectLoader(this.resources.map, keys, this.xref);
+        return objectLoader.load();
+      }.bind(this));
+    },
+    getOperatorList: function Page_getOperatorList(handler, task, intent, renderInteractiveForms) {
+      var self = this;
+      var pdfManager = this.pdfManager;
+      var contentStreamPromise = pdfManager.ensure(this, 'getContentStream', []);
+      var resourcesPromise = this.loadResources(['ExtGState', 'ColorSpace', 'Pattern', 'Shading', 'XObject', 'Font']);
+      var partialEvaluator = new PartialEvaluator(pdfManager, this.xref, handler, this.pageIndex, this.idFactory, this.fontCache, this.builtInCMapCache, this.evaluatorOptions);
+      var dataPromises = Promise.all([contentStreamPromise, resourcesPromise]);
+      var pageListPromise = dataPromises.then(function (data) {
+        var contentStream = data[0];
+        var opList = new OperatorList(intent, handler, self.pageIndex);
+        handler.send('StartRenderPage', {
+          transparency: partialEvaluator.hasBlendModes(self.resources),
+          pageIndex: self.pageIndex,
+          intent: intent
+        });
+        return partialEvaluator.getOperatorList(contentStream, task, self.resources, opList).then(function () {
+          return opList;
+        });
+      });
+      var annotationsPromise = pdfManager.ensure(this, 'annotations');
+      return Promise.all([pageListPromise, annotationsPromise]).then(function (datas) {
+        var pageOpList = datas[0];
+        var annotations = datas[1];
+        if (annotations.length === 0) {
+          pageOpList.flush(true);
+          return pageOpList;
+        }
+        var i,
+            ii,
+            opListPromises = [];
+        for (i = 0, ii = annotations.length; i < ii; i++) {
+          if (isAnnotationRenderable(annotations[i], intent)) {
+            opListPromises.push(annotations[i].getOperatorList(partialEvaluator, task, renderInteractiveForms));
+          }
+        }
+        return Promise.all(opListPromises).then(function (opLists) {
+          pageOpList.addOp(OPS.beginAnnotations, []);
+          for (i = 0, ii = opLists.length; i < ii; i++) {
+            pageOpList.addOpList(opLists[i]);
+          }
+          pageOpList.addOp(OPS.endAnnotations, []);
+          pageOpList.flush(true);
+          return pageOpList;
+        });
+      });
+    },
+    extractTextContent: function Page_extractTextContent(handler, task, normalizeWhitespace, combineTextItems) {
+      var self = this;
+      var pdfManager = this.pdfManager;
+      var contentStreamPromise = pdfManager.ensure(this, 'getContentStream', []);
+      var resourcesPromise = this.loadResources(['ExtGState', 'XObject', 'Font']);
+      var dataPromises = Promise.all([contentStreamPromise, resourcesPromise]);
+      return dataPromises.then(function (data) {
+        var contentStream = data[0];
+        var partialEvaluator = new PartialEvaluator(pdfManager, self.xref, handler, self.pageIndex, self.idFactory, self.fontCache, self.builtInCMapCache, self.evaluatorOptions);
+        return partialEvaluator.getTextContent(contentStream, task, self.resources, null, normalizeWhitespace, combineTextItems);
+      });
+    },
+    getAnnotationsData: function Page_getAnnotationsData(intent) {
+      var annotations = this.annotations;
+      var annotationsData = [];
+      for (var i = 0, n = annotations.length; i < n; ++i) {
+        if (!intent || isAnnotationRenderable(annotations[i], intent)) {
+          annotationsData.push(annotations[i].data);
+        }
+      }
+      return annotationsData;
+    },
+    get annotations() {
+      var annotations = [];
+      var annotationRefs = this.getInheritedPageProp('Annots') || [];
+      var annotationFactory = new AnnotationFactory();
+      for (var i = 0, n = annotationRefs.length; i < n; ++i) {
+        var annotationRef = annotationRefs[i];
+        var annotation = annotationFactory.create(this.xref, annotationRef, this.pdfManager, this.idFactory);
+        if (annotation) {
+          annotations.push(annotation);
+        }
+      }
+      return shadow(this, 'annotations', annotations);
     }
-    if (found) {
-     stream.skip(9);
-     var ch;
-     do {
-      ch = stream.getByte();
-     } while (isSpace(ch));
-     var str = '';
-     while (ch >= 0x20 && ch <= 0x39) {
-      str += String.fromCharCode(ch);
-      ch = stream.getByte();
-     }
-     startXRef = parseInt(str, 10);
-     if (isNaN(startXRef)) {
-      startXRef = 0;
-     }
+  };
+  return Page;
+}();
+var PDFDocument = function PDFDocumentClosure() {
+  var FINGERPRINT_FIRST_BYTES = 1024;
+  var EMPTY_FINGERPRINT = '\x00\x00\x00\x00\x00\x00\x00' + '\x00\x00\x00\x00\x00\x00\x00\x00\x00';
+  function PDFDocument(pdfManager, arg) {
+    var stream;
+    if (isStream(arg)) {
+      stream = arg;
+    } else if (isArrayBuffer(arg)) {
+      stream = new Stream(arg);
+    } else {
+      error('PDFDocument: Unknown argument type');
     }
-   }
-   return shadow(this, 'startXRef', startXRef);
-  },
-  get mainXRefEntriesOffset() {
-   var mainXRefEntriesOffset = 0;
-   var linearization = this.linearization;
-   if (linearization) {
-    mainXRefEntriesOffset = linearization.mainXRefEntriesOffset;
-   }
-   return shadow(this, 'mainXRefEntriesOffset', mainXRefEntriesOffset);
-  },
-  checkHeader: function PDFDocument_checkHeader() {
-   var stream = this.stream;
-   stream.reset();
-   if (find(stream, '%PDF-', 1024)) {
-    stream.moveStart();
-    var MAX_VERSION_LENGTH = 12;
-    var version = '', ch;
-    while ((ch = stream.getByte()) > 0x20) {
-     if (version.length >= MAX_VERSION_LENGTH) {
-      break;
-     }
-     version += String.fromCharCode(ch);
+    assert(stream.length > 0, 'stream must have data');
+    this.pdfManager = pdfManager;
+    this.stream = stream;
+    this.xref = new XRef(stream, pdfManager);
+  }
+  function find(stream, needle, limit, backwards) {
+    var pos = stream.pos;
+    var end = stream.end;
+    var strBuf = [];
+    if (pos + limit > end) {
+      limit = end - pos;
     }
-    if (!this.pdfFormatVersion) {
-     this.pdfFormatVersion = version.substring(5);
+    for (var n = 0; n < limit; ++n) {
+      strBuf.push(String.fromCharCode(stream.getByte()));
     }
-    return;
-   }
-  },
-  parseStartXRef: function PDFDocument_parseStartXRef() {
-   var startXRef = this.startXRef;
-   this.xref.setStartXRef(startXRef);
-  },
-  setup: function PDFDocument_setup(recoveryMode) {
-   this.xref.parse(recoveryMode);
-   var self = this;
-   var pageFactory = {
-    createPage: function (pageIndex, dict, ref, fontCache, builtInCMapCache) {
-     return new Page(self.pdfManager, self.xref, pageIndex, dict, ref, fontCache, builtInCMapCache);
+    var str = strBuf.join('');
+    stream.pos = pos;
+    var index = backwards ? str.lastIndexOf(needle) : str.indexOf(needle);
+    if (index === -1) {
+      return false;
     }
-   };
-   this.catalog = new Catalog(this.pdfManager, this.xref, pageFactory);
-  },
-  get numPages() {
-   var linearization = this.linearization;
-   var num = linearization ? linearization.numPages : this.catalog.numPages;
-   return shadow(this, 'numPages', num);
-  },
-  get documentInfo() {
-   var docInfo = {
-    PDFFormatVersion: this.pdfFormatVersion,
-    IsAcroFormPresent: !!this.acroForm,
-    IsXFAPresent: !!this.xfa
-   };
-   var infoDict;
-   try {
-    infoDict = this.xref.trailer.get('Info');
-   } catch (err) {
-    if (err instanceof MissingDataException) {
-     throw err;
+    stream.pos += index;
+    return true;
+  }
+  var DocumentInfoValidators = {
+    get entries() {
+      return shadow(this, 'entries', {
+        Title: isString,
+        Author: isString,
+        Subject: isString,
+        Keywords: isString,
+        Creator: isString,
+        Producer: isString,
+        CreationDate: isString,
+        ModDate: isString,
+        Trapped: isName
+      });
     }
-    info('The document information dictionary is invalid.');
-   }
-   if (infoDict) {
-    var validEntries = DocumentInfoValidators.entries;
-    for (var key in validEntries) {
-     if (infoDict.has(key)) {
-      var value = infoDict.get(key);
-      if (validEntries[key](value)) {
-       docInfo[key] = typeof value !== 'string' ? value : stringToPDFString(value);
+  };
+  PDFDocument.prototype = {
+    parse: function PDFDocument_parse(recoveryMode) {
+      this.setup(recoveryMode);
+      var version = this.catalog.catDict.get('Version');
+      if (isName(version)) {
+        this.pdfFormatVersion = version.name;
+      }
+      try {
+        this.acroForm = this.catalog.catDict.get('AcroForm');
+        if (this.acroForm) {
+          this.xfa = this.acroForm.get('XFA');
+          var fields = this.acroForm.get('Fields');
+          if ((!fields || !isArray(fields) || fields.length === 0) && !this.xfa) {
+            this.acroForm = null;
+          }
+        }
+      } catch (ex) {
+        if (ex instanceof MissingDataException) {
+          throw ex;
+        }
+        info('Something wrong with AcroForm entry');
+        this.acroForm = null;
+      }
+    },
+    get linearization() {
+      var linearization = null;
+      if (this.stream.length) {
+        try {
+          linearization = Linearization.create(this.stream);
+        } catch (err) {
+          if (err instanceof MissingDataException) {
+            throw err;
+          }
+          info(err);
+        }
+      }
+      return shadow(this, 'linearization', linearization);
+    },
+    get startXRef() {
+      var stream = this.stream;
+      var startXRef = 0;
+      var linearization = this.linearization;
+      if (linearization) {
+        stream.reset();
+        if (find(stream, 'endobj', 1024)) {
+          startXRef = stream.pos + 6;
+        }
       } else {
-       info('Bad value in document info for "' + key + '"');
+        var step = 1024;
+        var found = false,
+            pos = stream.end;
+        while (!found && pos > 0) {
+          pos -= step - 'startxref'.length;
+          if (pos < 0) {
+            pos = 0;
+          }
+          stream.pos = pos;
+          found = find(stream, 'startxref', step, true);
+        }
+        if (found) {
+          stream.skip(9);
+          var ch;
+          do {
+            ch = stream.getByte();
+          } while (isSpace(ch));
+          var str = '';
+          while (ch >= 0x20 && ch <= 0x39) {
+            str += String.fromCharCode(ch);
+            ch = stream.getByte();
+          }
+          startXRef = parseInt(str, 10);
+          if (isNaN(startXRef)) {
+            startXRef = 0;
+          }
+        }
       }
-     }
-    }
-   }
-   return shadow(this, 'documentInfo', docInfo);
-  },
-  get fingerprint() {
-   var xref = this.xref, hash, fileID = '';
-   var idArray = xref.trailer.get('ID');
-   if (idArray && isArray(idArray) && idArray[0] && isString(idArray[0]) && idArray[0] !== EMPTY_FINGERPRINT) {
-    hash = stringToBytes(idArray[0]);
-   } else {
-    if (this.stream.ensureRange) {
-     this.stream.ensureRange(0, Math.min(FINGERPRINT_FIRST_BYTES, this.stream.end));
+      return shadow(this, 'startXRef', startXRef);
+    },
+    get mainXRefEntriesOffset() {
+      var mainXRefEntriesOffset = 0;
+      var linearization = this.linearization;
+      if (linearization) {
+        mainXRefEntriesOffset = linearization.mainXRefEntriesOffset;
+      }
+      return shadow(this, 'mainXRefEntriesOffset', mainXRefEntriesOffset);
+    },
+    checkHeader: function PDFDocument_checkHeader() {
+      var stream = this.stream;
+      stream.reset();
+      if (find(stream, '%PDF-', 1024)) {
+        stream.moveStart();
+        var MAX_VERSION_LENGTH = 12;
+        var version = '',
+            ch;
+        while ((ch = stream.getByte()) > 0x20) {
+          if (version.length >= MAX_VERSION_LENGTH) {
+            break;
+          }
+          version += String.fromCharCode(ch);
+        }
+        if (!this.pdfFormatVersion) {
+          this.pdfFormatVersion = version.substring(5);
+        }
+        return;
+      }
+    },
+    parseStartXRef: function PDFDocument_parseStartXRef() {
+      var startXRef = this.startXRef;
+      this.xref.setStartXRef(startXRef);
+    },
+    setup: function PDFDocument_setup(recoveryMode) {
+      this.xref.parse(recoveryMode);
+      var self = this;
+      var pageFactory = {
+        createPage: function (pageIndex, dict, ref, fontCache, builtInCMapCache) {
+          return new Page(self.pdfManager, self.xref, pageIndex, dict, ref, fontCache, builtInCMapCache);
+        }
+      };
+      this.catalog = new Catalog(this.pdfManager, this.xref, pageFactory);
+    },
+    get numPages() {
+      var linearization = this.linearization;
+      var num = linearization ? linearization.numPages : this.catalog.numPages;
+      return shadow(this, 'numPages', num);
+    },
+    get documentInfo() {
+      var docInfo = {
+        PDFFormatVersion: this.pdfFormatVersion,
+        IsAcroFormPresent: !!this.acroForm,
+        IsXFAPresent: !!this.xfa
+      };
+      var infoDict;
+      try {
+        infoDict = this.xref.trailer.get('Info');
+      } catch (err) {
+        if (err instanceof MissingDataException) {
+          throw err;
+        }
+        info('The document information dictionary is invalid.');
+      }
+      if (infoDict) {
+        var validEntries = DocumentInfoValidators.entries;
+        for (var key in validEntries) {
+          if (infoDict.has(key)) {
+            var value = infoDict.get(key);
+            if (validEntries[key](value)) {
+              docInfo[key] = typeof value !== 'string' ? value : stringToPDFString(value);
+            } else {
+              info('Bad value in document info for "' + key + '"');
+            }
+          }
+        }
+      }
+      return shadow(this, 'documentInfo', docInfo);
+    },
+    get fingerprint() {
+      var xref = this.xref,
+          hash,
+          fileID = '';
+      var idArray = xref.trailer.get('ID');
+      if (idArray && isArray(idArray) && idArray[0] && isString(idArray[0]) && idArray[0] !== EMPTY_FINGERPRINT) {
+        hash = stringToBytes(idArray[0]);
+      } else {
+        if (this.stream.ensureRange) {
+          this.stream.ensureRange(0, Math.min(FINGERPRINT_FIRST_BYTES, this.stream.end));
+        }
+        hash = calculateMD5(this.stream.bytes.subarray(0, FINGERPRINT_FIRST_BYTES), 0, FINGERPRINT_FIRST_BYTES);
+      }
+      for (var i = 0, n = hash.length; i < n; i++) {
+        var hex = hash[i].toString(16);
+        fileID += hex.length === 1 ? '0' + hex : hex;
+      }
+      return shadow(this, 'fingerprint', fileID);
+    },
+    getPage: function PDFDocument_getPage(pageIndex) {
+      return this.catalog.getPage(pageIndex);
+    },
+    cleanup: function PDFDocument_cleanup() {
+      return this.catalog.cleanup();
     }
-    hash = calculateMD5(this.stream.bytes.subarray(0, FINGERPRINT_FIRST_BYTES), 0, FINGERPRINT_FIRST_BYTES);
-   }
-   for (var i = 0, n = hash.length; i < n; i++) {
-    var hex = hash[i].toString(16);
-    fileID += hex.length === 1 ? '0' + hex : hex;
-   }
-   return shadow(this, 'fingerprint', fileID);
-  },
-  getPage: function PDFDocument_getPage(pageIndex) {
-   return this.catalog.getPage(pageIndex);
-  },
-  cleanup: function PDFDocument_cleanup() {
-   return this.catalog.cleanup();
-  }
- };
- return PDFDocument;
+  };
+  return PDFDocument;
 }();
 exports.Page = Page;
 exports.PDFDocument = PDFDocument;

File diff suppressed because it is too large
+ 1 - 1794
lib/core/encodings.js


+ 2739 - 2837
lib/core/evaluator.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var coreStream = require('./stream.js');
@@ -91,2889 +92,2790 @@ var reverseIfRtl = coreUnicode.reverseIfRtl;
 var getUnicodeForGlyph = coreUnicode.getUnicodeForGlyph;
 var getGlyphsUnicode = coreGlyphList.getGlyphsUnicode;
 var PartialEvaluator = function PartialEvaluatorClosure() {
- var DefaultPartialEvaluatorOptions = {
-  forceDataSchema: false,
-  maxImageSize: -1,
-  disableFontFace: false,
-  disableNativeImageDecoder: false
- };
- function NativeImageDecoder(xref, resources, handler, forceDataSchema) {
-  this.xref = xref;
-  this.resources = resources;
-  this.handler = handler;
-  this.forceDataSchema = forceDataSchema;
- }
- NativeImageDecoder.prototype = {
-  canDecode: function (image) {
-   return image instanceof JpegStream && NativeImageDecoder.isDecodable(image, this.xref, this.resources);
-  },
-  decode: function (image) {
-   var dict = image.dict;
-   var colorSpace = dict.get('ColorSpace', 'CS');
-   colorSpace = ColorSpace.parse(colorSpace, this.xref, this.resources);
-   var numComps = colorSpace.numComps;
-   var decodePromise = this.handler.sendWithPromise('JpegDecode', [
-    image.getIR(this.forceDataSchema),
-    numComps
-   ]);
-   return decodePromise.then(function (message) {
-    var data = message.data;
-    return new Stream(data, 0, data.length, image.dict);
-   });
+  var DefaultPartialEvaluatorOptions = {
+    forceDataSchema: false,
+    maxImageSize: -1,
+    disableFontFace: false,
+    disableNativeImageDecoder: false
+  };
+  function NativeImageDecoder(xref, resources, handler, forceDataSchema) {
+    this.xref = xref;
+    this.resources = resources;
+    this.handler = handler;
+    this.forceDataSchema = forceDataSchema;
   }
- };
- NativeImageDecoder.isSupported = function NativeImageDecoder_isSupported(image, xref, res) {
-  var dict = image.dict;
-  if (dict.has('DecodeParms') || dict.has('DP')) {
-   return false;
-  }
-  var cs = ColorSpace.parse(dict.get('ColorSpace', 'CS'), xref, res);
-  return (cs.name === 'DeviceGray' || cs.name === 'DeviceRGB') && cs.isDefaultDecode(dict.getArray('Decode', 'D'));
- };
- NativeImageDecoder.isDecodable = function NativeImageDecoder_isDecodable(image, xref, res) {
-  var dict = image.dict;
-  if (dict.has('DecodeParms') || dict.has('DP')) {
-   return false;
-  }
-  var cs = ColorSpace.parse(dict.get('ColorSpace', 'CS'), xref, res);
-  return (cs.numComps === 1 || cs.numComps === 3) && cs.isDefaultDecode(dict.getArray('Decode', 'D'));
- };
- function PartialEvaluator(pdfManager, xref, handler, pageIndex, idFactory, fontCache, builtInCMapCache, options) {
-  this.pdfManager = pdfManager;
-  this.xref = xref;
-  this.handler = handler;
-  this.pageIndex = pageIndex;
-  this.idFactory = idFactory;
-  this.fontCache = fontCache;
-  this.builtInCMapCache = builtInCMapCache;
-  this.options = options || DefaultPartialEvaluatorOptions;
-  this.fetchBuiltInCMap = function (name) {
-   var cachedCMap = builtInCMapCache[name];
-   if (cachedCMap) {
-    return Promise.resolve(cachedCMap);
-   }
-   return handler.sendWithPromise('FetchBuiltInCMap', { name: name }).then(function (data) {
-    if (data.compressionType !== CMapCompressionType.NONE) {
-     builtInCMapCache[name] = data;
+  NativeImageDecoder.prototype = {
+    canDecode: function (image) {
+      return image instanceof JpegStream && NativeImageDecoder.isDecodable(image, this.xref, this.resources);
+    },
+    decode: function (image) {
+      var dict = image.dict;
+      var colorSpace = dict.get('ColorSpace', 'CS');
+      colorSpace = ColorSpace.parse(colorSpace, this.xref, this.resources);
+      var numComps = colorSpace.numComps;
+      var decodePromise = this.handler.sendWithPromise('JpegDecode', [image.getIR(this.forceDataSchema), numComps]);
+      return decodePromise.then(function (message) {
+        var data = message.data;
+        return new Stream(data, 0, data.length, image.dict);
+      });
     }
-    return data;
-   });
   };
- }
- var TIME_SLOT_DURATION_MS = 20;
- var CHECK_TIME_EVERY = 100;
- function TimeSlotManager() {
-  this.reset();
- }
- TimeSlotManager.prototype = {
-  check: function TimeSlotManager_check() {
-   if (++this.checked < CHECK_TIME_EVERY) {
-    return false;
-   }
-   this.checked = 0;
-   return this.endTime <= Date.now();
-  },
-  reset: function TimeSlotManager_reset() {
-   this.endTime = Date.now() + TIME_SLOT_DURATION_MS;
-   this.checked = 0;
+  NativeImageDecoder.isSupported = function NativeImageDecoder_isSupported(image, xref, res) {
+    var dict = image.dict;
+    if (dict.has('DecodeParms') || dict.has('DP')) {
+      return false;
+    }
+    var cs = ColorSpace.parse(dict.get('ColorSpace', 'CS'), xref, res);
+    return (cs.name === 'DeviceGray' || cs.name === 'DeviceRGB') && cs.isDefaultDecode(dict.getArray('Decode', 'D'));
+  };
+  NativeImageDecoder.isDecodable = function NativeImageDecoder_isDecodable(image, xref, res) {
+    var dict = image.dict;
+    if (dict.has('DecodeParms') || dict.has('DP')) {
+      return false;
+    }
+    var cs = ColorSpace.parse(dict.get('ColorSpace', 'CS'), xref, res);
+    return (cs.numComps === 1 || cs.numComps === 3) && cs.isDefaultDecode(dict.getArray('Decode', 'D'));
+  };
+  function PartialEvaluator(pdfManager, xref, handler, pageIndex, idFactory, fontCache, builtInCMapCache, options) {
+    this.pdfManager = pdfManager;
+    this.xref = xref;
+    this.handler = handler;
+    this.pageIndex = pageIndex;
+    this.idFactory = idFactory;
+    this.fontCache = fontCache;
+    this.builtInCMapCache = builtInCMapCache;
+    this.options = options || DefaultPartialEvaluatorOptions;
+    this.fetchBuiltInCMap = function (name) {
+      var cachedCMap = builtInCMapCache[name];
+      if (cachedCMap) {
+        return Promise.resolve(cachedCMap);
+      }
+      return handler.sendWithPromise('FetchBuiltInCMap', { name: name }).then(function (data) {
+        if (data.compressionType !== CMapCompressionType.NONE) {
+          builtInCMapCache[name] = data;
+        }
+        return data;
+      });
+    };
   }
- };
- var deferred = Promise.resolve();
- var TILING_PATTERN = 1, SHADING_PATTERN = 2;
- PartialEvaluator.prototype = {
-  hasBlendModes: function PartialEvaluator_hasBlendModes(resources) {
-   if (!isDict(resources)) {
-    return false;
-   }
-   var processed = Object.create(null);
-   if (resources.objId) {
-    processed[resources.objId] = true;
-   }
-   var nodes = [resources], xref = this.xref;
-   while (nodes.length) {
-    var key, i, ii;
-    var node = nodes.shift();
-    var graphicStates = node.get('ExtGState');
-    if (isDict(graphicStates)) {
-     var graphicStatesKeys = graphicStates.getKeys();
-     for (i = 0, ii = graphicStatesKeys.length; i < ii; i++) {
-      key = graphicStatesKeys[i];
-      var graphicState = graphicStates.get(key);
-      var bm = graphicState.get('BM');
-      if (isName(bm) && bm.name !== 'Normal') {
-       return true;
-      }
-     }
-    }
-    var xObjects = node.get('XObject');
-    if (!isDict(xObjects)) {
-     continue;
-    }
-    var xObjectsKeys = xObjects.getKeys();
-    for (i = 0, ii = xObjectsKeys.length; i < ii; i++) {
-     key = xObjectsKeys[i];
-     var xObject = xObjects.getRaw(key);
-     if (isRef(xObject)) {
-      if (processed[xObject.toString()]) {
-       continue;
-      }
-      xObject = xref.fetch(xObject);
-     }
-     if (!isStream(xObject)) {
-      continue;
-     }
-     if (xObject.dict.objId) {
-      if (processed[xObject.dict.objId]) {
-       continue;
-      }
-      processed[xObject.dict.objId] = true;
-     }
-     var xResources = xObject.dict.get('Resources');
-     if (isDict(xResources) && (!xResources.objId || !processed[xResources.objId])) {
-      nodes.push(xResources);
-      if (xResources.objId) {
-       processed[xResources.objId] = true;
-      }
-     }
-    }
-   }
-   return false;
-  },
-  buildFormXObject: function PartialEvaluator_buildFormXObject(resources, xobj, smask, operatorList, task, initialState) {
-   var matrix = xobj.dict.getArray('Matrix');
-   var bbox = xobj.dict.getArray('BBox');
-   var group = xobj.dict.get('Group');
-   if (group) {
-    var groupOptions = {
-     matrix: matrix,
-     bbox: bbox,
-     smask: smask,
-     isolated: false,
-     knockout: false
-    };
-    var groupSubtype = group.get('S');
-    var colorSpace;
-    if (isName(groupSubtype, 'Transparency')) {
-     groupOptions.isolated = group.get('I') || false;
-     groupOptions.knockout = group.get('K') || false;
-     colorSpace = group.has('CS') ? ColorSpace.parse(group.get('CS'), this.xref, resources) : null;
-    }
-    if (smask && smask.backdrop) {
-     colorSpace = colorSpace || ColorSpace.singletons.rgb;
-     smask.backdrop = colorSpace.getRgb(smask.backdrop, 0);
-    }
-    operatorList.addOp(OPS.beginGroup, [groupOptions]);
-   }
-   operatorList.addOp(OPS.paintFormXObjectBegin, [
-    matrix,
-    bbox
-   ]);
-   return this.getOperatorList(xobj, task, xobj.dict.get('Resources') || resources, operatorList, initialState).then(function () {
-    operatorList.addOp(OPS.paintFormXObjectEnd, []);
-    if (group) {
-     operatorList.addOp(OPS.endGroup, [groupOptions]);
-    }
-   });
-  },
-  buildPaintImageXObject: function PartialEvaluator_buildPaintImageXObject(resources, image, inline, operatorList, cacheKey, imageCache) {
-   var self = this;
-   var dict = image.dict;
-   var w = dict.get('Width', 'W');
-   var h = dict.get('Height', 'H');
-   if (!(w && isNum(w)) || !(h && isNum(h))) {
-    warn('Image dimensions are missing, or not numbers.');
-    return;
-   }
-   var maxImageSize = this.options.maxImageSize;
-   if (maxImageSize !== -1 && w * h > maxImageSize) {
-    warn('Image exceeded maximum allowed size and was removed.');
-    return;
-   }
-   var imageMask = dict.get('ImageMask', 'IM') || false;
-   var imgData, args;
-   if (imageMask) {
-    var width = dict.get('Width', 'W');
-    var height = dict.get('Height', 'H');
-    var bitStrideLength = width + 7 >> 3;
-    var imgArray = image.getBytes(bitStrideLength * height);
-    var decode = dict.getArray('Decode', 'D');
-    var inverseDecode = !!decode && decode[0] > 0;
-    imgData = PDFImage.createMask(imgArray, width, height, image instanceof DecodeStream, inverseDecode);
-    imgData.cached = true;
-    args = [imgData];
-    operatorList.addOp(OPS.paintImageMaskXObject, args);
-    if (cacheKey) {
-     imageCache[cacheKey] = {
-      fn: OPS.paintImageMaskXObject,
-      args: args
-     };
-    }
-    return;
-   }
-   var softMask = dict.get('SMask', 'SM') || false;
-   var mask = dict.get('Mask') || false;
-   var SMALL_IMAGE_DIMENSIONS = 200;
-   if (inline && !softMask && !mask && !(image instanceof JpegStream) && w + h < SMALL_IMAGE_DIMENSIONS) {
-    var imageObj = new PDFImage(this.xref, resources, image, inline, null, null);
-    imgData = imageObj.createImageData(true);
-    operatorList.addOp(OPS.paintInlineImageXObject, [imgData]);
-    return;
-   }
-   var useNativeImageDecoder = !this.options.disableNativeImageDecoder;
-   var objId = 'img_' + this.idFactory.createObjId();
-   operatorList.addDependency(objId);
-   args = [
-    objId,
-    w,
-    h
-   ];
-   if (useNativeImageDecoder && !softMask && !mask && image instanceof JpegStream && NativeImageDecoder.isSupported(image, this.xref, resources)) {
-    operatorList.addOp(OPS.paintJpegXObject, args);
-    this.handler.send('obj', [
-     objId,
-     this.pageIndex,
-     'JpegStream',
-     image.getIR(this.options.forceDataSchema)
-    ]);
-    return;
-   }
-   var nativeImageDecoder = null;
-   if (useNativeImageDecoder && (image instanceof JpegStream || mask instanceof JpegStream || softMask instanceof JpegStream)) {
-    nativeImageDecoder = new NativeImageDecoder(self.xref, resources, self.handler, self.options.forceDataSchema);
-   }
-   PDFImage.buildImage(self.handler, self.xref, resources, image, inline, nativeImageDecoder).then(function (imageObj) {
-    var imgData = imageObj.createImageData(false);
-    self.handler.send('obj', [
-     objId,
-     self.pageIndex,
-     'Image',
-     imgData
-    ], [imgData.data.buffer]);
-   }).then(undefined, function (reason) {
-    warn('Unable to decode image: ' + reason);
-    self.handler.send('obj', [
-     objId,
-     self.pageIndex,
-     'Image',
-     null
-    ]);
-   });
-   operatorList.addOp(OPS.paintImageXObject, args);
-   if (cacheKey) {
-    imageCache[cacheKey] = {
-     fn: OPS.paintImageXObject,
-     args: args
-    };
-   }
-  },
-  handleSMask: function PartialEvaluator_handleSmask(smask, resources, operatorList, task, stateManager) {
-   var smaskContent = smask.get('G');
-   var smaskOptions = {
-    subtype: smask.get('S').name,
-    backdrop: smask.get('BC')
-   };
-   var transferObj = smask.get('TR');
-   if (isPDFFunction(transferObj)) {
-    var transferFn = PDFFunction.parse(this.xref, transferObj);
-    var transferMap = new Uint8Array(256);
-    var tmp = new Float32Array(1);
-    for (var i = 0; i < 256; i++) {
-     tmp[0] = i / 255;
-     transferFn(tmp, 0, tmp, 0);
-     transferMap[i] = tmp[0] * 255 | 0;
-    }
-    smaskOptions.transferMap = transferMap;
-   }
-   return this.buildFormXObject(resources, smaskContent, smaskOptions, operatorList, task, stateManager.state.clone());
-  },
-  handleTilingType: function PartialEvaluator_handleTilingType(fn, args, resources, pattern, patternDict, operatorList, task) {
-   var tilingOpList = new OperatorList();
-   var resourcesArray = [
-    patternDict.get('Resources'),
-    resources
-   ];
-   var patternResources = Dict.merge(this.xref, resourcesArray);
-   return this.getOperatorList(pattern, task, patternResources, tilingOpList).then(function () {
-    operatorList.addDependencies(tilingOpList.dependencies);
-    operatorList.addOp(fn, getTilingPatternIR({
-     fnArray: tilingOpList.fnArray,
-     argsArray: tilingOpList.argsArray
-    }, patternDict, args));
-   });
-  },
-  handleSetFont: function PartialEvaluator_handleSetFont(resources, fontArgs, fontRef, operatorList, task, state) {
-   var fontName;
-   if (fontArgs) {
-    fontArgs = fontArgs.slice();
-    fontName = fontArgs[0].name;
-   }
-   var self = this;
-   return this.loadFont(fontName, fontRef, resources).then(function (translated) {
-    if (!translated.font.isType3Font) {
-     return translated;
-    }
-    return translated.loadType3Data(self, resources, operatorList, task).then(function () {
-     return translated;
-    }, function (reason) {
-     self.handler.send('UnsupportedFeature', { featureId: UNSUPPORTED_FEATURES.font });
-     return new TranslatedFont('g_font_error', new ErrorFont('Type3 font load error: ' + reason), translated.font);
-    });
-   }).then(function (translated) {
-    state.font = translated.font;
-    translated.send(self.handler);
-    return translated.loadedName;
-   });
-  },
-  handleText: function PartialEvaluator_handleText(chars, state) {
-   var font = state.font;
-   var glyphs = font.charsToGlyphs(chars);
-   var isAddToPathSet = !!(state.textRenderingMode & TextRenderingMode.ADD_TO_PATH_FLAG);
-   if (font.data && (isAddToPathSet || this.options.disableFontFace)) {
-    var buildPath = function (fontChar) {
-     if (!font.renderer.hasBuiltPath(fontChar)) {
-      var path = font.renderer.getPathJs(fontChar);
-      this.handler.send('commonobj', [
-       font.loadedName + '_path_' + fontChar,
-       'FontPath',
-       path
-      ]);
-     }
-    }.bind(this);
-    for (var i = 0, ii = glyphs.length; i < ii; i++) {
-     var glyph = glyphs[i];
-     buildPath(glyph.fontChar);
-     var accent = glyph.accent;
-     if (accent && accent.fontChar) {
-      buildPath(accent.fontChar);
-     }
-    }
-   }
-   return glyphs;
-  },
-  setGState: function PartialEvaluator_setGState(resources, gState, operatorList, task, stateManager) {
-   var gStateObj = [];
-   var gStateKeys = gState.getKeys();
-   var self = this;
-   var promise = Promise.resolve();
-   for (var i = 0, ii = gStateKeys.length; i < ii; i++) {
-    var key = gStateKeys[i];
-    var value = gState.get(key);
-    switch (key) {
-    case 'Type':
-     break;
-    case 'LW':
-    case 'LC':
-    case 'LJ':
-    case 'ML':
-    case 'D':
-    case 'RI':
-    case 'FL':
-    case 'CA':
-    case 'ca':
-     gStateObj.push([
-      key,
-      value
-     ]);
-     break;
-    case 'Font':
-     promise = promise.then(function () {
-      return self.handleSetFont(resources, null, value[0], operatorList, task, stateManager.state).then(function (loadedName) {
-       operatorList.addDependency(loadedName);
-       gStateObj.push([
-        key,
-        [
-         loadedName,
-         value[1]
-        ]
-       ]);
+  var TIME_SLOT_DURATION_MS = 20;
+  var CHECK_TIME_EVERY = 100;
+  function TimeSlotManager() {
+    this.reset();
+  }
+  TimeSlotManager.prototype = {
+    check: function TimeSlotManager_check() {
+      if (++this.checked < CHECK_TIME_EVERY) {
+        return false;
+      }
+      this.checked = 0;
+      return this.endTime <= Date.now();
+    },
+    reset: function TimeSlotManager_reset() {
+      this.endTime = Date.now() + TIME_SLOT_DURATION_MS;
+      this.checked = 0;
+    }
+  };
+  var deferred = Promise.resolve();
+  var TILING_PATTERN = 1,
+      SHADING_PATTERN = 2;
+  PartialEvaluator.prototype = {
+    hasBlendModes: function PartialEvaluator_hasBlendModes(resources) {
+      if (!isDict(resources)) {
+        return false;
+      }
+      var processed = Object.create(null);
+      if (resources.objId) {
+        processed[resources.objId] = true;
+      }
+      var nodes = [resources],
+          xref = this.xref;
+      while (nodes.length) {
+        var key, i, ii;
+        var node = nodes.shift();
+        var graphicStates = node.get('ExtGState');
+        if (isDict(graphicStates)) {
+          var graphicStatesKeys = graphicStates.getKeys();
+          for (i = 0, ii = graphicStatesKeys.length; i < ii; i++) {
+            key = graphicStatesKeys[i];
+            var graphicState = graphicStates.get(key);
+            var bm = graphicState.get('BM');
+            if (isName(bm) && bm.name !== 'Normal') {
+              return true;
+            }
+          }
+        }
+        var xObjects = node.get('XObject');
+        if (!isDict(xObjects)) {
+          continue;
+        }
+        var xObjectsKeys = xObjects.getKeys();
+        for (i = 0, ii = xObjectsKeys.length; i < ii; i++) {
+          key = xObjectsKeys[i];
+          var xObject = xObjects.getRaw(key);
+          if (isRef(xObject)) {
+            if (processed[xObject.toString()]) {
+              continue;
+            }
+            xObject = xref.fetch(xObject);
+          }
+          if (!isStream(xObject)) {
+            continue;
+          }
+          if (xObject.dict.objId) {
+            if (processed[xObject.dict.objId]) {
+              continue;
+            }
+            processed[xObject.dict.objId] = true;
+          }
+          var xResources = xObject.dict.get('Resources');
+          if (isDict(xResources) && (!xResources.objId || !processed[xResources.objId])) {
+            nodes.push(xResources);
+            if (xResources.objId) {
+              processed[xResources.objId] = true;
+            }
+          }
+        }
+      }
+      return false;
+    },
+    buildFormXObject: function PartialEvaluator_buildFormXObject(resources, xobj, smask, operatorList, task, initialState) {
+      var matrix = xobj.dict.getArray('Matrix');
+      var bbox = xobj.dict.getArray('BBox');
+      var group = xobj.dict.get('Group');
+      if (group) {
+        var groupOptions = {
+          matrix: matrix,
+          bbox: bbox,
+          smask: smask,
+          isolated: false,
+          knockout: false
+        };
+        var groupSubtype = group.get('S');
+        var colorSpace;
+        if (isName(groupSubtype, 'Transparency')) {
+          groupOptions.isolated = group.get('I') || false;
+          groupOptions.knockout = group.get('K') || false;
+          colorSpace = group.has('CS') ? ColorSpace.parse(group.get('CS'), this.xref, resources) : null;
+        }
+        if (smask && smask.backdrop) {
+          colorSpace = colorSpace || ColorSpace.singletons.rgb;
+          smask.backdrop = colorSpace.getRgb(smask.backdrop, 0);
+        }
+        operatorList.addOp(OPS.beginGroup, [groupOptions]);
+      }
+      operatorList.addOp(OPS.paintFormXObjectBegin, [matrix, bbox]);
+      return this.getOperatorList(xobj, task, xobj.dict.get('Resources') || resources, operatorList, initialState).then(function () {
+        operatorList.addOp(OPS.paintFormXObjectEnd, []);
+        if (group) {
+          operatorList.addOp(OPS.endGroup, [groupOptions]);
+        }
       });
-     });
-     break;
-    case 'BM':
-     gStateObj.push([
-      key,
-      value
-     ]);
-     break;
-    case 'SMask':
-     if (isName(value, 'None')) {
-      gStateObj.push([
-       key,
-       false
-      ]);
-      break;
-     }
-     if (isDict(value)) {
-      promise = promise.then(function (dict) {
-       return self.handleSMask(dict, resources, operatorList, task, stateManager);
-      }.bind(this, value));
-      gStateObj.push([
-       key,
-       true
-      ]);
-     } else {
-      warn('Unsupported SMask type');
-     }
-     break;
-    case 'OP':
-    case 'op':
-    case 'OPM':
-    case 'BG':
-    case 'BG2':
-    case 'UCR':
-    case 'UCR2':
-    case 'TR':
-    case 'TR2':
-    case 'HT':
-    case 'SM':
-    case 'SA':
-    case 'AIS':
-    case 'TK':
-     info('graphic state operator ' + key);
-     break;
-    default:
-     info('Unknown graphic state operator ' + key);
-     break;
-    }
-   }
-   return promise.then(function () {
-    if (gStateObj.length > 0) {
-     operatorList.addOp(OPS.setGState, [gStateObj]);
-    }
-   });
-  },
-  loadFont: function PartialEvaluator_loadFont(fontName, font, resources) {
-   function errorFont() {
-    return Promise.resolve(new TranslatedFont('g_font_error', new ErrorFont('Font ' + fontName + ' is not available'), font));
-   }
-   var fontRef, xref = this.xref;
-   if (font) {
-    assert(isRef(font));
-    fontRef = font;
-   } else {
-    var fontRes = resources.get('Font');
-    if (fontRes) {
-     fontRef = fontRes.getRaw(fontName);
-    } else {
-     warn('fontRes not available');
-     return errorFont();
-    }
-   }
-   if (!fontRef) {
-    warn('fontRef not available');
-    return errorFont();
-   }
-   if (this.fontCache.has(fontRef)) {
-    return this.fontCache.get(fontRef);
-   }
-   font = xref.fetchIfRef(fontRef);
-   if (!isDict(font)) {
-    return errorFont();
-   }
-   if (font.translated) {
-    return font.translated;
-   }
-   var fontCapability = createPromiseCapability();
-   var preEvaluatedFont = this.preEvaluateFont(font);
-   var descriptor = preEvaluatedFont.descriptor;
-   var fontRefIsRef = isRef(fontRef), fontID;
-   if (fontRefIsRef) {
-    fontID = fontRef.toString();
-   }
-   if (isDict(descriptor)) {
-    if (!descriptor.fontAliases) {
-     descriptor.fontAliases = Object.create(null);
-    }
-    var fontAliases = descriptor.fontAliases;
-    var hash = preEvaluatedFont.hash;
-    if (fontAliases[hash]) {
-     var aliasFontRef = fontAliases[hash].aliasRef;
-     if (fontRefIsRef && aliasFontRef && this.fontCache.has(aliasFontRef)) {
-      this.fontCache.putAlias(fontRef, aliasFontRef);
-      return this.fontCache.get(fontRef);
-     }
-    } else {
-     fontAliases[hash] = { fontID: Font.getFontID() };
-    }
-    if (fontRefIsRef) {
-     fontAliases[hash].aliasRef = fontRef;
-    }
-    fontID = fontAliases[hash].fontID;
-   }
-   if (fontRefIsRef) {
-    this.fontCache.put(fontRef, fontCapability.promise);
-   } else {
-    if (!fontID) {
-     fontID = this.idFactory.createObjId();
-    }
-    this.fontCache.put('id_' + fontID, fontCapability.promise);
-   }
-   assert(fontID, 'The "fontID" must be defined.');
-   font.loadedName = 'g_' + this.pdfManager.docId + '_f' + fontID;
-   font.translated = fontCapability.promise;
-   var translatedPromise;
-   try {
-    translatedPromise = this.translateFont(preEvaluatedFont);
-   } catch (e) {
-    translatedPromise = Promise.reject(e);
-   }
-   var self = this;
-   translatedPromise.then(function (translatedFont) {
-    if (translatedFont.fontType !== undefined) {
-     var xrefFontStats = xref.stats.fontTypes;
-     xrefFontStats[translatedFont.fontType] = true;
-    }
-    fontCapability.resolve(new TranslatedFont(font.loadedName, translatedFont, font));
-   }, function (reason) {
-    self.handler.send('UnsupportedFeature', { featureId: UNSUPPORTED_FEATURES.font });
-    try {
-     var descriptor = preEvaluatedFont.descriptor;
-     var fontFile3 = descriptor && descriptor.get('FontFile3');
-     var subtype = fontFile3 && fontFile3.get('Subtype');
-     var fontType = getFontType(preEvaluatedFont.type, subtype && subtype.name);
-     var xrefFontStats = xref.stats.fontTypes;
-     xrefFontStats[fontType] = true;
-    } catch (ex) {
-    }
-    fontCapability.resolve(new TranslatedFont(font.loadedName, new ErrorFont(reason instanceof Error ? reason.message : reason), font));
-   });
-   return fontCapability.promise;
-  },
-  buildPath: function PartialEvaluator_buildPath(operatorList, fn, args) {
-   var lastIndex = operatorList.length - 1;
-   if (!args) {
-    args = [];
-   }
-   if (lastIndex < 0 || operatorList.fnArray[lastIndex] !== OPS.constructPath) {
-    operatorList.addOp(OPS.constructPath, [
-     [fn],
-     args
-    ]);
-   } else {
-    var opArgs = operatorList.argsArray[lastIndex];
-    opArgs[0].push(fn);
-    Array.prototype.push.apply(opArgs[1], args);
-   }
-  },
-  handleColorN: function PartialEvaluator_handleColorN(operatorList, fn, args, cs, patterns, resources, task) {
-   var patternName = args[args.length - 1];
-   var pattern;
-   if (isName(patternName) && (pattern = patterns.get(patternName.name))) {
-    var dict = isStream(pattern) ? pattern.dict : pattern;
-    var typeNum = dict.get('PatternType');
-    if (typeNum === TILING_PATTERN) {
-     var color = cs.base ? cs.base.getRgb(args, 0) : null;
-     return this.handleTilingType(fn, color, resources, pattern, dict, operatorList, task);
-    } else if (typeNum === SHADING_PATTERN) {
-     var shading = dict.get('Shading');
-     var matrix = dict.getArray('Matrix');
-     pattern = Pattern.parseShading(shading, matrix, this.xref, resources, this.handler);
-     operatorList.addOp(fn, pattern.getIR());
-     return Promise.resolve();
-    }
-    return Promise.reject('Unknown PatternType: ' + typeNum);
-   }
-   operatorList.addOp(fn, args);
-   return Promise.resolve();
-  },
-  getOperatorList: function PartialEvaluator_getOperatorList(stream, task, resources, operatorList, initialState) {
-   var self = this;
-   var xref = this.xref;
-   var imageCache = Object.create(null);
-   assert(operatorList);
-   resources = resources || Dict.empty;
-   var xobjs = resources.get('XObject') || Dict.empty;
-   var patterns = resources.get('Pattern') || Dict.empty;
-   var stateManager = new StateManager(initialState || new EvalState());
-   var preprocessor = new EvaluatorPreprocessor(stream, xref, stateManager);
-   var timeSlotManager = new TimeSlotManager();
-   return new Promise(function promiseBody(resolve, reject) {
-    var next = function (promise) {
-     promise.then(function () {
-      try {
-       promiseBody(resolve, reject);
-      } catch (ex) {
-       reject(ex);
-      }
-     }, reject);
-    };
-    task.ensureNotTerminated();
-    timeSlotManager.reset();
-    var stop, operation = {}, i, ii, cs;
-    while (!(stop = timeSlotManager.check())) {
-     operation.args = null;
-     if (!preprocessor.read(operation)) {
-      break;
-     }
-     var args = operation.args;
-     var fn = operation.fn;
-     switch (fn | 0) {
-     case OPS.paintXObject:
-      if (args[0].code) {
-       break;
-      }
-      var name = args[0].name;
-      if (!name) {
-       warn('XObject must be referred to by name.');
-       continue;
-      }
-      if (imageCache[name] !== undefined) {
-       operatorList.addOp(imageCache[name].fn, imageCache[name].args);
-       args = null;
-       continue;
-      }
-      var xobj = xobjs.get(name);
-      if (xobj) {
-       assert(isStream(xobj), 'XObject should be a stream');
-       var type = xobj.dict.get('Subtype');
-       assert(isName(type), 'XObject should have a Name subtype');
-       if (type.name === 'Form') {
-        stateManager.save();
-        next(self.buildFormXObject(resources, xobj, null, operatorList, task, stateManager.state.clone()).then(function () {
-         stateManager.restore();
-        }));
+    },
+    buildPaintImageXObject: function PartialEvaluator_buildPaintImageXObject(resources, image, inline, operatorList, cacheKey, imageCache) {
+      var self = this;
+      var dict = image.dict;
+      var w = dict.get('Width', 'W');
+      var h = dict.get('Height', 'H');
+      if (!(w && isNum(w)) || !(h && isNum(h))) {
+        warn('Image dimensions are missing, or not numbers.');
         return;
-       } else if (type.name === 'Image') {
-        self.buildPaintImageXObject(resources, xobj, false, operatorList, name, imageCache);
-        args = null;
-        continue;
-       } else if (type.name === 'PS') {
-        info('Ignored XObject subtype PS');
-        continue;
-       } else {
-        error('Unhandled XObject subtype ' + type.name);
-       }
       }
-      break;
-     case OPS.setFont:
-      var fontSize = args[1];
-      next(self.handleSetFont(resources, args, null, operatorList, task, stateManager.state).then(function (loadedName) {
-       operatorList.addDependency(loadedName);
-       operatorList.addOp(OPS.setFont, [
-        loadedName,
-        fontSize
-       ]);
-      }));
-      return;
-     case OPS.endInlineImage:
-      var cacheKey = args[0].cacheKey;
+      var maxImageSize = this.options.maxImageSize;
+      if (maxImageSize !== -1 && w * h > maxImageSize) {
+        warn('Image exceeded maximum allowed size and was removed.');
+        return;
+      }
+      var imageMask = dict.get('ImageMask', 'IM') || false;
+      var imgData, args;
+      if (imageMask) {
+        var width = dict.get('Width', 'W');
+        var height = dict.get('Height', 'H');
+        var bitStrideLength = width + 7 >> 3;
+        var imgArray = image.getBytes(bitStrideLength * height);
+        var decode = dict.getArray('Decode', 'D');
+        var inverseDecode = !!decode && decode[0] > 0;
+        imgData = PDFImage.createMask(imgArray, width, height, image instanceof DecodeStream, inverseDecode);
+        imgData.cached = true;
+        args = [imgData];
+        operatorList.addOp(OPS.paintImageMaskXObject, args);
+        if (cacheKey) {
+          imageCache[cacheKey] = {
+            fn: OPS.paintImageMaskXObject,
+            args: args
+          };
+        }
+        return;
+      }
+      var softMask = dict.get('SMask', 'SM') || false;
+      var mask = dict.get('Mask') || false;
+      var SMALL_IMAGE_DIMENSIONS = 200;
+      if (inline && !softMask && !mask && !(image instanceof JpegStream) && w + h < SMALL_IMAGE_DIMENSIONS) {
+        var imageObj = new PDFImage(this.xref, resources, image, inline, null, null);
+        imgData = imageObj.createImageData(true);
+        operatorList.addOp(OPS.paintInlineImageXObject, [imgData]);
+        return;
+      }
+      var useNativeImageDecoder = !this.options.disableNativeImageDecoder;
+      var objId = 'img_' + this.idFactory.createObjId();
+      operatorList.addDependency(objId);
+      args = [objId, w, h];
+      if (useNativeImageDecoder && !softMask && !mask && image instanceof JpegStream && NativeImageDecoder.isSupported(image, this.xref, resources)) {
+        operatorList.addOp(OPS.paintJpegXObject, args);
+        this.handler.send('obj', [objId, this.pageIndex, 'JpegStream', image.getIR(this.options.forceDataSchema)]);
+        return;
+      }
+      var nativeImageDecoder = null;
+      if (useNativeImageDecoder && (image instanceof JpegStream || mask instanceof JpegStream || softMask instanceof JpegStream)) {
+        nativeImageDecoder = new NativeImageDecoder(self.xref, resources, self.handler, self.options.forceDataSchema);
+      }
+      PDFImage.buildImage(self.handler, self.xref, resources, image, inline, nativeImageDecoder).then(function (imageObj) {
+        var imgData = imageObj.createImageData(false);
+        self.handler.send('obj', [objId, self.pageIndex, 'Image', imgData], [imgData.data.buffer]);
+      }).then(undefined, function (reason) {
+        warn('Unable to decode image: ' + reason);
+        self.handler.send('obj', [objId, self.pageIndex, 'Image', null]);
+      });
+      operatorList.addOp(OPS.paintImageXObject, args);
       if (cacheKey) {
-       var cacheEntry = imageCache[cacheKey];
-       if (cacheEntry !== undefined) {
-        operatorList.addOp(cacheEntry.fn, cacheEntry.args);
-        args = null;
-        continue;
-       }
+        imageCache[cacheKey] = {
+          fn: OPS.paintImageXObject,
+          args: args
+        };
       }
-      self.buildPaintImageXObject(resources, args[0], true, operatorList, cacheKey, imageCache);
-      args = null;
-      continue;
-     case OPS.showText:
-      args[0] = self.handleText(args[0], stateManager.state);
-      break;
-     case OPS.showSpacedText:
-      var arr = args[0];
-      var combinedGlyphs = [];
-      var arrLength = arr.length;
-      var state = stateManager.state;
-      for (i = 0; i < arrLength; ++i) {
-       var arrItem = arr[i];
-       if (isString(arrItem)) {
-        Array.prototype.push.apply(combinedGlyphs, self.handleText(arrItem, state));
-       } else if (isNum(arrItem)) {
-        combinedGlyphs.push(arrItem);
-       }
-      }
-      args[0] = combinedGlyphs;
-      fn = OPS.showText;
-      break;
-     case OPS.nextLineShowText:
-      operatorList.addOp(OPS.nextLine);
-      args[0] = self.handleText(args[0], stateManager.state);
-      fn = OPS.showText;
-      break;
-     case OPS.nextLineSetSpacingShowText:
-      operatorList.addOp(OPS.nextLine);
-      operatorList.addOp(OPS.setWordSpacing, [args.shift()]);
-      operatorList.addOp(OPS.setCharSpacing, [args.shift()]);
-      args[0] = self.handleText(args[0], stateManager.state);
-      fn = OPS.showText;
-      break;
-     case OPS.setTextRenderingMode:
-      stateManager.state.textRenderingMode = args[0];
-      break;
-     case OPS.setFillColorSpace:
-      stateManager.state.fillColorSpace = ColorSpace.parse(args[0], xref, resources);
-      continue;
-     case OPS.setStrokeColorSpace:
-      stateManager.state.strokeColorSpace = ColorSpace.parse(args[0], xref, resources);
-      continue;
-     case OPS.setFillColor:
-      cs = stateManager.state.fillColorSpace;
-      args = cs.getRgb(args, 0);
-      fn = OPS.setFillRGBColor;
-      break;
-     case OPS.setStrokeColor:
-      cs = stateManager.state.strokeColorSpace;
-      args = cs.getRgb(args, 0);
-      fn = OPS.setStrokeRGBColor;
-      break;
-     case OPS.setFillGray:
-      stateManager.state.fillColorSpace = ColorSpace.singletons.gray;
-      args = ColorSpace.singletons.gray.getRgb(args, 0);
-      fn = OPS.setFillRGBColor;
-      break;
-     case OPS.setStrokeGray:
-      stateManager.state.strokeColorSpace = ColorSpace.singletons.gray;
-      args = ColorSpace.singletons.gray.getRgb(args, 0);
-      fn = OPS.setStrokeRGBColor;
-      break;
-     case OPS.setFillCMYKColor:
-      stateManager.state.fillColorSpace = ColorSpace.singletons.cmyk;
-      args = ColorSpace.singletons.cmyk.getRgb(args, 0);
-      fn = OPS.setFillRGBColor;
-      break;
-     case OPS.setStrokeCMYKColor:
-      stateManager.state.strokeColorSpace = ColorSpace.singletons.cmyk;
-      args = ColorSpace.singletons.cmyk.getRgb(args, 0);
-      fn = OPS.setStrokeRGBColor;
-      break;
-     case OPS.setFillRGBColor:
-      stateManager.state.fillColorSpace = ColorSpace.singletons.rgb;
-      args = ColorSpace.singletons.rgb.getRgb(args, 0);
-      break;
-     case OPS.setStrokeRGBColor:
-      stateManager.state.strokeColorSpace = ColorSpace.singletons.rgb;
-      args = ColorSpace.singletons.rgb.getRgb(args, 0);
-      break;
-     case OPS.setFillColorN:
-      cs = stateManager.state.fillColorSpace;
-      if (cs.name === 'Pattern') {
-       next(self.handleColorN(operatorList, OPS.setFillColorN, args, cs, patterns, resources, task));
-       return;
-      }
-      args = cs.getRgb(args, 0);
-      fn = OPS.setFillRGBColor;
-      break;
-     case OPS.setStrokeColorN:
-      cs = stateManager.state.strokeColorSpace;
-      if (cs.name === 'Pattern') {
-       next(self.handleColorN(operatorList, OPS.setStrokeColorN, args, cs, patterns, resources, task));
-       return;
-      }
-      args = cs.getRgb(args, 0);
-      fn = OPS.setStrokeRGBColor;
-      break;
-     case OPS.shadingFill:
-      var shadingRes = resources.get('Shading');
-      assert(shadingRes, 'No shading resource found');
-      var shading = shadingRes.get(args[0].name);
-      assert(shading, 'No shading object found');
-      var shadingFill = Pattern.parseShading(shading, null, xref, resources, self.handler);
-      var patternIR = shadingFill.getIR();
-      args = [patternIR];
-      fn = OPS.shadingFill;
-      break;
-     case OPS.setGState:
-      var dictName = args[0];
-      var extGState = resources.get('ExtGState');
-      if (!isDict(extGState) || !extGState.has(dictName.name)) {
-       break;
-      }
-      var gState = extGState.get(dictName.name);
-      next(self.setGState(resources, gState, operatorList, task, stateManager));
-      return;
-     case OPS.moveTo:
-     case OPS.lineTo:
-     case OPS.curveTo:
-     case OPS.curveTo2:
-     case OPS.curveTo3:
-     case OPS.closePath:
-      self.buildPath(operatorList, fn, args);
-      continue;
-     case OPS.rectangle:
-      self.buildPath(operatorList, fn, args);
-      continue;
-     case OPS.markPoint:
-     case OPS.markPointProps:
-     case OPS.beginMarkedContent:
-     case OPS.beginMarkedContentProps:
-     case OPS.endMarkedContent:
-     case OPS.beginCompat:
-     case OPS.endCompat:
-      continue;
-     default:
-      if (args !== null) {
-       for (i = 0, ii = args.length; i < ii; i++) {
-        if (args[i] instanceof Dict) {
-         break;
-        }
-       }
-       if (i < ii) {
-        warn('getOperatorList - ignoring operator: ' + fn);
-        continue;
-       }
+    },
+    handleSMask: function PartialEvaluator_handleSmask(smask, resources, operatorList, task, stateManager) {
+      var smaskContent = smask.get('G');
+      var smaskOptions = {
+        subtype: smask.get('S').name,
+        backdrop: smask.get('BC')
+      };
+      var transferObj = smask.get('TR');
+      if (isPDFFunction(transferObj)) {
+        var transferFn = PDFFunction.parse(this.xref, transferObj);
+        var transferMap = new Uint8Array(256);
+        var tmp = new Float32Array(1);
+        for (var i = 0; i < 256; i++) {
+          tmp[0] = i / 255;
+          transferFn(tmp, 0, tmp, 0);
+          transferMap[i] = tmp[0] * 255 | 0;
+        }
+        smaskOptions.transferMap = transferMap;
       }
-     }
-     operatorList.addOp(fn, args);
-    }
-    if (stop) {
-     next(deferred);
-     return;
-    }
-    for (i = 0, ii = preprocessor.savedStatesDepth; i < ii; i++) {
-     operatorList.addOp(OPS.restore, []);
-    }
-    resolve();
-   });
-  },
-  getTextContent: function PartialEvaluator_getTextContent(stream, task, resources, stateManager, normalizeWhitespace, combineTextItems) {
-   stateManager = stateManager || new StateManager(new TextState());
-   var WhitespaceRegexp = /\s/g;
-   var textContent = {
-    items: [],
-    styles: Object.create(null)
-   };
-   var textContentItem = {
-    initialized: false,
-    str: [],
-    width: 0,
-    height: 0,
-    vertical: false,
-    lastAdvanceWidth: 0,
-    lastAdvanceHeight: 0,
-    textAdvanceScale: 0,
-    spaceWidth: 0,
-    fakeSpaceMin: Infinity,
-    fakeMultiSpaceMin: Infinity,
-    fakeMultiSpaceMax: -0,
-    textRunBreakAllowed: false,
-    transform: null,
-    fontName: null
-   };
-   var SPACE_FACTOR = 0.3;
-   var MULTI_SPACE_FACTOR = 1.5;
-   var MULTI_SPACE_FACTOR_MAX = 4;
-   var self = this;
-   var xref = this.xref;
-   resources = xref.fetchIfRef(resources) || Dict.empty;
-   var xobjs = null;
-   var xobjsCache = Object.create(null);
-   var preprocessor = new EvaluatorPreprocessor(stream, xref, stateManager);
-   var textState;
-   function ensureTextContentItem() {
-    if (textContentItem.initialized) {
-     return textContentItem;
-    }
-    var font = textState.font;
-    if (!(font.loadedName in textContent.styles)) {
-     textContent.styles[font.loadedName] = {
-      fontFamily: font.fallbackName,
-      ascent: font.ascent,
-      descent: font.descent,
-      vertical: font.vertical
-     };
-    }
-    textContentItem.fontName = font.loadedName;
-    var tsm = [
-     textState.fontSize * textState.textHScale,
-     0,
-     0,
-     textState.fontSize,
-     0,
-     textState.textRise
-    ];
-    if (font.isType3Font && textState.fontMatrix !== FONT_IDENTITY_MATRIX && textState.fontSize === 1) {
-     var glyphHeight = font.bbox[3] - font.bbox[1];
-     if (glyphHeight > 0) {
-      glyphHeight = glyphHeight * textState.fontMatrix[3];
-      tsm[3] *= glyphHeight;
-     }
-    }
-    var trm = Util.transform(textState.ctm, Util.transform(textState.textMatrix, tsm));
-    textContentItem.transform = trm;
-    if (!font.vertical) {
-     textContentItem.width = 0;
-     textContentItem.height = Math.sqrt(trm[2] * trm[2] + trm[3] * trm[3]);
-     textContentItem.vertical = false;
-    } else {
-     textContentItem.width = Math.sqrt(trm[0] * trm[0] + trm[1] * trm[1]);
-     textContentItem.height = 0;
-     textContentItem.vertical = true;
-    }
-    var a = textState.textLineMatrix[0];
-    var b = textState.textLineMatrix[1];
-    var scaleLineX = Math.sqrt(a * a + b * b);
-    a = textState.ctm[0];
-    b = textState.ctm[1];
-    var scaleCtmX = Math.sqrt(a * a + b * b);
-    textContentItem.textAdvanceScale = scaleCtmX * scaleLineX;
-    textContentItem.lastAdvanceWidth = 0;
-    textContentItem.lastAdvanceHeight = 0;
-    var spaceWidth = font.spaceWidth / 1000 * textState.fontSize;
-    if (spaceWidth) {
-     textContentItem.spaceWidth = spaceWidth;
-     textContentItem.fakeSpaceMin = spaceWidth * SPACE_FACTOR;
-     textContentItem.fakeMultiSpaceMin = spaceWidth * MULTI_SPACE_FACTOR;
-     textContentItem.fakeMultiSpaceMax = spaceWidth * MULTI_SPACE_FACTOR_MAX;
-     textContentItem.textRunBreakAllowed = !font.isMonospace;
-    } else {
-     textContentItem.spaceWidth = 0;
-     textContentItem.fakeSpaceMin = Infinity;
-     textContentItem.fakeMultiSpaceMin = Infinity;
-     textContentItem.fakeMultiSpaceMax = 0;
-     textContentItem.textRunBreakAllowed = false;
-    }
-    textContentItem.initialized = true;
-    return textContentItem;
-   }
-   function replaceWhitespace(str) {
-    var i = 0, ii = str.length, code;
-    while (i < ii && (code = str.charCodeAt(i)) >= 0x20 && code <= 0x7F) {
-     i++;
-    }
-    return i < ii ? str.replace(WhitespaceRegexp, ' ') : str;
-   }
-   function runBidiTransform(textChunk) {
-    var str = textChunk.str.join('');
-    var bidiResult = bidi(str, -1, textChunk.vertical);
-    return {
-     str: normalizeWhitespace ? replaceWhitespace(bidiResult.str) : bidiResult.str,
-     dir: bidiResult.dir,
-     width: textChunk.width,
-     height: textChunk.height,
-     transform: textChunk.transform,
-     fontName: textChunk.fontName
-    };
-   }
-   function handleSetFont(fontName, fontRef) {
-    return self.loadFont(fontName, fontRef, resources).then(function (translated) {
-     textState.font = translated.font;
-     textState.fontMatrix = translated.font.fontMatrix || FONT_IDENTITY_MATRIX;
-    });
-   }
-   function buildTextContentItem(chars) {
-    var font = textState.font;
-    var textChunk = ensureTextContentItem();
-    var width = 0;
-    var height = 0;
-    var glyphs = font.charsToGlyphs(chars);
-    for (var i = 0; i < glyphs.length; i++) {
-     var glyph = glyphs[i];
-     var glyphWidth = null;
-     if (font.vertical && glyph.vmetric) {
-      glyphWidth = glyph.vmetric[0];
-     } else {
-      glyphWidth = glyph.width;
-     }
-     var glyphUnicode = glyph.unicode;
-     var NormalizedUnicodes = getNormalizedUnicodes();
-     if (NormalizedUnicodes[glyphUnicode] !== undefined) {
-      glyphUnicode = NormalizedUnicodes[glyphUnicode];
-     }
-     glyphUnicode = reverseIfRtl(glyphUnicode);
-     var charSpacing = textState.charSpacing;
-     if (glyph.isSpace) {
-      var wordSpacing = textState.wordSpacing;
-      charSpacing += wordSpacing;
-      if (wordSpacing > 0) {
-       addFakeSpaces(wordSpacing, textChunk.str);
-      }
-     }
-     var tx = 0;
-     var ty = 0;
-     if (!font.vertical) {
-      var w0 = glyphWidth * textState.fontMatrix[0];
-      tx = (w0 * textState.fontSize + charSpacing) * textState.textHScale;
-      width += tx;
-     } else {
-      var w1 = glyphWidth * textState.fontMatrix[0];
-      ty = w1 * textState.fontSize + charSpacing;
-      height += ty;
-     }
-     textState.translateTextMatrix(tx, ty);
-     textChunk.str.push(glyphUnicode);
-    }
-    if (!font.vertical) {
-     textChunk.lastAdvanceWidth = width;
-     textChunk.width += width;
-    } else {
-     textChunk.lastAdvanceHeight = height;
-     textChunk.height += Math.abs(height);
-    }
-    return textChunk;
-   }
-   function addFakeSpaces(width, strBuf) {
-    if (width < textContentItem.fakeSpaceMin) {
-     return;
-    }
-    if (width < textContentItem.fakeMultiSpaceMin) {
-     strBuf.push(' ');
-     return;
-    }
-    var fakeSpaces = Math.round(width / textContentItem.spaceWidth);
-    while (fakeSpaces-- > 0) {
-     strBuf.push(' ');
-    }
-   }
-   function flushTextContentItem() {
-    if (!textContentItem.initialized) {
-     return;
-    }
-    textContentItem.width *= textContentItem.textAdvanceScale;
-    textContentItem.height *= textContentItem.textAdvanceScale;
-    textContent.items.push(runBidiTransform(textContentItem));
-    textContentItem.initialized = false;
-    textContentItem.str.length = 0;
-   }
-   var timeSlotManager = new TimeSlotManager();
-   return new Promise(function promiseBody(resolve, reject) {
-    var next = function (promise) {
-     promise.then(function () {
+      return this.buildFormXObject(resources, smaskContent, smaskOptions, operatorList, task, stateManager.state.clone());
+    },
+    handleTilingType: function PartialEvaluator_handleTilingType(fn, args, resources, pattern, patternDict, operatorList, task) {
+      var tilingOpList = new OperatorList();
+      var resourcesArray = [patternDict.get('Resources'), resources];
+      var patternResources = Dict.merge(this.xref, resourcesArray);
+      return this.getOperatorList(pattern, task, patternResources, tilingOpList).then(function () {
+        operatorList.addDependencies(tilingOpList.dependencies);
+        operatorList.addOp(fn, getTilingPatternIR({
+          fnArray: tilingOpList.fnArray,
+          argsArray: tilingOpList.argsArray
+        }, patternDict, args));
+      });
+    },
+    handleSetFont: function PartialEvaluator_handleSetFont(resources, fontArgs, fontRef, operatorList, task, state) {
+      var fontName;
+      if (fontArgs) {
+        fontArgs = fontArgs.slice();
+        fontName = fontArgs[0].name;
+      }
+      var self = this;
+      return this.loadFont(fontName, fontRef, resources).then(function (translated) {
+        if (!translated.font.isType3Font) {
+          return translated;
+        }
+        return translated.loadType3Data(self, resources, operatorList, task).then(function () {
+          return translated;
+        }, function (reason) {
+          self.handler.send('UnsupportedFeature', { featureId: UNSUPPORTED_FEATURES.font });
+          return new TranslatedFont('g_font_error', new ErrorFont('Type3 font load error: ' + reason), translated.font);
+        });
+      }).then(function (translated) {
+        state.font = translated.font;
+        translated.send(self.handler);
+        return translated.loadedName;
+      });
+    },
+    handleText: function PartialEvaluator_handleText(chars, state) {
+      var font = state.font;
+      var glyphs = font.charsToGlyphs(chars);
+      var isAddToPathSet = !!(state.textRenderingMode & TextRenderingMode.ADD_TO_PATH_FLAG);
+      if (font.data && (isAddToPathSet || this.options.disableFontFace)) {
+        var buildPath = function (fontChar) {
+          if (!font.renderer.hasBuiltPath(fontChar)) {
+            var path = font.renderer.getPathJs(fontChar);
+            this.handler.send('commonobj', [font.loadedName + '_path_' + fontChar, 'FontPath', path]);
+          }
+        }.bind(this);
+        for (var i = 0, ii = glyphs.length; i < ii; i++) {
+          var glyph = glyphs[i];
+          buildPath(glyph.fontChar);
+          var accent = glyph.accent;
+          if (accent && accent.fontChar) {
+            buildPath(accent.fontChar);
+          }
+        }
+      }
+      return glyphs;
+    },
+    setGState: function PartialEvaluator_setGState(resources, gState, operatorList, task, stateManager) {
+      var gStateObj = [];
+      var gStateKeys = gState.getKeys();
+      var self = this;
+      var promise = Promise.resolve();
+      for (var i = 0, ii = gStateKeys.length; i < ii; i++) {
+        var key = gStateKeys[i];
+        var value = gState.get(key);
+        switch (key) {
+          case 'Type':
+            break;
+          case 'LW':
+          case 'LC':
+          case 'LJ':
+          case 'ML':
+          case 'D':
+          case 'RI':
+          case 'FL':
+          case 'CA':
+          case 'ca':
+            gStateObj.push([key, value]);
+            break;
+          case 'Font':
+            promise = promise.then(function () {
+              return self.handleSetFont(resources, null, value[0], operatorList, task, stateManager.state).then(function (loadedName) {
+                operatorList.addDependency(loadedName);
+                gStateObj.push([key, [loadedName, value[1]]]);
+              });
+            });
+            break;
+          case 'BM':
+            gStateObj.push([key, value]);
+            break;
+          case 'SMask':
+            if (isName(value, 'None')) {
+              gStateObj.push([key, false]);
+              break;
+            }
+            if (isDict(value)) {
+              promise = promise.then(function (dict) {
+                return self.handleSMask(dict, resources, operatorList, task, stateManager);
+              }.bind(this, value));
+              gStateObj.push([key, true]);
+            } else {
+              warn('Unsupported SMask type');
+            }
+            break;
+          case 'OP':
+          case 'op':
+          case 'OPM':
+          case 'BG':
+          case 'BG2':
+          case 'UCR':
+          case 'UCR2':
+          case 'TR':
+          case 'TR2':
+          case 'HT':
+          case 'SM':
+          case 'SA':
+          case 'AIS':
+          case 'TK':
+            info('graphic state operator ' + key);
+            break;
+          default:
+            info('Unknown graphic state operator ' + key);
+            break;
+        }
+      }
+      return promise.then(function () {
+        if (gStateObj.length > 0) {
+          operatorList.addOp(OPS.setGState, [gStateObj]);
+        }
+      });
+    },
+    loadFont: function PartialEvaluator_loadFont(fontName, font, resources) {
+      function errorFont() {
+        return Promise.resolve(new TranslatedFont('g_font_error', new ErrorFont('Font ' + fontName + ' is not available'), font));
+      }
+      var fontRef,
+          xref = this.xref;
+      if (font) {
+        assert(isRef(font));
+        fontRef = font;
+      } else {
+        var fontRes = resources.get('Font');
+        if (fontRes) {
+          fontRef = fontRes.getRaw(fontName);
+        } else {
+          warn('fontRes not available');
+          return errorFont();
+        }
+      }
+      if (!fontRef) {
+        warn('fontRef not available');
+        return errorFont();
+      }
+      if (this.fontCache.has(fontRef)) {
+        return this.fontCache.get(fontRef);
+      }
+      font = xref.fetchIfRef(fontRef);
+      if (!isDict(font)) {
+        return errorFont();
+      }
+      if (font.translated) {
+        return font.translated;
+      }
+      var fontCapability = createPromiseCapability();
+      var preEvaluatedFont = this.preEvaluateFont(font);
+      var descriptor = preEvaluatedFont.descriptor;
+      var fontRefIsRef = isRef(fontRef),
+          fontID;
+      if (fontRefIsRef) {
+        fontID = fontRef.toString();
+      }
+      if (isDict(descriptor)) {
+        if (!descriptor.fontAliases) {
+          descriptor.fontAliases = Object.create(null);
+        }
+        var fontAliases = descriptor.fontAliases;
+        var hash = preEvaluatedFont.hash;
+        if (fontAliases[hash]) {
+          var aliasFontRef = fontAliases[hash].aliasRef;
+          if (fontRefIsRef && aliasFontRef && this.fontCache.has(aliasFontRef)) {
+            this.fontCache.putAlias(fontRef, aliasFontRef);
+            return this.fontCache.get(fontRef);
+          }
+        } else {
+          fontAliases[hash] = { fontID: Font.getFontID() };
+        }
+        if (fontRefIsRef) {
+          fontAliases[hash].aliasRef = fontRef;
+        }
+        fontID = fontAliases[hash].fontID;
+      }
+      if (fontRefIsRef) {
+        this.fontCache.put(fontRef, fontCapability.promise);
+      } else {
+        if (!fontID) {
+          fontID = this.idFactory.createObjId();
+        }
+        this.fontCache.put('id_' + fontID, fontCapability.promise);
+      }
+      assert(fontID, 'The "fontID" must be defined.');
+      font.loadedName = 'g_' + this.pdfManager.docId + '_f' + fontID;
+      font.translated = fontCapability.promise;
+      var translatedPromise;
       try {
-       promiseBody(resolve, reject);
-      } catch (ex) {
-       reject(ex);
-      }
-     }, reject);
-    };
-    task.ensureNotTerminated();
-    timeSlotManager.reset();
-    var stop, operation = {}, args = [];
-    while (!(stop = timeSlotManager.check())) {
-     args.length = 0;
-     operation.args = args;
-     if (!preprocessor.read(operation)) {
-      break;
-     }
-     textState = stateManager.state;
-     var fn = operation.fn;
-     args = operation.args;
-     var advance, diff;
-     switch (fn | 0) {
-     case OPS.setFont:
-      var fontNameArg = args[0].name, fontSizeArg = args[1];
-      if (textState.font && fontNameArg === textState.fontName && fontSizeArg === textState.fontSize) {
-       break;
-      }
-      flushTextContentItem();
-      textState.fontName = fontNameArg;
-      textState.fontSize = fontSizeArg;
-      next(handleSetFont(fontNameArg, null));
-      return;
-     case OPS.setTextRise:
-      flushTextContentItem();
-      textState.textRise = args[0];
-      break;
-     case OPS.setHScale:
-      flushTextContentItem();
-      textState.textHScale = args[0] / 100;
-      break;
-     case OPS.setLeading:
-      flushTextContentItem();
-      textState.leading = args[0];
-      break;
-     case OPS.moveText:
-      var isSameTextLine = !textState.font ? false : (textState.font.vertical ? args[0] : args[1]) === 0;
-      advance = args[0] - args[1];
-      if (combineTextItems && isSameTextLine && textContentItem.initialized && advance > 0 && advance <= textContentItem.fakeMultiSpaceMax) {
-       textState.translateTextLineMatrix(args[0], args[1]);
-       textContentItem.width += args[0] - textContentItem.lastAdvanceWidth;
-       textContentItem.height += args[1] - textContentItem.lastAdvanceHeight;
-       diff = args[0] - textContentItem.lastAdvanceWidth - (args[1] - textContentItem.lastAdvanceHeight);
-       addFakeSpaces(diff, textContentItem.str);
-       break;
-      }
-      flushTextContentItem();
-      textState.translateTextLineMatrix(args[0], args[1]);
-      textState.textMatrix = textState.textLineMatrix.slice();
-      break;
-     case OPS.setLeadingMoveText:
-      flushTextContentItem();
-      textState.leading = -args[1];
-      textState.translateTextLineMatrix(args[0], args[1]);
-      textState.textMatrix = textState.textLineMatrix.slice();
-      break;
-     case OPS.nextLine:
-      flushTextContentItem();
-      textState.carriageReturn();
-      break;
-     case OPS.setTextMatrix:
-      advance = textState.calcTextLineMatrixAdvance(args[0], args[1], args[2], args[3], args[4], args[5]);
-      if (combineTextItems && advance !== null && textContentItem.initialized && advance.value > 0 && advance.value <= textContentItem.fakeMultiSpaceMax) {
-       textState.translateTextLineMatrix(advance.width, advance.height);
-       textContentItem.width += advance.width - textContentItem.lastAdvanceWidth;
-       textContentItem.height += advance.height - textContentItem.lastAdvanceHeight;
-       diff = advance.width - textContentItem.lastAdvanceWidth - (advance.height - textContentItem.lastAdvanceHeight);
-       addFakeSpaces(diff, textContentItem.str);
-       break;
-      }
-      flushTextContentItem();
-      textState.setTextMatrix(args[0], args[1], args[2], args[3], args[4], args[5]);
-      textState.setTextLineMatrix(args[0], args[1], args[2], args[3], args[4], args[5]);
-      break;
-     case OPS.setCharSpacing:
-      textState.charSpacing = args[0];
-      break;
-     case OPS.setWordSpacing:
-      textState.wordSpacing = args[0];
-      break;
-     case OPS.beginText:
-      flushTextContentItem();
-      textState.textMatrix = IDENTITY_MATRIX.slice();
-      textState.textLineMatrix = IDENTITY_MATRIX.slice();
-      break;
-     case OPS.showSpacedText:
-      var items = args[0];
-      var offset;
-      for (var j = 0, jj = items.length; j < jj; j++) {
-       if (typeof items[j] === 'string') {
-        buildTextContentItem(items[j]);
-       } else if (isNum(items[j])) {
-        ensureTextContentItem();
-        advance = items[j] * textState.fontSize / 1000;
-        var breakTextRun = false;
-        if (textState.font.vertical) {
-         offset = advance;
-         textState.translateTextMatrix(0, offset);
-         breakTextRun = textContentItem.textRunBreakAllowed && advance > textContentItem.fakeMultiSpaceMax;
-         if (!breakTextRun) {
-          textContentItem.height += offset;
-         }
+        translatedPromise = this.translateFont(preEvaluatedFont);
+      } catch (e) {
+        translatedPromise = Promise.reject(e);
+      }
+      var self = this;
+      translatedPromise.then(function (translatedFont) {
+        if (translatedFont.fontType !== undefined) {
+          var xrefFontStats = xref.stats.fontTypes;
+          xrefFontStats[translatedFont.fontType] = true;
+        }
+        fontCapability.resolve(new TranslatedFont(font.loadedName, translatedFont, font));
+      }, function (reason) {
+        self.handler.send('UnsupportedFeature', { featureId: UNSUPPORTED_FEATURES.font });
+        try {
+          var descriptor = preEvaluatedFont.descriptor;
+          var fontFile3 = descriptor && descriptor.get('FontFile3');
+          var subtype = fontFile3 && fontFile3.get('Subtype');
+          var fontType = getFontType(preEvaluatedFont.type, subtype && subtype.name);
+          var xrefFontStats = xref.stats.fontTypes;
+          xrefFontStats[fontType] = true;
+        } catch (ex) {}
+        fontCapability.resolve(new TranslatedFont(font.loadedName, new ErrorFont(reason instanceof Error ? reason.message : reason), font));
+      });
+      return fontCapability.promise;
+    },
+    buildPath: function PartialEvaluator_buildPath(operatorList, fn, args) {
+      var lastIndex = operatorList.length - 1;
+      if (!args) {
+        args = [];
+      }
+      if (lastIndex < 0 || operatorList.fnArray[lastIndex] !== OPS.constructPath) {
+        operatorList.addOp(OPS.constructPath, [[fn], args]);
+      } else {
+        var opArgs = operatorList.argsArray[lastIndex];
+        opArgs[0].push(fn);
+        Array.prototype.push.apply(opArgs[1], args);
+      }
+    },
+    handleColorN: function PartialEvaluator_handleColorN(operatorList, fn, args, cs, patterns, resources, task) {
+      var patternName = args[args.length - 1];
+      var pattern;
+      if (isName(patternName) && (pattern = patterns.get(patternName.name))) {
+        var dict = isStream(pattern) ? pattern.dict : pattern;
+        var typeNum = dict.get('PatternType');
+        if (typeNum === TILING_PATTERN) {
+          var color = cs.base ? cs.base.getRgb(args, 0) : null;
+          return this.handleTilingType(fn, color, resources, pattern, dict, operatorList, task);
+        } else if (typeNum === SHADING_PATTERN) {
+          var shading = dict.get('Shading');
+          var matrix = dict.getArray('Matrix');
+          pattern = Pattern.parseShading(shading, matrix, this.xref, resources, this.handler);
+          operatorList.addOp(fn, pattern.getIR());
+          return Promise.resolve();
+        }
+        return Promise.reject('Unknown PatternType: ' + typeNum);
+      }
+      operatorList.addOp(fn, args);
+      return Promise.resolve();
+    },
+    getOperatorList: function PartialEvaluator_getOperatorList(stream, task, resources, operatorList, initialState) {
+      var self = this;
+      var xref = this.xref;
+      var imageCache = Object.create(null);
+      assert(operatorList);
+      resources = resources || Dict.empty;
+      var xobjs = resources.get('XObject') || Dict.empty;
+      var patterns = resources.get('Pattern') || Dict.empty;
+      var stateManager = new StateManager(initialState || new EvalState());
+      var preprocessor = new EvaluatorPreprocessor(stream, xref, stateManager);
+      var timeSlotManager = new TimeSlotManager();
+      return new Promise(function promiseBody(resolve, reject) {
+        var next = function (promise) {
+          promise.then(function () {
+            try {
+              promiseBody(resolve, reject);
+            } catch (ex) {
+              reject(ex);
+            }
+          }, reject);
+        };
+        task.ensureNotTerminated();
+        timeSlotManager.reset();
+        var stop,
+            operation = {},
+            i,
+            ii,
+            cs;
+        while (!(stop = timeSlotManager.check())) {
+          operation.args = null;
+          if (!preprocessor.read(operation)) {
+            break;
+          }
+          var args = operation.args;
+          var fn = operation.fn;
+          switch (fn | 0) {
+            case OPS.paintXObject:
+              if (args[0].code) {
+                break;
+              }
+              var name = args[0].name;
+              if (!name) {
+                warn('XObject must be referred to by name.');
+                continue;
+              }
+              if (imageCache[name] !== undefined) {
+                operatorList.addOp(imageCache[name].fn, imageCache[name].args);
+                args = null;
+                continue;
+              }
+              var xobj = xobjs.get(name);
+              if (xobj) {
+                assert(isStream(xobj), 'XObject should be a stream');
+                var type = xobj.dict.get('Subtype');
+                assert(isName(type), 'XObject should have a Name subtype');
+                if (type.name === 'Form') {
+                  stateManager.save();
+                  next(self.buildFormXObject(resources, xobj, null, operatorList, task, stateManager.state.clone()).then(function () {
+                    stateManager.restore();
+                  }));
+                  return;
+                } else if (type.name === 'Image') {
+                  self.buildPaintImageXObject(resources, xobj, false, operatorList, name, imageCache);
+                  args = null;
+                  continue;
+                } else if (type.name === 'PS') {
+                  info('Ignored XObject subtype PS');
+                  continue;
+                } else {
+                  error('Unhandled XObject subtype ' + type.name);
+                }
+              }
+              break;
+            case OPS.setFont:
+              var fontSize = args[1];
+              next(self.handleSetFont(resources, args, null, operatorList, task, stateManager.state).then(function (loadedName) {
+                operatorList.addDependency(loadedName);
+                operatorList.addOp(OPS.setFont, [loadedName, fontSize]);
+              }));
+              return;
+            case OPS.endInlineImage:
+              var cacheKey = args[0].cacheKey;
+              if (cacheKey) {
+                var cacheEntry = imageCache[cacheKey];
+                if (cacheEntry !== undefined) {
+                  operatorList.addOp(cacheEntry.fn, cacheEntry.args);
+                  args = null;
+                  continue;
+                }
+              }
+              self.buildPaintImageXObject(resources, args[0], true, operatorList, cacheKey, imageCache);
+              args = null;
+              continue;
+            case OPS.showText:
+              args[0] = self.handleText(args[0], stateManager.state);
+              break;
+            case OPS.showSpacedText:
+              var arr = args[0];
+              var combinedGlyphs = [];
+              var arrLength = arr.length;
+              var state = stateManager.state;
+              for (i = 0; i < arrLength; ++i) {
+                var arrItem = arr[i];
+                if (isString(arrItem)) {
+                  Array.prototype.push.apply(combinedGlyphs, self.handleText(arrItem, state));
+                } else if (isNum(arrItem)) {
+                  combinedGlyphs.push(arrItem);
+                }
+              }
+              args[0] = combinedGlyphs;
+              fn = OPS.showText;
+              break;
+            case OPS.nextLineShowText:
+              operatorList.addOp(OPS.nextLine);
+              args[0] = self.handleText(args[0], stateManager.state);
+              fn = OPS.showText;
+              break;
+            case OPS.nextLineSetSpacingShowText:
+              operatorList.addOp(OPS.nextLine);
+              operatorList.addOp(OPS.setWordSpacing, [args.shift()]);
+              operatorList.addOp(OPS.setCharSpacing, [args.shift()]);
+              args[0] = self.handleText(args[0], stateManager.state);
+              fn = OPS.showText;
+              break;
+            case OPS.setTextRenderingMode:
+              stateManager.state.textRenderingMode = args[0];
+              break;
+            case OPS.setFillColorSpace:
+              stateManager.state.fillColorSpace = ColorSpace.parse(args[0], xref, resources);
+              continue;
+            case OPS.setStrokeColorSpace:
+              stateManager.state.strokeColorSpace = ColorSpace.parse(args[0], xref, resources);
+              continue;
+            case OPS.setFillColor:
+              cs = stateManager.state.fillColorSpace;
+              args = cs.getRgb(args, 0);
+              fn = OPS.setFillRGBColor;
+              break;
+            case OPS.setStrokeColor:
+              cs = stateManager.state.strokeColorSpace;
+              args = cs.getRgb(args, 0);
+              fn = OPS.setStrokeRGBColor;
+              break;
+            case OPS.setFillGray:
+              stateManager.state.fillColorSpace = ColorSpace.singletons.gray;
+              args = ColorSpace.singletons.gray.getRgb(args, 0);
+              fn = OPS.setFillRGBColor;
+              break;
+            case OPS.setStrokeGray:
+              stateManager.state.strokeColorSpace = ColorSpace.singletons.gray;
+              args = ColorSpace.singletons.gray.getRgb(args, 0);
+              fn = OPS.setStrokeRGBColor;
+              break;
+            case OPS.setFillCMYKColor:
+              stateManager.state.fillColorSpace = ColorSpace.singletons.cmyk;
+              args = ColorSpace.singletons.cmyk.getRgb(args, 0);
+              fn = OPS.setFillRGBColor;
+              break;
+            case OPS.setStrokeCMYKColor:
+              stateManager.state.strokeColorSpace = ColorSpace.singletons.cmyk;
+              args = ColorSpace.singletons.cmyk.getRgb(args, 0);
+              fn = OPS.setStrokeRGBColor;
+              break;
+            case OPS.setFillRGBColor:
+              stateManager.state.fillColorSpace = ColorSpace.singletons.rgb;
+              args = ColorSpace.singletons.rgb.getRgb(args, 0);
+              break;
+            case OPS.setStrokeRGBColor:
+              stateManager.state.strokeColorSpace = ColorSpace.singletons.rgb;
+              args = ColorSpace.singletons.rgb.getRgb(args, 0);
+              break;
+            case OPS.setFillColorN:
+              cs = stateManager.state.fillColorSpace;
+              if (cs.name === 'Pattern') {
+                next(self.handleColorN(operatorList, OPS.setFillColorN, args, cs, patterns, resources, task));
+                return;
+              }
+              args = cs.getRgb(args, 0);
+              fn = OPS.setFillRGBColor;
+              break;
+            case OPS.setStrokeColorN:
+              cs = stateManager.state.strokeColorSpace;
+              if (cs.name === 'Pattern') {
+                next(self.handleColorN(operatorList, OPS.setStrokeColorN, args, cs, patterns, resources, task));
+                return;
+              }
+              args = cs.getRgb(args, 0);
+              fn = OPS.setStrokeRGBColor;
+              break;
+            case OPS.shadingFill:
+              var shadingRes = resources.get('Shading');
+              assert(shadingRes, 'No shading resource found');
+              var shading = shadingRes.get(args[0].name);
+              assert(shading, 'No shading object found');
+              var shadingFill = Pattern.parseShading(shading, null, xref, resources, self.handler);
+              var patternIR = shadingFill.getIR();
+              args = [patternIR];
+              fn = OPS.shadingFill;
+              break;
+            case OPS.setGState:
+              var dictName = args[0];
+              var extGState = resources.get('ExtGState');
+              if (!isDict(extGState) || !extGState.has(dictName.name)) {
+                break;
+              }
+              var gState = extGState.get(dictName.name);
+              next(self.setGState(resources, gState, operatorList, task, stateManager));
+              return;
+            case OPS.moveTo:
+            case OPS.lineTo:
+            case OPS.curveTo:
+            case OPS.curveTo2:
+            case OPS.curveTo3:
+            case OPS.closePath:
+              self.buildPath(operatorList, fn, args);
+              continue;
+            case OPS.rectangle:
+              self.buildPath(operatorList, fn, args);
+              continue;
+            case OPS.markPoint:
+            case OPS.markPointProps:
+            case OPS.beginMarkedContent:
+            case OPS.beginMarkedContentProps:
+            case OPS.endMarkedContent:
+            case OPS.beginCompat:
+            case OPS.endCompat:
+              continue;
+            default:
+              if (args !== null) {
+                for (i = 0, ii = args.length; i < ii; i++) {
+                  if (args[i] instanceof Dict) {
+                    break;
+                  }
+                }
+                if (i < ii) {
+                  warn('getOperatorList - ignoring operator: ' + fn);
+                  continue;
+                }
+              }
+          }
+          operatorList.addOp(fn, args);
+        }
+        if (stop) {
+          next(deferred);
+          return;
+        }
+        for (i = 0, ii = preprocessor.savedStatesDepth; i < ii; i++) {
+          operatorList.addOp(OPS.restore, []);
+        }
+        resolve();
+      });
+    },
+    getTextContent: function PartialEvaluator_getTextContent(stream, task, resources, stateManager, normalizeWhitespace, combineTextItems) {
+      stateManager = stateManager || new StateManager(new TextState());
+      var WhitespaceRegexp = /\s/g;
+      var textContent = {
+        items: [],
+        styles: Object.create(null)
+      };
+      var textContentItem = {
+        initialized: false,
+        str: [],
+        width: 0,
+        height: 0,
+        vertical: false,
+        lastAdvanceWidth: 0,
+        lastAdvanceHeight: 0,
+        textAdvanceScale: 0,
+        spaceWidth: 0,
+        fakeSpaceMin: Infinity,
+        fakeMultiSpaceMin: Infinity,
+        fakeMultiSpaceMax: -0,
+        textRunBreakAllowed: false,
+        transform: null,
+        fontName: null
+      };
+      var SPACE_FACTOR = 0.3;
+      var MULTI_SPACE_FACTOR = 1.5;
+      var MULTI_SPACE_FACTOR_MAX = 4;
+      var self = this;
+      var xref = this.xref;
+      resources = xref.fetchIfRef(resources) || Dict.empty;
+      var xobjs = null;
+      var xobjsCache = Object.create(null);
+      var preprocessor = new EvaluatorPreprocessor(stream, xref, stateManager);
+      var textState;
+      function ensureTextContentItem() {
+        if (textContentItem.initialized) {
+          return textContentItem;
+        }
+        var font = textState.font;
+        if (!(font.loadedName in textContent.styles)) {
+          textContent.styles[font.loadedName] = {
+            fontFamily: font.fallbackName,
+            ascent: font.ascent,
+            descent: font.descent,
+            vertical: font.vertical
+          };
+        }
+        textContentItem.fontName = font.loadedName;
+        var tsm = [textState.fontSize * textState.textHScale, 0, 0, textState.fontSize, 0, textState.textRise];
+        if (font.isType3Font && textState.fontMatrix !== FONT_IDENTITY_MATRIX && textState.fontSize === 1) {
+          var glyphHeight = font.bbox[3] - font.bbox[1];
+          if (glyphHeight > 0) {
+            glyphHeight = glyphHeight * textState.fontMatrix[3];
+            tsm[3] *= glyphHeight;
+          }
+        }
+        var trm = Util.transform(textState.ctm, Util.transform(textState.textMatrix, tsm));
+        textContentItem.transform = trm;
+        if (!font.vertical) {
+          textContentItem.width = 0;
+          textContentItem.height = Math.sqrt(trm[2] * trm[2] + trm[3] * trm[3]);
+          textContentItem.vertical = false;
         } else {
-         advance = -advance;
-         offset = advance * textState.textHScale;
-         textState.translateTextMatrix(offset, 0);
-         breakTextRun = textContentItem.textRunBreakAllowed && advance > textContentItem.fakeMultiSpaceMax;
-         if (!breakTextRun) {
-          textContentItem.width += offset;
-         }
+          textContentItem.width = Math.sqrt(trm[0] * trm[0] + trm[1] * trm[1]);
+          textContentItem.height = 0;
+          textContentItem.vertical = true;
         }
-        if (breakTextRun) {
-         flushTextContentItem();
-        } else if (advance > 0) {
-         addFakeSpaces(advance, textContentItem.str);
+        var a = textState.textLineMatrix[0];
+        var b = textState.textLineMatrix[1];
+        var scaleLineX = Math.sqrt(a * a + b * b);
+        a = textState.ctm[0];
+        b = textState.ctm[1];
+        var scaleCtmX = Math.sqrt(a * a + b * b);
+        textContentItem.textAdvanceScale = scaleCtmX * scaleLineX;
+        textContentItem.lastAdvanceWidth = 0;
+        textContentItem.lastAdvanceHeight = 0;
+        var spaceWidth = font.spaceWidth / 1000 * textState.fontSize;
+        if (spaceWidth) {
+          textContentItem.spaceWidth = spaceWidth;
+          textContentItem.fakeSpaceMin = spaceWidth * SPACE_FACTOR;
+          textContentItem.fakeMultiSpaceMin = spaceWidth * MULTI_SPACE_FACTOR;
+          textContentItem.fakeMultiSpaceMax = spaceWidth * MULTI_SPACE_FACTOR_MAX;
+          textContentItem.textRunBreakAllowed = !font.isMonospace;
+        } else {
+          textContentItem.spaceWidth = 0;
+          textContentItem.fakeSpaceMin = Infinity;
+          textContentItem.fakeMultiSpaceMin = Infinity;
+          textContentItem.fakeMultiSpaceMax = 0;
+          textContentItem.textRunBreakAllowed = false;
         }
-       }
+        textContentItem.initialized = true;
+        return textContentItem;
       }
-      break;
-     case OPS.showText:
-      buildTextContentItem(args[0]);
-      break;
-     case OPS.nextLineShowText:
-      flushTextContentItem();
-      textState.carriageReturn();
-      buildTextContentItem(args[0]);
-      break;
-     case OPS.nextLineSetSpacingShowText:
-      flushTextContentItem();
-      textState.wordSpacing = args[0];
-      textState.charSpacing = args[1];
-      textState.carriageReturn();
-      buildTextContentItem(args[2]);
-      break;
-     case OPS.paintXObject:
-      flushTextContentItem();
-      if (args[0].code) {
-       break;
-      }
-      if (!xobjs) {
-       xobjs = resources.get('XObject') || Dict.empty;
-      }
-      var name = args[0].name;
-      if (xobjsCache.key === name) {
-       if (xobjsCache.texts) {
-        Util.appendToArray(textContent.items, xobjsCache.texts.items);
-        Util.extendObj(textContent.styles, xobjsCache.texts.styles);
-       }
-       break;
-      }
-      var xobj = xobjs.get(name);
-      if (!xobj) {
-       break;
-      }
-      assert(isStream(xobj), 'XObject should be a stream');
-      var type = xobj.dict.get('Subtype');
-      assert(isName(type), 'XObject should have a Name subtype');
-      if (type.name !== 'Form') {
-       xobjsCache.key = name;
-       xobjsCache.texts = null;
-       break;
-      }
-      stateManager.save();
-      var matrix = xobj.dict.getArray('Matrix');
-      if (isArray(matrix) && matrix.length === 6) {
-       stateManager.transform(matrix);
-      }
-      next(self.getTextContent(xobj, task, xobj.dict.get('Resources') || resources, stateManager, normalizeWhitespace, combineTextItems).then(function (formTextContent) {
-       Util.appendToArray(textContent.items, formTextContent.items);
-       Util.extendObj(textContent.styles, formTextContent.styles);
-       stateManager.restore();
-       xobjsCache.key = name;
-       xobjsCache.texts = formTextContent;
-      }));
-      return;
-     case OPS.setGState:
-      flushTextContentItem();
-      var dictName = args[0];
-      var extGState = resources.get('ExtGState');
-      if (!isDict(extGState) || !isName(dictName)) {
-       break;
-      }
-      var gState = extGState.get(dictName.name);
-      if (!isDict(gState)) {
-       break;
-      }
-      var gStateFont = gState.get('Font');
-      if (gStateFont) {
-       textState.fontName = null;
-       textState.fontSize = gStateFont[1];
-       next(handleSetFont(null, gStateFont[0]));
-       return;
+      function replaceWhitespace(str) {
+        var i = 0,
+            ii = str.length,
+            code;
+        while (i < ii && (code = str.charCodeAt(i)) >= 0x20 && code <= 0x7F) {
+          i++;
+        }
+        return i < ii ? str.replace(WhitespaceRegexp, ' ') : str;
       }
-      break;
-     }
-    }
-    if (stop) {
-     next(deferred);
-     return;
-    }
-    flushTextContentItem();
-    resolve(textContent);
-   });
-  },
-  extractDataStructures: function PartialEvaluator_extractDataStructures(dict, baseDict, properties) {
-   var xref = this.xref;
-   var toUnicode = dict.get('ToUnicode') || baseDict.get('ToUnicode');
-   var toUnicodePromise = toUnicode ? this.readToUnicode(toUnicode) : Promise.resolve(undefined);
-   if (properties.composite) {
-    var cidSystemInfo = dict.get('CIDSystemInfo');
-    if (isDict(cidSystemInfo)) {
-     properties.cidSystemInfo = {
-      registry: cidSystemInfo.get('Registry'),
-      ordering: cidSystemInfo.get('Ordering'),
-      supplement: cidSystemInfo.get('Supplement')
-     };
-    }
-    var cidToGidMap = dict.get('CIDToGIDMap');
-    if (isStream(cidToGidMap)) {
-     properties.cidToGidMap = this.readCidToGidMap(cidToGidMap);
-    }
-   }
-   var differences = [];
-   var baseEncodingName = null;
-   var encoding;
-   if (dict.has('Encoding')) {
-    encoding = dict.get('Encoding');
-    if (isDict(encoding)) {
-     baseEncodingName = encoding.get('BaseEncoding');
-     baseEncodingName = isName(baseEncodingName) ? baseEncodingName.name : null;
-     if (encoding.has('Differences')) {
-      var diffEncoding = encoding.get('Differences');
-      var index = 0;
-      for (var j = 0, jj = diffEncoding.length; j < jj; j++) {
-       var data = xref.fetchIfRef(diffEncoding[j]);
-       if (isNum(data)) {
-        index = data;
-       } else if (isName(data)) {
-        differences[index++] = data.name;
-       } else {
-        error('Invalid entry in \'Differences\' array: ' + data);
-       }
-      }
-     }
-    } else if (isName(encoding)) {
-     baseEncodingName = encoding.name;
-    } else {
-     error('Encoding is not a Name nor a Dict');
-    }
-    if (baseEncodingName !== 'MacRomanEncoding' && baseEncodingName !== 'MacExpertEncoding' && baseEncodingName !== 'WinAnsiEncoding') {
-     baseEncodingName = null;
-    }
-   }
-   if (baseEncodingName) {
-    properties.defaultEncoding = getEncoding(baseEncodingName).slice();
-   } else {
-    var isSymbolicFont = !!(properties.flags & FontFlags.Symbolic);
-    var isNonsymbolicFont = !!(properties.flags & FontFlags.Nonsymbolic);
-    encoding = StandardEncoding;
-    if (properties.type === 'TrueType' && !isNonsymbolicFont) {
-     encoding = WinAnsiEncoding;
-    }
-    if (isSymbolicFont) {
-     encoding = MacRomanEncoding;
-     if (!properties.file) {
-      if (/Symbol/i.test(properties.name)) {
-       encoding = SymbolSetEncoding;
-      } else if (/Dingbats/i.test(properties.name)) {
-       encoding = ZapfDingbatsEncoding;
-      }
-     }
-    }
-    properties.defaultEncoding = encoding;
-   }
-   properties.differences = differences;
-   properties.baseEncodingName = baseEncodingName;
-   properties.hasEncoding = !!baseEncodingName || differences.length > 0;
-   properties.dict = dict;
-   return toUnicodePromise.then(function (toUnicode) {
-    properties.toUnicode = toUnicode;
-    return this.buildToUnicode(properties);
-   }.bind(this)).then(function (toUnicode) {
-    properties.toUnicode = toUnicode;
-    return properties;
-   });
-  },
-  buildToUnicode: function PartialEvaluator_buildToUnicode(properties) {
-   properties.hasIncludedToUnicodeMap = !!properties.toUnicode && properties.toUnicode.length > 0;
-   if (properties.hasIncludedToUnicodeMap) {
-    return Promise.resolve(properties.toUnicode);
-   }
-   var toUnicode, charcode, glyphName;
-   if (!properties.composite) {
-    toUnicode = [];
-    var encoding = properties.defaultEncoding.slice();
-    var baseEncodingName = properties.baseEncodingName;
-    var differences = properties.differences;
-    for (charcode in differences) {
-     glyphName = differences[charcode];
-     if (glyphName === '.notdef') {
-      continue;
-     }
-     encoding[charcode] = glyphName;
-    }
-    var glyphsUnicodeMap = getGlyphsUnicode();
-    for (charcode in encoding) {
-     glyphName = encoding[charcode];
-     if (glyphName === '') {
-      continue;
-     } else if (glyphsUnicodeMap[glyphName] === undefined) {
-      var code = 0;
-      switch (glyphName[0]) {
-      case 'G':
-       if (glyphName.length === 3) {
-        code = parseInt(glyphName.substr(1), 16);
-       }
-       break;
-      case 'g':
-       if (glyphName.length === 5) {
-        code = parseInt(glyphName.substr(1), 16);
-       }
-       break;
-      case 'C':
-      case 'c':
-       if (glyphName.length >= 3) {
-        code = +glyphName.substr(1);
-       }
-       break;
-      default:
-       var unicode = getUnicodeForGlyph(glyphName, glyphsUnicodeMap);
-       if (unicode !== -1) {
-        code = unicode;
-       }
-      }
-      if (code) {
-       if (baseEncodingName && code === +charcode) {
-        var baseEncoding = getEncoding(baseEncodingName);
-        if (baseEncoding && (glyphName = baseEncoding[charcode])) {
-         toUnicode[charcode] = String.fromCharCode(glyphsUnicodeMap[glyphName]);
-         continue;
-        }
-       }
-       toUnicode[charcode] = String.fromCharCode(code);
-      }
-      continue;
-     }
-     toUnicode[charcode] = String.fromCharCode(glyphsUnicodeMap[glyphName]);
-    }
-    return Promise.resolve(new ToUnicodeMap(toUnicode));
-   }
-   if (properties.composite && (properties.cMap.builtInCMap && !(properties.cMap instanceof IdentityCMap) || properties.cidSystemInfo.registry === 'Adobe' && (properties.cidSystemInfo.ordering === 'GB1' || properties.cidSystemInfo.ordering === 'CNS1' || properties.cidSystemInfo.ordering === 'Japan1' || properties.cidSystemInfo.ordering === 'Korea1'))) {
-    var registry = properties.cidSystemInfo.registry;
-    var ordering = properties.cidSystemInfo.ordering;
-    var ucs2CMapName = Name.get(registry + '-' + ordering + '-UCS2');
-    return CMapFactory.create({
-     encoding: ucs2CMapName,
-     fetchBuiltInCMap: this.fetchBuiltInCMap,
-     useCMap: null
-    }).then(function (ucs2CMap) {
-     var cMap = properties.cMap;
-     toUnicode = [];
-     cMap.forEach(function (charcode, cid) {
-      assert(cid <= 0xffff, 'Max size of CID is 65,535');
-      var ucs2 = ucs2CMap.lookup(cid);
-      if (ucs2) {
-       toUnicode[charcode] = String.fromCharCode((ucs2.charCodeAt(0) << 8) + ucs2.charCodeAt(1));
-      }
-     });
-     return new ToUnicodeMap(toUnicode);
-    });
-   }
-   return Promise.resolve(new IdentityToUnicodeMap(properties.firstChar, properties.lastChar));
-  },
-  readToUnicode: function PartialEvaluator_readToUnicode(toUnicode) {
-   var cmapObj = toUnicode;
-   if (isName(cmapObj)) {
-    return CMapFactory.create({
-     encoding: cmapObj,
-     fetchBuiltInCMap: this.fetchBuiltInCMap,
-     useCMap: null
-    }).then(function (cmap) {
-     if (cmap instanceof IdentityCMap) {
-      return new IdentityToUnicodeMap(0, 0xFFFF);
-     }
-     return new ToUnicodeMap(cmap.getMap());
-    });
-   } else if (isStream(cmapObj)) {
-    return CMapFactory.create({
-     encoding: cmapObj,
-     fetchBuiltInCMap: this.fetchBuiltInCMap,
-     useCMap: null
-    }).then(function (cmap) {
-     if (cmap instanceof IdentityCMap) {
-      return new IdentityToUnicodeMap(0, 0xFFFF);
-     }
-     var map = new Array(cmap.length);
-     cmap.forEach(function (charCode, token) {
-      var str = [];
-      for (var k = 0; k < token.length; k += 2) {
-       var w1 = token.charCodeAt(k) << 8 | token.charCodeAt(k + 1);
-       if ((w1 & 0xF800) !== 0xD800) {
-        str.push(w1);
-        continue;
-       }
-       k += 2;
-       var w2 = token.charCodeAt(k) << 8 | token.charCodeAt(k + 1);
-       str.push(((w1 & 0x3ff) << 10) + (w2 & 0x3ff) + 0x10000);
-      }
-      map[charCode] = String.fromCharCode.apply(String, str);
-     });
-     return new ToUnicodeMap(map);
-    });
-   }
-   return Promise.resolve(null);
-  },
-  readCidToGidMap: function PartialEvaluator_readCidToGidMap(cidToGidStream) {
-   var glyphsData = cidToGidStream.getBytes();
-   var result = [];
-   for (var j = 0, jj = glyphsData.length; j < jj; j++) {
-    var glyphID = glyphsData[j++] << 8 | glyphsData[j];
-    if (glyphID === 0) {
-     continue;
-    }
-    var code = j >> 1;
-    result[code] = glyphID;
-   }
-   return result;
-  },
-  extractWidths: function PartialEvaluator_extractWidths(dict, descriptor, properties) {
-   var xref = this.xref;
-   var glyphsWidths = [];
-   var defaultWidth = 0;
-   var glyphsVMetrics = [];
-   var defaultVMetrics;
-   var i, ii, j, jj, start, code, widths;
-   if (properties.composite) {
-    defaultWidth = dict.get('DW') || 1000;
-    widths = dict.get('W');
-    if (widths) {
-     for (i = 0, ii = widths.length; i < ii; i++) {
-      start = xref.fetchIfRef(widths[i++]);
-      code = xref.fetchIfRef(widths[i]);
-      if (isArray(code)) {
-       for (j = 0, jj = code.length; j < jj; j++) {
-        glyphsWidths[start++] = xref.fetchIfRef(code[j]);
-       }
+      function runBidiTransform(textChunk) {
+        var str = textChunk.str.join('');
+        var bidiResult = bidi(str, -1, textChunk.vertical);
+        return {
+          str: normalizeWhitespace ? replaceWhitespace(bidiResult.str) : bidiResult.str,
+          dir: bidiResult.dir,
+          width: textChunk.width,
+          height: textChunk.height,
+          transform: textChunk.transform,
+          fontName: textChunk.fontName
+        };
+      }
+      function handleSetFont(fontName, fontRef) {
+        return self.loadFont(fontName, fontRef, resources).then(function (translated) {
+          textState.font = translated.font;
+          textState.fontMatrix = translated.font.fontMatrix || FONT_IDENTITY_MATRIX;
+        });
+      }
+      function buildTextContentItem(chars) {
+        var font = textState.font;
+        var textChunk = ensureTextContentItem();
+        var width = 0;
+        var height = 0;
+        var glyphs = font.charsToGlyphs(chars);
+        for (var i = 0; i < glyphs.length; i++) {
+          var glyph = glyphs[i];
+          var glyphWidth = null;
+          if (font.vertical && glyph.vmetric) {
+            glyphWidth = glyph.vmetric[0];
+          } else {
+            glyphWidth = glyph.width;
+          }
+          var glyphUnicode = glyph.unicode;
+          var NormalizedUnicodes = getNormalizedUnicodes();
+          if (NormalizedUnicodes[glyphUnicode] !== undefined) {
+            glyphUnicode = NormalizedUnicodes[glyphUnicode];
+          }
+          glyphUnicode = reverseIfRtl(glyphUnicode);
+          var charSpacing = textState.charSpacing;
+          if (glyph.isSpace) {
+            var wordSpacing = textState.wordSpacing;
+            charSpacing += wordSpacing;
+            if (wordSpacing > 0) {
+              addFakeSpaces(wordSpacing, textChunk.str);
+            }
+          }
+          var tx = 0;
+          var ty = 0;
+          if (!font.vertical) {
+            var w0 = glyphWidth * textState.fontMatrix[0];
+            tx = (w0 * textState.fontSize + charSpacing) * textState.textHScale;
+            width += tx;
+          } else {
+            var w1 = glyphWidth * textState.fontMatrix[0];
+            ty = w1 * textState.fontSize + charSpacing;
+            height += ty;
+          }
+          textState.translateTextMatrix(tx, ty);
+          textChunk.str.push(glyphUnicode);
+        }
+        if (!font.vertical) {
+          textChunk.lastAdvanceWidth = width;
+          textChunk.width += width;
+        } else {
+          textChunk.lastAdvanceHeight = height;
+          textChunk.height += Math.abs(height);
+        }
+        return textChunk;
+      }
+      function addFakeSpaces(width, strBuf) {
+        if (width < textContentItem.fakeSpaceMin) {
+          return;
+        }
+        if (width < textContentItem.fakeMultiSpaceMin) {
+          strBuf.push(' ');
+          return;
+        }
+        var fakeSpaces = Math.round(width / textContentItem.spaceWidth);
+        while (fakeSpaces-- > 0) {
+          strBuf.push(' ');
+        }
+      }
+      function flushTextContentItem() {
+        if (!textContentItem.initialized) {
+          return;
+        }
+        textContentItem.width *= textContentItem.textAdvanceScale;
+        textContentItem.height *= textContentItem.textAdvanceScale;
+        textContent.items.push(runBidiTransform(textContentItem));
+        textContentItem.initialized = false;
+        textContentItem.str.length = 0;
+      }
+      var timeSlotManager = new TimeSlotManager();
+      return new Promise(function promiseBody(resolve, reject) {
+        var next = function (promise) {
+          promise.then(function () {
+            try {
+              promiseBody(resolve, reject);
+            } catch (ex) {
+              reject(ex);
+            }
+          }, reject);
+        };
+        task.ensureNotTerminated();
+        timeSlotManager.reset();
+        var stop,
+            operation = {},
+            args = [];
+        while (!(stop = timeSlotManager.check())) {
+          args.length = 0;
+          operation.args = args;
+          if (!preprocessor.read(operation)) {
+            break;
+          }
+          textState = stateManager.state;
+          var fn = operation.fn;
+          args = operation.args;
+          var advance, diff;
+          switch (fn | 0) {
+            case OPS.setFont:
+              var fontNameArg = args[0].name,
+                  fontSizeArg = args[1];
+              if (textState.font && fontNameArg === textState.fontName && fontSizeArg === textState.fontSize) {
+                break;
+              }
+              flushTextContentItem();
+              textState.fontName = fontNameArg;
+              textState.fontSize = fontSizeArg;
+              next(handleSetFont(fontNameArg, null));
+              return;
+            case OPS.setTextRise:
+              flushTextContentItem();
+              textState.textRise = args[0];
+              break;
+            case OPS.setHScale:
+              flushTextContentItem();
+              textState.textHScale = args[0] / 100;
+              break;
+            case OPS.setLeading:
+              flushTextContentItem();
+              textState.leading = args[0];
+              break;
+            case OPS.moveText:
+              var isSameTextLine = !textState.font ? false : (textState.font.vertical ? args[0] : args[1]) === 0;
+              advance = args[0] - args[1];
+              if (combineTextItems && isSameTextLine && textContentItem.initialized && advance > 0 && advance <= textContentItem.fakeMultiSpaceMax) {
+                textState.translateTextLineMatrix(args[0], args[1]);
+                textContentItem.width += args[0] - textContentItem.lastAdvanceWidth;
+                textContentItem.height += args[1] - textContentItem.lastAdvanceHeight;
+                diff = args[0] - textContentItem.lastAdvanceWidth - (args[1] - textContentItem.lastAdvanceHeight);
+                addFakeSpaces(diff, textContentItem.str);
+                break;
+              }
+              flushTextContentItem();
+              textState.translateTextLineMatrix(args[0], args[1]);
+              textState.textMatrix = textState.textLineMatrix.slice();
+              break;
+            case OPS.setLeadingMoveText:
+              flushTextContentItem();
+              textState.leading = -args[1];
+              textState.translateTextLineMatrix(args[0], args[1]);
+              textState.textMatrix = textState.textLineMatrix.slice();
+              break;
+            case OPS.nextLine:
+              flushTextContentItem();
+              textState.carriageReturn();
+              break;
+            case OPS.setTextMatrix:
+              advance = textState.calcTextLineMatrixAdvance(args[0], args[1], args[2], args[3], args[4], args[5]);
+              if (combineTextItems && advance !== null && textContentItem.initialized && advance.value > 0 && advance.value <= textContentItem.fakeMultiSpaceMax) {
+                textState.translateTextLineMatrix(advance.width, advance.height);
+                textContentItem.width += advance.width - textContentItem.lastAdvanceWidth;
+                textContentItem.height += advance.height - textContentItem.lastAdvanceHeight;
+                diff = advance.width - textContentItem.lastAdvanceWidth - (advance.height - textContentItem.lastAdvanceHeight);
+                addFakeSpaces(diff, textContentItem.str);
+                break;
+              }
+              flushTextContentItem();
+              textState.setTextMatrix(args[0], args[1], args[2], args[3], args[4], args[5]);
+              textState.setTextLineMatrix(args[0], args[1], args[2], args[3], args[4], args[5]);
+              break;
+            case OPS.setCharSpacing:
+              textState.charSpacing = args[0];
+              break;
+            case OPS.setWordSpacing:
+              textState.wordSpacing = args[0];
+              break;
+            case OPS.beginText:
+              flushTextContentItem();
+              textState.textMatrix = IDENTITY_MATRIX.slice();
+              textState.textLineMatrix = IDENTITY_MATRIX.slice();
+              break;
+            case OPS.showSpacedText:
+              var items = args[0];
+              var offset;
+              for (var j = 0, jj = items.length; j < jj; j++) {
+                if (typeof items[j] === 'string') {
+                  buildTextContentItem(items[j]);
+                } else if (isNum(items[j])) {
+                  ensureTextContentItem();
+                  advance = items[j] * textState.fontSize / 1000;
+                  var breakTextRun = false;
+                  if (textState.font.vertical) {
+                    offset = advance;
+                    textState.translateTextMatrix(0, offset);
+                    breakTextRun = textContentItem.textRunBreakAllowed && advance > textContentItem.fakeMultiSpaceMax;
+                    if (!breakTextRun) {
+                      textContentItem.height += offset;
+                    }
+                  } else {
+                    advance = -advance;
+                    offset = advance * textState.textHScale;
+                    textState.translateTextMatrix(offset, 0);
+                    breakTextRun = textContentItem.textRunBreakAllowed && advance > textContentItem.fakeMultiSpaceMax;
+                    if (!breakTextRun) {
+                      textContentItem.width += offset;
+                    }
+                  }
+                  if (breakTextRun) {
+                    flushTextContentItem();
+                  } else if (advance > 0) {
+                    addFakeSpaces(advance, textContentItem.str);
+                  }
+                }
+              }
+              break;
+            case OPS.showText:
+              buildTextContentItem(args[0]);
+              break;
+            case OPS.nextLineShowText:
+              flushTextContentItem();
+              textState.carriageReturn();
+              buildTextContentItem(args[0]);
+              break;
+            case OPS.nextLineSetSpacingShowText:
+              flushTextContentItem();
+              textState.wordSpacing = args[0];
+              textState.charSpacing = args[1];
+              textState.carriageReturn();
+              buildTextContentItem(args[2]);
+              break;
+            case OPS.paintXObject:
+              flushTextContentItem();
+              if (args[0].code) {
+                break;
+              }
+              if (!xobjs) {
+                xobjs = resources.get('XObject') || Dict.empty;
+              }
+              var name = args[0].name;
+              if (xobjsCache.key === name) {
+                if (xobjsCache.texts) {
+                  Util.appendToArray(textContent.items, xobjsCache.texts.items);
+                  Util.extendObj(textContent.styles, xobjsCache.texts.styles);
+                }
+                break;
+              }
+              var xobj = xobjs.get(name);
+              if (!xobj) {
+                break;
+              }
+              assert(isStream(xobj), 'XObject should be a stream');
+              var type = xobj.dict.get('Subtype');
+              assert(isName(type), 'XObject should have a Name subtype');
+              if (type.name !== 'Form') {
+                xobjsCache.key = name;
+                xobjsCache.texts = null;
+                break;
+              }
+              stateManager.save();
+              var matrix = xobj.dict.getArray('Matrix');
+              if (isArray(matrix) && matrix.length === 6) {
+                stateManager.transform(matrix);
+              }
+              next(self.getTextContent(xobj, task, xobj.dict.get('Resources') || resources, stateManager, normalizeWhitespace, combineTextItems).then(function (formTextContent) {
+                Util.appendToArray(textContent.items, formTextContent.items);
+                Util.extendObj(textContent.styles, formTextContent.styles);
+                stateManager.restore();
+                xobjsCache.key = name;
+                xobjsCache.texts = formTextContent;
+              }));
+              return;
+            case OPS.setGState:
+              flushTextContentItem();
+              var dictName = args[0];
+              var extGState = resources.get('ExtGState');
+              if (!isDict(extGState) || !isName(dictName)) {
+                break;
+              }
+              var gState = extGState.get(dictName.name);
+              if (!isDict(gState)) {
+                break;
+              }
+              var gStateFont = gState.get('Font');
+              if (gStateFont) {
+                textState.fontName = null;
+                textState.fontSize = gStateFont[1];
+                next(handleSetFont(null, gStateFont[0]));
+                return;
+              }
+              break;
+          }
+        }
+        if (stop) {
+          next(deferred);
+          return;
+        }
+        flushTextContentItem();
+        resolve(textContent);
+      });
+    },
+    extractDataStructures: function PartialEvaluator_extractDataStructures(dict, baseDict, properties) {
+      var xref = this.xref;
+      var toUnicode = dict.get('ToUnicode') || baseDict.get('ToUnicode');
+      var toUnicodePromise = toUnicode ? this.readToUnicode(toUnicode) : Promise.resolve(undefined);
+      if (properties.composite) {
+        var cidSystemInfo = dict.get('CIDSystemInfo');
+        if (isDict(cidSystemInfo)) {
+          properties.cidSystemInfo = {
+            registry: cidSystemInfo.get('Registry'),
+            ordering: cidSystemInfo.get('Ordering'),
+            supplement: cidSystemInfo.get('Supplement')
+          };
+        }
+        var cidToGidMap = dict.get('CIDToGIDMap');
+        if (isStream(cidToGidMap)) {
+          properties.cidToGidMap = this.readCidToGidMap(cidToGidMap);
+        }
+      }
+      var differences = [];
+      var baseEncodingName = null;
+      var encoding;
+      if (dict.has('Encoding')) {
+        encoding = dict.get('Encoding');
+        if (isDict(encoding)) {
+          baseEncodingName = encoding.get('BaseEncoding');
+          baseEncodingName = isName(baseEncodingName) ? baseEncodingName.name : null;
+          if (encoding.has('Differences')) {
+            var diffEncoding = encoding.get('Differences');
+            var index = 0;
+            for (var j = 0, jj = diffEncoding.length; j < jj; j++) {
+              var data = xref.fetchIfRef(diffEncoding[j]);
+              if (isNum(data)) {
+                index = data;
+              } else if (isName(data)) {
+                differences[index++] = data.name;
+              } else {
+                error('Invalid entry in \'Differences\' array: ' + data);
+              }
+            }
+          }
+        } else if (isName(encoding)) {
+          baseEncodingName = encoding.name;
+        } else {
+          error('Encoding is not a Name nor a Dict');
+        }
+        if (baseEncodingName !== 'MacRomanEncoding' && baseEncodingName !== 'MacExpertEncoding' && baseEncodingName !== 'WinAnsiEncoding') {
+          baseEncodingName = null;
+        }
+      }
+      if (baseEncodingName) {
+        properties.defaultEncoding = getEncoding(baseEncodingName).slice();
       } else {
-       var width = xref.fetchIfRef(widths[++i]);
-       for (j = start; j <= code; j++) {
-        glyphsWidths[j] = width;
-       }
+        var isSymbolicFont = !!(properties.flags & FontFlags.Symbolic);
+        var isNonsymbolicFont = !!(properties.flags & FontFlags.Nonsymbolic);
+        encoding = StandardEncoding;
+        if (properties.type === 'TrueType' && !isNonsymbolicFont) {
+          encoding = WinAnsiEncoding;
+        }
+        if (isSymbolicFont) {
+          encoding = MacRomanEncoding;
+          if (!properties.file) {
+            if (/Symbol/i.test(properties.name)) {
+              encoding = SymbolSetEncoding;
+            } else if (/Dingbats/i.test(properties.name)) {
+              encoding = ZapfDingbatsEncoding;
+            }
+          }
+        }
+        properties.defaultEncoding = encoding;
       }
-     }
-    }
-    if (properties.vertical) {
-     var vmetrics = dict.getArray('DW2') || [
-      880,
-      -1000
-     ];
-     defaultVMetrics = [
-      vmetrics[1],
-      defaultWidth * 0.5,
-      vmetrics[0]
-     ];
-     vmetrics = dict.get('W2');
-     if (vmetrics) {
-      for (i = 0, ii = vmetrics.length; i < ii; i++) {
-       start = xref.fetchIfRef(vmetrics[i++]);
-       code = xref.fetchIfRef(vmetrics[i]);
-       if (isArray(code)) {
-        for (j = 0, jj = code.length; j < jj; j++) {
-         glyphsVMetrics[start++] = [
-          xref.fetchIfRef(code[j++]),
-          xref.fetchIfRef(code[j++]),
-          xref.fetchIfRef(code[j])
-         ];
-        }
-       } else {
-        var vmetric = [
-         xref.fetchIfRef(vmetrics[++i]),
-         xref.fetchIfRef(vmetrics[++i]),
-         xref.fetchIfRef(vmetrics[++i])
-        ];
-        for (j = start; j <= code; j++) {
-         glyphsVMetrics[j] = vmetric;
-        }
-       }
-      }
-     }
-    }
-   } else {
-    var firstChar = properties.firstChar;
-    widths = dict.get('Widths');
-    if (widths) {
-     j = firstChar;
-     for (i = 0, ii = widths.length; i < ii; i++) {
-      glyphsWidths[j++] = xref.fetchIfRef(widths[i]);
-     }
-     defaultWidth = parseFloat(descriptor.get('MissingWidth')) || 0;
-    } else {
-     var baseFontName = dict.get('BaseFont');
-     if (isName(baseFontName)) {
-      var metrics = this.getBaseFontMetrics(baseFontName.name);
-      glyphsWidths = this.buildCharCodeToWidth(metrics.widths, properties);
-      defaultWidth = metrics.defaultWidth;
-     }
-    }
-   }
-   var isMonospace = true;
-   var firstWidth = defaultWidth;
-   for (var glyph in glyphsWidths) {
-    var glyphWidth = glyphsWidths[glyph];
-    if (!glyphWidth) {
-     continue;
-    }
-    if (!firstWidth) {
-     firstWidth = glyphWidth;
-     continue;
-    }
-    if (firstWidth !== glyphWidth) {
-     isMonospace = false;
-     break;
-    }
-   }
-   if (isMonospace) {
-    properties.flags |= FontFlags.FixedPitch;
-   }
-   properties.defaultWidth = defaultWidth;
-   properties.widths = glyphsWidths;
-   properties.defaultVMetrics = defaultVMetrics;
-   properties.vmetrics = glyphsVMetrics;
-  },
-  isSerifFont: function PartialEvaluator_isSerifFont(baseFontName) {
-   var fontNameWoStyle = baseFontName.split('-')[0];
-   return fontNameWoStyle in getSerifFonts() || fontNameWoStyle.search(/serif/gi) !== -1;
-  },
-  getBaseFontMetrics: function PartialEvaluator_getBaseFontMetrics(name) {
-   var defaultWidth = 0;
-   var widths = [];
-   var monospace = false;
-   var stdFontMap = getStdFontMap();
-   var lookupName = stdFontMap[name] || name;
-   var Metrics = getMetrics();
-   if (!(lookupName in Metrics)) {
-    if (this.isSerifFont(name)) {
-     lookupName = 'Times-Roman';
-    } else {
-     lookupName = 'Helvetica';
-    }
-   }
-   var glyphWidths = Metrics[lookupName];
-   if (isNum(glyphWidths)) {
-    defaultWidth = glyphWidths;
-    monospace = true;
-   } else {
-    widths = glyphWidths();
-   }
-   return {
-    defaultWidth: defaultWidth,
-    monospace: monospace,
-    widths: widths
-   };
-  },
-  buildCharCodeToWidth: function PartialEvaluator_bulildCharCodeToWidth(widthsByGlyphName, properties) {
-   var widths = Object.create(null);
-   var differences = properties.differences;
-   var encoding = properties.defaultEncoding;
-   for (var charCode = 0; charCode < 256; charCode++) {
-    if (charCode in differences && widthsByGlyphName[differences[charCode]]) {
-     widths[charCode] = widthsByGlyphName[differences[charCode]];
-     continue;
-    }
-    if (charCode in encoding && widthsByGlyphName[encoding[charCode]]) {
-     widths[charCode] = widthsByGlyphName[encoding[charCode]];
-     continue;
-    }
-   }
-   return widths;
-  },
-  preEvaluateFont: function PartialEvaluator_preEvaluateFont(dict) {
-   var baseDict = dict;
-   var type = dict.get('Subtype');
-   assert(isName(type), 'invalid font Subtype');
-   var composite = false;
-   var uint8array;
-   if (type.name === 'Type0') {
-    var df = dict.get('DescendantFonts');
-    assert(df, 'Descendant fonts are not specified');
-    dict = isArray(df) ? this.xref.fetchIfRef(df[0]) : df;
-    type = dict.get('Subtype');
-    assert(isName(type), 'invalid font Subtype');
-    composite = true;
-   }
-   var descriptor = dict.get('FontDescriptor');
-   if (descriptor) {
-    var hash = new MurmurHash3_64();
-    var encoding = baseDict.getRaw('Encoding');
-    if (isName(encoding)) {
-     hash.update(encoding.name);
-    } else if (isRef(encoding)) {
-     hash.update(encoding.toString());
-    } else if (isDict(encoding)) {
-     var keys = encoding.getKeys();
-     for (var i = 0, ii = keys.length; i < ii; i++) {
-      var entry = encoding.getRaw(keys[i]);
-      if (isName(entry)) {
-       hash.update(entry.name);
-      } else if (isRef(entry)) {
-       hash.update(entry.toString());
-      } else if (isArray(entry)) {
-       var diffLength = entry.length, diffBuf = new Array(diffLength);
-       for (var j = 0; j < diffLength; j++) {
-        var diffEntry = entry[j];
-        if (isName(diffEntry)) {
-         diffBuf[j] = diffEntry.name;
-        } else if (isNum(diffEntry) || isRef(diffEntry)) {
-         diffBuf[j] = diffEntry.toString();
-        }
-       }
-       hash.update(diffBuf.join());
-      }
-     }
-    }
-    var toUnicode = dict.get('ToUnicode') || baseDict.get('ToUnicode');
-    if (isStream(toUnicode)) {
-     var stream = toUnicode.str || toUnicode;
-     uint8array = stream.buffer ? new Uint8Array(stream.buffer.buffer, 0, stream.bufferLength) : new Uint8Array(stream.bytes.buffer, stream.start, stream.end - stream.start);
-     hash.update(uint8array);
-    } else if (isName(toUnicode)) {
-     hash.update(toUnicode.name);
-    }
-    var widths = dict.get('Widths') || baseDict.get('Widths');
-    if (widths) {
-     uint8array = new Uint8Array(new Uint32Array(widths).buffer);
-     hash.update(uint8array);
-    }
-   }
-   return {
-    descriptor: descriptor,
-    dict: dict,
-    baseDict: baseDict,
-    composite: composite,
-    type: type.name,
-    hash: hash ? hash.hexdigest() : ''
-   };
-  },
-  translateFont: function PartialEvaluator_translateFont(preEvaluatedFont) {
-   var baseDict = preEvaluatedFont.baseDict;
-   var dict = preEvaluatedFont.dict;
-   var composite = preEvaluatedFont.composite;
-   var descriptor = preEvaluatedFont.descriptor;
-   var type = preEvaluatedFont.type;
-   var maxCharIndex = composite ? 0xFFFF : 0xFF;
-   var properties;
-   if (!descriptor) {
-    if (type === 'Type3') {
-     descriptor = new Dict(null);
-     descriptor.set('FontName', Name.get(type));
-     descriptor.set('FontBBox', dict.getArray('FontBBox'));
-    } else {
-     var baseFontName = dict.get('BaseFont');
-     assert(isName(baseFontName), 'Base font is not specified');
-     baseFontName = baseFontName.name.replace(/[,_]/g, '-');
-     var metrics = this.getBaseFontMetrics(baseFontName);
-     var fontNameWoStyle = baseFontName.split('-')[0];
-     var flags = (this.isSerifFont(fontNameWoStyle) ? FontFlags.Serif : 0) | (metrics.monospace ? FontFlags.FixedPitch : 0) | (getSymbolsFonts()[fontNameWoStyle] ? FontFlags.Symbolic : FontFlags.Nonsymbolic);
-     properties = {
-      type: type,
-      name: baseFontName,
-      widths: metrics.widths,
-      defaultWidth: metrics.defaultWidth,
-      flags: flags,
-      firstChar: 0,
-      lastChar: maxCharIndex
-     };
-     return this.extractDataStructures(dict, dict, properties).then(function (properties) {
-      properties.widths = this.buildCharCodeToWidth(metrics.widths, properties);
-      return new Font(baseFontName, null, properties);
-     }.bind(this));
-    }
-   }
-   var firstChar = dict.get('FirstChar') || 0;
-   var lastChar = dict.get('LastChar') || maxCharIndex;
-   var fontName = descriptor.get('FontName');
-   var baseFont = dict.get('BaseFont');
-   if (isString(fontName)) {
-    fontName = Name.get(fontName);
-   }
-   if (isString(baseFont)) {
-    baseFont = Name.get(baseFont);
-   }
-   if (type !== 'Type3') {
-    var fontNameStr = fontName && fontName.name;
-    var baseFontStr = baseFont && baseFont.name;
-    if (fontNameStr !== baseFontStr) {
-     info('The FontDescriptor\'s FontName is "' + fontNameStr + '" but should be the same as the Font\'s BaseFont "' + baseFontStr + '"');
-     if (fontNameStr && baseFontStr && baseFontStr.indexOf(fontNameStr) === 0) {
-      fontName = baseFont;
-     }
-    }
-   }
-   fontName = fontName || baseFont;
-   assert(isName(fontName), 'invalid font name');
-   var fontFile = descriptor.get('FontFile', 'FontFile2', 'FontFile3');
-   if (fontFile) {
-    if (fontFile.dict) {
-     var subtype = fontFile.dict.get('Subtype');
-     if (subtype) {
-      subtype = subtype.name;
-     }
-     var length1 = fontFile.dict.get('Length1');
-     var length2 = fontFile.dict.get('Length2');
-     var length3 = fontFile.dict.get('Length3');
-    }
-   }
-   properties = {
-    type: type,
-    name: fontName.name,
-    subtype: subtype,
-    file: fontFile,
-    length1: length1,
-    length2: length2,
-    length3: length3,
-    loadedName: baseDict.loadedName,
-    composite: composite,
-    wideChars: composite,
-    fixedPitch: false,
-    fontMatrix: dict.getArray('FontMatrix') || FONT_IDENTITY_MATRIX,
-    firstChar: firstChar || 0,
-    lastChar: lastChar || maxCharIndex,
-    bbox: descriptor.getArray('FontBBox'),
-    ascent: descriptor.get('Ascent'),
-    descent: descriptor.get('Descent'),
-    xHeight: descriptor.get('XHeight'),
-    capHeight: descriptor.get('CapHeight'),
-    flags: descriptor.get('Flags'),
-    italicAngle: descriptor.get('ItalicAngle'),
-    coded: false
-   };
-   var cMapPromise;
-   if (composite) {
-    var cidEncoding = baseDict.get('Encoding');
-    if (isName(cidEncoding)) {
-     properties.cidEncoding = cidEncoding.name;
-    }
-    cMapPromise = CMapFactory.create({
-     encoding: cidEncoding,
-     fetchBuiltInCMap: this.fetchBuiltInCMap,
-     useCMap: null
-    }).then(function (cMap) {
-     properties.cMap = cMap;
-     properties.vertical = properties.cMap.vertical;
-    });
-   } else {
-    cMapPromise = Promise.resolve(undefined);
-   }
-   return cMapPromise.then(function () {
-    return this.extractDataStructures(dict, baseDict, properties);
-   }.bind(this)).then(function (properties) {
-    this.extractWidths(dict, descriptor, properties);
-    if (type === 'Type3') {
-     properties.isType3Font = true;
+      properties.differences = differences;
+      properties.baseEncodingName = baseEncodingName;
+      properties.hasEncoding = !!baseEncodingName || differences.length > 0;
+      properties.dict = dict;
+      return toUnicodePromise.then(function (toUnicode) {
+        properties.toUnicode = toUnicode;
+        return this.buildToUnicode(properties);
+      }.bind(this)).then(function (toUnicode) {
+        properties.toUnicode = toUnicode;
+        return properties;
+      });
+    },
+    buildToUnicode: function PartialEvaluator_buildToUnicode(properties) {
+      properties.hasIncludedToUnicodeMap = !!properties.toUnicode && properties.toUnicode.length > 0;
+      if (properties.hasIncludedToUnicodeMap) {
+        return Promise.resolve(properties.toUnicode);
+      }
+      var toUnicode, charcode, glyphName;
+      if (!properties.composite) {
+        toUnicode = [];
+        var encoding = properties.defaultEncoding.slice();
+        var baseEncodingName = properties.baseEncodingName;
+        var differences = properties.differences;
+        for (charcode in differences) {
+          glyphName = differences[charcode];
+          if (glyphName === '.notdef') {
+            continue;
+          }
+          encoding[charcode] = glyphName;
+        }
+        var glyphsUnicodeMap = getGlyphsUnicode();
+        for (charcode in encoding) {
+          glyphName = encoding[charcode];
+          if (glyphName === '') {
+            continue;
+          } else if (glyphsUnicodeMap[glyphName] === undefined) {
+            var code = 0;
+            switch (glyphName[0]) {
+              case 'G':
+                if (glyphName.length === 3) {
+                  code = parseInt(glyphName.substr(1), 16);
+                }
+                break;
+              case 'g':
+                if (glyphName.length === 5) {
+                  code = parseInt(glyphName.substr(1), 16);
+                }
+                break;
+              case 'C':
+              case 'c':
+                if (glyphName.length >= 3) {
+                  code = +glyphName.substr(1);
+                }
+                break;
+              default:
+                var unicode = getUnicodeForGlyph(glyphName, glyphsUnicodeMap);
+                if (unicode !== -1) {
+                  code = unicode;
+                }
+            }
+            if (code) {
+              if (baseEncodingName && code === +charcode) {
+                var baseEncoding = getEncoding(baseEncodingName);
+                if (baseEncoding && (glyphName = baseEncoding[charcode])) {
+                  toUnicode[charcode] = String.fromCharCode(glyphsUnicodeMap[glyphName]);
+                  continue;
+                }
+              }
+              toUnicode[charcode] = String.fromCharCode(code);
+            }
+            continue;
+          }
+          toUnicode[charcode] = String.fromCharCode(glyphsUnicodeMap[glyphName]);
+        }
+        return Promise.resolve(new ToUnicodeMap(toUnicode));
+      }
+      if (properties.composite && (properties.cMap.builtInCMap && !(properties.cMap instanceof IdentityCMap) || properties.cidSystemInfo.registry === 'Adobe' && (properties.cidSystemInfo.ordering === 'GB1' || properties.cidSystemInfo.ordering === 'CNS1' || properties.cidSystemInfo.ordering === 'Japan1' || properties.cidSystemInfo.ordering === 'Korea1'))) {
+        var registry = properties.cidSystemInfo.registry;
+        var ordering = properties.cidSystemInfo.ordering;
+        var ucs2CMapName = Name.get(registry + '-' + ordering + '-UCS2');
+        return CMapFactory.create({
+          encoding: ucs2CMapName,
+          fetchBuiltInCMap: this.fetchBuiltInCMap,
+          useCMap: null
+        }).then(function (ucs2CMap) {
+          var cMap = properties.cMap;
+          toUnicode = [];
+          cMap.forEach(function (charcode, cid) {
+            assert(cid <= 0xffff, 'Max size of CID is 65,535');
+            var ucs2 = ucs2CMap.lookup(cid);
+            if (ucs2) {
+              toUnicode[charcode] = String.fromCharCode((ucs2.charCodeAt(0) << 8) + ucs2.charCodeAt(1));
+            }
+          });
+          return new ToUnicodeMap(toUnicode);
+        });
+      }
+      return Promise.resolve(new IdentityToUnicodeMap(properties.firstChar, properties.lastChar));
+    },
+    readToUnicode: function PartialEvaluator_readToUnicode(toUnicode) {
+      var cmapObj = toUnicode;
+      if (isName(cmapObj)) {
+        return CMapFactory.create({
+          encoding: cmapObj,
+          fetchBuiltInCMap: this.fetchBuiltInCMap,
+          useCMap: null
+        }).then(function (cmap) {
+          if (cmap instanceof IdentityCMap) {
+            return new IdentityToUnicodeMap(0, 0xFFFF);
+          }
+          return new ToUnicodeMap(cmap.getMap());
+        });
+      } else if (isStream(cmapObj)) {
+        return CMapFactory.create({
+          encoding: cmapObj,
+          fetchBuiltInCMap: this.fetchBuiltInCMap,
+          useCMap: null
+        }).then(function (cmap) {
+          if (cmap instanceof IdentityCMap) {
+            return new IdentityToUnicodeMap(0, 0xFFFF);
+          }
+          var map = new Array(cmap.length);
+          cmap.forEach(function (charCode, token) {
+            var str = [];
+            for (var k = 0; k < token.length; k += 2) {
+              var w1 = token.charCodeAt(k) << 8 | token.charCodeAt(k + 1);
+              if ((w1 & 0xF800) !== 0xD800) {
+                str.push(w1);
+                continue;
+              }
+              k += 2;
+              var w2 = token.charCodeAt(k) << 8 | token.charCodeAt(k + 1);
+              str.push(((w1 & 0x3ff) << 10) + (w2 & 0x3ff) + 0x10000);
+            }
+            map[charCode] = String.fromCharCode.apply(String, str);
+          });
+          return new ToUnicodeMap(map);
+        });
+      }
+      return Promise.resolve(null);
+    },
+    readCidToGidMap: function PartialEvaluator_readCidToGidMap(cidToGidStream) {
+      var glyphsData = cidToGidStream.getBytes();
+      var result = [];
+      for (var j = 0, jj = glyphsData.length; j < jj; j++) {
+        var glyphID = glyphsData[j++] << 8 | glyphsData[j];
+        if (glyphID === 0) {
+          continue;
+        }
+        var code = j >> 1;
+        result[code] = glyphID;
+      }
+      return result;
+    },
+    extractWidths: function PartialEvaluator_extractWidths(dict, descriptor, properties) {
+      var xref = this.xref;
+      var glyphsWidths = [];
+      var defaultWidth = 0;
+      var glyphsVMetrics = [];
+      var defaultVMetrics;
+      var i, ii, j, jj, start, code, widths;
+      if (properties.composite) {
+        defaultWidth = dict.get('DW') || 1000;
+        widths = dict.get('W');
+        if (widths) {
+          for (i = 0, ii = widths.length; i < ii; i++) {
+            start = xref.fetchIfRef(widths[i++]);
+            code = xref.fetchIfRef(widths[i]);
+            if (isArray(code)) {
+              for (j = 0, jj = code.length; j < jj; j++) {
+                glyphsWidths[start++] = xref.fetchIfRef(code[j]);
+              }
+            } else {
+              var width = xref.fetchIfRef(widths[++i]);
+              for (j = start; j <= code; j++) {
+                glyphsWidths[j] = width;
+              }
+            }
+          }
+        }
+        if (properties.vertical) {
+          var vmetrics = dict.getArray('DW2') || [880, -1000];
+          defaultVMetrics = [vmetrics[1], defaultWidth * 0.5, vmetrics[0]];
+          vmetrics = dict.get('W2');
+          if (vmetrics) {
+            for (i = 0, ii = vmetrics.length; i < ii; i++) {
+              start = xref.fetchIfRef(vmetrics[i++]);
+              code = xref.fetchIfRef(vmetrics[i]);
+              if (isArray(code)) {
+                for (j = 0, jj = code.length; j < jj; j++) {
+                  glyphsVMetrics[start++] = [xref.fetchIfRef(code[j++]), xref.fetchIfRef(code[j++]), xref.fetchIfRef(code[j])];
+                }
+              } else {
+                var vmetric = [xref.fetchIfRef(vmetrics[++i]), xref.fetchIfRef(vmetrics[++i]), xref.fetchIfRef(vmetrics[++i])];
+                for (j = start; j <= code; j++) {
+                  glyphsVMetrics[j] = vmetric;
+                }
+              }
+            }
+          }
+        }
+      } else {
+        var firstChar = properties.firstChar;
+        widths = dict.get('Widths');
+        if (widths) {
+          j = firstChar;
+          for (i = 0, ii = widths.length; i < ii; i++) {
+            glyphsWidths[j++] = xref.fetchIfRef(widths[i]);
+          }
+          defaultWidth = parseFloat(descriptor.get('MissingWidth')) || 0;
+        } else {
+          var baseFontName = dict.get('BaseFont');
+          if (isName(baseFontName)) {
+            var metrics = this.getBaseFontMetrics(baseFontName.name);
+            glyphsWidths = this.buildCharCodeToWidth(metrics.widths, properties);
+            defaultWidth = metrics.defaultWidth;
+          }
+        }
+      }
+      var isMonospace = true;
+      var firstWidth = defaultWidth;
+      for (var glyph in glyphsWidths) {
+        var glyphWidth = glyphsWidths[glyph];
+        if (!glyphWidth) {
+          continue;
+        }
+        if (!firstWidth) {
+          firstWidth = glyphWidth;
+          continue;
+        }
+        if (firstWidth !== glyphWidth) {
+          isMonospace = false;
+          break;
+        }
+      }
+      if (isMonospace) {
+        properties.flags |= FontFlags.FixedPitch;
+      }
+      properties.defaultWidth = defaultWidth;
+      properties.widths = glyphsWidths;
+      properties.defaultVMetrics = defaultVMetrics;
+      properties.vmetrics = glyphsVMetrics;
+    },
+    isSerifFont: function PartialEvaluator_isSerifFont(baseFontName) {
+      var fontNameWoStyle = baseFontName.split('-')[0];
+      return fontNameWoStyle in getSerifFonts() || fontNameWoStyle.search(/serif/gi) !== -1;
+    },
+    getBaseFontMetrics: function PartialEvaluator_getBaseFontMetrics(name) {
+      var defaultWidth = 0;
+      var widths = [];
+      var monospace = false;
+      var stdFontMap = getStdFontMap();
+      var lookupName = stdFontMap[name] || name;
+      var Metrics = getMetrics();
+      if (!(lookupName in Metrics)) {
+        if (this.isSerifFont(name)) {
+          lookupName = 'Times-Roman';
+        } else {
+          lookupName = 'Helvetica';
+        }
+      }
+      var glyphWidths = Metrics[lookupName];
+      if (isNum(glyphWidths)) {
+        defaultWidth = glyphWidths;
+        monospace = true;
+      } else {
+        widths = glyphWidths();
+      }
+      return {
+        defaultWidth: defaultWidth,
+        monospace: monospace,
+        widths: widths
+      };
+    },
+    buildCharCodeToWidth: function PartialEvaluator_bulildCharCodeToWidth(widthsByGlyphName, properties) {
+      var widths = Object.create(null);
+      var differences = properties.differences;
+      var encoding = properties.defaultEncoding;
+      for (var charCode = 0; charCode < 256; charCode++) {
+        if (charCode in differences && widthsByGlyphName[differences[charCode]]) {
+          widths[charCode] = widthsByGlyphName[differences[charCode]];
+          continue;
+        }
+        if (charCode in encoding && widthsByGlyphName[encoding[charCode]]) {
+          widths[charCode] = widthsByGlyphName[encoding[charCode]];
+          continue;
+        }
+      }
+      return widths;
+    },
+    preEvaluateFont: function PartialEvaluator_preEvaluateFont(dict) {
+      var baseDict = dict;
+      var type = dict.get('Subtype');
+      assert(isName(type), 'invalid font Subtype');
+      var composite = false;
+      var uint8array;
+      if (type.name === 'Type0') {
+        var df = dict.get('DescendantFonts');
+        assert(df, 'Descendant fonts are not specified');
+        dict = isArray(df) ? this.xref.fetchIfRef(df[0]) : df;
+        type = dict.get('Subtype');
+        assert(isName(type), 'invalid font Subtype');
+        composite = true;
+      }
+      var descriptor = dict.get('FontDescriptor');
+      if (descriptor) {
+        var hash = new MurmurHash3_64();
+        var encoding = baseDict.getRaw('Encoding');
+        if (isName(encoding)) {
+          hash.update(encoding.name);
+        } else if (isRef(encoding)) {
+          hash.update(encoding.toString());
+        } else if (isDict(encoding)) {
+          var keys = encoding.getKeys();
+          for (var i = 0, ii = keys.length; i < ii; i++) {
+            var entry = encoding.getRaw(keys[i]);
+            if (isName(entry)) {
+              hash.update(entry.name);
+            } else if (isRef(entry)) {
+              hash.update(entry.toString());
+            } else if (isArray(entry)) {
+              var diffLength = entry.length,
+                  diffBuf = new Array(diffLength);
+              for (var j = 0; j < diffLength; j++) {
+                var diffEntry = entry[j];
+                if (isName(diffEntry)) {
+                  diffBuf[j] = diffEntry.name;
+                } else if (isNum(diffEntry) || isRef(diffEntry)) {
+                  diffBuf[j] = diffEntry.toString();
+                }
+              }
+              hash.update(diffBuf.join());
+            }
+          }
+        }
+        var toUnicode = dict.get('ToUnicode') || baseDict.get('ToUnicode');
+        if (isStream(toUnicode)) {
+          var stream = toUnicode.str || toUnicode;
+          uint8array = stream.buffer ? new Uint8Array(stream.buffer.buffer, 0, stream.bufferLength) : new Uint8Array(stream.bytes.buffer, stream.start, stream.end - stream.start);
+          hash.update(uint8array);
+        } else if (isName(toUnicode)) {
+          hash.update(toUnicode.name);
+        }
+        var widths = dict.get('Widths') || baseDict.get('Widths');
+        if (widths) {
+          uint8array = new Uint8Array(new Uint32Array(widths).buffer);
+          hash.update(uint8array);
+        }
+      }
+      return {
+        descriptor: descriptor,
+        dict: dict,
+        baseDict: baseDict,
+        composite: composite,
+        type: type.name,
+        hash: hash ? hash.hexdigest() : ''
+      };
+    },
+    translateFont: function PartialEvaluator_translateFont(preEvaluatedFont) {
+      var baseDict = preEvaluatedFont.baseDict;
+      var dict = preEvaluatedFont.dict;
+      var composite = preEvaluatedFont.composite;
+      var descriptor = preEvaluatedFont.descriptor;
+      var type = preEvaluatedFont.type;
+      var maxCharIndex = composite ? 0xFFFF : 0xFF;
+      var properties;
+      if (!descriptor) {
+        if (type === 'Type3') {
+          descriptor = new Dict(null);
+          descriptor.set('FontName', Name.get(type));
+          descriptor.set('FontBBox', dict.getArray('FontBBox'));
+        } else {
+          var baseFontName = dict.get('BaseFont');
+          assert(isName(baseFontName), 'Base font is not specified');
+          baseFontName = baseFontName.name.replace(/[,_]/g, '-');
+          var metrics = this.getBaseFontMetrics(baseFontName);
+          var fontNameWoStyle = baseFontName.split('-')[0];
+          var flags = (this.isSerifFont(fontNameWoStyle) ? FontFlags.Serif : 0) | (metrics.monospace ? FontFlags.FixedPitch : 0) | (getSymbolsFonts()[fontNameWoStyle] ? FontFlags.Symbolic : FontFlags.Nonsymbolic);
+          properties = {
+            type: type,
+            name: baseFontName,
+            widths: metrics.widths,
+            defaultWidth: metrics.defaultWidth,
+            flags: flags,
+            firstChar: 0,
+            lastChar: maxCharIndex
+          };
+          return this.extractDataStructures(dict, dict, properties).then(function (properties) {
+            properties.widths = this.buildCharCodeToWidth(metrics.widths, properties);
+            return new Font(baseFontName, null, properties);
+          }.bind(this));
+        }
+      }
+      var firstChar = dict.get('FirstChar') || 0;
+      var lastChar = dict.get('LastChar') || maxCharIndex;
+      var fontName = descriptor.get('FontName');
+      var baseFont = dict.get('BaseFont');
+      if (isString(fontName)) {
+        fontName = Name.get(fontName);
+      }
+      if (isString(baseFont)) {
+        baseFont = Name.get(baseFont);
+      }
+      if (type !== 'Type3') {
+        var fontNameStr = fontName && fontName.name;
+        var baseFontStr = baseFont && baseFont.name;
+        if (fontNameStr !== baseFontStr) {
+          info('The FontDescriptor\'s FontName is "' + fontNameStr + '" but should be the same as the Font\'s BaseFont "' + baseFontStr + '"');
+          if (fontNameStr && baseFontStr && baseFontStr.indexOf(fontNameStr) === 0) {
+            fontName = baseFont;
+          }
+        }
+      }
+      fontName = fontName || baseFont;
+      assert(isName(fontName), 'invalid font name');
+      var fontFile = descriptor.get('FontFile', 'FontFile2', 'FontFile3');
+      if (fontFile) {
+        if (fontFile.dict) {
+          var subtype = fontFile.dict.get('Subtype');
+          if (subtype) {
+            subtype = subtype.name;
+          }
+          var length1 = fontFile.dict.get('Length1');
+          var length2 = fontFile.dict.get('Length2');
+          var length3 = fontFile.dict.get('Length3');
+        }
+      }
+      properties = {
+        type: type,
+        name: fontName.name,
+        subtype: subtype,
+        file: fontFile,
+        length1: length1,
+        length2: length2,
+        length3: length3,
+        loadedName: baseDict.loadedName,
+        composite: composite,
+        wideChars: composite,
+        fixedPitch: false,
+        fontMatrix: dict.getArray('FontMatrix') || FONT_IDENTITY_MATRIX,
+        firstChar: firstChar || 0,
+        lastChar: lastChar || maxCharIndex,
+        bbox: descriptor.getArray('FontBBox'),
+        ascent: descriptor.get('Ascent'),
+        descent: descriptor.get('Descent'),
+        xHeight: descriptor.get('XHeight'),
+        capHeight: descriptor.get('CapHeight'),
+        flags: descriptor.get('Flags'),
+        italicAngle: descriptor.get('ItalicAngle'),
+        coded: false
+      };
+      var cMapPromise;
+      if (composite) {
+        var cidEncoding = baseDict.get('Encoding');
+        if (isName(cidEncoding)) {
+          properties.cidEncoding = cidEncoding.name;
+        }
+        cMapPromise = CMapFactory.create({
+          encoding: cidEncoding,
+          fetchBuiltInCMap: this.fetchBuiltInCMap,
+          useCMap: null
+        }).then(function (cMap) {
+          properties.cMap = cMap;
+          properties.vertical = properties.cMap.vertical;
+        });
+      } else {
+        cMapPromise = Promise.resolve(undefined);
+      }
+      return cMapPromise.then(function () {
+        return this.extractDataStructures(dict, baseDict, properties);
+      }.bind(this)).then(function (properties) {
+        this.extractWidths(dict, descriptor, properties);
+        if (type === 'Type3') {
+          properties.isType3Font = true;
+        }
+        return new Font(fontName.name, fontFile, properties);
+      }.bind(this));
     }
-    return new Font(fontName.name, fontFile, properties);
-   }.bind(this));
-  }
- };
- return PartialEvaluator;
+  };
+  return PartialEvaluator;
 }();
 var TranslatedFont = function TranslatedFontClosure() {
- function TranslatedFont(loadedName, font, dict) {
-  this.loadedName = loadedName;
-  this.font = font;
-  this.dict = dict;
-  this.type3Loaded = null;
-  this.sent = false;
- }
- TranslatedFont.prototype = {
-  send: function (handler) {
-   if (this.sent) {
-    return;
-   }
-   var fontData = this.font.exportData();
-   handler.send('commonobj', [
-    this.loadedName,
-    'Font',
-    fontData
-   ]);
-   this.sent = true;
-  },
-  loadType3Data: function (evaluator, resources, parentOperatorList, task) {
-   assert(this.font.isType3Font);
-   if (this.type3Loaded) {
-    return this.type3Loaded;
-   }
-   var translatedFont = this.font;
-   var loadCharProcsPromise = Promise.resolve();
-   var charProcs = this.dict.get('CharProcs');
-   var fontResources = this.dict.get('Resources') || resources;
-   var charProcKeys = charProcs.getKeys();
-   var charProcOperatorList = Object.create(null);
-   for (var i = 0, n = charProcKeys.length; i < n; ++i) {
-    loadCharProcsPromise = loadCharProcsPromise.then(function (key) {
-     var glyphStream = charProcs.get(key);
-     var operatorList = new OperatorList();
-     return evaluator.getOperatorList(glyphStream, task, fontResources, operatorList).then(function () {
-      charProcOperatorList[key] = operatorList.getIR();
-      parentOperatorList.addDependencies(operatorList.dependencies);
-     }, function (reason) {
-      warn('Type3 font resource \"' + key + '\" is not available');
-      var operatorList = new OperatorList();
-      charProcOperatorList[key] = operatorList.getIR();
-     });
-    }.bind(this, charProcKeys[i]));
-   }
-   this.type3Loaded = loadCharProcsPromise.then(function () {
-    translatedFont.charProcOperatorList = charProcOperatorList;
-   });
-   return this.type3Loaded;
+  function TranslatedFont(loadedName, font, dict) {
+    this.loadedName = loadedName;
+    this.font = font;
+    this.dict = dict;
+    this.type3Loaded = null;
+    this.sent = false;
   }
- };
- return TranslatedFont;
+  TranslatedFont.prototype = {
+    send: function (handler) {
+      if (this.sent) {
+        return;
+      }
+      var fontData = this.font.exportData();
+      handler.send('commonobj', [this.loadedName, 'Font', fontData]);
+      this.sent = true;
+    },
+    loadType3Data: function (evaluator, resources, parentOperatorList, task) {
+      assert(this.font.isType3Font);
+      if (this.type3Loaded) {
+        return this.type3Loaded;
+      }
+      var translatedFont = this.font;
+      var loadCharProcsPromise = Promise.resolve();
+      var charProcs = this.dict.get('CharProcs');
+      var fontResources = this.dict.get('Resources') || resources;
+      var charProcKeys = charProcs.getKeys();
+      var charProcOperatorList = Object.create(null);
+      for (var i = 0, n = charProcKeys.length; i < n; ++i) {
+        loadCharProcsPromise = loadCharProcsPromise.then(function (key) {
+          var glyphStream = charProcs.get(key);
+          var operatorList = new OperatorList();
+          return evaluator.getOperatorList(glyphStream, task, fontResources, operatorList).then(function () {
+            charProcOperatorList[key] = operatorList.getIR();
+            parentOperatorList.addDependencies(operatorList.dependencies);
+          }, function (reason) {
+            warn('Type3 font resource \"' + key + '\" is not available');
+            var operatorList = new OperatorList();
+            charProcOperatorList[key] = operatorList.getIR();
+          });
+        }.bind(this, charProcKeys[i]));
+      }
+      this.type3Loaded = loadCharProcsPromise.then(function () {
+        translatedFont.charProcOperatorList = charProcOperatorList;
+      });
+      return this.type3Loaded;
+    }
+  };
+  return TranslatedFont;
 }();
 var OperatorList = function OperatorListClosure() {
- var CHUNK_SIZE = 1000;
- var CHUNK_SIZE_ABOUT = CHUNK_SIZE - 5;
- function getTransfers(queue) {
-  var transfers = [];
-  var fnArray = queue.fnArray, argsArray = queue.argsArray;
-  for (var i = 0, ii = queue.length; i < ii; i++) {
-   switch (fnArray[i]) {
-   case OPS.paintInlineImageXObject:
-   case OPS.paintInlineImageXObjectGroup:
-   case OPS.paintImageMaskXObject:
-    var arg = argsArray[i][0];
-    if (!arg.cached) {
-     transfers.push(arg.data.buffer);
+  var CHUNK_SIZE = 1000;
+  var CHUNK_SIZE_ABOUT = CHUNK_SIZE - 5;
+  function getTransfers(queue) {
+    var transfers = [];
+    var fnArray = queue.fnArray,
+        argsArray = queue.argsArray;
+    for (var i = 0, ii = queue.length; i < ii; i++) {
+      switch (fnArray[i]) {
+        case OPS.paintInlineImageXObject:
+        case OPS.paintInlineImageXObjectGroup:
+        case OPS.paintImageMaskXObject:
+          var arg = argsArray[i][0];
+          if (!arg.cached) {
+            transfers.push(arg.data.buffer);
+          }
+          break;
+      }
     }
-    break;
-   }
+    return transfers;
   }
-  return transfers;
- }
- function OperatorList(intent, messageHandler, pageIndex) {
-  this.messageHandler = messageHandler;
-  this.fnArray = [];
-  this.argsArray = [];
-  this.dependencies = Object.create(null);
-  this._totalLength = 0;
-  this.pageIndex = pageIndex;
-  this.intent = intent;
- }
- OperatorList.prototype = {
-  get length() {
-   return this.argsArray.length;
-  },
-  get totalLength() {
-   return this._totalLength + this.length;
-  },
-  addOp: function (fn, args) {
-   this.fnArray.push(fn);
-   this.argsArray.push(args);
-   if (this.messageHandler) {
-    if (this.fnArray.length >= CHUNK_SIZE) {
-     this.flush();
-    } else if (this.fnArray.length >= CHUNK_SIZE_ABOUT && (fn === OPS.restore || fn === OPS.endText)) {
-     this.flush();
-    }
-   }
-  },
-  addDependency: function (dependency) {
-   if (dependency in this.dependencies) {
-    return;
-   }
-   this.dependencies[dependency] = true;
-   this.addOp(OPS.dependency, [dependency]);
-  },
-  addDependencies: function (dependencies) {
-   for (var key in dependencies) {
-    this.addDependency(key);
-   }
-  },
-  addOpList: function (opList) {
-   Util.extendObj(this.dependencies, opList.dependencies);
-   for (var i = 0, ii = opList.length; i < ii; i++) {
-    this.addOp(opList.fnArray[i], opList.argsArray[i]);
-   }
-  },
-  getIR: function () {
-   return {
-    fnArray: this.fnArray,
-    argsArray: this.argsArray,
-    length: this.length
-   };
-  },
-  flush: function (lastChunk) {
-   if (this.intent !== 'oplist') {
-    new QueueOptimizer().optimize(this);
-   }
-   var transfers = getTransfers(this);
-   var length = this.length;
-   this._totalLength += length;
-   this.messageHandler.send('RenderPageChunk', {
-    operatorList: {
-     fnArray: this.fnArray,
-     argsArray: this.argsArray,
-     lastChunk: lastChunk,
-     length: length
-    },
-    pageIndex: this.pageIndex,
-    intent: this.intent
-   }, transfers);
-   this.dependencies = Object.create(null);
-   this.fnArray.length = 0;
-   this.argsArray.length = 0;
+  function OperatorList(intent, messageHandler, pageIndex) {
+    this.messageHandler = messageHandler;
+    this.fnArray = [];
+    this.argsArray = [];
+    this.dependencies = Object.create(null);
+    this._totalLength = 0;
+    this.pageIndex = pageIndex;
+    this.intent = intent;
   }
- };
- return OperatorList;
+  OperatorList.prototype = {
+    get length() {
+      return this.argsArray.length;
+    },
+    get totalLength() {
+      return this._totalLength + this.length;
+    },
+    addOp: function (fn, args) {
+      this.fnArray.push(fn);
+      this.argsArray.push(args);
+      if (this.messageHandler) {
+        if (this.fnArray.length >= CHUNK_SIZE) {
+          this.flush();
+        } else if (this.fnArray.length >= CHUNK_SIZE_ABOUT && (fn === OPS.restore || fn === OPS.endText)) {
+          this.flush();
+        }
+      }
+    },
+    addDependency: function (dependency) {
+      if (dependency in this.dependencies) {
+        return;
+      }
+      this.dependencies[dependency] = true;
+      this.addOp(OPS.dependency, [dependency]);
+    },
+    addDependencies: function (dependencies) {
+      for (var key in dependencies) {
+        this.addDependency(key);
+      }
+    },
+    addOpList: function (opList) {
+      Util.extendObj(this.dependencies, opList.dependencies);
+      for (var i = 0, ii = opList.length; i < ii; i++) {
+        this.addOp(opList.fnArray[i], opList.argsArray[i]);
+      }
+    },
+    getIR: function () {
+      return {
+        fnArray: this.fnArray,
+        argsArray: this.argsArray,
+        length: this.length
+      };
+    },
+    flush: function (lastChunk) {
+      if (this.intent !== 'oplist') {
+        new QueueOptimizer().optimize(this);
+      }
+      var transfers = getTransfers(this);
+      var length = this.length;
+      this._totalLength += length;
+      this.messageHandler.send('RenderPageChunk', {
+        operatorList: {
+          fnArray: this.fnArray,
+          argsArray: this.argsArray,
+          lastChunk: lastChunk,
+          length: length
+        },
+        pageIndex: this.pageIndex,
+        intent: this.intent
+      }, transfers);
+      this.dependencies = Object.create(null);
+      this.fnArray.length = 0;
+      this.argsArray.length = 0;
+    }
+  };
+  return OperatorList;
 }();
 var StateManager = function StateManagerClosure() {
- function StateManager(initialState) {
-  this.state = initialState;
-  this.stateStack = [];
- }
- StateManager.prototype = {
-  save: function () {
-   var old = this.state;
-   this.stateStack.push(this.state);
-   this.state = old.clone();
-  },
-  restore: function () {
-   var prev = this.stateStack.pop();
-   if (prev) {
-    this.state = prev;
-   }
-  },
-  transform: function (args) {
-   this.state.ctm = Util.transform(this.state.ctm, args);
+  function StateManager(initialState) {
+    this.state = initialState;
+    this.stateStack = [];
   }
- };
- return StateManager;
+  StateManager.prototype = {
+    save: function () {
+      var old = this.state;
+      this.stateStack.push(this.state);
+      this.state = old.clone();
+    },
+    restore: function () {
+      var prev = this.stateStack.pop();
+      if (prev) {
+        this.state = prev;
+      }
+    },
+    transform: function (args) {
+      this.state.ctm = Util.transform(this.state.ctm, args);
+    }
+  };
+  return StateManager;
 }();
 var TextState = function TextStateClosure() {
- function TextState() {
-  this.ctm = new Float32Array(IDENTITY_MATRIX);
-  this.fontName = null;
-  this.fontSize = 0;
-  this.font = null;
-  this.fontMatrix = FONT_IDENTITY_MATRIX;
-  this.textMatrix = IDENTITY_MATRIX.slice();
-  this.textLineMatrix = IDENTITY_MATRIX.slice();
-  this.charSpacing = 0;
-  this.wordSpacing = 0;
-  this.leading = 0;
-  this.textHScale = 1;
-  this.textRise = 0;
- }
- TextState.prototype = {
-  setTextMatrix: function TextState_setTextMatrix(a, b, c, d, e, f) {
-   var m = this.textMatrix;
-   m[0] = a;
-   m[1] = b;
-   m[2] = c;
-   m[3] = d;
-   m[4] = e;
-   m[5] = f;
-  },
-  setTextLineMatrix: function TextState_setTextMatrix(a, b, c, d, e, f) {
-   var m = this.textLineMatrix;
-   m[0] = a;
-   m[1] = b;
-   m[2] = c;
-   m[3] = d;
-   m[4] = e;
-   m[5] = f;
-  },
-  translateTextMatrix: function TextState_translateTextMatrix(x, y) {
-   var m = this.textMatrix;
-   m[4] = m[0] * x + m[2] * y + m[4];
-   m[5] = m[1] * x + m[3] * y + m[5];
-  },
-  translateTextLineMatrix: function TextState_translateTextMatrix(x, y) {
-   var m = this.textLineMatrix;
-   m[4] = m[0] * x + m[2] * y + m[4];
-   m[5] = m[1] * x + m[3] * y + m[5];
-  },
-  calcTextLineMatrixAdvance: function TextState_calcTextLineMatrixAdvance(a, b, c, d, e, f) {
-   var font = this.font;
-   if (!font) {
-    return null;
-   }
-   var m = this.textLineMatrix;
-   if (!(a === m[0] && b === m[1] && c === m[2] && d === m[3])) {
-    return null;
-   }
-   var txDiff = e - m[4], tyDiff = f - m[5];
-   if (font.vertical && txDiff !== 0 || !font.vertical && tyDiff !== 0) {
-    return null;
-   }
-   var tx, ty, denominator = a * d - b * c;
-   if (font.vertical) {
-    tx = -tyDiff * c / denominator;
-    ty = tyDiff * a / denominator;
-   } else {
-    tx = txDiff * d / denominator;
-    ty = -txDiff * b / denominator;
-   }
-   return {
-    width: tx,
-    height: ty,
-    value: font.vertical ? ty : tx
-   };
-  },
-  calcRenderMatrix: function TextState_calcRendeMatrix(ctm) {
-   var tsm = [
-    this.fontSize * this.textHScale,
-    0,
-    0,
-    this.fontSize,
-    0,
-    this.textRise
-   ];
-   return Util.transform(ctm, Util.transform(this.textMatrix, tsm));
-  },
-  carriageReturn: function TextState_carriageReturn() {
-   this.translateTextLineMatrix(0, -this.leading);
-   this.textMatrix = this.textLineMatrix.slice();
-  },
-  clone: function TextState_clone() {
-   var clone = Object.create(this);
-   clone.textMatrix = this.textMatrix.slice();
-   clone.textLineMatrix = this.textLineMatrix.slice();
-   clone.fontMatrix = this.fontMatrix.slice();
-   return clone;
+  function TextState() {
+    this.ctm = new Float32Array(IDENTITY_MATRIX);
+    this.fontName = null;
+    this.fontSize = 0;
+    this.font = null;
+    this.fontMatrix = FONT_IDENTITY_MATRIX;
+    this.textMatrix = IDENTITY_MATRIX.slice();
+    this.textLineMatrix = IDENTITY_MATRIX.slice();
+    this.charSpacing = 0;
+    this.wordSpacing = 0;
+    this.leading = 0;
+    this.textHScale = 1;
+    this.textRise = 0;
   }
- };
- return TextState;
+  TextState.prototype = {
+    setTextMatrix: function TextState_setTextMatrix(a, b, c, d, e, f) {
+      var m = this.textMatrix;
+      m[0] = a;
+      m[1] = b;
+      m[2] = c;
+      m[3] = d;
+      m[4] = e;
+      m[5] = f;
+    },
+    setTextLineMatrix: function TextState_setTextMatrix(a, b, c, d, e, f) {
+      var m = this.textLineMatrix;
+      m[0] = a;
+      m[1] = b;
+      m[2] = c;
+      m[3] = d;
+      m[4] = e;
+      m[5] = f;
+    },
+    translateTextMatrix: function TextState_translateTextMatrix(x, y) {
+      var m = this.textMatrix;
+      m[4] = m[0] * x + m[2] * y + m[4];
+      m[5] = m[1] * x + m[3] * y + m[5];
+    },
+    translateTextLineMatrix: function TextState_translateTextMatrix(x, y) {
+      var m = this.textLineMatrix;
+      m[4] = m[0] * x + m[2] * y + m[4];
+      m[5] = m[1] * x + m[3] * y + m[5];
+    },
+    calcTextLineMatrixAdvance: function TextState_calcTextLineMatrixAdvance(a, b, c, d, e, f) {
+      var font = this.font;
+      if (!font) {
+        return null;
+      }
+      var m = this.textLineMatrix;
+      if (!(a === m[0] && b === m[1] && c === m[2] && d === m[3])) {
+        return null;
+      }
+      var txDiff = e - m[4],
+          tyDiff = f - m[5];
+      if (font.vertical && txDiff !== 0 || !font.vertical && tyDiff !== 0) {
+        return null;
+      }
+      var tx,
+          ty,
+          denominator = a * d - b * c;
+      if (font.vertical) {
+        tx = -tyDiff * c / denominator;
+        ty = tyDiff * a / denominator;
+      } else {
+        tx = txDiff * d / denominator;
+        ty = -txDiff * b / denominator;
+      }
+      return {
+        width: tx,
+        height: ty,
+        value: font.vertical ? ty : tx
+      };
+    },
+    calcRenderMatrix: function TextState_calcRendeMatrix(ctm) {
+      var tsm = [this.fontSize * this.textHScale, 0, 0, this.fontSize, 0, this.textRise];
+      return Util.transform(ctm, Util.transform(this.textMatrix, tsm));
+    },
+    carriageReturn: function TextState_carriageReturn() {
+      this.translateTextLineMatrix(0, -this.leading);
+      this.textMatrix = this.textLineMatrix.slice();
+    },
+    clone: function TextState_clone() {
+      var clone = Object.create(this);
+      clone.textMatrix = this.textMatrix.slice();
+      clone.textLineMatrix = this.textLineMatrix.slice();
+      clone.fontMatrix = this.fontMatrix.slice();
+      return clone;
+    }
+  };
+  return TextState;
 }();
 var EvalState = function EvalStateClosure() {
- function EvalState() {
-  this.ctm = new Float32Array(IDENTITY_MATRIX);
-  this.font = null;
-  this.textRenderingMode = TextRenderingMode.FILL;
-  this.fillColorSpace = ColorSpace.singletons.gray;
-  this.strokeColorSpace = ColorSpace.singletons.gray;
- }
- EvalState.prototype = {
-  clone: function CanvasExtraState_clone() {
-   return Object.create(this);
+  function EvalState() {
+    this.ctm = new Float32Array(IDENTITY_MATRIX);
+    this.font = null;
+    this.textRenderingMode = TextRenderingMode.FILL;
+    this.fillColorSpace = ColorSpace.singletons.gray;
+    this.strokeColorSpace = ColorSpace.singletons.gray;
   }
- };
- return EvalState;
+  EvalState.prototype = {
+    clone: function CanvasExtraState_clone() {
+      return Object.create(this);
+    }
+  };
+  return EvalState;
 }();
 var EvaluatorPreprocessor = function EvaluatorPreprocessorClosure() {
- var getOPMap = getLookupTableFactory(function (t) {
-  t['w'] = {
-   id: OPS.setLineWidth,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['J'] = {
-   id: OPS.setLineCap,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['j'] = {
-   id: OPS.setLineJoin,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['M'] = {
-   id: OPS.setMiterLimit,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['d'] = {
-   id: OPS.setDash,
-   numArgs: 2,
-   variableArgs: false
-  };
-  t['ri'] = {
-   id: OPS.setRenderingIntent,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['i'] = {
-   id: OPS.setFlatness,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['gs'] = {
-   id: OPS.setGState,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['q'] = {
-   id: OPS.save,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['Q'] = {
-   id: OPS.restore,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['cm'] = {
-   id: OPS.transform,
-   numArgs: 6,
-   variableArgs: false
-  };
-  t['m'] = {
-   id: OPS.moveTo,
-   numArgs: 2,
-   variableArgs: false
-  };
-  t['l'] = {
-   id: OPS.lineTo,
-   numArgs: 2,
-   variableArgs: false
-  };
-  t['c'] = {
-   id: OPS.curveTo,
-   numArgs: 6,
-   variableArgs: false
-  };
-  t['v'] = {
-   id: OPS.curveTo2,
-   numArgs: 4,
-   variableArgs: false
-  };
-  t['y'] = {
-   id: OPS.curveTo3,
-   numArgs: 4,
-   variableArgs: false
-  };
-  t['h'] = {
-   id: OPS.closePath,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['re'] = {
-   id: OPS.rectangle,
-   numArgs: 4,
-   variableArgs: false
-  };
-  t['S'] = {
-   id: OPS.stroke,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['s'] = {
-   id: OPS.closeStroke,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['f'] = {
-   id: OPS.fill,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['F'] = {
-   id: OPS.fill,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['f*'] = {
-   id: OPS.eoFill,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['B'] = {
-   id: OPS.fillStroke,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['B*'] = {
-   id: OPS.eoFillStroke,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['b'] = {
-   id: OPS.closeFillStroke,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['b*'] = {
-   id: OPS.closeEOFillStroke,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['n'] = {
-   id: OPS.endPath,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['W'] = {
-   id: OPS.clip,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['W*'] = {
-   id: OPS.eoClip,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['BT'] = {
-   id: OPS.beginText,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['ET'] = {
-   id: OPS.endText,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['Tc'] = {
-   id: OPS.setCharSpacing,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['Tw'] = {
-   id: OPS.setWordSpacing,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['Tz'] = {
-   id: OPS.setHScale,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['TL'] = {
-   id: OPS.setLeading,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['Tf'] = {
-   id: OPS.setFont,
-   numArgs: 2,
-   variableArgs: false
-  };
-  t['Tr'] = {
-   id: OPS.setTextRenderingMode,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['Ts'] = {
-   id: OPS.setTextRise,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['Td'] = {
-   id: OPS.moveText,
-   numArgs: 2,
-   variableArgs: false
-  };
-  t['TD'] = {
-   id: OPS.setLeadingMoveText,
-   numArgs: 2,
-   variableArgs: false
-  };
-  t['Tm'] = {
-   id: OPS.setTextMatrix,
-   numArgs: 6,
-   variableArgs: false
-  };
-  t['T*'] = {
-   id: OPS.nextLine,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['Tj'] = {
-   id: OPS.showText,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['TJ'] = {
-   id: OPS.showSpacedText,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['\''] = {
-   id: OPS.nextLineShowText,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['"'] = {
-   id: OPS.nextLineSetSpacingShowText,
-   numArgs: 3,
-   variableArgs: false
-  };
-  t['d0'] = {
-   id: OPS.setCharWidth,
-   numArgs: 2,
-   variableArgs: false
-  };
-  t['d1'] = {
-   id: OPS.setCharWidthAndBounds,
-   numArgs: 6,
-   variableArgs: false
-  };
-  t['CS'] = {
-   id: OPS.setStrokeColorSpace,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['cs'] = {
-   id: OPS.setFillColorSpace,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['SC'] = {
-   id: OPS.setStrokeColor,
-   numArgs: 4,
-   variableArgs: true
-  };
-  t['SCN'] = {
-   id: OPS.setStrokeColorN,
-   numArgs: 33,
-   variableArgs: true
-  };
-  t['sc'] = {
-   id: OPS.setFillColor,
-   numArgs: 4,
-   variableArgs: true
-  };
-  t['scn'] = {
-   id: OPS.setFillColorN,
-   numArgs: 33,
-   variableArgs: true
-  };
-  t['G'] = {
-   id: OPS.setStrokeGray,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['g'] = {
-   id: OPS.setFillGray,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['RG'] = {
-   id: OPS.setStrokeRGBColor,
-   numArgs: 3,
-   variableArgs: false
-  };
-  t['rg'] = {
-   id: OPS.setFillRGBColor,
-   numArgs: 3,
-   variableArgs: false
-  };
-  t['K'] = {
-   id: OPS.setStrokeCMYKColor,
-   numArgs: 4,
-   variableArgs: false
-  };
-  t['k'] = {
-   id: OPS.setFillCMYKColor,
-   numArgs: 4,
-   variableArgs: false
-  };
-  t['sh'] = {
-   id: OPS.shadingFill,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['BI'] = {
-   id: OPS.beginInlineImage,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['ID'] = {
-   id: OPS.beginImageData,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['EI'] = {
-   id: OPS.endInlineImage,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['Do'] = {
-   id: OPS.paintXObject,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['MP'] = {
-   id: OPS.markPoint,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['DP'] = {
-   id: OPS.markPointProps,
-   numArgs: 2,
-   variableArgs: false
-  };
-  t['BMC'] = {
-   id: OPS.beginMarkedContent,
-   numArgs: 1,
-   variableArgs: false
-  };
-  t['BDC'] = {
-   id: OPS.beginMarkedContentProps,
-   numArgs: 2,
-   variableArgs: false
-  };
-  t['EMC'] = {
-   id: OPS.endMarkedContent,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['BX'] = {
-   id: OPS.beginCompat,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['EX'] = {
-   id: OPS.endCompat,
-   numArgs: 0,
-   variableArgs: false
-  };
-  t['BM'] = null;
-  t['BD'] = null;
-  t['true'] = null;
-  t['fa'] = null;
-  t['fal'] = null;
-  t['fals'] = null;
-  t['false'] = null;
-  t['nu'] = null;
-  t['nul'] = null;
-  t['null'] = null;
- });
- function EvaluatorPreprocessor(stream, xref, stateManager) {
-  this.opMap = getOPMap();
-  this.parser = new Parser(new Lexer(stream, this.opMap), false, xref);
-  this.stateManager = stateManager;
-  this.nonProcessedArgs = [];
- }
- EvaluatorPreprocessor.prototype = {
-  get savedStatesDepth() {
-   return this.stateManager.stateStack.length;
-  },
-  read: function EvaluatorPreprocessor_read(operation) {
-   var args = operation.args;
-   while (true) {
-    var obj = this.parser.getObj();
-    if (isCmd(obj)) {
-     var cmd = obj.cmd;
-     var opSpec = this.opMap[cmd];
-     if (!opSpec) {
-      warn('Unknown command "' + cmd + '"');
-      continue;
-     }
-     var fn = opSpec.id;
-     var numArgs = opSpec.numArgs;
-     var argsLength = args !== null ? args.length : 0;
-     if (!opSpec.variableArgs) {
-      if (argsLength !== numArgs) {
-       var nonProcessedArgs = this.nonProcessedArgs;
-       while (argsLength > numArgs) {
-        nonProcessedArgs.push(args.shift());
-        argsLength--;
-       }
-       while (argsLength < numArgs && nonProcessedArgs.length !== 0) {
-        if (args === null) {
-         args = [];
-        }
-        args.unshift(nonProcessedArgs.pop());
-        argsLength++;
-       }
-      }
-      if (argsLength < numArgs) {
-       warn('Skipping command ' + fn + ': expected ' + numArgs + ' args, but received ' + argsLength + ' args.');
-       if (args !== null) {
-        args.length = 0;
-       }
-       continue;
-      }
-     } else if (argsLength > numArgs) {
-      info('Command ' + fn + ': expected [0,' + numArgs + '] args, but received ' + argsLength + ' args.');
-     }
-     this.preprocessCommand(fn, args);
-     operation.fn = fn;
-     operation.args = args;
-     return true;
-    }
-    if (isEOF(obj)) {
-     return false;
-    }
-    if (obj !== null) {
-     if (args === null) {
-      args = [];
-     }
-     args.push(obj);
-     assert(args.length <= 33, 'Too many arguments');
-    }
-   }
-  },
-  preprocessCommand: function EvaluatorPreprocessor_preprocessCommand(fn, args) {
-   switch (fn | 0) {
-   case OPS.save:
-    this.stateManager.save();
-    break;
-   case OPS.restore:
-    this.stateManager.restore();
-    break;
-   case OPS.transform:
-    this.stateManager.transform(args);
-    break;
-   }
+  var getOPMap = getLookupTableFactory(function (t) {
+    t['w'] = {
+      id: OPS.setLineWidth,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['J'] = {
+      id: OPS.setLineCap,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['j'] = {
+      id: OPS.setLineJoin,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['M'] = {
+      id: OPS.setMiterLimit,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['d'] = {
+      id: OPS.setDash,
+      numArgs: 2,
+      variableArgs: false
+    };
+    t['ri'] = {
+      id: OPS.setRenderingIntent,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['i'] = {
+      id: OPS.setFlatness,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['gs'] = {
+      id: OPS.setGState,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['q'] = {
+      id: OPS.save,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['Q'] = {
+      id: OPS.restore,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['cm'] = {
+      id: OPS.transform,
+      numArgs: 6,
+      variableArgs: false
+    };
+    t['m'] = {
+      id: OPS.moveTo,
+      numArgs: 2,
+      variableArgs: false
+    };
+    t['l'] = {
+      id: OPS.lineTo,
+      numArgs: 2,
+      variableArgs: false
+    };
+    t['c'] = {
+      id: OPS.curveTo,
+      numArgs: 6,
+      variableArgs: false
+    };
+    t['v'] = {
+      id: OPS.curveTo2,
+      numArgs: 4,
+      variableArgs: false
+    };
+    t['y'] = {
+      id: OPS.curveTo3,
+      numArgs: 4,
+      variableArgs: false
+    };
+    t['h'] = {
+      id: OPS.closePath,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['re'] = {
+      id: OPS.rectangle,
+      numArgs: 4,
+      variableArgs: false
+    };
+    t['S'] = {
+      id: OPS.stroke,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['s'] = {
+      id: OPS.closeStroke,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['f'] = {
+      id: OPS.fill,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['F'] = {
+      id: OPS.fill,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['f*'] = {
+      id: OPS.eoFill,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['B'] = {
+      id: OPS.fillStroke,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['B*'] = {
+      id: OPS.eoFillStroke,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['b'] = {
+      id: OPS.closeFillStroke,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['b*'] = {
+      id: OPS.closeEOFillStroke,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['n'] = {
+      id: OPS.endPath,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['W'] = {
+      id: OPS.clip,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['W*'] = {
+      id: OPS.eoClip,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['BT'] = {
+      id: OPS.beginText,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['ET'] = {
+      id: OPS.endText,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['Tc'] = {
+      id: OPS.setCharSpacing,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['Tw'] = {
+      id: OPS.setWordSpacing,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['Tz'] = {
+      id: OPS.setHScale,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['TL'] = {
+      id: OPS.setLeading,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['Tf'] = {
+      id: OPS.setFont,
+      numArgs: 2,
+      variableArgs: false
+    };
+    t['Tr'] = {
+      id: OPS.setTextRenderingMode,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['Ts'] = {
+      id: OPS.setTextRise,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['Td'] = {
+      id: OPS.moveText,
+      numArgs: 2,
+      variableArgs: false
+    };
+    t['TD'] = {
+      id: OPS.setLeadingMoveText,
+      numArgs: 2,
+      variableArgs: false
+    };
+    t['Tm'] = {
+      id: OPS.setTextMatrix,
+      numArgs: 6,
+      variableArgs: false
+    };
+    t['T*'] = {
+      id: OPS.nextLine,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['Tj'] = {
+      id: OPS.showText,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['TJ'] = {
+      id: OPS.showSpacedText,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['\''] = {
+      id: OPS.nextLineShowText,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['"'] = {
+      id: OPS.nextLineSetSpacingShowText,
+      numArgs: 3,
+      variableArgs: false
+    };
+    t['d0'] = {
+      id: OPS.setCharWidth,
+      numArgs: 2,
+      variableArgs: false
+    };
+    t['d1'] = {
+      id: OPS.setCharWidthAndBounds,
+      numArgs: 6,
+      variableArgs: false
+    };
+    t['CS'] = {
+      id: OPS.setStrokeColorSpace,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['cs'] = {
+      id: OPS.setFillColorSpace,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['SC'] = {
+      id: OPS.setStrokeColor,
+      numArgs: 4,
+      variableArgs: true
+    };
+    t['SCN'] = {
+      id: OPS.setStrokeColorN,
+      numArgs: 33,
+      variableArgs: true
+    };
+    t['sc'] = {
+      id: OPS.setFillColor,
+      numArgs: 4,
+      variableArgs: true
+    };
+    t['scn'] = {
+      id: OPS.setFillColorN,
+      numArgs: 33,
+      variableArgs: true
+    };
+    t['G'] = {
+      id: OPS.setStrokeGray,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['g'] = {
+      id: OPS.setFillGray,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['RG'] = {
+      id: OPS.setStrokeRGBColor,
+      numArgs: 3,
+      variableArgs: false
+    };
+    t['rg'] = {
+      id: OPS.setFillRGBColor,
+      numArgs: 3,
+      variableArgs: false
+    };
+    t['K'] = {
+      id: OPS.setStrokeCMYKColor,
+      numArgs: 4,
+      variableArgs: false
+    };
+    t['k'] = {
+      id: OPS.setFillCMYKColor,
+      numArgs: 4,
+      variableArgs: false
+    };
+    t['sh'] = {
+      id: OPS.shadingFill,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['BI'] = {
+      id: OPS.beginInlineImage,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['ID'] = {
+      id: OPS.beginImageData,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['EI'] = {
+      id: OPS.endInlineImage,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['Do'] = {
+      id: OPS.paintXObject,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['MP'] = {
+      id: OPS.markPoint,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['DP'] = {
+      id: OPS.markPointProps,
+      numArgs: 2,
+      variableArgs: false
+    };
+    t['BMC'] = {
+      id: OPS.beginMarkedContent,
+      numArgs: 1,
+      variableArgs: false
+    };
+    t['BDC'] = {
+      id: OPS.beginMarkedContentProps,
+      numArgs: 2,
+      variableArgs: false
+    };
+    t['EMC'] = {
+      id: OPS.endMarkedContent,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['BX'] = {
+      id: OPS.beginCompat,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['EX'] = {
+      id: OPS.endCompat,
+      numArgs: 0,
+      variableArgs: false
+    };
+    t['BM'] = null;
+    t['BD'] = null;
+    t['true'] = null;
+    t['fa'] = null;
+    t['fal'] = null;
+    t['fals'] = null;
+    t['false'] = null;
+    t['nu'] = null;
+    t['nul'] = null;
+    t['null'] = null;
+  });
+  function EvaluatorPreprocessor(stream, xref, stateManager) {
+    this.opMap = getOPMap();
+    this.parser = new Parser(new Lexer(stream, this.opMap), false, xref);
+    this.stateManager = stateManager;
+    this.nonProcessedArgs = [];
   }
- };
- return EvaluatorPreprocessor;
+  EvaluatorPreprocessor.prototype = {
+    get savedStatesDepth() {
+      return this.stateManager.stateStack.length;
+    },
+    read: function EvaluatorPreprocessor_read(operation) {
+      var args = operation.args;
+      while (true) {
+        var obj = this.parser.getObj();
+        if (isCmd(obj)) {
+          var cmd = obj.cmd;
+          var opSpec = this.opMap[cmd];
+          if (!opSpec) {
+            warn('Unknown command "' + cmd + '"');
+            continue;
+          }
+          var fn = opSpec.id;
+          var numArgs = opSpec.numArgs;
+          var argsLength = args !== null ? args.length : 0;
+          if (!opSpec.variableArgs) {
+            if (argsLength !== numArgs) {
+              var nonProcessedArgs = this.nonProcessedArgs;
+              while (argsLength > numArgs) {
+                nonProcessedArgs.push(args.shift());
+                argsLength--;
+              }
+              while (argsLength < numArgs && nonProcessedArgs.length !== 0) {
+                if (args === null) {
+                  args = [];
+                }
+                args.unshift(nonProcessedArgs.pop());
+                argsLength++;
+              }
+            }
+            if (argsLength < numArgs) {
+              warn('Skipping command ' + fn + ': expected ' + numArgs + ' args, but received ' + argsLength + ' args.');
+              if (args !== null) {
+                args.length = 0;
+              }
+              continue;
+            }
+          } else if (argsLength > numArgs) {
+            info('Command ' + fn + ': expected [0,' + numArgs + '] args, but received ' + argsLength + ' args.');
+          }
+          this.preprocessCommand(fn, args);
+          operation.fn = fn;
+          operation.args = args;
+          return true;
+        }
+        if (isEOF(obj)) {
+          return false;
+        }
+        if (obj !== null) {
+          if (args === null) {
+            args = [];
+          }
+          args.push(obj);
+          assert(args.length <= 33, 'Too many arguments');
+        }
+      }
+    },
+    preprocessCommand: function EvaluatorPreprocessor_preprocessCommand(fn, args) {
+      switch (fn | 0) {
+        case OPS.save:
+          this.stateManager.save();
+          break;
+        case OPS.restore:
+          this.stateManager.restore();
+          break;
+        case OPS.transform:
+          this.stateManager.transform(args);
+          break;
+      }
+    }
+  };
+  return EvaluatorPreprocessor;
 }();
 var QueueOptimizer = function QueueOptimizerClosure() {
- function addState(parentState, pattern, fn) {
-  var state = parentState;
-  for (var i = 0, ii = pattern.length - 1; i < ii; i++) {
-   var item = pattern[i];
-   state = state[item] || (state[item] = []);
-  }
-  state[pattern[pattern.length - 1]] = fn;
- }
- function handlePaintSolidColorImageMask(iFirstSave, count, fnArray, argsArray) {
-  var iFirstPIMXO = iFirstSave + 2;
-  for (var i = 0; i < count; i++) {
-   var arg = argsArray[iFirstPIMXO + 4 * i];
-   var imageMask = arg.length === 1 && arg[0];
-   if (imageMask && imageMask.width === 1 && imageMask.height === 1 && (!imageMask.data.length || imageMask.data.length === 1 && imageMask.data[0] === 0)) {
-    fnArray[iFirstPIMXO + 4 * i] = OPS.paintSolidColorImageMask;
-    continue;
-   }
-   break;
-  }
-  return count - i;
- }
- var InitialState = [];
- addState(InitialState, [
-  OPS.save,
-  OPS.transform,
-  OPS.paintInlineImageXObject,
-  OPS.restore
- ], function foundInlineImageGroup(context) {
-  var MIN_IMAGES_IN_INLINE_IMAGES_BLOCK = 10;
-  var MAX_IMAGES_IN_INLINE_IMAGES_BLOCK = 200;
-  var MAX_WIDTH = 1000;
-  var IMAGE_PADDING = 1;
-  var fnArray = context.fnArray, argsArray = context.argsArray;
-  var curr = context.iCurr;
-  var iFirstSave = curr - 3;
-  var iFirstTransform = curr - 2;
-  var iFirstPIIXO = curr - 1;
-  var i = iFirstSave + 4;
-  var ii = fnArray.length;
-  while (i + 3 < ii) {
-   if (fnArray[i] !== OPS.save || fnArray[i + 1] !== OPS.transform || fnArray[i + 2] !== OPS.paintInlineImageXObject || fnArray[i + 3] !== OPS.restore) {
-    break;
-   }
-   i += 4;
-  }
-  var count = Math.min((i - iFirstSave) / 4, MAX_IMAGES_IN_INLINE_IMAGES_BLOCK);
-  if (count < MIN_IMAGES_IN_INLINE_IMAGES_BLOCK) {
-   return i;
-  }
-  var maxX = 0;
-  var map = [], maxLineHeight = 0;
-  var currentX = IMAGE_PADDING, currentY = IMAGE_PADDING;
-  var q;
-  for (q = 0; q < count; q++) {
-   var transform = argsArray[iFirstTransform + (q << 2)];
-   var img = argsArray[iFirstPIIXO + (q << 2)][0];
-   if (currentX + img.width > MAX_WIDTH) {
-    maxX = Math.max(maxX, currentX);
-    currentY += maxLineHeight + 2 * IMAGE_PADDING;
-    currentX = 0;
-    maxLineHeight = 0;
-   }
-   map.push({
-    transform: transform,
-    x: currentX,
-    y: currentY,
-    w: img.width,
-    h: img.height
-   });
-   currentX += img.width + 2 * IMAGE_PADDING;
-   maxLineHeight = Math.max(maxLineHeight, img.height);
-  }
-  var imgWidth = Math.max(maxX, currentX) + IMAGE_PADDING;
-  var imgHeight = currentY + maxLineHeight + IMAGE_PADDING;
-  var imgData = new Uint8Array(imgWidth * imgHeight * 4);
-  var imgRowSize = imgWidth << 2;
-  for (q = 0; q < count; q++) {
-   var data = argsArray[iFirstPIIXO + (q << 2)][0].data;
-   var rowSize = map[q].w << 2;
-   var dataOffset = 0;
-   var offset = map[q].x + map[q].y * imgWidth << 2;
-   imgData.set(data.subarray(0, rowSize), offset - imgRowSize);
-   for (var k = 0, kk = map[q].h; k < kk; k++) {
-    imgData.set(data.subarray(dataOffset, dataOffset + rowSize), offset);
-    dataOffset += rowSize;
-    offset += imgRowSize;
-   }
-   imgData.set(data.subarray(dataOffset - rowSize, dataOffset), offset);
-   while (offset >= 0) {
-    data[offset - 4] = data[offset];
-    data[offset - 3] = data[offset + 1];
-    data[offset - 2] = data[offset + 2];
-    data[offset - 1] = data[offset + 3];
-    data[offset + rowSize] = data[offset + rowSize - 4];
-    data[offset + rowSize + 1] = data[offset + rowSize - 3];
-    data[offset + rowSize + 2] = data[offset + rowSize - 2];
-    data[offset + rowSize + 3] = data[offset + rowSize - 1];
-    offset -= imgRowSize;
-   }
-  }
-  fnArray.splice(iFirstSave, count * 4, OPS.paintInlineImageXObjectGroup);
-  argsArray.splice(iFirstSave, count * 4, [
-   {
-    width: imgWidth,
-    height: imgHeight,
-    kind: ImageKind.RGBA_32BPP,
-    data: imgData
-   },
-   map
-  ]);
-  return iFirstSave + 1;
- });
- addState(InitialState, [
-  OPS.save,
-  OPS.transform,
-  OPS.paintImageMaskXObject,
-  OPS.restore
- ], function foundImageMaskGroup(context) {
-  var MIN_IMAGES_IN_MASKS_BLOCK = 10;
-  var MAX_IMAGES_IN_MASKS_BLOCK = 100;
-  var MAX_SAME_IMAGES_IN_MASKS_BLOCK = 1000;
-  var fnArray = context.fnArray, argsArray = context.argsArray;
-  var curr = context.iCurr;
-  var iFirstSave = curr - 3;
-  var iFirstTransform = curr - 2;
-  var iFirstPIMXO = curr - 1;
-  var i = iFirstSave + 4;
-  var ii = fnArray.length;
-  while (i + 3 < ii) {
-   if (fnArray[i] !== OPS.save || fnArray[i + 1] !== OPS.transform || fnArray[i + 2] !== OPS.paintImageMaskXObject || fnArray[i + 3] !== OPS.restore) {
-    break;
-   }
-   i += 4;
-  }
-  var count = (i - iFirstSave) / 4;
-  count = handlePaintSolidColorImageMask(iFirstSave, count, fnArray, argsArray);
-  if (count < MIN_IMAGES_IN_MASKS_BLOCK) {
-   return i;
-  }
-  var q;
-  var isSameImage = false;
-  var iTransform, transformArgs;
-  var firstPIMXOArg0 = argsArray[iFirstPIMXO][0];
-  if (argsArray[iFirstTransform][1] === 0 && argsArray[iFirstTransform][2] === 0) {
-   isSameImage = true;
-   var firstTransformArg0 = argsArray[iFirstTransform][0];
-   var firstTransformArg3 = argsArray[iFirstTransform][3];
-   iTransform = iFirstTransform + 4;
-   var iPIMXO = iFirstPIMXO + 4;
-   for (q = 1; q < count; q++, iTransform += 4, iPIMXO += 4) {
-    transformArgs = argsArray[iTransform];
-    if (argsArray[iPIMXO][0] !== firstPIMXOArg0 || transformArgs[0] !== firstTransformArg0 || transformArgs[1] !== 0 || transformArgs[2] !== 0 || transformArgs[3] !== firstTransformArg3) {
-     if (q < MIN_IMAGES_IN_MASKS_BLOCK) {
-      isSameImage = false;
-     } else {
-      count = q;
-     }
-     break;
+  function addState(parentState, pattern, fn) {
+    var state = parentState;
+    for (var i = 0, ii = pattern.length - 1; i < ii; i++) {
+      var item = pattern[i];
+      state = state[item] || (state[item] = []);
     }
-   }
-  }
-  if (isSameImage) {
-   count = Math.min(count, MAX_SAME_IMAGES_IN_MASKS_BLOCK);
-   var positions = new Float32Array(count * 2);
-   iTransform = iFirstTransform;
-   for (q = 0; q < count; q++, iTransform += 4) {
-    transformArgs = argsArray[iTransform];
-    positions[q << 1] = transformArgs[4];
-    positions[(q << 1) + 1] = transformArgs[5];
-   }
-   fnArray.splice(iFirstSave, count * 4, OPS.paintImageMaskXObjectRepeat);
-   argsArray.splice(iFirstSave, count * 4, [
-    firstPIMXOArg0,
-    firstTransformArg0,
-    firstTransformArg3,
-    positions
-   ]);
-  } else {
-   count = Math.min(count, MAX_IMAGES_IN_MASKS_BLOCK);
-   var images = [];
-   for (q = 0; q < count; q++) {
-    transformArgs = argsArray[iFirstTransform + (q << 2)];
-    var maskParams = argsArray[iFirstPIMXO + (q << 2)][0];
-    images.push({
-     data: maskParams.data,
-     width: maskParams.width,
-     height: maskParams.height,
-     transform: transformArgs
-    });
-   }
-   fnArray.splice(iFirstSave, count * 4, OPS.paintImageMaskXObjectGroup);
-   argsArray.splice(iFirstSave, count * 4, [images]);
-  }
-  return iFirstSave + 1;
- });
- addState(InitialState, [
-  OPS.save,
-  OPS.transform,
-  OPS.paintImageXObject,
-  OPS.restore
- ], function (context) {
-  var MIN_IMAGES_IN_BLOCK = 3;
-  var MAX_IMAGES_IN_BLOCK = 1000;
-  var fnArray = context.fnArray, argsArray = context.argsArray;
-  var curr = context.iCurr;
-  var iFirstSave = curr - 3;
-  var iFirstTransform = curr - 2;
-  var iFirstPIXO = curr - 1;
-  var iFirstRestore = curr;
-  if (argsArray[iFirstTransform][1] !== 0 || argsArray[iFirstTransform][2] !== 0) {
-   return iFirstRestore + 1;
+    state[pattern[pattern.length - 1]] = fn;
   }
-  var firstPIXOArg0 = argsArray[iFirstPIXO][0];
-  var firstTransformArg0 = argsArray[iFirstTransform][0];
-  var firstTransformArg3 = argsArray[iFirstTransform][3];
-  var i = iFirstSave + 4;
-  var ii = fnArray.length;
-  while (i + 3 < ii) {
-   if (fnArray[i] !== OPS.save || fnArray[i + 1] !== OPS.transform || fnArray[i + 2] !== OPS.paintImageXObject || fnArray[i + 3] !== OPS.restore) {
-    break;
-   }
-   if (argsArray[i + 1][0] !== firstTransformArg0 || argsArray[i + 1][1] !== 0 || argsArray[i + 1][2] !== 0 || argsArray[i + 1][3] !== firstTransformArg3) {
-    break;
-   }
-   if (argsArray[i + 2][0] !== firstPIXOArg0) {
-    break;
-   }
-   i += 4;
-  }
-  var count = Math.min((i - iFirstSave) / 4, MAX_IMAGES_IN_BLOCK);
-  if (count < MIN_IMAGES_IN_BLOCK) {
-   return i;
-  }
-  var positions = new Float32Array(count * 2);
-  var iTransform = iFirstTransform;
-  for (var q = 0; q < count; q++, iTransform += 4) {
-   var transformArgs = argsArray[iTransform];
-   positions[q << 1] = transformArgs[4];
-   positions[(q << 1) + 1] = transformArgs[5];
-  }
-  var args = [
-   firstPIXOArg0,
-   firstTransformArg0,
-   firstTransformArg3,
-   positions
-  ];
-  fnArray.splice(iFirstSave, count * 4, OPS.paintImageXObjectRepeat);
-  argsArray.splice(iFirstSave, count * 4, args);
-  return iFirstSave + 1;
- });
- addState(InitialState, [
-  OPS.beginText,
-  OPS.setFont,
-  OPS.setTextMatrix,
-  OPS.showText,
-  OPS.endText
- ], function (context) {
-  var MIN_CHARS_IN_BLOCK = 3;
-  var MAX_CHARS_IN_BLOCK = 1000;
-  var fnArray = context.fnArray, argsArray = context.argsArray;
-  var curr = context.iCurr;
-  var iFirstBeginText = curr - 4;
-  var iFirstSetFont = curr - 3;
-  var iFirstSetTextMatrix = curr - 2;
-  var iFirstShowText = curr - 1;
-  var iFirstEndText = curr;
-  var firstSetFontArg0 = argsArray[iFirstSetFont][0];
-  var firstSetFontArg1 = argsArray[iFirstSetFont][1];
-  var i = iFirstBeginText + 5;
-  var ii = fnArray.length;
-  while (i + 4 < ii) {
-   if (fnArray[i] !== OPS.beginText || fnArray[i + 1] !== OPS.setFont || fnArray[i + 2] !== OPS.setTextMatrix || fnArray[i + 3] !== OPS.showText || fnArray[i + 4] !== OPS.endText) {
-    break;
-   }
-   if (argsArray[i + 1][0] !== firstSetFontArg0 || argsArray[i + 1][1] !== firstSetFontArg1) {
-    break;
-   }
-   i += 5;
-  }
-  var count = Math.min((i - iFirstBeginText) / 5, MAX_CHARS_IN_BLOCK);
-  if (count < MIN_CHARS_IN_BLOCK) {
-   return i;
-  }
-  var iFirst = iFirstBeginText;
-  if (iFirstBeginText >= 4 && fnArray[iFirstBeginText - 4] === fnArray[iFirstSetFont] && fnArray[iFirstBeginText - 3] === fnArray[iFirstSetTextMatrix] && fnArray[iFirstBeginText - 2] === fnArray[iFirstShowText] && fnArray[iFirstBeginText - 1] === fnArray[iFirstEndText] && argsArray[iFirstBeginText - 4][0] === firstSetFontArg0 && argsArray[iFirstBeginText - 4][1] === firstSetFontArg1) {
-   count++;
-   iFirst -= 5;
-  }
-  var iEndText = iFirst + 4;
-  for (var q = 1; q < count; q++) {
-   fnArray.splice(iEndText, 3);
-   argsArray.splice(iEndText, 3);
-   iEndText += 2;
+  function handlePaintSolidColorImageMask(iFirstSave, count, fnArray, argsArray) {
+    var iFirstPIMXO = iFirstSave + 2;
+    for (var i = 0; i < count; i++) {
+      var arg = argsArray[iFirstPIMXO + 4 * i];
+      var imageMask = arg.length === 1 && arg[0];
+      if (imageMask && imageMask.width === 1 && imageMask.height === 1 && (!imageMask.data.length || imageMask.data.length === 1 && imageMask.data[0] === 0)) {
+        fnArray[iFirstPIMXO + 4 * i] = OPS.paintSolidColorImageMask;
+        continue;
+      }
+      break;
+    }
+    return count - i;
   }
-  return iEndText + 1;
- });
- function QueueOptimizer() {
- }
- QueueOptimizer.prototype = {
-  optimize: function QueueOptimizer_optimize(queue) {
-   var fnArray = queue.fnArray, argsArray = queue.argsArray;
-   var context = {
-    iCurr: 0,
-    fnArray: fnArray,
-    argsArray: argsArray
-   };
-   var state;
-   var i = 0, ii = fnArray.length;
-   while (i < ii) {
-    state = (state || InitialState)[fnArray[i]];
-    if (typeof state === 'function') {
-     context.iCurr = i;
-     i = state(context);
-     state = undefined;
-     ii = context.fnArray.length;
+  var InitialState = [];
+  addState(InitialState, [OPS.save, OPS.transform, OPS.paintInlineImageXObject, OPS.restore], function foundInlineImageGroup(context) {
+    var MIN_IMAGES_IN_INLINE_IMAGES_BLOCK = 10;
+    var MAX_IMAGES_IN_INLINE_IMAGES_BLOCK = 200;
+    var MAX_WIDTH = 1000;
+    var IMAGE_PADDING = 1;
+    var fnArray = context.fnArray,
+        argsArray = context.argsArray;
+    var curr = context.iCurr;
+    var iFirstSave = curr - 3;
+    var iFirstTransform = curr - 2;
+    var iFirstPIIXO = curr - 1;
+    var i = iFirstSave + 4;
+    var ii = fnArray.length;
+    while (i + 3 < ii) {
+      if (fnArray[i] !== OPS.save || fnArray[i + 1] !== OPS.transform || fnArray[i + 2] !== OPS.paintInlineImageXObject || fnArray[i + 3] !== OPS.restore) {
+        break;
+      }
+      i += 4;
+    }
+    var count = Math.min((i - iFirstSave) / 4, MAX_IMAGES_IN_INLINE_IMAGES_BLOCK);
+    if (count < MIN_IMAGES_IN_INLINE_IMAGES_BLOCK) {
+      return i;
+    }
+    var maxX = 0;
+    var map = [],
+        maxLineHeight = 0;
+    var currentX = IMAGE_PADDING,
+        currentY = IMAGE_PADDING;
+    var q;
+    for (q = 0; q < count; q++) {
+      var transform = argsArray[iFirstTransform + (q << 2)];
+      var img = argsArray[iFirstPIIXO + (q << 2)][0];
+      if (currentX + img.width > MAX_WIDTH) {
+        maxX = Math.max(maxX, currentX);
+        currentY += maxLineHeight + 2 * IMAGE_PADDING;
+        currentX = 0;
+        maxLineHeight = 0;
+      }
+      map.push({
+        transform: transform,
+        x: currentX,
+        y: currentY,
+        w: img.width,
+        h: img.height
+      });
+      currentX += img.width + 2 * IMAGE_PADDING;
+      maxLineHeight = Math.max(maxLineHeight, img.height);
+    }
+    var imgWidth = Math.max(maxX, currentX) + IMAGE_PADDING;
+    var imgHeight = currentY + maxLineHeight + IMAGE_PADDING;
+    var imgData = new Uint8Array(imgWidth * imgHeight * 4);
+    var imgRowSize = imgWidth << 2;
+    for (q = 0; q < count; q++) {
+      var data = argsArray[iFirstPIIXO + (q << 2)][0].data;
+      var rowSize = map[q].w << 2;
+      var dataOffset = 0;
+      var offset = map[q].x + map[q].y * imgWidth << 2;
+      imgData.set(data.subarray(0, rowSize), offset - imgRowSize);
+      for (var k = 0, kk = map[q].h; k < kk; k++) {
+        imgData.set(data.subarray(dataOffset, dataOffset + rowSize), offset);
+        dataOffset += rowSize;
+        offset += imgRowSize;
+      }
+      imgData.set(data.subarray(dataOffset - rowSize, dataOffset), offset);
+      while (offset >= 0) {
+        data[offset - 4] = data[offset];
+        data[offset - 3] = data[offset + 1];
+        data[offset - 2] = data[offset + 2];
+        data[offset - 1] = data[offset + 3];
+        data[offset + rowSize] = data[offset + rowSize - 4];
+        data[offset + rowSize + 1] = data[offset + rowSize - 3];
+        data[offset + rowSize + 2] = data[offset + rowSize - 2];
+        data[offset + rowSize + 3] = data[offset + rowSize - 1];
+        offset -= imgRowSize;
+      }
+    }
+    fnArray.splice(iFirstSave, count * 4, OPS.paintInlineImageXObjectGroup);
+    argsArray.splice(iFirstSave, count * 4, [{
+      width: imgWidth,
+      height: imgHeight,
+      kind: ImageKind.RGBA_32BPP,
+      data: imgData
+    }, map]);
+    return iFirstSave + 1;
+  });
+  addState(InitialState, [OPS.save, OPS.transform, OPS.paintImageMaskXObject, OPS.restore], function foundImageMaskGroup(context) {
+    var MIN_IMAGES_IN_MASKS_BLOCK = 10;
+    var MAX_IMAGES_IN_MASKS_BLOCK = 100;
+    var MAX_SAME_IMAGES_IN_MASKS_BLOCK = 1000;
+    var fnArray = context.fnArray,
+        argsArray = context.argsArray;
+    var curr = context.iCurr;
+    var iFirstSave = curr - 3;
+    var iFirstTransform = curr - 2;
+    var iFirstPIMXO = curr - 1;
+    var i = iFirstSave + 4;
+    var ii = fnArray.length;
+    while (i + 3 < ii) {
+      if (fnArray[i] !== OPS.save || fnArray[i + 1] !== OPS.transform || fnArray[i + 2] !== OPS.paintImageMaskXObject || fnArray[i + 3] !== OPS.restore) {
+        break;
+      }
+      i += 4;
+    }
+    var count = (i - iFirstSave) / 4;
+    count = handlePaintSolidColorImageMask(iFirstSave, count, fnArray, argsArray);
+    if (count < MIN_IMAGES_IN_MASKS_BLOCK) {
+      return i;
+    }
+    var q;
+    var isSameImage = false;
+    var iTransform, transformArgs;
+    var firstPIMXOArg0 = argsArray[iFirstPIMXO][0];
+    if (argsArray[iFirstTransform][1] === 0 && argsArray[iFirstTransform][2] === 0) {
+      isSameImage = true;
+      var firstTransformArg0 = argsArray[iFirstTransform][0];
+      var firstTransformArg3 = argsArray[iFirstTransform][3];
+      iTransform = iFirstTransform + 4;
+      var iPIMXO = iFirstPIMXO + 4;
+      for (q = 1; q < count; q++, iTransform += 4, iPIMXO += 4) {
+        transformArgs = argsArray[iTransform];
+        if (argsArray[iPIMXO][0] !== firstPIMXOArg0 || transformArgs[0] !== firstTransformArg0 || transformArgs[1] !== 0 || transformArgs[2] !== 0 || transformArgs[3] !== firstTransformArg3) {
+          if (q < MIN_IMAGES_IN_MASKS_BLOCK) {
+            isSameImage = false;
+          } else {
+            count = q;
+          }
+          break;
+        }
+      }
+    }
+    if (isSameImage) {
+      count = Math.min(count, MAX_SAME_IMAGES_IN_MASKS_BLOCK);
+      var positions = new Float32Array(count * 2);
+      iTransform = iFirstTransform;
+      for (q = 0; q < count; q++, iTransform += 4) {
+        transformArgs = argsArray[iTransform];
+        positions[q << 1] = transformArgs[4];
+        positions[(q << 1) + 1] = transformArgs[5];
+      }
+      fnArray.splice(iFirstSave, count * 4, OPS.paintImageMaskXObjectRepeat);
+      argsArray.splice(iFirstSave, count * 4, [firstPIMXOArg0, firstTransformArg0, firstTransformArg3, positions]);
     } else {
-     i++;
+      count = Math.min(count, MAX_IMAGES_IN_MASKS_BLOCK);
+      var images = [];
+      for (q = 0; q < count; q++) {
+        transformArgs = argsArray[iFirstTransform + (q << 2)];
+        var maskParams = argsArray[iFirstPIMXO + (q << 2)][0];
+        images.push({
+          data: maskParams.data,
+          width: maskParams.width,
+          height: maskParams.height,
+          transform: transformArgs
+        });
+      }
+      fnArray.splice(iFirstSave, count * 4, OPS.paintImageMaskXObjectGroup);
+      argsArray.splice(iFirstSave, count * 4, [images]);
+    }
+    return iFirstSave + 1;
+  });
+  addState(InitialState, [OPS.save, OPS.transform, OPS.paintImageXObject, OPS.restore], function (context) {
+    var MIN_IMAGES_IN_BLOCK = 3;
+    var MAX_IMAGES_IN_BLOCK = 1000;
+    var fnArray = context.fnArray,
+        argsArray = context.argsArray;
+    var curr = context.iCurr;
+    var iFirstSave = curr - 3;
+    var iFirstTransform = curr - 2;
+    var iFirstPIXO = curr - 1;
+    var iFirstRestore = curr;
+    if (argsArray[iFirstTransform][1] !== 0 || argsArray[iFirstTransform][2] !== 0) {
+      return iFirstRestore + 1;
+    }
+    var firstPIXOArg0 = argsArray[iFirstPIXO][0];
+    var firstTransformArg0 = argsArray[iFirstTransform][0];
+    var firstTransformArg3 = argsArray[iFirstTransform][3];
+    var i = iFirstSave + 4;
+    var ii = fnArray.length;
+    while (i + 3 < ii) {
+      if (fnArray[i] !== OPS.save || fnArray[i + 1] !== OPS.transform || fnArray[i + 2] !== OPS.paintImageXObject || fnArray[i + 3] !== OPS.restore) {
+        break;
+      }
+      if (argsArray[i + 1][0] !== firstTransformArg0 || argsArray[i + 1][1] !== 0 || argsArray[i + 1][2] !== 0 || argsArray[i + 1][3] !== firstTransformArg3) {
+        break;
+      }
+      if (argsArray[i + 2][0] !== firstPIXOArg0) {
+        break;
+      }
+      i += 4;
+    }
+    var count = Math.min((i - iFirstSave) / 4, MAX_IMAGES_IN_BLOCK);
+    if (count < MIN_IMAGES_IN_BLOCK) {
+      return i;
+    }
+    var positions = new Float32Array(count * 2);
+    var iTransform = iFirstTransform;
+    for (var q = 0; q < count; q++, iTransform += 4) {
+      var transformArgs = argsArray[iTransform];
+      positions[q << 1] = transformArgs[4];
+      positions[(q << 1) + 1] = transformArgs[5];
+    }
+    var args = [firstPIXOArg0, firstTransformArg0, firstTransformArg3, positions];
+    fnArray.splice(iFirstSave, count * 4, OPS.paintImageXObjectRepeat);
+    argsArray.splice(iFirstSave, count * 4, args);
+    return iFirstSave + 1;
+  });
+  addState(InitialState, [OPS.beginText, OPS.setFont, OPS.setTextMatrix, OPS.showText, OPS.endText], function (context) {
+    var MIN_CHARS_IN_BLOCK = 3;
+    var MAX_CHARS_IN_BLOCK = 1000;
+    var fnArray = context.fnArray,
+        argsArray = context.argsArray;
+    var curr = context.iCurr;
+    var iFirstBeginText = curr - 4;
+    var iFirstSetFont = curr - 3;
+    var iFirstSetTextMatrix = curr - 2;
+    var iFirstShowText = curr - 1;
+    var iFirstEndText = curr;
+    var firstSetFontArg0 = argsArray[iFirstSetFont][0];
+    var firstSetFontArg1 = argsArray[iFirstSetFont][1];
+    var i = iFirstBeginText + 5;
+    var ii = fnArray.length;
+    while (i + 4 < ii) {
+      if (fnArray[i] !== OPS.beginText || fnArray[i + 1] !== OPS.setFont || fnArray[i + 2] !== OPS.setTextMatrix || fnArray[i + 3] !== OPS.showText || fnArray[i + 4] !== OPS.endText) {
+        break;
+      }
+      if (argsArray[i + 1][0] !== firstSetFontArg0 || argsArray[i + 1][1] !== firstSetFontArg1) {
+        break;
+      }
+      i += 5;
+    }
+    var count = Math.min((i - iFirstBeginText) / 5, MAX_CHARS_IN_BLOCK);
+    if (count < MIN_CHARS_IN_BLOCK) {
+      return i;
+    }
+    var iFirst = iFirstBeginText;
+    if (iFirstBeginText >= 4 && fnArray[iFirstBeginText - 4] === fnArray[iFirstSetFont] && fnArray[iFirstBeginText - 3] === fnArray[iFirstSetTextMatrix] && fnArray[iFirstBeginText - 2] === fnArray[iFirstShowText] && fnArray[iFirstBeginText - 1] === fnArray[iFirstEndText] && argsArray[iFirstBeginText - 4][0] === firstSetFontArg0 && argsArray[iFirstBeginText - 4][1] === firstSetFontArg1) {
+      count++;
+      iFirst -= 5;
+    }
+    var iEndText = iFirst + 4;
+    for (var q = 1; q < count; q++) {
+      fnArray.splice(iEndText, 3);
+      argsArray.splice(iEndText, 3);
+      iEndText += 2;
+    }
+    return iEndText + 1;
+  });
+  function QueueOptimizer() {}
+  QueueOptimizer.prototype = {
+    optimize: function QueueOptimizer_optimize(queue) {
+      var fnArray = queue.fnArray,
+          argsArray = queue.argsArray;
+      var context = {
+        iCurr: 0,
+        fnArray: fnArray,
+        argsArray: argsArray
+      };
+      var state;
+      var i = 0,
+          ii = fnArray.length;
+      while (i < ii) {
+        state = (state || InitialState)[fnArray[i]];
+        if (typeof state === 'function') {
+          context.iCurr = i;
+          i = state(context);
+          state = undefined;
+          ii = context.fnArray.length;
+        } else {
+          i++;
+        }
+      }
     }
-   }
-  }
- };
- return QueueOptimizer;
+  };
+  return QueueOptimizer;
 }();
 exports.OperatorList = OperatorList;
 exports.PartialEvaluator = PartialEvaluator;

+ 686 - 735
lib/core/font_renderer.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var coreStream = require('./stream.js');
 var coreGlyphList = require('./glyphlist.js');
@@ -26,769 +27,719 @@ var getGlyphsUnicode = coreGlyphList.getGlyphsUnicode;
 var StandardEncoding = coreEncodings.StandardEncoding;
 var CFFParser = coreCFFParser.CFFParser;
 var FontRendererFactory = function FontRendererFactoryClosure() {
- function getLong(data, offset) {
-  return data[offset] << 24 | data[offset + 1] << 16 | data[offset + 2] << 8 | data[offset + 3];
- }
- function getUshort(data, offset) {
-  return data[offset] << 8 | data[offset + 1];
- }
- function parseCmap(data, start, end) {
-  var offset = getUshort(data, start + 2) === 1 ? getLong(data, start + 8) : getLong(data, start + 16);
-  var format = getUshort(data, start + offset);
-  var ranges, p, i;
-  if (format === 4) {
-   getUshort(data, start + offset + 2);
-   var segCount = getUshort(data, start + offset + 6) >> 1;
-   p = start + offset + 14;
-   ranges = [];
-   for (i = 0; i < segCount; i++, p += 2) {
-    ranges[i] = { end: getUshort(data, p) };
-   }
-   p += 2;
-   for (i = 0; i < segCount; i++, p += 2) {
-    ranges[i].start = getUshort(data, p);
-   }
-   for (i = 0; i < segCount; i++, p += 2) {
-    ranges[i].idDelta = getUshort(data, p);
-   }
-   for (i = 0; i < segCount; i++, p += 2) {
-    var idOffset = getUshort(data, p);
-    if (idOffset === 0) {
-     continue;
-    }
-    ranges[i].ids = [];
-    for (var j = 0, jj = ranges[i].end - ranges[i].start + 1; j < jj; j++) {
-     ranges[i].ids[j] = getUshort(data, p + idOffset);
-     idOffset += 2;
-    }
-   }
-   return ranges;
-  } else if (format === 12) {
-   getLong(data, start + offset + 4);
-   var groups = getLong(data, start + offset + 12);
-   p = start + offset + 16;
-   ranges = [];
-   for (i = 0; i < groups; i++) {
-    ranges.push({
-     start: getLong(data, p),
-     end: getLong(data, p + 4),
-     idDelta: getLong(data, p + 8) - getLong(data, p)
-    });
-    p += 12;
-   }
-   return ranges;
-  }
-  error('not supported cmap: ' + format);
- }
- function parseCff(data, start, end, seacAnalysisEnabled) {
-  var properties = {};
-  var parser = new CFFParser(new Stream(data, start, end - start), properties, seacAnalysisEnabled);
-  var cff = parser.parse();
-  return {
-   glyphs: cff.charStrings.objects,
-   subrs: cff.topDict.privateDict && cff.topDict.privateDict.subrsIndex && cff.topDict.privateDict.subrsIndex.objects,
-   gsubrs: cff.globalSubrIndex && cff.globalSubrIndex.objects
-  };
- }
- function parseGlyfTable(glyf, loca, isGlyphLocationsLong) {
-  var itemSize, itemDecode;
-  if (isGlyphLocationsLong) {
-   itemSize = 4;
-   itemDecode = function fontItemDecodeLong(data, offset) {
+  function getLong(data, offset) {
     return data[offset] << 24 | data[offset + 1] << 16 | data[offset + 2] << 8 | data[offset + 3];
-   };
-  } else {
-   itemSize = 2;
-   itemDecode = function fontItemDecode(data, offset) {
-    return data[offset] << 9 | data[offset + 1] << 1;
-   };
-  }
-  var glyphs = [];
-  var startOffset = itemDecode(loca, 0);
-  for (var j = itemSize; j < loca.length; j += itemSize) {
-   var endOffset = itemDecode(loca, j);
-   glyphs.push(glyf.subarray(startOffset, endOffset));
-   startOffset = endOffset;
-  }
-  return glyphs;
- }
- function lookupCmap(ranges, unicode) {
-  var code = unicode.charCodeAt(0), gid = 0;
-  var l = 0, r = ranges.length - 1;
-  while (l < r) {
-   var c = l + r + 1 >> 1;
-   if (code < ranges[c].start) {
-    r = c - 1;
-   } else {
-    l = c;
-   }
-  }
-  if (ranges[l].start <= code && code <= ranges[l].end) {
-   gid = ranges[l].idDelta + (ranges[l].ids ? ranges[l].ids[code - ranges[l].start] : code) & 0xFFFF;
   }
-  return {
-   charCode: code,
-   glyphId: gid
-  };
- }
- function compileGlyf(code, cmds, font) {
-  function moveTo(x, y) {
-   cmds.push({
-    cmd: 'moveTo',
-    args: [
-     x,
-     y
-    ]
-   });
+  function getUshort(data, offset) {
+    return data[offset] << 8 | data[offset + 1];
   }
-  function lineTo(x, y) {
-   cmds.push({
-    cmd: 'lineTo',
-    args: [
-     x,
-     y
-    ]
-   });
+  function parseCmap(data, start, end) {
+    var offset = getUshort(data, start + 2) === 1 ? getLong(data, start + 8) : getLong(data, start + 16);
+    var format = getUshort(data, start + offset);
+    var ranges, p, i;
+    if (format === 4) {
+      getUshort(data, start + offset + 2);
+      var segCount = getUshort(data, start + offset + 6) >> 1;
+      p = start + offset + 14;
+      ranges = [];
+      for (i = 0; i < segCount; i++, p += 2) {
+        ranges[i] = { end: getUshort(data, p) };
+      }
+      p += 2;
+      for (i = 0; i < segCount; i++, p += 2) {
+        ranges[i].start = getUshort(data, p);
+      }
+      for (i = 0; i < segCount; i++, p += 2) {
+        ranges[i].idDelta = getUshort(data, p);
+      }
+      for (i = 0; i < segCount; i++, p += 2) {
+        var idOffset = getUshort(data, p);
+        if (idOffset === 0) {
+          continue;
+        }
+        ranges[i].ids = [];
+        for (var j = 0, jj = ranges[i].end - ranges[i].start + 1; j < jj; j++) {
+          ranges[i].ids[j] = getUshort(data, p + idOffset);
+          idOffset += 2;
+        }
+      }
+      return ranges;
+    } else if (format === 12) {
+      getLong(data, start + offset + 4);
+      var groups = getLong(data, start + offset + 12);
+      p = start + offset + 16;
+      ranges = [];
+      for (i = 0; i < groups; i++) {
+        ranges.push({
+          start: getLong(data, p),
+          end: getLong(data, p + 4),
+          idDelta: getLong(data, p + 8) - getLong(data, p)
+        });
+        p += 12;
+      }
+      return ranges;
+    }
+    error('not supported cmap: ' + format);
   }
-  function quadraticCurveTo(xa, ya, x, y) {
-   cmds.push({
-    cmd: 'quadraticCurveTo',
-    args: [
-     xa,
-     ya,
-     x,
-     y
-    ]
-   });
+  function parseCff(data, start, end, seacAnalysisEnabled) {
+    var properties = {};
+    var parser = new CFFParser(new Stream(data, start, end - start), properties, seacAnalysisEnabled);
+    var cff = parser.parse();
+    return {
+      glyphs: cff.charStrings.objects,
+      subrs: cff.topDict.privateDict && cff.topDict.privateDict.subrsIndex && cff.topDict.privateDict.subrsIndex.objects,
+      gsubrs: cff.globalSubrIndex && cff.globalSubrIndex.objects
+    };
   }
-  var i = 0;
-  var numberOfContours = (code[i] << 24 | code[i + 1] << 16) >> 16;
-  var flags;
-  var x = 0, y = 0;
-  i += 10;
-  if (numberOfContours < 0) {
-   do {
-    flags = code[i] << 8 | code[i + 1];
-    var glyphIndex = code[i + 2] << 8 | code[i + 3];
-    i += 4;
-    var arg1, arg2;
-    if (flags & 0x01) {
-     arg1 = (code[i] << 24 | code[i + 1] << 16) >> 16;
-     arg2 = (code[i + 2] << 24 | code[i + 3] << 16) >> 16;
-     i += 4;
+  function parseGlyfTable(glyf, loca, isGlyphLocationsLong) {
+    var itemSize, itemDecode;
+    if (isGlyphLocationsLong) {
+      itemSize = 4;
+      itemDecode = function fontItemDecodeLong(data, offset) {
+        return data[offset] << 24 | data[offset + 1] << 16 | data[offset + 2] << 8 | data[offset + 3];
+      };
     } else {
-     arg1 = code[i++];
-     arg2 = code[i++];
+      itemSize = 2;
+      itemDecode = function fontItemDecode(data, offset) {
+        return data[offset] << 9 | data[offset + 1] << 1;
+      };
     }
-    if (flags & 0x02) {
-     x = arg1;
-     y = arg2;
-    } else {
-     x = 0;
-     y = 0;
-    }
-    var scaleX = 1, scaleY = 1, scale01 = 0, scale10 = 0;
-    if (flags & 0x08) {
-     scaleX = scaleY = (code[i] << 24 | code[i + 1] << 16) / 1073741824;
-     i += 2;
-    } else if (flags & 0x40) {
-     scaleX = (code[i] << 24 | code[i + 1] << 16) / 1073741824;
-     scaleY = (code[i + 2] << 24 | code[i + 3] << 16) / 1073741824;
-     i += 4;
-    } else if (flags & 0x80) {
-     scaleX = (code[i] << 24 | code[i + 1] << 16) / 1073741824;
-     scale01 = (code[i + 2] << 24 | code[i + 3] << 16) / 1073741824;
-     scale10 = (code[i + 4] << 24 | code[i + 5] << 16) / 1073741824;
-     scaleY = (code[i + 6] << 24 | code[i + 7] << 16) / 1073741824;
-     i += 8;
+    var glyphs = [];
+    var startOffset = itemDecode(loca, 0);
+    for (var j = itemSize; j < loca.length; j += itemSize) {
+      var endOffset = itemDecode(loca, j);
+      glyphs.push(glyf.subarray(startOffset, endOffset));
+      startOffset = endOffset;
     }
-    var subglyph = font.glyphs[glyphIndex];
-    if (subglyph) {
-     cmds.push({ cmd: 'save' });
-     cmds.push({
-      cmd: 'transform',
-      args: [
-       scaleX,
-       scale01,
-       scale10,
-       scaleY,
-       x,
-       y
-      ]
-     });
-     compileGlyf(subglyph, cmds, font);
-     cmds.push({ cmd: 'restore' });
+    return glyphs;
+  }
+  function lookupCmap(ranges, unicode) {
+    var code = unicode.charCodeAt(0),
+        gid = 0;
+    var l = 0,
+        r = ranges.length - 1;
+    while (l < r) {
+      var c = l + r + 1 >> 1;
+      if (code < ranges[c].start) {
+        r = c - 1;
+      } else {
+        l = c;
+      }
     }
-   } while (flags & 0x20);
-  } else {
-   var endPtsOfContours = [];
-   var j, jj;
-   for (j = 0; j < numberOfContours; j++) {
-    endPtsOfContours.push(code[i] << 8 | code[i + 1]);
-    i += 2;
-   }
-   var instructionLength = code[i] << 8 | code[i + 1];
-   i += 2 + instructionLength;
-   var numberOfPoints = endPtsOfContours[endPtsOfContours.length - 1] + 1;
-   var points = [];
-   while (points.length < numberOfPoints) {
-    flags = code[i++];
-    var repeat = 1;
-    if (flags & 0x08) {
-     repeat += code[i++];
+    if (ranges[l].start <= code && code <= ranges[l].end) {
+      gid = ranges[l].idDelta + (ranges[l].ids ? ranges[l].ids[code - ranges[l].start] : code) & 0xFFFF;
     }
-    while (repeat-- > 0) {
-     points.push({ flags: flags });
+    return {
+      charCode: code,
+      glyphId: gid
+    };
+  }
+  function compileGlyf(code, cmds, font) {
+    function moveTo(x, y) {
+      cmds.push({
+        cmd: 'moveTo',
+        args: [x, y]
+      });
     }
-   }
-   for (j = 0; j < numberOfPoints; j++) {
-    switch (points[j].flags & 0x12) {
-    case 0x00:
-     x += (code[i] << 24 | code[i + 1] << 16) >> 16;
-     i += 2;
-     break;
-    case 0x02:
-     x -= code[i++];
-     break;
-    case 0x12:
-     x += code[i++];
-     break;
+    function lineTo(x, y) {
+      cmds.push({
+        cmd: 'lineTo',
+        args: [x, y]
+      });
     }
-    points[j].x = x;
-   }
-   for (j = 0; j < numberOfPoints; j++) {
-    switch (points[j].flags & 0x24) {
-    case 0x00:
-     y += (code[i] << 24 | code[i + 1] << 16) >> 16;
-     i += 2;
-     break;
-    case 0x04:
-     y -= code[i++];
-     break;
-    case 0x24:
-     y += code[i++];
-     break;
+    function quadraticCurveTo(xa, ya, x, y) {
+      cmds.push({
+        cmd: 'quadraticCurveTo',
+        args: [xa, ya, x, y]
+      });
     }
-    points[j].y = y;
-   }
-   var startPoint = 0;
-   for (i = 0; i < numberOfContours; i++) {
-    var endPoint = endPtsOfContours[i];
-    var contour = points.slice(startPoint, endPoint + 1);
-    if (contour[0].flags & 1) {
-     contour.push(contour[0]);
-    } else if (contour[contour.length - 1].flags & 1) {
-     contour.unshift(contour[contour.length - 1]);
+    var i = 0;
+    var numberOfContours = (code[i] << 24 | code[i + 1] << 16) >> 16;
+    var flags;
+    var x = 0,
+        y = 0;
+    i += 10;
+    if (numberOfContours < 0) {
+      do {
+        flags = code[i] << 8 | code[i + 1];
+        var glyphIndex = code[i + 2] << 8 | code[i + 3];
+        i += 4;
+        var arg1, arg2;
+        if (flags & 0x01) {
+          arg1 = (code[i] << 24 | code[i + 1] << 16) >> 16;
+          arg2 = (code[i + 2] << 24 | code[i + 3] << 16) >> 16;
+          i += 4;
+        } else {
+          arg1 = code[i++];
+          arg2 = code[i++];
+        }
+        if (flags & 0x02) {
+          x = arg1;
+          y = arg2;
+        } else {
+          x = 0;
+          y = 0;
+        }
+        var scaleX = 1,
+            scaleY = 1,
+            scale01 = 0,
+            scale10 = 0;
+        if (flags & 0x08) {
+          scaleX = scaleY = (code[i] << 24 | code[i + 1] << 16) / 1073741824;
+          i += 2;
+        } else if (flags & 0x40) {
+          scaleX = (code[i] << 24 | code[i + 1] << 16) / 1073741824;
+          scaleY = (code[i + 2] << 24 | code[i + 3] << 16) / 1073741824;
+          i += 4;
+        } else if (flags & 0x80) {
+          scaleX = (code[i] << 24 | code[i + 1] << 16) / 1073741824;
+          scale01 = (code[i + 2] << 24 | code[i + 3] << 16) / 1073741824;
+          scale10 = (code[i + 4] << 24 | code[i + 5] << 16) / 1073741824;
+          scaleY = (code[i + 6] << 24 | code[i + 7] << 16) / 1073741824;
+          i += 8;
+        }
+        var subglyph = font.glyphs[glyphIndex];
+        if (subglyph) {
+          cmds.push({ cmd: 'save' });
+          cmds.push({
+            cmd: 'transform',
+            args: [scaleX, scale01, scale10, scaleY, x, y]
+          });
+          compileGlyf(subglyph, cmds, font);
+          cmds.push({ cmd: 'restore' });
+        }
+      } while (flags & 0x20);
     } else {
-     var p = {
-      flags: 1,
-      x: (contour[0].x + contour[contour.length - 1].x) / 2,
-      y: (contour[0].y + contour[contour.length - 1].y) / 2
-     };
-     contour.unshift(p);
-     contour.push(p);
-    }
-    moveTo(contour[0].x, contour[0].y);
-    for (j = 1, jj = contour.length; j < jj; j++) {
-     if (contour[j].flags & 1) {
-      lineTo(contour[j].x, contour[j].y);
-     } else if (contour[j + 1].flags & 1) {
-      quadraticCurveTo(contour[j].x, contour[j].y, contour[j + 1].x, contour[j + 1].y);
-      j++;
-     } else {
-      quadraticCurveTo(contour[j].x, contour[j].y, (contour[j].x + contour[j + 1].x) / 2, (contour[j].y + contour[j + 1].y) / 2);
-     }
+      var endPtsOfContours = [];
+      var j, jj;
+      for (j = 0; j < numberOfContours; j++) {
+        endPtsOfContours.push(code[i] << 8 | code[i + 1]);
+        i += 2;
+      }
+      var instructionLength = code[i] << 8 | code[i + 1];
+      i += 2 + instructionLength;
+      var numberOfPoints = endPtsOfContours[endPtsOfContours.length - 1] + 1;
+      var points = [];
+      while (points.length < numberOfPoints) {
+        flags = code[i++];
+        var repeat = 1;
+        if (flags & 0x08) {
+          repeat += code[i++];
+        }
+        while (repeat-- > 0) {
+          points.push({ flags: flags });
+        }
+      }
+      for (j = 0; j < numberOfPoints; j++) {
+        switch (points[j].flags & 0x12) {
+          case 0x00:
+            x += (code[i] << 24 | code[i + 1] << 16) >> 16;
+            i += 2;
+            break;
+          case 0x02:
+            x -= code[i++];
+            break;
+          case 0x12:
+            x += code[i++];
+            break;
+        }
+        points[j].x = x;
+      }
+      for (j = 0; j < numberOfPoints; j++) {
+        switch (points[j].flags & 0x24) {
+          case 0x00:
+            y += (code[i] << 24 | code[i + 1] << 16) >> 16;
+            i += 2;
+            break;
+          case 0x04:
+            y -= code[i++];
+            break;
+          case 0x24:
+            y += code[i++];
+            break;
+        }
+        points[j].y = y;
+      }
+      var startPoint = 0;
+      for (i = 0; i < numberOfContours; i++) {
+        var endPoint = endPtsOfContours[i];
+        var contour = points.slice(startPoint, endPoint + 1);
+        if (contour[0].flags & 1) {
+          contour.push(contour[0]);
+        } else if (contour[contour.length - 1].flags & 1) {
+          contour.unshift(contour[contour.length - 1]);
+        } else {
+          var p = {
+            flags: 1,
+            x: (contour[0].x + contour[contour.length - 1].x) / 2,
+            y: (contour[0].y + contour[contour.length - 1].y) / 2
+          };
+          contour.unshift(p);
+          contour.push(p);
+        }
+        moveTo(contour[0].x, contour[0].y);
+        for (j = 1, jj = contour.length; j < jj; j++) {
+          if (contour[j].flags & 1) {
+            lineTo(contour[j].x, contour[j].y);
+          } else if (contour[j + 1].flags & 1) {
+            quadraticCurveTo(contour[j].x, contour[j].y, contour[j + 1].x, contour[j + 1].y);
+            j++;
+          } else {
+            quadraticCurveTo(contour[j].x, contour[j].y, (contour[j].x + contour[j + 1].x) / 2, (contour[j].y + contour[j + 1].y) / 2);
+          }
+        }
+        startPoint = endPoint + 1;
+      }
     }
-    startPoint = endPoint + 1;
-   }
-  }
- }
- function compileCharString(code, cmds, font) {
-  var stack = [];
-  var x = 0, y = 0;
-  var stems = 0;
-  function moveTo(x, y) {
-   cmds.push({
-    cmd: 'moveTo',
-    args: [
-     x,
-     y
-    ]
-   });
   }
-  function lineTo(x, y) {
-   cmds.push({
-    cmd: 'lineTo',
-    args: [
-     x,
-     y
-    ]
-   });
+  function compileCharString(code, cmds, font) {
+    var stack = [];
+    var x = 0,
+        y = 0;
+    var stems = 0;
+    function moveTo(x, y) {
+      cmds.push({
+        cmd: 'moveTo',
+        args: [x, y]
+      });
+    }
+    function lineTo(x, y) {
+      cmds.push({
+        cmd: 'lineTo',
+        args: [x, y]
+      });
+    }
+    function bezierCurveTo(x1, y1, x2, y2, x, y) {
+      cmds.push({
+        cmd: 'bezierCurveTo',
+        args: [x1, y1, x2, y2, x, y]
+      });
+    }
+    function parse(code) {
+      var i = 0;
+      while (i < code.length) {
+        var stackClean = false;
+        var v = code[i++];
+        var xa, xb, ya, yb, y1, y2, y3, n, subrCode;
+        switch (v) {
+          case 1:
+            stems += stack.length >> 1;
+            stackClean = true;
+            break;
+          case 3:
+            stems += stack.length >> 1;
+            stackClean = true;
+            break;
+          case 4:
+            y += stack.pop();
+            moveTo(x, y);
+            stackClean = true;
+            break;
+          case 5:
+            while (stack.length > 0) {
+              x += stack.shift();
+              y += stack.shift();
+              lineTo(x, y);
+            }
+            break;
+          case 6:
+            while (stack.length > 0) {
+              x += stack.shift();
+              lineTo(x, y);
+              if (stack.length === 0) {
+                break;
+              }
+              y += stack.shift();
+              lineTo(x, y);
+            }
+            break;
+          case 7:
+            while (stack.length > 0) {
+              y += stack.shift();
+              lineTo(x, y);
+              if (stack.length === 0) {
+                break;
+              }
+              x += stack.shift();
+              lineTo(x, y);
+            }
+            break;
+          case 8:
+            while (stack.length > 0) {
+              xa = x + stack.shift();
+              ya = y + stack.shift();
+              xb = xa + stack.shift();
+              yb = ya + stack.shift();
+              x = xb + stack.shift();
+              y = yb + stack.shift();
+              bezierCurveTo(xa, ya, xb, yb, x, y);
+            }
+            break;
+          case 10:
+            n = stack.pop() + font.subrsBias;
+            subrCode = font.subrs[n];
+            if (subrCode) {
+              parse(subrCode);
+            }
+            break;
+          case 11:
+            return;
+          case 12:
+            v = code[i++];
+            switch (v) {
+              case 34:
+                xa = x + stack.shift();
+                xb = xa + stack.shift();
+                y1 = y + stack.shift();
+                x = xb + stack.shift();
+                bezierCurveTo(xa, y, xb, y1, x, y1);
+                xa = x + stack.shift();
+                xb = xa + stack.shift();
+                x = xb + stack.shift();
+                bezierCurveTo(xa, y1, xb, y, x, y);
+                break;
+              case 35:
+                xa = x + stack.shift();
+                ya = y + stack.shift();
+                xb = xa + stack.shift();
+                yb = ya + stack.shift();
+                x = xb + stack.shift();
+                y = yb + stack.shift();
+                bezierCurveTo(xa, ya, xb, yb, x, y);
+                xa = x + stack.shift();
+                ya = y + stack.shift();
+                xb = xa + stack.shift();
+                yb = ya + stack.shift();
+                x = xb + stack.shift();
+                y = yb + stack.shift();
+                bezierCurveTo(xa, ya, xb, yb, x, y);
+                stack.pop();
+                break;
+              case 36:
+                xa = x + stack.shift();
+                y1 = y + stack.shift();
+                xb = xa + stack.shift();
+                y2 = y1 + stack.shift();
+                x = xb + stack.shift();
+                bezierCurveTo(xa, y1, xb, y2, x, y2);
+                xa = x + stack.shift();
+                xb = xa + stack.shift();
+                y3 = y2 + stack.shift();
+                x = xb + stack.shift();
+                bezierCurveTo(xa, y2, xb, y3, x, y);
+                break;
+              case 37:
+                var x0 = x,
+                    y0 = y;
+                xa = x + stack.shift();
+                ya = y + stack.shift();
+                xb = xa + stack.shift();
+                yb = ya + stack.shift();
+                x = xb + stack.shift();
+                y = yb + stack.shift();
+                bezierCurveTo(xa, ya, xb, yb, x, y);
+                xa = x + stack.shift();
+                ya = y + stack.shift();
+                xb = xa + stack.shift();
+                yb = ya + stack.shift();
+                x = xb;
+                y = yb;
+                if (Math.abs(x - x0) > Math.abs(y - y0)) {
+                  x += stack.shift();
+                } else {
+                  y += stack.shift();
+                }
+                bezierCurveTo(xa, ya, xb, yb, x, y);
+                break;
+              default:
+                error('unknown operator: 12 ' + v);
+            }
+            break;
+          case 14:
+            if (stack.length >= 4) {
+              var achar = stack.pop();
+              var bchar = stack.pop();
+              y = stack.pop();
+              x = stack.pop();
+              cmds.push({ cmd: 'save' });
+              cmds.push({
+                cmd: 'translate',
+                args: [x, y]
+              });
+              var cmap = lookupCmap(font.cmap, String.fromCharCode(font.glyphNameMap[StandardEncoding[achar]]));
+              compileCharString(font.glyphs[cmap.glyphId], cmds, font);
+              cmds.push({ cmd: 'restore' });
+              cmap = lookupCmap(font.cmap, String.fromCharCode(font.glyphNameMap[StandardEncoding[bchar]]));
+              compileCharString(font.glyphs[cmap.glyphId], cmds, font);
+            }
+            return;
+          case 18:
+            stems += stack.length >> 1;
+            stackClean = true;
+            break;
+          case 19:
+            stems += stack.length >> 1;
+            i += stems + 7 >> 3;
+            stackClean = true;
+            break;
+          case 20:
+            stems += stack.length >> 1;
+            i += stems + 7 >> 3;
+            stackClean = true;
+            break;
+          case 21:
+            y += stack.pop();
+            x += stack.pop();
+            moveTo(x, y);
+            stackClean = true;
+            break;
+          case 22:
+            x += stack.pop();
+            moveTo(x, y);
+            stackClean = true;
+            break;
+          case 23:
+            stems += stack.length >> 1;
+            stackClean = true;
+            break;
+          case 24:
+            while (stack.length > 2) {
+              xa = x + stack.shift();
+              ya = y + stack.shift();
+              xb = xa + stack.shift();
+              yb = ya + stack.shift();
+              x = xb + stack.shift();
+              y = yb + stack.shift();
+              bezierCurveTo(xa, ya, xb, yb, x, y);
+            }
+            x += stack.shift();
+            y += stack.shift();
+            lineTo(x, y);
+            break;
+          case 25:
+            while (stack.length > 6) {
+              x += stack.shift();
+              y += stack.shift();
+              lineTo(x, y);
+            }
+            xa = x + stack.shift();
+            ya = y + stack.shift();
+            xb = xa + stack.shift();
+            yb = ya + stack.shift();
+            x = xb + stack.shift();
+            y = yb + stack.shift();
+            bezierCurveTo(xa, ya, xb, yb, x, y);
+            break;
+          case 26:
+            if (stack.length % 2) {
+              x += stack.shift();
+            }
+            while (stack.length > 0) {
+              xa = x;
+              ya = y + stack.shift();
+              xb = xa + stack.shift();
+              yb = ya + stack.shift();
+              x = xb;
+              y = yb + stack.shift();
+              bezierCurveTo(xa, ya, xb, yb, x, y);
+            }
+            break;
+          case 27:
+            if (stack.length % 2) {
+              y += stack.shift();
+            }
+            while (stack.length > 0) {
+              xa = x + stack.shift();
+              ya = y;
+              xb = xa + stack.shift();
+              yb = ya + stack.shift();
+              x = xb + stack.shift();
+              y = yb;
+              bezierCurveTo(xa, ya, xb, yb, x, y);
+            }
+            break;
+          case 28:
+            stack.push((code[i] << 24 | code[i + 1] << 16) >> 16);
+            i += 2;
+            break;
+          case 29:
+            n = stack.pop() + font.gsubrsBias;
+            subrCode = font.gsubrs[n];
+            if (subrCode) {
+              parse(subrCode);
+            }
+            break;
+          case 30:
+            while (stack.length > 0) {
+              xa = x;
+              ya = y + stack.shift();
+              xb = xa + stack.shift();
+              yb = ya + stack.shift();
+              x = xb + stack.shift();
+              y = yb + (stack.length === 1 ? stack.shift() : 0);
+              bezierCurveTo(xa, ya, xb, yb, x, y);
+              if (stack.length === 0) {
+                break;
+              }
+              xa = x + stack.shift();
+              ya = y;
+              xb = xa + stack.shift();
+              yb = ya + stack.shift();
+              y = yb + stack.shift();
+              x = xb + (stack.length === 1 ? stack.shift() : 0);
+              bezierCurveTo(xa, ya, xb, yb, x, y);
+            }
+            break;
+          case 31:
+            while (stack.length > 0) {
+              xa = x + stack.shift();
+              ya = y;
+              xb = xa + stack.shift();
+              yb = ya + stack.shift();
+              y = yb + stack.shift();
+              x = xb + (stack.length === 1 ? stack.shift() : 0);
+              bezierCurveTo(xa, ya, xb, yb, x, y);
+              if (stack.length === 0) {
+                break;
+              }
+              xa = x;
+              ya = y + stack.shift();
+              xb = xa + stack.shift();
+              yb = ya + stack.shift();
+              x = xb + stack.shift();
+              y = yb + (stack.length === 1 ? stack.shift() : 0);
+              bezierCurveTo(xa, ya, xb, yb, x, y);
+            }
+            break;
+          default:
+            if (v < 32) {
+              error('unknown operator: ' + v);
+            }
+            if (v < 247) {
+              stack.push(v - 139);
+            } else if (v < 251) {
+              stack.push((v - 247) * 256 + code[i++] + 108);
+            } else if (v < 255) {
+              stack.push(-(v - 251) * 256 - code[i++] - 108);
+            } else {
+              stack.push((code[i] << 24 | code[i + 1] << 16 | code[i + 2] << 8 | code[i + 3]) / 65536);
+              i += 4;
+            }
+            break;
+        }
+        if (stackClean) {
+          stack.length = 0;
+        }
+      }
+    }
+    parse(code);
   }
-  function bezierCurveTo(x1, y1, x2, y2, x, y) {
-   cmds.push({
-    cmd: 'bezierCurveTo',
-    args: [
-     x1,
-     y1,
-     x2,
-     y2,
-     x,
-     y
-    ]
-   });
+  var noop = '';
+  function CompiledFont(fontMatrix) {
+    this.compiledGlyphs = Object.create(null);
+    this.compiledCharCodeToGlyphId = Object.create(null);
+    this.fontMatrix = fontMatrix;
   }
-  function parse(code) {
-   var i = 0;
-   while (i < code.length) {
-    var stackClean = false;
-    var v = code[i++];
-    var xa, xb, ya, yb, y1, y2, y3, n, subrCode;
-    switch (v) {
-    case 1:
-     stems += stack.length >> 1;
-     stackClean = true;
-     break;
-    case 3:
-     stems += stack.length >> 1;
-     stackClean = true;
-     break;
-    case 4:
-     y += stack.pop();
-     moveTo(x, y);
-     stackClean = true;
-     break;
-    case 5:
-     while (stack.length > 0) {
-      x += stack.shift();
-      y += stack.shift();
-      lineTo(x, y);
-     }
-     break;
-    case 6:
-     while (stack.length > 0) {
-      x += stack.shift();
-      lineTo(x, y);
-      if (stack.length === 0) {
-       break;
+  CompiledFont.prototype = {
+    getPathJs: function (unicode) {
+      var cmap = lookupCmap(this.cmap, unicode);
+      var fn = this.compiledGlyphs[cmap.glyphId];
+      if (!fn) {
+        fn = this.compileGlyph(this.glyphs[cmap.glyphId]);
+        this.compiledGlyphs[cmap.glyphId] = fn;
       }
-      y += stack.shift();
-      lineTo(x, y);
-     }
-     break;
-    case 7:
-     while (stack.length > 0) {
-      y += stack.shift();
-      lineTo(x, y);
-      if (stack.length === 0) {
-       break;
+      if (this.compiledCharCodeToGlyphId[cmap.charCode] === undefined) {
+        this.compiledCharCodeToGlyphId[cmap.charCode] = cmap.glyphId;
       }
-      x += stack.shift();
-      lineTo(x, y);
-     }
-     break;
-    case 8:
-     while (stack.length > 0) {
-      xa = x + stack.shift();
-      ya = y + stack.shift();
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb + stack.shift();
-      y = yb + stack.shift();
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-     }
-     break;
-    case 10:
-     n = stack.pop() + font.subrsBias;
-     subrCode = font.subrs[n];
-     if (subrCode) {
-      parse(subrCode);
-     }
-     break;
-    case 11:
-     return;
-    case 12:
-     v = code[i++];
-     switch (v) {
-     case 34:
-      xa = x + stack.shift();
-      xb = xa + stack.shift();
-      y1 = y + stack.shift();
-      x = xb + stack.shift();
-      bezierCurveTo(xa, y, xb, y1, x, y1);
-      xa = x + stack.shift();
-      xb = xa + stack.shift();
-      x = xb + stack.shift();
-      bezierCurveTo(xa, y1, xb, y, x, y);
-      break;
-     case 35:
-      xa = x + stack.shift();
-      ya = y + stack.shift();
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb + stack.shift();
-      y = yb + stack.shift();
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-      xa = x + stack.shift();
-      ya = y + stack.shift();
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb + stack.shift();
-      y = yb + stack.shift();
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-      stack.pop();
-      break;
-     case 36:
-      xa = x + stack.shift();
-      y1 = y + stack.shift();
-      xb = xa + stack.shift();
-      y2 = y1 + stack.shift();
-      x = xb + stack.shift();
-      bezierCurveTo(xa, y1, xb, y2, x, y2);
-      xa = x + stack.shift();
-      xb = xa + stack.shift();
-      y3 = y2 + stack.shift();
-      x = xb + stack.shift();
-      bezierCurveTo(xa, y2, xb, y3, x, y);
-      break;
-     case 37:
-      var x0 = x, y0 = y;
-      xa = x + stack.shift();
-      ya = y + stack.shift();
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb + stack.shift();
-      y = yb + stack.shift();
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-      xa = x + stack.shift();
-      ya = y + stack.shift();
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb;
-      y = yb;
-      if (Math.abs(x - x0) > Math.abs(y - y0)) {
-       x += stack.shift();
-      } else {
-       y += stack.shift();
+      return fn;
+    },
+    compileGlyph: function (code) {
+      if (!code || code.length === 0 || code[0] === 14) {
+        return noop;
       }
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-      break;
-     default:
-      error('unknown operator: 12 ' + v);
-     }
-     break;
-    case 14:
-     if (stack.length >= 4) {
-      var achar = stack.pop();
-      var bchar = stack.pop();
-      y = stack.pop();
-      x = stack.pop();
+      var cmds = [];
       cmds.push({ cmd: 'save' });
       cmds.push({
-       cmd: 'translate',
-       args: [
-        x,
-        y
-       ]
+        cmd: 'transform',
+        args: this.fontMatrix.slice()
       });
-      var cmap = lookupCmap(font.cmap, String.fromCharCode(font.glyphNameMap[StandardEncoding[achar]]));
-      compileCharString(font.glyphs[cmap.glyphId], cmds, font);
+      cmds.push({
+        cmd: 'scale',
+        args: ['size', '-size']
+      });
+      this.compileGlyphImpl(code, cmds);
       cmds.push({ cmd: 'restore' });
-      cmap = lookupCmap(font.cmap, String.fromCharCode(font.glyphNameMap[StandardEncoding[bchar]]));
-      compileCharString(font.glyphs[cmap.glyphId], cmds, font);
-     }
-     return;
-    case 18:
-     stems += stack.length >> 1;
-     stackClean = true;
-     break;
-    case 19:
-     stems += stack.length >> 1;
-     i += stems + 7 >> 3;
-     stackClean = true;
-     break;
-    case 20:
-     stems += stack.length >> 1;
-     i += stems + 7 >> 3;
-     stackClean = true;
-     break;
-    case 21:
-     y += stack.pop();
-     x += stack.pop();
-     moveTo(x, y);
-     stackClean = true;
-     break;
-    case 22:
-     x += stack.pop();
-     moveTo(x, y);
-     stackClean = true;
-     break;
-    case 23:
-     stems += stack.length >> 1;
-     stackClean = true;
-     break;
-    case 24:
-     while (stack.length > 2) {
-      xa = x + stack.shift();
-      ya = y + stack.shift();
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb + stack.shift();
-      y = yb + stack.shift();
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-     }
-     x += stack.shift();
-     y += stack.shift();
-     lineTo(x, y);
-     break;
-    case 25:
-     while (stack.length > 6) {
-      x += stack.shift();
-      y += stack.shift();
-      lineTo(x, y);
-     }
-     xa = x + stack.shift();
-     ya = y + stack.shift();
-     xb = xa + stack.shift();
-     yb = ya + stack.shift();
-     x = xb + stack.shift();
-     y = yb + stack.shift();
-     bezierCurveTo(xa, ya, xb, yb, x, y);
-     break;
-    case 26:
-     if (stack.length % 2) {
-      x += stack.shift();
-     }
-     while (stack.length > 0) {
-      xa = x;
-      ya = y + stack.shift();
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb;
-      y = yb + stack.shift();
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-     }
-     break;
-    case 27:
-     if (stack.length % 2) {
-      y += stack.shift();
-     }
-     while (stack.length > 0) {
-      xa = x + stack.shift();
-      ya = y;
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb + stack.shift();
-      y = yb;
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-     }
-     break;
-    case 28:
-     stack.push((code[i] << 24 | code[i + 1] << 16) >> 16);
-     i += 2;
-     break;
-    case 29:
-     n = stack.pop() + font.gsubrsBias;
-     subrCode = font.gsubrs[n];
-     if (subrCode) {
-      parse(subrCode);
-     }
-     break;
-    case 30:
-     while (stack.length > 0) {
-      xa = x;
-      ya = y + stack.shift();
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb + stack.shift();
-      y = yb + (stack.length === 1 ? stack.shift() : 0);
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-      if (stack.length === 0) {
-       break;
-      }
-      xa = x + stack.shift();
-      ya = y;
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      y = yb + stack.shift();
-      x = xb + (stack.length === 1 ? stack.shift() : 0);
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-     }
-     break;
-    case 31:
-     while (stack.length > 0) {
-      xa = x + stack.shift();
-      ya = y;
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      y = yb + stack.shift();
-      x = xb + (stack.length === 1 ? stack.shift() : 0);
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-      if (stack.length === 0) {
-       break;
-      }
-      xa = x;
-      ya = y + stack.shift();
-      xb = xa + stack.shift();
-      yb = ya + stack.shift();
-      x = xb + stack.shift();
-      y = yb + (stack.length === 1 ? stack.shift() : 0);
-      bezierCurveTo(xa, ya, xb, yb, x, y);
-     }
-     break;
-    default:
-     if (v < 32) {
-      error('unknown operator: ' + v);
-     }
-     if (v < 247) {
-      stack.push(v - 139);
-     } else if (v < 251) {
-      stack.push((v - 247) * 256 + code[i++] + 108);
-     } else if (v < 255) {
-      stack.push(-(v - 251) * 256 - code[i++] - 108);
-     } else {
-      stack.push((code[i] << 24 | code[i + 1] << 16 | code[i + 2] << 8 | code[i + 3]) / 65536);
-      i += 4;
-     }
-     break;
+      return cmds;
+    },
+    compileGlyphImpl: function () {
+      error('Children classes should implement this.');
+    },
+    hasBuiltPath: function (unicode) {
+      var cmap = lookupCmap(this.cmap, unicode);
+      return this.compiledGlyphs[cmap.glyphId] !== undefined && this.compiledCharCodeToGlyphId[cmap.charCode] !== undefined;
     }
-    if (stackClean) {
-     stack.length = 0;
-    }
-   }
-  }
-  parse(code);
- }
- var noop = '';
- function CompiledFont(fontMatrix) {
-  this.compiledGlyphs = Object.create(null);
-  this.compiledCharCodeToGlyphId = Object.create(null);
-  this.fontMatrix = fontMatrix;
- }
- CompiledFont.prototype = {
-  getPathJs: function (unicode) {
-   var cmap = lookupCmap(this.cmap, unicode);
-   var fn = this.compiledGlyphs[cmap.glyphId];
-   if (!fn) {
-    fn = this.compileGlyph(this.glyphs[cmap.glyphId]);
-    this.compiledGlyphs[cmap.glyphId] = fn;
-   }
-   if (this.compiledCharCodeToGlyphId[cmap.charCode] === undefined) {
-    this.compiledCharCodeToGlyphId[cmap.charCode] = cmap.glyphId;
-   }
-   return fn;
-  },
-  compileGlyph: function (code) {
-   if (!code || code.length === 0 || code[0] === 14) {
-    return noop;
-   }
-   var cmds = [];
-   cmds.push({ cmd: 'save' });
-   cmds.push({
-    cmd: 'transform',
-    args: this.fontMatrix.slice()
-   });
-   cmds.push({
-    cmd: 'scale',
-    args: [
-     'size',
-     '-size'
-    ]
-   });
-   this.compileGlyphImpl(code, cmds);
-   cmds.push({ cmd: 'restore' });
-   return cmds;
-  },
-  compileGlyphImpl: function () {
-   error('Children classes should implement this.');
-  },
-  hasBuiltPath: function (unicode) {
-   var cmap = lookupCmap(this.cmap, unicode);
-   return this.compiledGlyphs[cmap.glyphId] !== undefined && this.compiledCharCodeToGlyphId[cmap.charCode] !== undefined;
-  }
- };
- function TrueTypeCompiled(glyphs, cmap, fontMatrix) {
-  fontMatrix = fontMatrix || [
-   0.000488,
-   0,
-   0,
-   0.000488,
-   0,
-   0
-  ];
-  CompiledFont.call(this, fontMatrix);
-  this.glyphs = glyphs;
-  this.cmap = cmap;
- }
- Util.inherit(TrueTypeCompiled, CompiledFont, {
-  compileGlyphImpl: function (code, cmds) {
-   compileGlyf(code, cmds, this);
-  }
- });
- function Type2Compiled(cffInfo, cmap, fontMatrix, glyphNameMap) {
-  fontMatrix = fontMatrix || [
-   0.001,
-   0,
-   0,
-   0.001,
-   0,
-   0
-  ];
-  CompiledFont.call(this, fontMatrix);
-  this.glyphs = cffInfo.glyphs;
-  this.gsubrs = cffInfo.gsubrs || [];
-  this.subrs = cffInfo.subrs || [];
-  this.cmap = cmap;
-  this.glyphNameMap = glyphNameMap || getGlyphsUnicode();
-  this.gsubrsBias = this.gsubrs.length < 1240 ? 107 : this.gsubrs.length < 33900 ? 1131 : 32768;
-  this.subrsBias = this.subrs.length < 1240 ? 107 : this.subrs.length < 33900 ? 1131 : 32768;
- }
- Util.inherit(Type2Compiled, CompiledFont, {
-  compileGlyphImpl: function (code, cmds) {
-   compileCharString(code, cmds, this);
+  };
+  function TrueTypeCompiled(glyphs, cmap, fontMatrix) {
+    fontMatrix = fontMatrix || [0.000488, 0, 0, 0.000488, 0, 0];
+    CompiledFont.call(this, fontMatrix);
+    this.glyphs = glyphs;
+    this.cmap = cmap;
   }
- });
- return {
-  create: function FontRendererFactory_create(font, seacAnalysisEnabled) {
-   var data = new Uint8Array(font.data);
-   var cmap, glyf, loca, cff, indexToLocFormat, unitsPerEm;
-   var numTables = getUshort(data, 4);
-   for (var i = 0, p = 12; i < numTables; i++, p += 16) {
-    var tag = bytesToString(data.subarray(p, p + 4));
-    var offset = getLong(data, p + 8);
-    var length = getLong(data, p + 12);
-    switch (tag) {
-    case 'cmap':
-     cmap = parseCmap(data, offset, offset + length);
-     break;
-    case 'glyf':
-     glyf = data.subarray(offset, offset + length);
-     break;
-    case 'loca':
-     loca = data.subarray(offset, offset + length);
-     break;
-    case 'head':
-     unitsPerEm = getUshort(data, offset + 18);
-     indexToLocFormat = getUshort(data, offset + 50);
-     break;
-    case 'CFF ':
-     cff = parseCff(data, offset, offset + length, seacAnalysisEnabled);
-     break;
+  Util.inherit(TrueTypeCompiled, CompiledFont, {
+    compileGlyphImpl: function (code, cmds) {
+      compileGlyf(code, cmds, this);
     }
-   }
-   if (glyf) {
-    var fontMatrix = !unitsPerEm ? font.fontMatrix : [
-     1 / unitsPerEm,
-     0,
-     0,
-     1 / unitsPerEm,
-     0,
-     0
-    ];
-    return new TrueTypeCompiled(parseGlyfTable(glyf, loca, indexToLocFormat), cmap, fontMatrix);
-   }
-   return new Type2Compiled(cff, cmap, font.fontMatrix, font.glyphNameMap);
+  });
+  function Type2Compiled(cffInfo, cmap, fontMatrix, glyphNameMap) {
+    fontMatrix = fontMatrix || [0.001, 0, 0, 0.001, 0, 0];
+    CompiledFont.call(this, fontMatrix);
+    this.glyphs = cffInfo.glyphs;
+    this.gsubrs = cffInfo.gsubrs || [];
+    this.subrs = cffInfo.subrs || [];
+    this.cmap = cmap;
+    this.glyphNameMap = glyphNameMap || getGlyphsUnicode();
+    this.gsubrsBias = this.gsubrs.length < 1240 ? 107 : this.gsubrs.length < 33900 ? 1131 : 32768;
+    this.subrsBias = this.subrs.length < 1240 ? 107 : this.subrs.length < 33900 ? 1131 : 32768;
   }
- };
+  Util.inherit(Type2Compiled, CompiledFont, {
+    compileGlyphImpl: function (code, cmds) {
+      compileCharString(code, cmds, this);
+    }
+  });
+  return {
+    create: function FontRendererFactory_create(font, seacAnalysisEnabled) {
+      var data = new Uint8Array(font.data);
+      var cmap, glyf, loca, cff, indexToLocFormat, unitsPerEm;
+      var numTables = getUshort(data, 4);
+      for (var i = 0, p = 12; i < numTables; i++, p += 16) {
+        var tag = bytesToString(data.subarray(p, p + 4));
+        var offset = getLong(data, p + 8);
+        var length = getLong(data, p + 12);
+        switch (tag) {
+          case 'cmap':
+            cmap = parseCmap(data, offset, offset + length);
+            break;
+          case 'glyf':
+            glyf = data.subarray(offset, offset + length);
+            break;
+          case 'loca':
+            loca = data.subarray(offset, offset + length);
+            break;
+          case 'head':
+            unitsPerEm = getUshort(data, offset + 18);
+            indexToLocFormat = getUshort(data, offset + 50);
+            break;
+          case 'CFF ':
+            cff = parseCff(data, offset, offset + length, seacAnalysisEnabled);
+            break;
+        }
+      }
+      if (glyf) {
+        var fontMatrix = !unitsPerEm ? font.fontMatrix : [1 / unitsPerEm, 0, 0, 1 / unitsPerEm, 0, 0];
+        return new TrueTypeCompiled(parseGlyfTable(glyf, loca, indexToLocFormat), cmap, fontMatrix);
+      }
+      return new Type2Compiled(cff, cmap, font.fontMatrix, font.glyphNameMap);
+    }
+  };
 }();
 exports.FontRendererFactory = FontRendererFactory;

File diff suppressed because it is too large
+ 10 - 269
lib/core/fonts.js


+ 921 - 950
lib/core/function.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var corePsParser = require('./ps_parser.js');
@@ -25,983 +26,953 @@ var isStream = corePrimitives.isStream;
 var PostScriptLexer = corePsParser.PostScriptLexer;
 var PostScriptParser = corePsParser.PostScriptParser;
 var PDFFunction = function PDFFunctionClosure() {
- var CONSTRUCT_SAMPLED = 0;
- var CONSTRUCT_INTERPOLATED = 2;
- var CONSTRUCT_STICHED = 3;
- var CONSTRUCT_POSTSCRIPT = 4;
- return {
-  getSampleArray: function PDFFunction_getSampleArray(size, outputSize, bps, str) {
-   var i, ii;
-   var length = 1;
-   for (i = 0, ii = size.length; i < ii; i++) {
-    length *= size[i];
-   }
-   length *= outputSize;
-   var array = new Array(length);
-   var codeSize = 0;
-   var codeBuf = 0;
-   var sampleMul = 1.0 / (Math.pow(2.0, bps) - 1);
-   var strBytes = str.getBytes((length * bps + 7) / 8);
-   var strIdx = 0;
-   for (i = 0; i < length; i++) {
-    while (codeSize < bps) {
-     codeBuf <<= 8;
-     codeBuf |= strBytes[strIdx++];
-     codeSize += 8;
-    }
-    codeSize -= bps;
-    array[i] = (codeBuf >> codeSize) * sampleMul;
-    codeBuf &= (1 << codeSize) - 1;
-   }
-   return array;
-  },
-  getIR: function PDFFunction_getIR(xref, fn) {
-   var dict = fn.dict;
-   if (!dict) {
-    dict = fn;
-   }
-   var types = [
-    this.constructSampled,
-    null,
-    this.constructInterpolated,
-    this.constructStiched,
-    this.constructPostScript
-   ];
-   var typeNum = dict.get('FunctionType');
-   var typeFn = types[typeNum];
-   if (!typeFn) {
-    error('Unknown type of function');
-   }
-   return typeFn.call(this, fn, dict, xref);
-  },
-  fromIR: function PDFFunction_fromIR(IR) {
-   var type = IR[0];
-   switch (type) {
-   case CONSTRUCT_SAMPLED:
-    return this.constructSampledFromIR(IR);
-   case CONSTRUCT_INTERPOLATED:
-    return this.constructInterpolatedFromIR(IR);
-   case CONSTRUCT_STICHED:
-    return this.constructStichedFromIR(IR);
-   default:
-    return this.constructPostScriptFromIR(IR);
-   }
-  },
-  parse: function PDFFunction_parse(xref, fn) {
-   var IR = this.getIR(xref, fn);
-   return this.fromIR(IR);
-  },
-  parseArray: function PDFFunction_parseArray(xref, fnObj) {
-   if (!isArray(fnObj)) {
-    return this.parse(xref, fnObj);
-   }
-   var fnArray = [];
-   for (var j = 0, jj = fnObj.length; j < jj; j++) {
-    var obj = xref.fetchIfRef(fnObj[j]);
-    fnArray.push(PDFFunction.parse(xref, obj));
-   }
-   return function (src, srcOffset, dest, destOffset) {
-    for (var i = 0, ii = fnArray.length; i < ii; i++) {
-     fnArray[i](src, srcOffset, dest, destOffset + i);
-    }
-   };
-  },
-  constructSampled: function PDFFunction_constructSampled(str, dict) {
-   function toMultiArray(arr) {
-    var inputLength = arr.length;
-    var out = [];
-    var index = 0;
-    for (var i = 0; i < inputLength; i += 2) {
-     out[index] = [
-      arr[i],
-      arr[i + 1]
-     ];
-     ++index;
-    }
-    return out;
-   }
-   var domain = dict.getArray('Domain');
-   var range = dict.getArray('Range');
-   if (!domain || !range) {
-    error('No domain or range');
-   }
-   var inputSize = domain.length / 2;
-   var outputSize = range.length / 2;
-   domain = toMultiArray(domain);
-   range = toMultiArray(range);
-   var size = dict.get('Size');
-   var bps = dict.get('BitsPerSample');
-   var order = dict.get('Order') || 1;
-   if (order !== 1) {
-    info('No support for cubic spline interpolation: ' + order);
-   }
-   var encode = dict.getArray('Encode');
-   if (!encode) {
-    encode = [];
-    for (var i = 0; i < inputSize; ++i) {
-     encode.push(0);
-     encode.push(size[i] - 1);
-    }
-   }
-   encode = toMultiArray(encode);
-   var decode = dict.getArray('Decode');
-   if (!decode) {
-    decode = range;
-   } else {
-    decode = toMultiArray(decode);
-   }
-   var samples = this.getSampleArray(size, outputSize, bps, str);
-   return [
-    CONSTRUCT_SAMPLED,
-    inputSize,
-    domain,
-    encode,
-    decode,
-    samples,
-    size,
-    outputSize,
-    Math.pow(2, bps) - 1,
-    range
-   ];
-  },
-  constructSampledFromIR: function PDFFunction_constructSampledFromIR(IR) {
-   function interpolate(x, xmin, xmax, ymin, ymax) {
-    return ymin + (x - xmin) * ((ymax - ymin) / (xmax - xmin));
-   }
-   return function constructSampledFromIRResult(src, srcOffset, dest, destOffset) {
-    var m = IR[1];
-    var domain = IR[2];
-    var encode = IR[3];
-    var decode = IR[4];
-    var samples = IR[5];
-    var size = IR[6];
-    var n = IR[7];
-    var range = IR[9];
-    var cubeVertices = 1 << m;
-    var cubeN = new Float64Array(cubeVertices);
-    var cubeVertex = new Uint32Array(cubeVertices);
-    var i, j;
-    for (j = 0; j < cubeVertices; j++) {
-     cubeN[j] = 1;
-    }
-    var k = n, pos = 1;
-    for (i = 0; i < m; ++i) {
-     var domain_2i = domain[i][0];
-     var domain_2i_1 = domain[i][1];
-     var xi = Math.min(Math.max(src[srcOffset + i], domain_2i), domain_2i_1);
-     var e = interpolate(xi, domain_2i, domain_2i_1, encode[i][0], encode[i][1]);
-     var size_i = size[i];
-     e = Math.min(Math.max(e, 0), size_i - 1);
-     var e0 = e < size_i - 1 ? Math.floor(e) : e - 1;
-     var n0 = e0 + 1 - e;
-     var n1 = e - e0;
-     var offset0 = e0 * k;
-     var offset1 = offset0 + k;
-     for (j = 0; j < cubeVertices; j++) {
-      if (j & pos) {
-       cubeN[j] *= n1;
-       cubeVertex[j] += offset1;
+  var CONSTRUCT_SAMPLED = 0;
+  var CONSTRUCT_INTERPOLATED = 2;
+  var CONSTRUCT_STICHED = 3;
+  var CONSTRUCT_POSTSCRIPT = 4;
+  return {
+    getSampleArray: function PDFFunction_getSampleArray(size, outputSize, bps, str) {
+      var i, ii;
+      var length = 1;
+      for (i = 0, ii = size.length; i < ii; i++) {
+        length *= size[i];
+      }
+      length *= outputSize;
+      var array = new Array(length);
+      var codeSize = 0;
+      var codeBuf = 0;
+      var sampleMul = 1.0 / (Math.pow(2.0, bps) - 1);
+      var strBytes = str.getBytes((length * bps + 7) / 8);
+      var strIdx = 0;
+      for (i = 0; i < length; i++) {
+        while (codeSize < bps) {
+          codeBuf <<= 8;
+          codeBuf |= strBytes[strIdx++];
+          codeSize += 8;
+        }
+        codeSize -= bps;
+        array[i] = (codeBuf >> codeSize) * sampleMul;
+        codeBuf &= (1 << codeSize) - 1;
+      }
+      return array;
+    },
+    getIR: function PDFFunction_getIR(xref, fn) {
+      var dict = fn.dict;
+      if (!dict) {
+        dict = fn;
+      }
+      var types = [this.constructSampled, null, this.constructInterpolated, this.constructStiched, this.constructPostScript];
+      var typeNum = dict.get('FunctionType');
+      var typeFn = types[typeNum];
+      if (!typeFn) {
+        error('Unknown type of function');
+      }
+      return typeFn.call(this, fn, dict, xref);
+    },
+    fromIR: function PDFFunction_fromIR(IR) {
+      var type = IR[0];
+      switch (type) {
+        case CONSTRUCT_SAMPLED:
+          return this.constructSampledFromIR(IR);
+        case CONSTRUCT_INTERPOLATED:
+          return this.constructInterpolatedFromIR(IR);
+        case CONSTRUCT_STICHED:
+          return this.constructStichedFromIR(IR);
+        default:
+          return this.constructPostScriptFromIR(IR);
+      }
+    },
+    parse: function PDFFunction_parse(xref, fn) {
+      var IR = this.getIR(xref, fn);
+      return this.fromIR(IR);
+    },
+    parseArray: function PDFFunction_parseArray(xref, fnObj) {
+      if (!isArray(fnObj)) {
+        return this.parse(xref, fnObj);
+      }
+      var fnArray = [];
+      for (var j = 0, jj = fnObj.length; j < jj; j++) {
+        var obj = xref.fetchIfRef(fnObj[j]);
+        fnArray.push(PDFFunction.parse(xref, obj));
+      }
+      return function (src, srcOffset, dest, destOffset) {
+        for (var i = 0, ii = fnArray.length; i < ii; i++) {
+          fnArray[i](src, srcOffset, dest, destOffset + i);
+        }
+      };
+    },
+    constructSampled: function PDFFunction_constructSampled(str, dict) {
+      function toMultiArray(arr) {
+        var inputLength = arr.length;
+        var out = [];
+        var index = 0;
+        for (var i = 0; i < inputLength; i += 2) {
+          out[index] = [arr[i], arr[i + 1]];
+          ++index;
+        }
+        return out;
+      }
+      var domain = dict.getArray('Domain');
+      var range = dict.getArray('Range');
+      if (!domain || !range) {
+        error('No domain or range');
+      }
+      var inputSize = domain.length / 2;
+      var outputSize = range.length / 2;
+      domain = toMultiArray(domain);
+      range = toMultiArray(range);
+      var size = dict.get('Size');
+      var bps = dict.get('BitsPerSample');
+      var order = dict.get('Order') || 1;
+      if (order !== 1) {
+        info('No support for cubic spline interpolation: ' + order);
+      }
+      var encode = dict.getArray('Encode');
+      if (!encode) {
+        encode = [];
+        for (var i = 0; i < inputSize; ++i) {
+          encode.push(0);
+          encode.push(size[i] - 1);
+        }
+      }
+      encode = toMultiArray(encode);
+      var decode = dict.getArray('Decode');
+      if (!decode) {
+        decode = range;
       } else {
-       cubeN[j] *= n0;
-       cubeVertex[j] += offset0;
+        decode = toMultiArray(decode);
       }
-     }
-     k *= size_i;
-     pos <<= 1;
-    }
-    for (j = 0; j < n; ++j) {
-     var rj = 0;
-     for (i = 0; i < cubeVertices; i++) {
-      rj += samples[cubeVertex[i] + j] * cubeN[i];
-     }
-     rj = interpolate(rj, 0, 1, decode[j][0], decode[j][1]);
-     dest[destOffset + j] = Math.min(Math.max(rj, range[j][0]), range[j][1]);
-    }
-   };
-  },
-  constructInterpolated: function PDFFunction_constructInterpolated(str, dict) {
-   var c0 = dict.getArray('C0') || [0];
-   var c1 = dict.getArray('C1') || [1];
-   var n = dict.get('N');
-   if (!isArray(c0) || !isArray(c1)) {
-    error('Illegal dictionary for interpolated function');
-   }
-   var length = c0.length;
-   var diff = [];
-   for (var i = 0; i < length; ++i) {
-    diff.push(c1[i] - c0[i]);
-   }
-   return [
-    CONSTRUCT_INTERPOLATED,
-    c0,
-    diff,
-    n
-   ];
-  },
-  constructInterpolatedFromIR: function PDFFunction_constructInterpolatedFromIR(IR) {
-   var c0 = IR[1];
-   var diff = IR[2];
-   var n = IR[3];
-   var length = diff.length;
-   return function constructInterpolatedFromIRResult(src, srcOffset, dest, destOffset) {
-    var x = n === 1 ? src[srcOffset] : Math.pow(src[srcOffset], n);
-    for (var j = 0; j < length; ++j) {
-     dest[destOffset + j] = c0[j] + x * diff[j];
-    }
-   };
-  },
-  constructStiched: function PDFFunction_constructStiched(fn, dict, xref) {
-   var domain = dict.getArray('Domain');
-   if (!domain) {
-    error('No domain');
-   }
-   var inputSize = domain.length / 2;
-   if (inputSize !== 1) {
-    error('Bad domain for stiched function');
-   }
-   var fnRefs = dict.get('Functions');
-   var fns = [];
-   for (var i = 0, ii = fnRefs.length; i < ii; ++i) {
-    fns.push(PDFFunction.getIR(xref, xref.fetchIfRef(fnRefs[i])));
-   }
-   var bounds = dict.getArray('Bounds');
-   var encode = dict.getArray('Encode');
-   return [
-    CONSTRUCT_STICHED,
-    domain,
-    bounds,
-    encode,
-    fns
-   ];
-  },
-  constructStichedFromIR: function PDFFunction_constructStichedFromIR(IR) {
-   var domain = IR[1];
-   var bounds = IR[2];
-   var encode = IR[3];
-   var fnsIR = IR[4];
-   var fns = [];
-   var tmpBuf = new Float32Array(1);
-   for (var i = 0, ii = fnsIR.length; i < ii; i++) {
-    fns.push(PDFFunction.fromIR(fnsIR[i]));
-   }
-   return function constructStichedFromIRResult(src, srcOffset, dest, destOffset) {
-    var clip = function constructStichedFromIRClip(v, min, max) {
-     if (v > max) {
-      v = max;
-     } else if (v < min) {
-      v = min;
-     }
-     return v;
-    };
-    var v = clip(src[srcOffset], domain[0], domain[1]);
-    for (var i = 0, ii = bounds.length; i < ii; ++i) {
-     if (v < bounds[i]) {
-      break;
-     }
-    }
-    var dmin = domain[0];
-    if (i > 0) {
-     dmin = bounds[i - 1];
-    }
-    var dmax = domain[1];
-    if (i < bounds.length) {
-     dmax = bounds[i];
-    }
-    var rmin = encode[2 * i];
-    var rmax = encode[2 * i + 1];
-    tmpBuf[0] = dmin === dmax ? rmin : rmin + (v - dmin) * (rmax - rmin) / (dmax - dmin);
-    fns[i](tmpBuf, 0, dest, destOffset);
-   };
-  },
-  constructPostScript: function PDFFunction_constructPostScript(fn, dict, xref) {
-   var domain = dict.getArray('Domain');
-   var range = dict.getArray('Range');
-   if (!domain) {
-    error('No domain.');
-   }
-   if (!range) {
-    error('No range.');
-   }
-   var lexer = new PostScriptLexer(fn);
-   var parser = new PostScriptParser(lexer);
-   var code = parser.parse();
-   return [
-    CONSTRUCT_POSTSCRIPT,
-    domain,
-    range,
-    code
-   ];
-  },
-  constructPostScriptFromIR: function PDFFunction_constructPostScriptFromIR(IR) {
-   var domain = IR[1];
-   var range = IR[2];
-   var code = IR[3];
-   var compiled = new PostScriptCompiler().compile(code, domain, range);
-   if (compiled) {
-    return new Function('src', 'srcOffset', 'dest', 'destOffset', compiled);
-   }
-   info('Unable to compile PS function');
-   var numOutputs = range.length >> 1;
-   var numInputs = domain.length >> 1;
-   var evaluator = new PostScriptEvaluator(code);
-   var cache = Object.create(null);
-   var MAX_CACHE_SIZE = 2048 * 4;
-   var cache_available = MAX_CACHE_SIZE;
-   var tmpBuf = new Float32Array(numInputs);
-   return function constructPostScriptFromIRResult(src, srcOffset, dest, destOffset) {
-    var i, value;
-    var key = '';
-    var input = tmpBuf;
-    for (i = 0; i < numInputs; i++) {
-     value = src[srcOffset + i];
-     input[i] = value;
-     key += value + '_';
-    }
-    var cachedValue = cache[key];
-    if (cachedValue !== undefined) {
-     dest.set(cachedValue, destOffset);
-     return;
-    }
-    var output = new Float32Array(numOutputs);
-    var stack = evaluator.execute(input);
-    var stackIndex = stack.length - numOutputs;
-    for (i = 0; i < numOutputs; i++) {
-     value = stack[stackIndex + i];
-     var bound = range[i * 2];
-     if (value < bound) {
-      value = bound;
-     } else {
-      bound = range[i * 2 + 1];
-      if (value > bound) {
-       value = bound;
-      }
-     }
-     output[i] = value;
-    }
-    if (cache_available > 0) {
-     cache_available--;
-     cache[key] = output;
+      var samples = this.getSampleArray(size, outputSize, bps, str);
+      return [CONSTRUCT_SAMPLED, inputSize, domain, encode, decode, samples, size, outputSize, Math.pow(2, bps) - 1, range];
+    },
+    constructSampledFromIR: function PDFFunction_constructSampledFromIR(IR) {
+      function interpolate(x, xmin, xmax, ymin, ymax) {
+        return ymin + (x - xmin) * ((ymax - ymin) / (xmax - xmin));
+      }
+      return function constructSampledFromIRResult(src, srcOffset, dest, destOffset) {
+        var m = IR[1];
+        var domain = IR[2];
+        var encode = IR[3];
+        var decode = IR[4];
+        var samples = IR[5];
+        var size = IR[6];
+        var n = IR[7];
+        var range = IR[9];
+        var cubeVertices = 1 << m;
+        var cubeN = new Float64Array(cubeVertices);
+        var cubeVertex = new Uint32Array(cubeVertices);
+        var i, j;
+        for (j = 0; j < cubeVertices; j++) {
+          cubeN[j] = 1;
+        }
+        var k = n,
+            pos = 1;
+        for (i = 0; i < m; ++i) {
+          var domain_2i = domain[i][0];
+          var domain_2i_1 = domain[i][1];
+          var xi = Math.min(Math.max(src[srcOffset + i], domain_2i), domain_2i_1);
+          var e = interpolate(xi, domain_2i, domain_2i_1, encode[i][0], encode[i][1]);
+          var size_i = size[i];
+          e = Math.min(Math.max(e, 0), size_i - 1);
+          var e0 = e < size_i - 1 ? Math.floor(e) : e - 1;
+          var n0 = e0 + 1 - e;
+          var n1 = e - e0;
+          var offset0 = e0 * k;
+          var offset1 = offset0 + k;
+          for (j = 0; j < cubeVertices; j++) {
+            if (j & pos) {
+              cubeN[j] *= n1;
+              cubeVertex[j] += offset1;
+            } else {
+              cubeN[j] *= n0;
+              cubeVertex[j] += offset0;
+            }
+          }
+          k *= size_i;
+          pos <<= 1;
+        }
+        for (j = 0; j < n; ++j) {
+          var rj = 0;
+          for (i = 0; i < cubeVertices; i++) {
+            rj += samples[cubeVertex[i] + j] * cubeN[i];
+          }
+          rj = interpolate(rj, 0, 1, decode[j][0], decode[j][1]);
+          dest[destOffset + j] = Math.min(Math.max(rj, range[j][0]), range[j][1]);
+        }
+      };
+    },
+    constructInterpolated: function PDFFunction_constructInterpolated(str, dict) {
+      var c0 = dict.getArray('C0') || [0];
+      var c1 = dict.getArray('C1') || [1];
+      var n = dict.get('N');
+      if (!isArray(c0) || !isArray(c1)) {
+        error('Illegal dictionary for interpolated function');
+      }
+      var length = c0.length;
+      var diff = [];
+      for (var i = 0; i < length; ++i) {
+        diff.push(c1[i] - c0[i]);
+      }
+      return [CONSTRUCT_INTERPOLATED, c0, diff, n];
+    },
+    constructInterpolatedFromIR: function PDFFunction_constructInterpolatedFromIR(IR) {
+      var c0 = IR[1];
+      var diff = IR[2];
+      var n = IR[3];
+      var length = diff.length;
+      return function constructInterpolatedFromIRResult(src, srcOffset, dest, destOffset) {
+        var x = n === 1 ? src[srcOffset] : Math.pow(src[srcOffset], n);
+        for (var j = 0; j < length; ++j) {
+          dest[destOffset + j] = c0[j] + x * diff[j];
+        }
+      };
+    },
+    constructStiched: function PDFFunction_constructStiched(fn, dict, xref) {
+      var domain = dict.getArray('Domain');
+      if (!domain) {
+        error('No domain');
+      }
+      var inputSize = domain.length / 2;
+      if (inputSize !== 1) {
+        error('Bad domain for stiched function');
+      }
+      var fnRefs = dict.get('Functions');
+      var fns = [];
+      for (var i = 0, ii = fnRefs.length; i < ii; ++i) {
+        fns.push(PDFFunction.getIR(xref, xref.fetchIfRef(fnRefs[i])));
+      }
+      var bounds = dict.getArray('Bounds');
+      var encode = dict.getArray('Encode');
+      return [CONSTRUCT_STICHED, domain, bounds, encode, fns];
+    },
+    constructStichedFromIR: function PDFFunction_constructStichedFromIR(IR) {
+      var domain = IR[1];
+      var bounds = IR[2];
+      var encode = IR[3];
+      var fnsIR = IR[4];
+      var fns = [];
+      var tmpBuf = new Float32Array(1);
+      for (var i = 0, ii = fnsIR.length; i < ii; i++) {
+        fns.push(PDFFunction.fromIR(fnsIR[i]));
+      }
+      return function constructStichedFromIRResult(src, srcOffset, dest, destOffset) {
+        var clip = function constructStichedFromIRClip(v, min, max) {
+          if (v > max) {
+            v = max;
+          } else if (v < min) {
+            v = min;
+          }
+          return v;
+        };
+        var v = clip(src[srcOffset], domain[0], domain[1]);
+        for (var i = 0, ii = bounds.length; i < ii; ++i) {
+          if (v < bounds[i]) {
+            break;
+          }
+        }
+        var dmin = domain[0];
+        if (i > 0) {
+          dmin = bounds[i - 1];
+        }
+        var dmax = domain[1];
+        if (i < bounds.length) {
+          dmax = bounds[i];
+        }
+        var rmin = encode[2 * i];
+        var rmax = encode[2 * i + 1];
+        tmpBuf[0] = dmin === dmax ? rmin : rmin + (v - dmin) * (rmax - rmin) / (dmax - dmin);
+        fns[i](tmpBuf, 0, dest, destOffset);
+      };
+    },
+    constructPostScript: function PDFFunction_constructPostScript(fn, dict, xref) {
+      var domain = dict.getArray('Domain');
+      var range = dict.getArray('Range');
+      if (!domain) {
+        error('No domain.');
+      }
+      if (!range) {
+        error('No range.');
+      }
+      var lexer = new PostScriptLexer(fn);
+      var parser = new PostScriptParser(lexer);
+      var code = parser.parse();
+      return [CONSTRUCT_POSTSCRIPT, domain, range, code];
+    },
+    constructPostScriptFromIR: function PDFFunction_constructPostScriptFromIR(IR) {
+      var domain = IR[1];
+      var range = IR[2];
+      var code = IR[3];
+      var compiled = new PostScriptCompiler().compile(code, domain, range);
+      if (compiled) {
+        return new Function('src', 'srcOffset', 'dest', 'destOffset', compiled);
+      }
+      info('Unable to compile PS function');
+      var numOutputs = range.length >> 1;
+      var numInputs = domain.length >> 1;
+      var evaluator = new PostScriptEvaluator(code);
+      var cache = Object.create(null);
+      var MAX_CACHE_SIZE = 2048 * 4;
+      var cache_available = MAX_CACHE_SIZE;
+      var tmpBuf = new Float32Array(numInputs);
+      return function constructPostScriptFromIRResult(src, srcOffset, dest, destOffset) {
+        var i, value;
+        var key = '';
+        var input = tmpBuf;
+        for (i = 0; i < numInputs; i++) {
+          value = src[srcOffset + i];
+          input[i] = value;
+          key += value + '_';
+        }
+        var cachedValue = cache[key];
+        if (cachedValue !== undefined) {
+          dest.set(cachedValue, destOffset);
+          return;
+        }
+        var output = new Float32Array(numOutputs);
+        var stack = evaluator.execute(input);
+        var stackIndex = stack.length - numOutputs;
+        for (i = 0; i < numOutputs; i++) {
+          value = stack[stackIndex + i];
+          var bound = range[i * 2];
+          if (value < bound) {
+            value = bound;
+          } else {
+            bound = range[i * 2 + 1];
+            if (value > bound) {
+              value = bound;
+            }
+          }
+          output[i] = value;
+        }
+        if (cache_available > 0) {
+          cache_available--;
+          cache[key] = output;
+        }
+        dest.set(output, destOffset);
+      };
     }
-    dest.set(output, destOffset);
-   };
-  }
- };
+  };
 }();
 function isPDFFunction(v) {
- var fnDict;
- if (typeof v !== 'object') {
-  return false;
- } else if (isDict(v)) {
-  fnDict = v;
- } else if (isStream(v)) {
-  fnDict = v.dict;
- } else {
-  return false;
- }
- return fnDict.has('FunctionType');
+  var fnDict;
+  if (typeof v !== 'object') {
+    return false;
+  } else if (isDict(v)) {
+    fnDict = v;
+  } else if (isStream(v)) {
+    fnDict = v.dict;
+  } else {
+    return false;
+  }
+  return fnDict.has('FunctionType');
 }
 var PostScriptStack = function PostScriptStackClosure() {
- var MAX_STACK_SIZE = 100;
- function PostScriptStack(initialStack) {
-  this.stack = !initialStack ? [] : Array.prototype.slice.call(initialStack, 0);
- }
- PostScriptStack.prototype = {
-  push: function PostScriptStack_push(value) {
-   if (this.stack.length >= MAX_STACK_SIZE) {
-    error('PostScript function stack overflow.');
-   }
-   this.stack.push(value);
-  },
-  pop: function PostScriptStack_pop() {
-   if (this.stack.length <= 0) {
-    error('PostScript function stack underflow.');
-   }
-   return this.stack.pop();
-  },
-  copy: function PostScriptStack_copy(n) {
-   if (this.stack.length + n >= MAX_STACK_SIZE) {
-    error('PostScript function stack overflow.');
-   }
-   var stack = this.stack;
-   for (var i = stack.length - n, j = n - 1; j >= 0; j--, i++) {
-    stack.push(stack[i]);
-   }
-  },
-  index: function PostScriptStack_index(n) {
-   this.push(this.stack[this.stack.length - n - 1]);
-  },
-  roll: function PostScriptStack_roll(n, p) {
-   var stack = this.stack;
-   var l = stack.length - n;
-   var r = stack.length - 1, c = l + (p - Math.floor(p / n) * n), i, j, t;
-   for (i = l, j = r; i < j; i++, j--) {
-    t = stack[i];
-    stack[i] = stack[j];
-    stack[j] = t;
-   }
-   for (i = l, j = c - 1; i < j; i++, j--) {
-    t = stack[i];
-    stack[i] = stack[j];
-    stack[j] = t;
-   }
-   for (i = c, j = r; i < j; i++, j--) {
-    t = stack[i];
-    stack[i] = stack[j];
-    stack[j] = t;
-   }
+  var MAX_STACK_SIZE = 100;
+  function PostScriptStack(initialStack) {
+    this.stack = !initialStack ? [] : Array.prototype.slice.call(initialStack, 0);
   }
- };
- return PostScriptStack;
+  PostScriptStack.prototype = {
+    push: function PostScriptStack_push(value) {
+      if (this.stack.length >= MAX_STACK_SIZE) {
+        error('PostScript function stack overflow.');
+      }
+      this.stack.push(value);
+    },
+    pop: function PostScriptStack_pop() {
+      if (this.stack.length <= 0) {
+        error('PostScript function stack underflow.');
+      }
+      return this.stack.pop();
+    },
+    copy: function PostScriptStack_copy(n) {
+      if (this.stack.length + n >= MAX_STACK_SIZE) {
+        error('PostScript function stack overflow.');
+      }
+      var stack = this.stack;
+      for (var i = stack.length - n, j = n - 1; j >= 0; j--, i++) {
+        stack.push(stack[i]);
+      }
+    },
+    index: function PostScriptStack_index(n) {
+      this.push(this.stack[this.stack.length - n - 1]);
+    },
+    roll: function PostScriptStack_roll(n, p) {
+      var stack = this.stack;
+      var l = stack.length - n;
+      var r = stack.length - 1,
+          c = l + (p - Math.floor(p / n) * n),
+          i,
+          j,
+          t;
+      for (i = l, j = r; i < j; i++, j--) {
+        t = stack[i];
+        stack[i] = stack[j];
+        stack[j] = t;
+      }
+      for (i = l, j = c - 1; i < j; i++, j--) {
+        t = stack[i];
+        stack[i] = stack[j];
+        stack[j] = t;
+      }
+      for (i = c, j = r; i < j; i++, j--) {
+        t = stack[i];
+        stack[i] = stack[j];
+        stack[j] = t;
+      }
+    }
+  };
+  return PostScriptStack;
 }();
 var PostScriptEvaluator = function PostScriptEvaluatorClosure() {
- function PostScriptEvaluator(operators) {
-  this.operators = operators;
- }
- PostScriptEvaluator.prototype = {
-  execute: function PostScriptEvaluator_execute(initialStack) {
-   var stack = new PostScriptStack(initialStack);
-   var counter = 0;
-   var operators = this.operators;
-   var length = operators.length;
-   var operator, a, b;
-   while (counter < length) {
-    operator = operators[counter++];
-    if (typeof operator === 'number') {
-     stack.push(operator);
-     continue;
-    }
-    switch (operator) {
-    case 'jz':
-     b = stack.pop();
-     a = stack.pop();
-     if (!a) {
-      counter = b;
-     }
-     break;
-    case 'j':
-     a = stack.pop();
-     counter = a;
-     break;
-    case 'abs':
-     a = stack.pop();
-     stack.push(Math.abs(a));
-     break;
-    case 'add':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a + b);
-     break;
-    case 'and':
-     b = stack.pop();
-     a = stack.pop();
-     if (isBool(a) && isBool(b)) {
-      stack.push(a && b);
-     } else {
-      stack.push(a & b);
-     }
-     break;
-    case 'atan':
-     a = stack.pop();
-     stack.push(Math.atan(a));
-     break;
-    case 'bitshift':
-     b = stack.pop();
-     a = stack.pop();
-     if (a > 0) {
-      stack.push(a << b);
-     } else {
-      stack.push(a >> b);
-     }
-     break;
-    case 'ceiling':
-     a = stack.pop();
-     stack.push(Math.ceil(a));
-     break;
-    case 'copy':
-     a = stack.pop();
-     stack.copy(a);
-     break;
-    case 'cos':
-     a = stack.pop();
-     stack.push(Math.cos(a));
-     break;
-    case 'cvi':
-     a = stack.pop() | 0;
-     stack.push(a);
-     break;
-    case 'cvr':
-     break;
-    case 'div':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a / b);
-     break;
-    case 'dup':
-     stack.copy(1);
-     break;
-    case 'eq':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a === b);
-     break;
-    case 'exch':
-     stack.roll(2, 1);
-     break;
-    case 'exp':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(Math.pow(a, b));
-     break;
-    case 'false':
-     stack.push(false);
-     break;
-    case 'floor':
-     a = stack.pop();
-     stack.push(Math.floor(a));
-     break;
-    case 'ge':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a >= b);
-     break;
-    case 'gt':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a > b);
-     break;
-    case 'idiv':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a / b | 0);
-     break;
-    case 'index':
-     a = stack.pop();
-     stack.index(a);
-     break;
-    case 'le':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a <= b);
-     break;
-    case 'ln':
-     a = stack.pop();
-     stack.push(Math.log(a));
-     break;
-    case 'log':
-     a = stack.pop();
-     stack.push(Math.log(a) / Math.LN10);
-     break;
-    case 'lt':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a < b);
-     break;
-    case 'mod':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a % b);
-     break;
-    case 'mul':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a * b);
-     break;
-    case 'ne':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a !== b);
-     break;
-    case 'neg':
-     a = stack.pop();
-     stack.push(-a);
-     break;
-    case 'not':
-     a = stack.pop();
-     if (isBool(a)) {
-      stack.push(!a);
-     } else {
-      stack.push(~a);
-     }
-     break;
-    case 'or':
-     b = stack.pop();
-     a = stack.pop();
-     if (isBool(a) && isBool(b)) {
-      stack.push(a || b);
-     } else {
-      stack.push(a | b);
-     }
-     break;
-    case 'pop':
-     stack.pop();
-     break;
-    case 'roll':
-     b = stack.pop();
-     a = stack.pop();
-     stack.roll(a, b);
-     break;
-    case 'round':
-     a = stack.pop();
-     stack.push(Math.round(a));
-     break;
-    case 'sin':
-     a = stack.pop();
-     stack.push(Math.sin(a));
-     break;
-    case 'sqrt':
-     a = stack.pop();
-     stack.push(Math.sqrt(a));
-     break;
-    case 'sub':
-     b = stack.pop();
-     a = stack.pop();
-     stack.push(a - b);
-     break;
-    case 'true':
-     stack.push(true);
-     break;
-    case 'truncate':
-     a = stack.pop();
-     a = a < 0 ? Math.ceil(a) : Math.floor(a);
-     stack.push(a);
-     break;
-    case 'xor':
-     b = stack.pop();
-     a = stack.pop();
-     if (isBool(a) && isBool(b)) {
-      stack.push(a !== b);
-     } else {
-      stack.push(a ^ b);
-     }
-     break;
-    default:
-     error('Unknown operator ' + operator);
-     break;
-    }
-   }
-   return stack.stack;
+  function PostScriptEvaluator(operators) {
+    this.operators = operators;
   }
- };
- return PostScriptEvaluator;
+  PostScriptEvaluator.prototype = {
+    execute: function PostScriptEvaluator_execute(initialStack) {
+      var stack = new PostScriptStack(initialStack);
+      var counter = 0;
+      var operators = this.operators;
+      var length = operators.length;
+      var operator, a, b;
+      while (counter < length) {
+        operator = operators[counter++];
+        if (typeof operator === 'number') {
+          stack.push(operator);
+          continue;
+        }
+        switch (operator) {
+          case 'jz':
+            b = stack.pop();
+            a = stack.pop();
+            if (!a) {
+              counter = b;
+            }
+            break;
+          case 'j':
+            a = stack.pop();
+            counter = a;
+            break;
+          case 'abs':
+            a = stack.pop();
+            stack.push(Math.abs(a));
+            break;
+          case 'add':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a + b);
+            break;
+          case 'and':
+            b = stack.pop();
+            a = stack.pop();
+            if (isBool(a) && isBool(b)) {
+              stack.push(a && b);
+            } else {
+              stack.push(a & b);
+            }
+            break;
+          case 'atan':
+            a = stack.pop();
+            stack.push(Math.atan(a));
+            break;
+          case 'bitshift':
+            b = stack.pop();
+            a = stack.pop();
+            if (a > 0) {
+              stack.push(a << b);
+            } else {
+              stack.push(a >> b);
+            }
+            break;
+          case 'ceiling':
+            a = stack.pop();
+            stack.push(Math.ceil(a));
+            break;
+          case 'copy':
+            a = stack.pop();
+            stack.copy(a);
+            break;
+          case 'cos':
+            a = stack.pop();
+            stack.push(Math.cos(a));
+            break;
+          case 'cvi':
+            a = stack.pop() | 0;
+            stack.push(a);
+            break;
+          case 'cvr':
+            break;
+          case 'div':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a / b);
+            break;
+          case 'dup':
+            stack.copy(1);
+            break;
+          case 'eq':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a === b);
+            break;
+          case 'exch':
+            stack.roll(2, 1);
+            break;
+          case 'exp':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(Math.pow(a, b));
+            break;
+          case 'false':
+            stack.push(false);
+            break;
+          case 'floor':
+            a = stack.pop();
+            stack.push(Math.floor(a));
+            break;
+          case 'ge':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a >= b);
+            break;
+          case 'gt':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a > b);
+            break;
+          case 'idiv':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a / b | 0);
+            break;
+          case 'index':
+            a = stack.pop();
+            stack.index(a);
+            break;
+          case 'le':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a <= b);
+            break;
+          case 'ln':
+            a = stack.pop();
+            stack.push(Math.log(a));
+            break;
+          case 'log':
+            a = stack.pop();
+            stack.push(Math.log(a) / Math.LN10);
+            break;
+          case 'lt':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a < b);
+            break;
+          case 'mod':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a % b);
+            break;
+          case 'mul':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a * b);
+            break;
+          case 'ne':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a !== b);
+            break;
+          case 'neg':
+            a = stack.pop();
+            stack.push(-a);
+            break;
+          case 'not':
+            a = stack.pop();
+            if (isBool(a)) {
+              stack.push(!a);
+            } else {
+              stack.push(~a);
+            }
+            break;
+          case 'or':
+            b = stack.pop();
+            a = stack.pop();
+            if (isBool(a) && isBool(b)) {
+              stack.push(a || b);
+            } else {
+              stack.push(a | b);
+            }
+            break;
+          case 'pop':
+            stack.pop();
+            break;
+          case 'roll':
+            b = stack.pop();
+            a = stack.pop();
+            stack.roll(a, b);
+            break;
+          case 'round':
+            a = stack.pop();
+            stack.push(Math.round(a));
+            break;
+          case 'sin':
+            a = stack.pop();
+            stack.push(Math.sin(a));
+            break;
+          case 'sqrt':
+            a = stack.pop();
+            stack.push(Math.sqrt(a));
+            break;
+          case 'sub':
+            b = stack.pop();
+            a = stack.pop();
+            stack.push(a - b);
+            break;
+          case 'true':
+            stack.push(true);
+            break;
+          case 'truncate':
+            a = stack.pop();
+            a = a < 0 ? Math.ceil(a) : Math.floor(a);
+            stack.push(a);
+            break;
+          case 'xor':
+            b = stack.pop();
+            a = stack.pop();
+            if (isBool(a) && isBool(b)) {
+              stack.push(a !== b);
+            } else {
+              stack.push(a ^ b);
+            }
+            break;
+          default:
+            error('Unknown operator ' + operator);
+            break;
+        }
+      }
+      return stack.stack;
+    }
+  };
+  return PostScriptEvaluator;
 }();
 var PostScriptCompiler = function PostScriptCompilerClosure() {
- function AstNode(type) {
-  this.type = type;
- }
- AstNode.prototype.visit = function (visitor) {
-  throw new Error('abstract method');
- };
- function AstArgument(index, min, max) {
-  AstNode.call(this, 'args');
-  this.index = index;
-  this.min = min;
-  this.max = max;
- }
- AstArgument.prototype = Object.create(AstNode.prototype);
- AstArgument.prototype.visit = function (visitor) {
-  visitor.visitArgument(this);
- };
- function AstLiteral(number) {
-  AstNode.call(this, 'literal');
-  this.number = number;
-  this.min = number;
-  this.max = number;
- }
- AstLiteral.prototype = Object.create(AstNode.prototype);
- AstLiteral.prototype.visit = function (visitor) {
-  visitor.visitLiteral(this);
- };
- function AstBinaryOperation(op, arg1, arg2, min, max) {
-  AstNode.call(this, 'binary');
-  this.op = op;
-  this.arg1 = arg1;
-  this.arg2 = arg2;
-  this.min = min;
-  this.max = max;
- }
- AstBinaryOperation.prototype = Object.create(AstNode.prototype);
- AstBinaryOperation.prototype.visit = function (visitor) {
-  visitor.visitBinaryOperation(this);
- };
- function AstMin(arg, max) {
-  AstNode.call(this, 'max');
-  this.arg = arg;
-  this.min = arg.min;
-  this.max = max;
- }
- AstMin.prototype = Object.create(AstNode.prototype);
- AstMin.prototype.visit = function (visitor) {
-  visitor.visitMin(this);
- };
- function AstVariable(index, min, max) {
-  AstNode.call(this, 'var');
-  this.index = index;
-  this.min = min;
-  this.max = max;
- }
- AstVariable.prototype = Object.create(AstNode.prototype);
- AstVariable.prototype.visit = function (visitor) {
-  visitor.visitVariable(this);
- };
- function AstVariableDefinition(variable, arg) {
-  AstNode.call(this, 'definition');
-  this.variable = variable;
-  this.arg = arg;
- }
- AstVariableDefinition.prototype = Object.create(AstNode.prototype);
- AstVariableDefinition.prototype.visit = function (visitor) {
-  visitor.visitVariableDefinition(this);
- };
- function ExpressionBuilderVisitor() {
-  this.parts = [];
- }
- ExpressionBuilderVisitor.prototype = {
-  visitArgument: function (arg) {
-   this.parts.push('Math.max(', arg.min, ', Math.min(', arg.max, ', src[srcOffset + ', arg.index, ']))');
-  },
-  visitVariable: function (variable) {
-   this.parts.push('v', variable.index);
-  },
-  visitLiteral: function (literal) {
-   this.parts.push(literal.number);
-  },
-  visitBinaryOperation: function (operation) {
-   this.parts.push('(');
-   operation.arg1.visit(this);
-   this.parts.push(' ', operation.op, ' ');
-   operation.arg2.visit(this);
-   this.parts.push(')');
-  },
-  visitVariableDefinition: function (definition) {
-   this.parts.push('var ');
-   definition.variable.visit(this);
-   this.parts.push(' = ');
-   definition.arg.visit(this);
-   this.parts.push(';');
-  },
-  visitMin: function (max) {
-   this.parts.push('Math.min(');
-   max.arg.visit(this);
-   this.parts.push(', ', max.max, ')');
-  },
-  toString: function () {
-   return this.parts.join('');
+  function AstNode(type) {
+    this.type = type;
   }
- };
- function buildAddOperation(num1, num2) {
-  if (num2.type === 'literal' && num2.number === 0) {
-   return num1;
+  AstNode.prototype.visit = function (visitor) {
+    throw new Error('abstract method');
+  };
+  function AstArgument(index, min, max) {
+    AstNode.call(this, 'args');
+    this.index = index;
+    this.min = min;
+    this.max = max;
   }
-  if (num1.type === 'literal' && num1.number === 0) {
-   return num2;
+  AstArgument.prototype = Object.create(AstNode.prototype);
+  AstArgument.prototype.visit = function (visitor) {
+    visitor.visitArgument(this);
+  };
+  function AstLiteral(number) {
+    AstNode.call(this, 'literal');
+    this.number = number;
+    this.min = number;
+    this.max = number;
   }
-  if (num2.type === 'literal' && num1.type === 'literal') {
-   return new AstLiteral(num1.number + num2.number);
+  AstLiteral.prototype = Object.create(AstNode.prototype);
+  AstLiteral.prototype.visit = function (visitor) {
+    visitor.visitLiteral(this);
+  };
+  function AstBinaryOperation(op, arg1, arg2, min, max) {
+    AstNode.call(this, 'binary');
+    this.op = op;
+    this.arg1 = arg1;
+    this.arg2 = arg2;
+    this.min = min;
+    this.max = max;
   }
-  return new AstBinaryOperation('+', num1, num2, num1.min + num2.min, num1.max + num2.max);
- }
- function buildMulOperation(num1, num2) {
-  if (num2.type === 'literal') {
-   if (num2.number === 0) {
-    return new AstLiteral(0);
-   } else if (num2.number === 1) {
-    return num1;
-   } else if (num1.type === 'literal') {
-    return new AstLiteral(num1.number * num2.number);
-   }
+  AstBinaryOperation.prototype = Object.create(AstNode.prototype);
+  AstBinaryOperation.prototype.visit = function (visitor) {
+    visitor.visitBinaryOperation(this);
+  };
+  function AstMin(arg, max) {
+    AstNode.call(this, 'max');
+    this.arg = arg;
+    this.min = arg.min;
+    this.max = max;
   }
-  if (num1.type === 'literal') {
-   if (num1.number === 0) {
-    return new AstLiteral(0);
-   } else if (num1.number === 1) {
-    return num2;
-   }
+  AstMin.prototype = Object.create(AstNode.prototype);
+  AstMin.prototype.visit = function (visitor) {
+    visitor.visitMin(this);
+  };
+  function AstVariable(index, min, max) {
+    AstNode.call(this, 'var');
+    this.index = index;
+    this.min = min;
+    this.max = max;
   }
-  var min = Math.min(num1.min * num2.min, num1.min * num2.max, num1.max * num2.min, num1.max * num2.max);
-  var max = Math.max(num1.min * num2.min, num1.min * num2.max, num1.max * num2.min, num1.max * num2.max);
-  return new AstBinaryOperation('*', num1, num2, min, max);
- }
- function buildSubOperation(num1, num2) {
-  if (num2.type === 'literal') {
-   if (num2.number === 0) {
-    return num1;
-   } else if (num1.type === 'literal') {
-    return new AstLiteral(num1.number - num2.number);
-   }
+  AstVariable.prototype = Object.create(AstNode.prototype);
+  AstVariable.prototype.visit = function (visitor) {
+    visitor.visitVariable(this);
+  };
+  function AstVariableDefinition(variable, arg) {
+    AstNode.call(this, 'definition');
+    this.variable = variable;
+    this.arg = arg;
   }
-  if (num2.type === 'binary' && num2.op === '-' && num1.type === 'literal' && num1.number === 1 && num2.arg1.type === 'literal' && num2.arg1.number === 1) {
-   return num2.arg2;
+  AstVariableDefinition.prototype = Object.create(AstNode.prototype);
+  AstVariableDefinition.prototype.visit = function (visitor) {
+    visitor.visitVariableDefinition(this);
+  };
+  function ExpressionBuilderVisitor() {
+    this.parts = [];
   }
-  return new AstBinaryOperation('-', num1, num2, num1.min - num2.max, num1.max - num2.min);
- }
- function buildMinOperation(num1, max) {
-  if (num1.min >= max) {
-   return new AstLiteral(max);
-  } else if (num1.max <= max) {
-   return num1;
+  ExpressionBuilderVisitor.prototype = {
+    visitArgument: function (arg) {
+      this.parts.push('Math.max(', arg.min, ', Math.min(', arg.max, ', src[srcOffset + ', arg.index, ']))');
+    },
+    visitVariable: function (variable) {
+      this.parts.push('v', variable.index);
+    },
+    visitLiteral: function (literal) {
+      this.parts.push(literal.number);
+    },
+    visitBinaryOperation: function (operation) {
+      this.parts.push('(');
+      operation.arg1.visit(this);
+      this.parts.push(' ', operation.op, ' ');
+      operation.arg2.visit(this);
+      this.parts.push(')');
+    },
+    visitVariableDefinition: function (definition) {
+      this.parts.push('var ');
+      definition.variable.visit(this);
+      this.parts.push(' = ');
+      definition.arg.visit(this);
+      this.parts.push(';');
+    },
+    visitMin: function (max) {
+      this.parts.push('Math.min(');
+      max.arg.visit(this);
+      this.parts.push(', ', max.max, ')');
+    },
+    toString: function () {
+      return this.parts.join('');
+    }
+  };
+  function buildAddOperation(num1, num2) {
+    if (num2.type === 'literal' && num2.number === 0) {
+      return num1;
+    }
+    if (num1.type === 'literal' && num1.number === 0) {
+      return num2;
+    }
+    if (num2.type === 'literal' && num1.type === 'literal') {
+      return new AstLiteral(num1.number + num2.number);
+    }
+    return new AstBinaryOperation('+', num1, num2, num1.min + num2.min, num1.max + num2.max);
   }
-  return new AstMin(num1, max);
- }
- function PostScriptCompiler() {
- }
- PostScriptCompiler.prototype = {
-  compile: function PostScriptCompiler_compile(code, domain, range) {
-   var stack = [];
-   var i, ii;
-   var instructions = [];
-   var inputSize = domain.length >> 1, outputSize = range.length >> 1;
-   var lastRegister = 0;
-   var n, j;
-   var num1, num2, ast1, ast2, tmpVar, item;
-   for (i = 0; i < inputSize; i++) {
-    stack.push(new AstArgument(i, domain[i * 2], domain[i * 2 + 1]));
-   }
-   for (i = 0, ii = code.length; i < ii; i++) {
-    item = code[i];
-    if (typeof item === 'number') {
-     stack.push(new AstLiteral(item));
-     continue;
+  function buildMulOperation(num1, num2) {
+    if (num2.type === 'literal') {
+      if (num2.number === 0) {
+        return new AstLiteral(0);
+      } else if (num2.number === 1) {
+        return num1;
+      } else if (num1.type === 'literal') {
+        return new AstLiteral(num1.number * num2.number);
+      }
     }
-    switch (item) {
-    case 'add':
-     if (stack.length < 2) {
-      return null;
-     }
-     num2 = stack.pop();
-     num1 = stack.pop();
-     stack.push(buildAddOperation(num1, num2));
-     break;
-    case 'cvr':
-     if (stack.length < 1) {
-      return null;
-     }
-     break;
-    case 'mul':
-     if (stack.length < 2) {
-      return null;
-     }
-     num2 = stack.pop();
-     num1 = stack.pop();
-     stack.push(buildMulOperation(num1, num2));
-     break;
-    case 'sub':
-     if (stack.length < 2) {
-      return null;
-     }
-     num2 = stack.pop();
-     num1 = stack.pop();
-     stack.push(buildSubOperation(num1, num2));
-     break;
-    case 'exch':
-     if (stack.length < 2) {
-      return null;
-     }
-     ast1 = stack.pop();
-     ast2 = stack.pop();
-     stack.push(ast1, ast2);
-     break;
-    case 'pop':
-     if (stack.length < 1) {
-      return null;
-     }
-     stack.pop();
-     break;
-    case 'index':
-     if (stack.length < 1) {
-      return null;
-     }
-     num1 = stack.pop();
-     if (num1.type !== 'literal') {
-      return null;
-     }
-     n = num1.number;
-     if (n < 0 || (n | 0) !== n || stack.length < n) {
-      return null;
-     }
-     ast1 = stack[stack.length - n - 1];
-     if (ast1.type === 'literal' || ast1.type === 'var') {
-      stack.push(ast1);
-      break;
-     }
-     tmpVar = new AstVariable(lastRegister++, ast1.min, ast1.max);
-     stack[stack.length - n - 1] = tmpVar;
-     stack.push(tmpVar);
-     instructions.push(new AstVariableDefinition(tmpVar, ast1));
-     break;
-    case 'dup':
-     if (stack.length < 1) {
-      return null;
-     }
-     if (typeof code[i + 1] === 'number' && code[i + 2] === 'gt' && code[i + 3] === i + 7 && code[i + 4] === 'jz' && code[i + 5] === 'pop' && code[i + 6] === code[i + 1]) {
-      num1 = stack.pop();
-      stack.push(buildMinOperation(num1, code[i + 1]));
-      i += 6;
-      break;
-     }
-     ast1 = stack[stack.length - 1];
-     if (ast1.type === 'literal' || ast1.type === 'var') {
-      stack.push(ast1);
-      break;
-     }
-     tmpVar = new AstVariable(lastRegister++, ast1.min, ast1.max);
-     stack[stack.length - 1] = tmpVar;
-     stack.push(tmpVar);
-     instructions.push(new AstVariableDefinition(tmpVar, ast1));
-     break;
-    case 'roll':
-     if (stack.length < 2) {
-      return null;
-     }
-     num2 = stack.pop();
-     num1 = stack.pop();
-     if (num2.type !== 'literal' || num1.type !== 'literal') {
-      return null;
-     }
-     j = num2.number;
-     n = num1.number;
-     if (n <= 0 || (n | 0) !== n || (j | 0) !== j || stack.length < n) {
-      return null;
-     }
-     j = (j % n + n) % n;
-     if (j === 0) {
-      break;
-     }
-     Array.prototype.push.apply(stack, stack.splice(stack.length - n, n - j));
-     break;
-    default:
-     return null;
+    if (num1.type === 'literal') {
+      if (num1.number === 0) {
+        return new AstLiteral(0);
+      } else if (num1.number === 1) {
+        return num2;
+      }
     }
-   }
-   if (stack.length !== outputSize) {
-    return null;
-   }
-   var result = [];
-   instructions.forEach(function (instruction) {
-    var statementBuilder = new ExpressionBuilderVisitor();
-    instruction.visit(statementBuilder);
-    result.push(statementBuilder.toString());
-   });
-   stack.forEach(function (expr, i) {
-    var statementBuilder = new ExpressionBuilderVisitor();
-    expr.visit(statementBuilder);
-    var min = range[i * 2], max = range[i * 2 + 1];
-    var out = [statementBuilder.toString()];
-    if (min > expr.min) {
-     out.unshift('Math.max(', min, ', ');
-     out.push(')');
+    var min = Math.min(num1.min * num2.min, num1.min * num2.max, num1.max * num2.min, num1.max * num2.max);
+    var max = Math.max(num1.min * num2.min, num1.min * num2.max, num1.max * num2.min, num1.max * num2.max);
+    return new AstBinaryOperation('*', num1, num2, min, max);
+  }
+  function buildSubOperation(num1, num2) {
+    if (num2.type === 'literal') {
+      if (num2.number === 0) {
+        return num1;
+      } else if (num1.type === 'literal') {
+        return new AstLiteral(num1.number - num2.number);
+      }
+    }
+    if (num2.type === 'binary' && num2.op === '-' && num1.type === 'literal' && num1.number === 1 && num2.arg1.type === 'literal' && num2.arg1.number === 1) {
+      return num2.arg2;
     }
-    if (max < expr.max) {
-     out.unshift('Math.min(', max, ', ');
-     out.push(')');
+    return new AstBinaryOperation('-', num1, num2, num1.min - num2.max, num1.max - num2.min);
+  }
+  function buildMinOperation(num1, max) {
+    if (num1.min >= max) {
+      return new AstLiteral(max);
+    } else if (num1.max <= max) {
+      return num1;
     }
-    out.unshift('dest[destOffset + ', i, '] = ');
-    out.push(';');
-    result.push(out.join(''));
-   });
-   return result.join('\n');
+    return new AstMin(num1, max);
   }
- };
- return PostScriptCompiler;
+  function PostScriptCompiler() {}
+  PostScriptCompiler.prototype = {
+    compile: function PostScriptCompiler_compile(code, domain, range) {
+      var stack = [];
+      var i, ii;
+      var instructions = [];
+      var inputSize = domain.length >> 1,
+          outputSize = range.length >> 1;
+      var lastRegister = 0;
+      var n, j;
+      var num1, num2, ast1, ast2, tmpVar, item;
+      for (i = 0; i < inputSize; i++) {
+        stack.push(new AstArgument(i, domain[i * 2], domain[i * 2 + 1]));
+      }
+      for (i = 0, ii = code.length; i < ii; i++) {
+        item = code[i];
+        if (typeof item === 'number') {
+          stack.push(new AstLiteral(item));
+          continue;
+        }
+        switch (item) {
+          case 'add':
+            if (stack.length < 2) {
+              return null;
+            }
+            num2 = stack.pop();
+            num1 = stack.pop();
+            stack.push(buildAddOperation(num1, num2));
+            break;
+          case 'cvr':
+            if (stack.length < 1) {
+              return null;
+            }
+            break;
+          case 'mul':
+            if (stack.length < 2) {
+              return null;
+            }
+            num2 = stack.pop();
+            num1 = stack.pop();
+            stack.push(buildMulOperation(num1, num2));
+            break;
+          case 'sub':
+            if (stack.length < 2) {
+              return null;
+            }
+            num2 = stack.pop();
+            num1 = stack.pop();
+            stack.push(buildSubOperation(num1, num2));
+            break;
+          case 'exch':
+            if (stack.length < 2) {
+              return null;
+            }
+            ast1 = stack.pop();
+            ast2 = stack.pop();
+            stack.push(ast1, ast2);
+            break;
+          case 'pop':
+            if (stack.length < 1) {
+              return null;
+            }
+            stack.pop();
+            break;
+          case 'index':
+            if (stack.length < 1) {
+              return null;
+            }
+            num1 = stack.pop();
+            if (num1.type !== 'literal') {
+              return null;
+            }
+            n = num1.number;
+            if (n < 0 || (n | 0) !== n || stack.length < n) {
+              return null;
+            }
+            ast1 = stack[stack.length - n - 1];
+            if (ast1.type === 'literal' || ast1.type === 'var') {
+              stack.push(ast1);
+              break;
+            }
+            tmpVar = new AstVariable(lastRegister++, ast1.min, ast1.max);
+            stack[stack.length - n - 1] = tmpVar;
+            stack.push(tmpVar);
+            instructions.push(new AstVariableDefinition(tmpVar, ast1));
+            break;
+          case 'dup':
+            if (stack.length < 1) {
+              return null;
+            }
+            if (typeof code[i + 1] === 'number' && code[i + 2] === 'gt' && code[i + 3] === i + 7 && code[i + 4] === 'jz' && code[i + 5] === 'pop' && code[i + 6] === code[i + 1]) {
+              num1 = stack.pop();
+              stack.push(buildMinOperation(num1, code[i + 1]));
+              i += 6;
+              break;
+            }
+            ast1 = stack[stack.length - 1];
+            if (ast1.type === 'literal' || ast1.type === 'var') {
+              stack.push(ast1);
+              break;
+            }
+            tmpVar = new AstVariable(lastRegister++, ast1.min, ast1.max);
+            stack[stack.length - 1] = tmpVar;
+            stack.push(tmpVar);
+            instructions.push(new AstVariableDefinition(tmpVar, ast1));
+            break;
+          case 'roll':
+            if (stack.length < 2) {
+              return null;
+            }
+            num2 = stack.pop();
+            num1 = stack.pop();
+            if (num2.type !== 'literal' || num1.type !== 'literal') {
+              return null;
+            }
+            j = num2.number;
+            n = num1.number;
+            if (n <= 0 || (n | 0) !== n || (j | 0) !== j || stack.length < n) {
+              return null;
+            }
+            j = (j % n + n) % n;
+            if (j === 0) {
+              break;
+            }
+            Array.prototype.push.apply(stack, stack.splice(stack.length - n, n - j));
+            break;
+          default:
+            return null;
+        }
+      }
+      if (stack.length !== outputSize) {
+        return null;
+      }
+      var result = [];
+      instructions.forEach(function (instruction) {
+        var statementBuilder = new ExpressionBuilderVisitor();
+        instruction.visit(statementBuilder);
+        result.push(statementBuilder.toString());
+      });
+      stack.forEach(function (expr, i) {
+        var statementBuilder = new ExpressionBuilderVisitor();
+        expr.visit(statementBuilder);
+        var min = range[i * 2],
+            max = range[i * 2 + 1];
+        var out = [statementBuilder.toString()];
+        if (min > expr.min) {
+          out.unshift('Math.max(', min, ', ');
+          out.push(')');
+        }
+        if (max < expr.max) {
+          out.unshift('Math.min(', max, ', ');
+          out.push(')');
+        }
+        out.unshift('dest[destOffset + ', i, '] = ');
+        out.push(';');
+        result.push(out.join(''));
+      });
+      return result.join('\n');
+    }
+  };
+  return PostScriptCompiler;
 }();
 exports.isPDFFunction = isPDFFunction;
 exports.PDFFunction = PDFFunction;

+ 4525 - 4524
lib/core/glyphlist.js

@@ -13,4535 +13,4536 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var getLookupTableFactory = sharedUtil.getLookupTableFactory;
 var getGlyphsUnicode = getLookupTableFactory(function (t) {
- t['A'] = 0x0041;
- t['AE'] = 0x00C6;
- t['AEacute'] = 0x01FC;
- t['AEmacron'] = 0x01E2;
- t['AEsmall'] = 0xF7E6;
- t['Aacute'] = 0x00C1;
- t['Aacutesmall'] = 0xF7E1;
- t['Abreve'] = 0x0102;
- t['Abreveacute'] = 0x1EAE;
- t['Abrevecyrillic'] = 0x04D0;
- t['Abrevedotbelow'] = 0x1EB6;
- t['Abrevegrave'] = 0x1EB0;
- t['Abrevehookabove'] = 0x1EB2;
- t['Abrevetilde'] = 0x1EB4;
- t['Acaron'] = 0x01CD;
- t['Acircle'] = 0x24B6;
- t['Acircumflex'] = 0x00C2;
- t['Acircumflexacute'] = 0x1EA4;
- t['Acircumflexdotbelow'] = 0x1EAC;
- t['Acircumflexgrave'] = 0x1EA6;
- t['Acircumflexhookabove'] = 0x1EA8;
- t['Acircumflexsmall'] = 0xF7E2;
- t['Acircumflextilde'] = 0x1EAA;
- t['Acute'] = 0xF6C9;
- t['Acutesmall'] = 0xF7B4;
- t['Acyrillic'] = 0x0410;
- t['Adblgrave'] = 0x0200;
- t['Adieresis'] = 0x00C4;
- t['Adieresiscyrillic'] = 0x04D2;
- t['Adieresismacron'] = 0x01DE;
- t['Adieresissmall'] = 0xF7E4;
- t['Adotbelow'] = 0x1EA0;
- t['Adotmacron'] = 0x01E0;
- t['Agrave'] = 0x00C0;
- t['Agravesmall'] = 0xF7E0;
- t['Ahookabove'] = 0x1EA2;
- t['Aiecyrillic'] = 0x04D4;
- t['Ainvertedbreve'] = 0x0202;
- t['Alpha'] = 0x0391;
- t['Alphatonos'] = 0x0386;
- t['Amacron'] = 0x0100;
- t['Amonospace'] = 0xFF21;
- t['Aogonek'] = 0x0104;
- t['Aring'] = 0x00C5;
- t['Aringacute'] = 0x01FA;
- t['Aringbelow'] = 0x1E00;
- t['Aringsmall'] = 0xF7E5;
- t['Asmall'] = 0xF761;
- t['Atilde'] = 0x00C3;
- t['Atildesmall'] = 0xF7E3;
- t['Aybarmenian'] = 0x0531;
- t['B'] = 0x0042;
- t['Bcircle'] = 0x24B7;
- t['Bdotaccent'] = 0x1E02;
- t['Bdotbelow'] = 0x1E04;
- t['Becyrillic'] = 0x0411;
- t['Benarmenian'] = 0x0532;
- t['Beta'] = 0x0392;
- t['Bhook'] = 0x0181;
- t['Blinebelow'] = 0x1E06;
- t['Bmonospace'] = 0xFF22;
- t['Brevesmall'] = 0xF6F4;
- t['Bsmall'] = 0xF762;
- t['Btopbar'] = 0x0182;
- t['C'] = 0x0043;
- t['Caarmenian'] = 0x053E;
- t['Cacute'] = 0x0106;
- t['Caron'] = 0xF6CA;
- t['Caronsmall'] = 0xF6F5;
- t['Ccaron'] = 0x010C;
- t['Ccedilla'] = 0x00C7;
- t['Ccedillaacute'] = 0x1E08;
- t['Ccedillasmall'] = 0xF7E7;
- t['Ccircle'] = 0x24B8;
- t['Ccircumflex'] = 0x0108;
- t['Cdot'] = 0x010A;
- t['Cdotaccent'] = 0x010A;
- t['Cedillasmall'] = 0xF7B8;
- t['Chaarmenian'] = 0x0549;
- t['Cheabkhasiancyrillic'] = 0x04BC;
- t['Checyrillic'] = 0x0427;
- t['Chedescenderabkhasiancyrillic'] = 0x04BE;
- t['Chedescendercyrillic'] = 0x04B6;
- t['Chedieresiscyrillic'] = 0x04F4;
- t['Cheharmenian'] = 0x0543;
- t['Chekhakassiancyrillic'] = 0x04CB;
- t['Cheverticalstrokecyrillic'] = 0x04B8;
- t['Chi'] = 0x03A7;
- t['Chook'] = 0x0187;
- t['Circumflexsmall'] = 0xF6F6;
- t['Cmonospace'] = 0xFF23;
- t['Coarmenian'] = 0x0551;
- t['Csmall'] = 0xF763;
- t['D'] = 0x0044;
- t['DZ'] = 0x01F1;
- t['DZcaron'] = 0x01C4;
- t['Daarmenian'] = 0x0534;
- t['Dafrican'] = 0x0189;
- t['Dcaron'] = 0x010E;
- t['Dcedilla'] = 0x1E10;
- t['Dcircle'] = 0x24B9;
- t['Dcircumflexbelow'] = 0x1E12;
- t['Dcroat'] = 0x0110;
- t['Ddotaccent'] = 0x1E0A;
- t['Ddotbelow'] = 0x1E0C;
- t['Decyrillic'] = 0x0414;
- t['Deicoptic'] = 0x03EE;
- t['Delta'] = 0x2206;
- t['Deltagreek'] = 0x0394;
- t['Dhook'] = 0x018A;
- t['Dieresis'] = 0xF6CB;
- t['DieresisAcute'] = 0xF6CC;
- t['DieresisGrave'] = 0xF6CD;
- t['Dieresissmall'] = 0xF7A8;
- t['Digammagreek'] = 0x03DC;
- t['Djecyrillic'] = 0x0402;
- t['Dlinebelow'] = 0x1E0E;
- t['Dmonospace'] = 0xFF24;
- t['Dotaccentsmall'] = 0xF6F7;
- t['Dslash'] = 0x0110;
- t['Dsmall'] = 0xF764;
- t['Dtopbar'] = 0x018B;
- t['Dz'] = 0x01F2;
- t['Dzcaron'] = 0x01C5;
- t['Dzeabkhasiancyrillic'] = 0x04E0;
- t['Dzecyrillic'] = 0x0405;
- t['Dzhecyrillic'] = 0x040F;
- t['E'] = 0x0045;
- t['Eacute'] = 0x00C9;
- t['Eacutesmall'] = 0xF7E9;
- t['Ebreve'] = 0x0114;
- t['Ecaron'] = 0x011A;
- t['Ecedillabreve'] = 0x1E1C;
- t['Echarmenian'] = 0x0535;
- t['Ecircle'] = 0x24BA;
- t['Ecircumflex'] = 0x00CA;
- t['Ecircumflexacute'] = 0x1EBE;
- t['Ecircumflexbelow'] = 0x1E18;
- t['Ecircumflexdotbelow'] = 0x1EC6;
- t['Ecircumflexgrave'] = 0x1EC0;
- t['Ecircumflexhookabove'] = 0x1EC2;
- t['Ecircumflexsmall'] = 0xF7EA;
- t['Ecircumflextilde'] = 0x1EC4;
- t['Ecyrillic'] = 0x0404;
- t['Edblgrave'] = 0x0204;
- t['Edieresis'] = 0x00CB;
- t['Edieresissmall'] = 0xF7EB;
- t['Edot'] = 0x0116;
- t['Edotaccent'] = 0x0116;
- t['Edotbelow'] = 0x1EB8;
- t['Efcyrillic'] = 0x0424;
- t['Egrave'] = 0x00C8;
- t['Egravesmall'] = 0xF7E8;
- t['Eharmenian'] = 0x0537;
- t['Ehookabove'] = 0x1EBA;
- t['Eightroman'] = 0x2167;
- t['Einvertedbreve'] = 0x0206;
- t['Eiotifiedcyrillic'] = 0x0464;
- t['Elcyrillic'] = 0x041B;
- t['Elevenroman'] = 0x216A;
- t['Emacron'] = 0x0112;
- t['Emacronacute'] = 0x1E16;
- t['Emacrongrave'] = 0x1E14;
- t['Emcyrillic'] = 0x041C;
- t['Emonospace'] = 0xFF25;
- t['Encyrillic'] = 0x041D;
- t['Endescendercyrillic'] = 0x04A2;
- t['Eng'] = 0x014A;
- t['Enghecyrillic'] = 0x04A4;
- t['Enhookcyrillic'] = 0x04C7;
- t['Eogonek'] = 0x0118;
- t['Eopen'] = 0x0190;
- t['Epsilon'] = 0x0395;
- t['Epsilontonos'] = 0x0388;
- t['Ercyrillic'] = 0x0420;
- t['Ereversed'] = 0x018E;
- t['Ereversedcyrillic'] = 0x042D;
- t['Escyrillic'] = 0x0421;
- t['Esdescendercyrillic'] = 0x04AA;
- t['Esh'] = 0x01A9;
- t['Esmall'] = 0xF765;
- t['Eta'] = 0x0397;
- t['Etarmenian'] = 0x0538;
- t['Etatonos'] = 0x0389;
- t['Eth'] = 0x00D0;
- t['Ethsmall'] = 0xF7F0;
- t['Etilde'] = 0x1EBC;
- t['Etildebelow'] = 0x1E1A;
- t['Euro'] = 0x20AC;
- t['Ezh'] = 0x01B7;
- t['Ezhcaron'] = 0x01EE;
- t['Ezhreversed'] = 0x01B8;
- t['F'] = 0x0046;
- t['Fcircle'] = 0x24BB;
- t['Fdotaccent'] = 0x1E1E;
- t['Feharmenian'] = 0x0556;
- t['Feicoptic'] = 0x03E4;
- t['Fhook'] = 0x0191;
- t['Fitacyrillic'] = 0x0472;
- t['Fiveroman'] = 0x2164;
- t['Fmonospace'] = 0xFF26;
- t['Fourroman'] = 0x2163;
- t['Fsmall'] = 0xF766;
- t['G'] = 0x0047;
- t['GBsquare'] = 0x3387;
- t['Gacute'] = 0x01F4;
- t['Gamma'] = 0x0393;
- t['Gammaafrican'] = 0x0194;
- t['Gangiacoptic'] = 0x03EA;
- t['Gbreve'] = 0x011E;
- t['Gcaron'] = 0x01E6;
- t['Gcedilla'] = 0x0122;
- t['Gcircle'] = 0x24BC;
- t['Gcircumflex'] = 0x011C;
- t['Gcommaaccent'] = 0x0122;
- t['Gdot'] = 0x0120;
- t['Gdotaccent'] = 0x0120;
- t['Gecyrillic'] = 0x0413;
- t['Ghadarmenian'] = 0x0542;
- t['Ghemiddlehookcyrillic'] = 0x0494;
- t['Ghestrokecyrillic'] = 0x0492;
- t['Gheupturncyrillic'] = 0x0490;
- t['Ghook'] = 0x0193;
- t['Gimarmenian'] = 0x0533;
- t['Gjecyrillic'] = 0x0403;
- t['Gmacron'] = 0x1E20;
- t['Gmonospace'] = 0xFF27;
- t['Grave'] = 0xF6CE;
- t['Gravesmall'] = 0xF760;
- t['Gsmall'] = 0xF767;
- t['Gsmallhook'] = 0x029B;
- t['Gstroke'] = 0x01E4;
- t['H'] = 0x0048;
- t['H18533'] = 0x25CF;
- t['H18543'] = 0x25AA;
- t['H18551'] = 0x25AB;
- t['H22073'] = 0x25A1;
- t['HPsquare'] = 0x33CB;
- t['Haabkhasiancyrillic'] = 0x04A8;
- t['Hadescendercyrillic'] = 0x04B2;
- t['Hardsigncyrillic'] = 0x042A;
- t['Hbar'] = 0x0126;
- t['Hbrevebelow'] = 0x1E2A;
- t['Hcedilla'] = 0x1E28;
- t['Hcircle'] = 0x24BD;
- t['Hcircumflex'] = 0x0124;
- t['Hdieresis'] = 0x1E26;
- t['Hdotaccent'] = 0x1E22;
- t['Hdotbelow'] = 0x1E24;
- t['Hmonospace'] = 0xFF28;
- t['Hoarmenian'] = 0x0540;
- t['Horicoptic'] = 0x03E8;
- t['Hsmall'] = 0xF768;
- t['Hungarumlaut'] = 0xF6CF;
- t['Hungarumlautsmall'] = 0xF6F8;
- t['Hzsquare'] = 0x3390;
- t['I'] = 0x0049;
- t['IAcyrillic'] = 0x042F;
- t['IJ'] = 0x0132;
- t['IUcyrillic'] = 0x042E;
- t['Iacute'] = 0x00CD;
- t['Iacutesmall'] = 0xF7ED;
- t['Ibreve'] = 0x012C;
- t['Icaron'] = 0x01CF;
- t['Icircle'] = 0x24BE;
- t['Icircumflex'] = 0x00CE;
- t['Icircumflexsmall'] = 0xF7EE;
- t['Icyrillic'] = 0x0406;
- t['Idblgrave'] = 0x0208;
- t['Idieresis'] = 0x00CF;
- t['Idieresisacute'] = 0x1E2E;
- t['Idieresiscyrillic'] = 0x04E4;
- t['Idieresissmall'] = 0xF7EF;
- t['Idot'] = 0x0130;
- t['Idotaccent'] = 0x0130;
- t['Idotbelow'] = 0x1ECA;
- t['Iebrevecyrillic'] = 0x04D6;
- t['Iecyrillic'] = 0x0415;
- t['Ifraktur'] = 0x2111;
- t['Igrave'] = 0x00CC;
- t['Igravesmall'] = 0xF7EC;
- t['Ihookabove'] = 0x1EC8;
- t['Iicyrillic'] = 0x0418;
- t['Iinvertedbreve'] = 0x020A;
- t['Iishortcyrillic'] = 0x0419;
- t['Imacron'] = 0x012A;
- t['Imacroncyrillic'] = 0x04E2;
- t['Imonospace'] = 0xFF29;
- t['Iniarmenian'] = 0x053B;
- t['Iocyrillic'] = 0x0401;
- t['Iogonek'] = 0x012E;
- t['Iota'] = 0x0399;
- t['Iotaafrican'] = 0x0196;
- t['Iotadieresis'] = 0x03AA;
- t['Iotatonos'] = 0x038A;
- t['Ismall'] = 0xF769;
- t['Istroke'] = 0x0197;
- t['Itilde'] = 0x0128;
- t['Itildebelow'] = 0x1E2C;
- t['Izhitsacyrillic'] = 0x0474;
- t['Izhitsadblgravecyrillic'] = 0x0476;
- t['J'] = 0x004A;
- t['Jaarmenian'] = 0x0541;
- t['Jcircle'] = 0x24BF;
- t['Jcircumflex'] = 0x0134;
- t['Jecyrillic'] = 0x0408;
- t['Jheharmenian'] = 0x054B;
- t['Jmonospace'] = 0xFF2A;
- t['Jsmall'] = 0xF76A;
- t['K'] = 0x004B;
- t['KBsquare'] = 0x3385;
- t['KKsquare'] = 0x33CD;
- t['Kabashkircyrillic'] = 0x04A0;
- t['Kacute'] = 0x1E30;
- t['Kacyrillic'] = 0x041A;
- t['Kadescendercyrillic'] = 0x049A;
- t['Kahookcyrillic'] = 0x04C3;
- t['Kappa'] = 0x039A;
- t['Kastrokecyrillic'] = 0x049E;
- t['Kaverticalstrokecyrillic'] = 0x049C;
- t['Kcaron'] = 0x01E8;
- t['Kcedilla'] = 0x0136;
- t['Kcircle'] = 0x24C0;
- t['Kcommaaccent'] = 0x0136;
- t['Kdotbelow'] = 0x1E32;
- t['Keharmenian'] = 0x0554;
- t['Kenarmenian'] = 0x053F;
- t['Khacyrillic'] = 0x0425;
- t['Kheicoptic'] = 0x03E6;
- t['Khook'] = 0x0198;
- t['Kjecyrillic'] = 0x040C;
- t['Klinebelow'] = 0x1E34;
- t['Kmonospace'] = 0xFF2B;
- t['Koppacyrillic'] = 0x0480;
- t['Koppagreek'] = 0x03DE;
- t['Ksicyrillic'] = 0x046E;
- t['Ksmall'] = 0xF76B;
- t['L'] = 0x004C;
- t['LJ'] = 0x01C7;
- t['LL'] = 0xF6BF;
- t['Lacute'] = 0x0139;
- t['Lambda'] = 0x039B;
- t['Lcaron'] = 0x013D;
- t['Lcedilla'] = 0x013B;
- t['Lcircle'] = 0x24C1;
- t['Lcircumflexbelow'] = 0x1E3C;
- t['Lcommaaccent'] = 0x013B;
- t['Ldot'] = 0x013F;
- t['Ldotaccent'] = 0x013F;
- t['Ldotbelow'] = 0x1E36;
- t['Ldotbelowmacron'] = 0x1E38;
- t['Liwnarmenian'] = 0x053C;
- t['Lj'] = 0x01C8;
- t['Ljecyrillic'] = 0x0409;
- t['Llinebelow'] = 0x1E3A;
- t['Lmonospace'] = 0xFF2C;
- t['Lslash'] = 0x0141;
- t['Lslashsmall'] = 0xF6F9;
- t['Lsmall'] = 0xF76C;
- t['M'] = 0x004D;
- t['MBsquare'] = 0x3386;
- t['Macron'] = 0xF6D0;
- t['Macronsmall'] = 0xF7AF;
- t['Macute'] = 0x1E3E;
- t['Mcircle'] = 0x24C2;
- t['Mdotaccent'] = 0x1E40;
- t['Mdotbelow'] = 0x1E42;
- t['Menarmenian'] = 0x0544;
- t['Mmonospace'] = 0xFF2D;
- t['Msmall'] = 0xF76D;
- t['Mturned'] = 0x019C;
- t['Mu'] = 0x039C;
- t['N'] = 0x004E;
- t['NJ'] = 0x01CA;
- t['Nacute'] = 0x0143;
- t['Ncaron'] = 0x0147;
- t['Ncedilla'] = 0x0145;
- t['Ncircle'] = 0x24C3;
- t['Ncircumflexbelow'] = 0x1E4A;
- t['Ncommaaccent'] = 0x0145;
- t['Ndotaccent'] = 0x1E44;
- t['Ndotbelow'] = 0x1E46;
- t['Nhookleft'] = 0x019D;
- t['Nineroman'] = 0x2168;
- t['Nj'] = 0x01CB;
- t['Njecyrillic'] = 0x040A;
- t['Nlinebelow'] = 0x1E48;
- t['Nmonospace'] = 0xFF2E;
- t['Nowarmenian'] = 0x0546;
- t['Nsmall'] = 0xF76E;
- t['Ntilde'] = 0x00D1;
- t['Ntildesmall'] = 0xF7F1;
- t['Nu'] = 0x039D;
- t['O'] = 0x004F;
- t['OE'] = 0x0152;
- t['OEsmall'] = 0xF6FA;
- t['Oacute'] = 0x00D3;
- t['Oacutesmall'] = 0xF7F3;
- t['Obarredcyrillic'] = 0x04E8;
- t['Obarreddieresiscyrillic'] = 0x04EA;
- t['Obreve'] = 0x014E;
- t['Ocaron'] = 0x01D1;
- t['Ocenteredtilde'] = 0x019F;
- t['Ocircle'] = 0x24C4;
- t['Ocircumflex'] = 0x00D4;
- t['Ocircumflexacute'] = 0x1ED0;
- t['Ocircumflexdotbelow'] = 0x1ED8;
- t['Ocircumflexgrave'] = 0x1ED2;
- t['Ocircumflexhookabove'] = 0x1ED4;
- t['Ocircumflexsmall'] = 0xF7F4;
- t['Ocircumflextilde'] = 0x1ED6;
- t['Ocyrillic'] = 0x041E;
- t['Odblacute'] = 0x0150;
- t['Odblgrave'] = 0x020C;
- t['Odieresis'] = 0x00D6;
- t['Odieresiscyrillic'] = 0x04E6;
- t['Odieresissmall'] = 0xF7F6;
- t['Odotbelow'] = 0x1ECC;
- t['Ogoneksmall'] = 0xF6FB;
- t['Ograve'] = 0x00D2;
- t['Ogravesmall'] = 0xF7F2;
- t['Oharmenian'] = 0x0555;
- t['Ohm'] = 0x2126;
- t['Ohookabove'] = 0x1ECE;
- t['Ohorn'] = 0x01A0;
- t['Ohornacute'] = 0x1EDA;
- t['Ohorndotbelow'] = 0x1EE2;
- t['Ohorngrave'] = 0x1EDC;
- t['Ohornhookabove'] = 0x1EDE;
- t['Ohorntilde'] = 0x1EE0;
- t['Ohungarumlaut'] = 0x0150;
- t['Oi'] = 0x01A2;
- t['Oinvertedbreve'] = 0x020E;
- t['Omacron'] = 0x014C;
- t['Omacronacute'] = 0x1E52;
- t['Omacrongrave'] = 0x1E50;
- t['Omega'] = 0x2126;
- t['Omegacyrillic'] = 0x0460;
- t['Omegagreek'] = 0x03A9;
- t['Omegaroundcyrillic'] = 0x047A;
- t['Omegatitlocyrillic'] = 0x047C;
- t['Omegatonos'] = 0x038F;
- t['Omicron'] = 0x039F;
- t['Omicrontonos'] = 0x038C;
- t['Omonospace'] = 0xFF2F;
- t['Oneroman'] = 0x2160;
- t['Oogonek'] = 0x01EA;
- t['Oogonekmacron'] = 0x01EC;
- t['Oopen'] = 0x0186;
- t['Oslash'] = 0x00D8;
- t['Oslashacute'] = 0x01FE;
- t['Oslashsmall'] = 0xF7F8;
- t['Osmall'] = 0xF76F;
- t['Ostrokeacute'] = 0x01FE;
- t['Otcyrillic'] = 0x047E;
- t['Otilde'] = 0x00D5;
- t['Otildeacute'] = 0x1E4C;
- t['Otildedieresis'] = 0x1E4E;
- t['Otildesmall'] = 0xF7F5;
- t['P'] = 0x0050;
- t['Pacute'] = 0x1E54;
- t['Pcircle'] = 0x24C5;
- t['Pdotaccent'] = 0x1E56;
- t['Pecyrillic'] = 0x041F;
- t['Peharmenian'] = 0x054A;
- t['Pemiddlehookcyrillic'] = 0x04A6;
- t['Phi'] = 0x03A6;
- t['Phook'] = 0x01A4;
- t['Pi'] = 0x03A0;
- t['Piwrarmenian'] = 0x0553;
- t['Pmonospace'] = 0xFF30;
- t['Psi'] = 0x03A8;
- t['Psicyrillic'] = 0x0470;
- t['Psmall'] = 0xF770;
- t['Q'] = 0x0051;
- t['Qcircle'] = 0x24C6;
- t['Qmonospace'] = 0xFF31;
- t['Qsmall'] = 0xF771;
- t['R'] = 0x0052;
- t['Raarmenian'] = 0x054C;
- t['Racute'] = 0x0154;
- t['Rcaron'] = 0x0158;
- t['Rcedilla'] = 0x0156;
- t['Rcircle'] = 0x24C7;
- t['Rcommaaccent'] = 0x0156;
- t['Rdblgrave'] = 0x0210;
- t['Rdotaccent'] = 0x1E58;
- t['Rdotbelow'] = 0x1E5A;
- t['Rdotbelowmacron'] = 0x1E5C;
- t['Reharmenian'] = 0x0550;
- t['Rfraktur'] = 0x211C;
- t['Rho'] = 0x03A1;
- t['Ringsmall'] = 0xF6FC;
- t['Rinvertedbreve'] = 0x0212;
- t['Rlinebelow'] = 0x1E5E;
- t['Rmonospace'] = 0xFF32;
- t['Rsmall'] = 0xF772;
- t['Rsmallinverted'] = 0x0281;
- t['Rsmallinvertedsuperior'] = 0x02B6;
- t['S'] = 0x0053;
- t['SF010000'] = 0x250C;
- t['SF020000'] = 0x2514;
- t['SF030000'] = 0x2510;
- t['SF040000'] = 0x2518;
- t['SF050000'] = 0x253C;
- t['SF060000'] = 0x252C;
- t['SF070000'] = 0x2534;
- t['SF080000'] = 0x251C;
- t['SF090000'] = 0x2524;
- t['SF100000'] = 0x2500;
- t['SF110000'] = 0x2502;
- t['SF190000'] = 0x2561;
- t['SF200000'] = 0x2562;
- t['SF210000'] = 0x2556;
- t['SF220000'] = 0x2555;
- t['SF230000'] = 0x2563;
- t['SF240000'] = 0x2551;
- t['SF250000'] = 0x2557;
- t['SF260000'] = 0x255D;
- t['SF270000'] = 0x255C;
- t['SF280000'] = 0x255B;
- t['SF360000'] = 0x255E;
- t['SF370000'] = 0x255F;
- t['SF380000'] = 0x255A;
- t['SF390000'] = 0x2554;
- t['SF400000'] = 0x2569;
- t['SF410000'] = 0x2566;
- t['SF420000'] = 0x2560;
- t['SF430000'] = 0x2550;
- t['SF440000'] = 0x256C;
- t['SF450000'] = 0x2567;
- t['SF460000'] = 0x2568;
- t['SF470000'] = 0x2564;
- t['SF480000'] = 0x2565;
- t['SF490000'] = 0x2559;
- t['SF500000'] = 0x2558;
- t['SF510000'] = 0x2552;
- t['SF520000'] = 0x2553;
- t['SF530000'] = 0x256B;
- t['SF540000'] = 0x256A;
- t['Sacute'] = 0x015A;
- t['Sacutedotaccent'] = 0x1E64;
- t['Sampigreek'] = 0x03E0;
- t['Scaron'] = 0x0160;
- t['Scarondotaccent'] = 0x1E66;
- t['Scaronsmall'] = 0xF6FD;
- t['Scedilla'] = 0x015E;
- t['Schwa'] = 0x018F;
- t['Schwacyrillic'] = 0x04D8;
- t['Schwadieresiscyrillic'] = 0x04DA;
- t['Scircle'] = 0x24C8;
- t['Scircumflex'] = 0x015C;
- t['Scommaaccent'] = 0x0218;
- t['Sdotaccent'] = 0x1E60;
- t['Sdotbelow'] = 0x1E62;
- t['Sdotbelowdotaccent'] = 0x1E68;
- t['Seharmenian'] = 0x054D;
- t['Sevenroman'] = 0x2166;
- t['Shaarmenian'] = 0x0547;
- t['Shacyrillic'] = 0x0428;
- t['Shchacyrillic'] = 0x0429;
- t['Sheicoptic'] = 0x03E2;
- t['Shhacyrillic'] = 0x04BA;
- t['Shimacoptic'] = 0x03EC;
- t['Sigma'] = 0x03A3;
- t['Sixroman'] = 0x2165;
- t['Smonospace'] = 0xFF33;
- t['Softsigncyrillic'] = 0x042C;
- t['Ssmall'] = 0xF773;
- t['Stigmagreek'] = 0x03DA;
- t['T'] = 0x0054;
- t['Tau'] = 0x03A4;
- t['Tbar'] = 0x0166;
- t['Tcaron'] = 0x0164;
- t['Tcedilla'] = 0x0162;
- t['Tcircle'] = 0x24C9;
- t['Tcircumflexbelow'] = 0x1E70;
- t['Tcommaaccent'] = 0x0162;
- t['Tdotaccent'] = 0x1E6A;
- t['Tdotbelow'] = 0x1E6C;
- t['Tecyrillic'] = 0x0422;
- t['Tedescendercyrillic'] = 0x04AC;
- t['Tenroman'] = 0x2169;
- t['Tetsecyrillic'] = 0x04B4;
- t['Theta'] = 0x0398;
- t['Thook'] = 0x01AC;
- t['Thorn'] = 0x00DE;
- t['Thornsmall'] = 0xF7FE;
- t['Threeroman'] = 0x2162;
- t['Tildesmall'] = 0xF6FE;
- t['Tiwnarmenian'] = 0x054F;
- t['Tlinebelow'] = 0x1E6E;
- t['Tmonospace'] = 0xFF34;
- t['Toarmenian'] = 0x0539;
- t['Tonefive'] = 0x01BC;
- t['Tonesix'] = 0x0184;
- t['Tonetwo'] = 0x01A7;
- t['Tretroflexhook'] = 0x01AE;
- t['Tsecyrillic'] = 0x0426;
- t['Tshecyrillic'] = 0x040B;
- t['Tsmall'] = 0xF774;
- t['Twelveroman'] = 0x216B;
- t['Tworoman'] = 0x2161;
- t['U'] = 0x0055;
- t['Uacute'] = 0x00DA;
- t['Uacutesmall'] = 0xF7FA;
- t['Ubreve'] = 0x016C;
- t['Ucaron'] = 0x01D3;
- t['Ucircle'] = 0x24CA;
- t['Ucircumflex'] = 0x00DB;
- t['Ucircumflexbelow'] = 0x1E76;
- t['Ucircumflexsmall'] = 0xF7FB;
- t['Ucyrillic'] = 0x0423;
- t['Udblacute'] = 0x0170;
- t['Udblgrave'] = 0x0214;
- t['Udieresis'] = 0x00DC;
- t['Udieresisacute'] = 0x01D7;
- t['Udieresisbelow'] = 0x1E72;
- t['Udieresiscaron'] = 0x01D9;
- t['Udieresiscyrillic'] = 0x04F0;
- t['Udieresisgrave'] = 0x01DB;
- t['Udieresismacron'] = 0x01D5;
- t['Udieresissmall'] = 0xF7FC;
- t['Udotbelow'] = 0x1EE4;
- t['Ugrave'] = 0x00D9;
- t['Ugravesmall'] = 0xF7F9;
- t['Uhookabove'] = 0x1EE6;
- t['Uhorn'] = 0x01AF;
- t['Uhornacute'] = 0x1EE8;
- t['Uhorndotbelow'] = 0x1EF0;
- t['Uhorngrave'] = 0x1EEA;
- t['Uhornhookabove'] = 0x1EEC;
- t['Uhorntilde'] = 0x1EEE;
- t['Uhungarumlaut'] = 0x0170;
- t['Uhungarumlautcyrillic'] = 0x04F2;
- t['Uinvertedbreve'] = 0x0216;
- t['Ukcyrillic'] = 0x0478;
- t['Umacron'] = 0x016A;
- t['Umacroncyrillic'] = 0x04EE;
- t['Umacrondieresis'] = 0x1E7A;
- t['Umonospace'] = 0xFF35;
- t['Uogonek'] = 0x0172;
- t['Upsilon'] = 0x03A5;
- t['Upsilon1'] = 0x03D2;
- t['Upsilonacutehooksymbolgreek'] = 0x03D3;
- t['Upsilonafrican'] = 0x01B1;
- t['Upsilondieresis'] = 0x03AB;
- t['Upsilondieresishooksymbolgreek'] = 0x03D4;
- t['Upsilonhooksymbol'] = 0x03D2;
- t['Upsilontonos'] = 0x038E;
- t['Uring'] = 0x016E;
- t['Ushortcyrillic'] = 0x040E;
- t['Usmall'] = 0xF775;
- t['Ustraightcyrillic'] = 0x04AE;
- t['Ustraightstrokecyrillic'] = 0x04B0;
- t['Utilde'] = 0x0168;
- t['Utildeacute'] = 0x1E78;
- t['Utildebelow'] = 0x1E74;
- t['V'] = 0x0056;
- t['Vcircle'] = 0x24CB;
- t['Vdotbelow'] = 0x1E7E;
- t['Vecyrillic'] = 0x0412;
- t['Vewarmenian'] = 0x054E;
- t['Vhook'] = 0x01B2;
- t['Vmonospace'] = 0xFF36;
- t['Voarmenian'] = 0x0548;
- t['Vsmall'] = 0xF776;
- t['Vtilde'] = 0x1E7C;
- t['W'] = 0x0057;
- t['Wacute'] = 0x1E82;
- t['Wcircle'] = 0x24CC;
- t['Wcircumflex'] = 0x0174;
- t['Wdieresis'] = 0x1E84;
- t['Wdotaccent'] = 0x1E86;
- t['Wdotbelow'] = 0x1E88;
- t['Wgrave'] = 0x1E80;
- t['Wmonospace'] = 0xFF37;
- t['Wsmall'] = 0xF777;
- t['X'] = 0x0058;
- t['Xcircle'] = 0x24CD;
- t['Xdieresis'] = 0x1E8C;
- t['Xdotaccent'] = 0x1E8A;
- t['Xeharmenian'] = 0x053D;
- t['Xi'] = 0x039E;
- t['Xmonospace'] = 0xFF38;
- t['Xsmall'] = 0xF778;
- t['Y'] = 0x0059;
- t['Yacute'] = 0x00DD;
- t['Yacutesmall'] = 0xF7FD;
- t['Yatcyrillic'] = 0x0462;
- t['Ycircle'] = 0x24CE;
- t['Ycircumflex'] = 0x0176;
- t['Ydieresis'] = 0x0178;
- t['Ydieresissmall'] = 0xF7FF;
- t['Ydotaccent'] = 0x1E8E;
- t['Ydotbelow'] = 0x1EF4;
- t['Yericyrillic'] = 0x042B;
- t['Yerudieresiscyrillic'] = 0x04F8;
- t['Ygrave'] = 0x1EF2;
- t['Yhook'] = 0x01B3;
- t['Yhookabove'] = 0x1EF6;
- t['Yiarmenian'] = 0x0545;
- t['Yicyrillic'] = 0x0407;
- t['Yiwnarmenian'] = 0x0552;
- t['Ymonospace'] = 0xFF39;
- t['Ysmall'] = 0xF779;
- t['Ytilde'] = 0x1EF8;
- t['Yusbigcyrillic'] = 0x046A;
- t['Yusbigiotifiedcyrillic'] = 0x046C;
- t['Yuslittlecyrillic'] = 0x0466;
- t['Yuslittleiotifiedcyrillic'] = 0x0468;
- t['Z'] = 0x005A;
- t['Zaarmenian'] = 0x0536;
- t['Zacute'] = 0x0179;
- t['Zcaron'] = 0x017D;
- t['Zcaronsmall'] = 0xF6FF;
- t['Zcircle'] = 0x24CF;
- t['Zcircumflex'] = 0x1E90;
- t['Zdot'] = 0x017B;
- t['Zdotaccent'] = 0x017B;
- t['Zdotbelow'] = 0x1E92;
- t['Zecyrillic'] = 0x0417;
- t['Zedescendercyrillic'] = 0x0498;
- t['Zedieresiscyrillic'] = 0x04DE;
- t['Zeta'] = 0x0396;
- t['Zhearmenian'] = 0x053A;
- t['Zhebrevecyrillic'] = 0x04C1;
- t['Zhecyrillic'] = 0x0416;
- t['Zhedescendercyrillic'] = 0x0496;
- t['Zhedieresiscyrillic'] = 0x04DC;
- t['Zlinebelow'] = 0x1E94;
- t['Zmonospace'] = 0xFF3A;
- t['Zsmall'] = 0xF77A;
- t['Zstroke'] = 0x01B5;
- t['a'] = 0x0061;
- t['aabengali'] = 0x0986;
- t['aacute'] = 0x00E1;
- t['aadeva'] = 0x0906;
- t['aagujarati'] = 0x0A86;
- t['aagurmukhi'] = 0x0A06;
- t['aamatragurmukhi'] = 0x0A3E;
- t['aarusquare'] = 0x3303;
- t['aavowelsignbengali'] = 0x09BE;
- t['aavowelsigndeva'] = 0x093E;
- t['aavowelsigngujarati'] = 0x0ABE;
- t['abbreviationmarkarmenian'] = 0x055F;
- t['abbreviationsigndeva'] = 0x0970;
- t['abengali'] = 0x0985;
- t['abopomofo'] = 0x311A;
- t['abreve'] = 0x0103;
- t['abreveacute'] = 0x1EAF;
- t['abrevecyrillic'] = 0x04D1;
- t['abrevedotbelow'] = 0x1EB7;
- t['abrevegrave'] = 0x1EB1;
- t['abrevehookabove'] = 0x1EB3;
- t['abrevetilde'] = 0x1EB5;
- t['acaron'] = 0x01CE;
- t['acircle'] = 0x24D0;
- t['acircumflex'] = 0x00E2;
- t['acircumflexacute'] = 0x1EA5;
- t['acircumflexdotbelow'] = 0x1EAD;
- t['acircumflexgrave'] = 0x1EA7;
- t['acircumflexhookabove'] = 0x1EA9;
- t['acircumflextilde'] = 0x1EAB;
- t['acute'] = 0x00B4;
- t['acutebelowcmb'] = 0x0317;
- t['acutecmb'] = 0x0301;
- t['acutecomb'] = 0x0301;
- t['acutedeva'] = 0x0954;
- t['acutelowmod'] = 0x02CF;
- t['acutetonecmb'] = 0x0341;
- t['acyrillic'] = 0x0430;
- t['adblgrave'] = 0x0201;
- t['addakgurmukhi'] = 0x0A71;
- t['adeva'] = 0x0905;
- t['adieresis'] = 0x00E4;
- t['adieresiscyrillic'] = 0x04D3;
- t['adieresismacron'] = 0x01DF;
- t['adotbelow'] = 0x1EA1;
- t['adotmacron'] = 0x01E1;
- t['ae'] = 0x00E6;
- t['aeacute'] = 0x01FD;
- t['aekorean'] = 0x3150;
- t['aemacron'] = 0x01E3;
- t['afii00208'] = 0x2015;
- t['afii08941'] = 0x20A4;
- t['afii10017'] = 0x0410;
- t['afii10018'] = 0x0411;
- t['afii10019'] = 0x0412;
- t['afii10020'] = 0x0413;
- t['afii10021'] = 0x0414;
- t['afii10022'] = 0x0415;
- t['afii10023'] = 0x0401;
- t['afii10024'] = 0x0416;
- t['afii10025'] = 0x0417;
- t['afii10026'] = 0x0418;
- t['afii10027'] = 0x0419;
- t['afii10028'] = 0x041A;
- t['afii10029'] = 0x041B;
- t['afii10030'] = 0x041C;
- t['afii10031'] = 0x041D;
- t['afii10032'] = 0x041E;
- t['afii10033'] = 0x041F;
- t['afii10034'] = 0x0420;
- t['afii10035'] = 0x0421;
- t['afii10036'] = 0x0422;
- t['afii10037'] = 0x0423;
- t['afii10038'] = 0x0424;
- t['afii10039'] = 0x0425;
- t['afii10040'] = 0x0426;
- t['afii10041'] = 0x0427;
- t['afii10042'] = 0x0428;
- t['afii10043'] = 0x0429;
- t['afii10044'] = 0x042A;
- t['afii10045'] = 0x042B;
- t['afii10046'] = 0x042C;
- t['afii10047'] = 0x042D;
- t['afii10048'] = 0x042E;
- t['afii10049'] = 0x042F;
- t['afii10050'] = 0x0490;
- t['afii10051'] = 0x0402;
- t['afii10052'] = 0x0403;
- t['afii10053'] = 0x0404;
- t['afii10054'] = 0x0405;
- t['afii10055'] = 0x0406;
- t['afii10056'] = 0x0407;
- t['afii10057'] = 0x0408;
- t['afii10058'] = 0x0409;
- t['afii10059'] = 0x040A;
- t['afii10060'] = 0x040B;
- t['afii10061'] = 0x040C;
- t['afii10062'] = 0x040E;
- t['afii10063'] = 0xF6C4;
- t['afii10064'] = 0xF6C5;
- t['afii10065'] = 0x0430;
- t['afii10066'] = 0x0431;
- t['afii10067'] = 0x0432;
- t['afii10068'] = 0x0433;
- t['afii10069'] = 0x0434;
- t['afii10070'] = 0x0435;
- t['afii10071'] = 0x0451;
- t['afii10072'] = 0x0436;
- t['afii10073'] = 0x0437;
- t['afii10074'] = 0x0438;
- t['afii10075'] = 0x0439;
- t['afii10076'] = 0x043A;
- t['afii10077'] = 0x043B;
- t['afii10078'] = 0x043C;
- t['afii10079'] = 0x043D;
- t['afii10080'] = 0x043E;
- t['afii10081'] = 0x043F;
- t['afii10082'] = 0x0440;
- t['afii10083'] = 0x0441;
- t['afii10084'] = 0x0442;
- t['afii10085'] = 0x0443;
- t['afii10086'] = 0x0444;
- t['afii10087'] = 0x0445;
- t['afii10088'] = 0x0446;
- t['afii10089'] = 0x0447;
- t['afii10090'] = 0x0448;
- t['afii10091'] = 0x0449;
- t['afii10092'] = 0x044A;
- t['afii10093'] = 0x044B;
- t['afii10094'] = 0x044C;
- t['afii10095'] = 0x044D;
- t['afii10096'] = 0x044E;
- t['afii10097'] = 0x044F;
- t['afii10098'] = 0x0491;
- t['afii10099'] = 0x0452;
- t['afii10100'] = 0x0453;
- t['afii10101'] = 0x0454;
- t['afii10102'] = 0x0455;
- t['afii10103'] = 0x0456;
- t['afii10104'] = 0x0457;
- t['afii10105'] = 0x0458;
- t['afii10106'] = 0x0459;
- t['afii10107'] = 0x045A;
- t['afii10108'] = 0x045B;
- t['afii10109'] = 0x045C;
- t['afii10110'] = 0x045E;
- t['afii10145'] = 0x040F;
- t['afii10146'] = 0x0462;
- t['afii10147'] = 0x0472;
- t['afii10148'] = 0x0474;
- t['afii10192'] = 0xF6C6;
- t['afii10193'] = 0x045F;
- t['afii10194'] = 0x0463;
- t['afii10195'] = 0x0473;
- t['afii10196'] = 0x0475;
- t['afii10831'] = 0xF6C7;
- t['afii10832'] = 0xF6C8;
- t['afii10846'] = 0x04D9;
- t['afii299'] = 0x200E;
- t['afii300'] = 0x200F;
- t['afii301'] = 0x200D;
- t['afii57381'] = 0x066A;
- t['afii57388'] = 0x060C;
- t['afii57392'] = 0x0660;
- t['afii57393'] = 0x0661;
- t['afii57394'] = 0x0662;
- t['afii57395'] = 0x0663;
- t['afii57396'] = 0x0664;
- t['afii57397'] = 0x0665;
- t['afii57398'] = 0x0666;
- t['afii57399'] = 0x0667;
- t['afii57400'] = 0x0668;
- t['afii57401'] = 0x0669;
- t['afii57403'] = 0x061B;
- t['afii57407'] = 0x061F;
- t['afii57409'] = 0x0621;
- t['afii57410'] = 0x0622;
- t['afii57411'] = 0x0623;
- t['afii57412'] = 0x0624;
- t['afii57413'] = 0x0625;
- t['afii57414'] = 0x0626;
- t['afii57415'] = 0x0627;
- t['afii57416'] = 0x0628;
- t['afii57417'] = 0x0629;
- t['afii57418'] = 0x062A;
- t['afii57419'] = 0x062B;
- t['afii57420'] = 0x062C;
- t['afii57421'] = 0x062D;
- t['afii57422'] = 0x062E;
- t['afii57423'] = 0x062F;
- t['afii57424'] = 0x0630;
- t['afii57425'] = 0x0631;
- t['afii57426'] = 0x0632;
- t['afii57427'] = 0x0633;
- t['afii57428'] = 0x0634;
- t['afii57429'] = 0x0635;
- t['afii57430'] = 0x0636;
- t['afii57431'] = 0x0637;
- t['afii57432'] = 0x0638;
- t['afii57433'] = 0x0639;
- t['afii57434'] = 0x063A;
- t['afii57440'] = 0x0640;
- t['afii57441'] = 0x0641;
- t['afii57442'] = 0x0642;
- t['afii57443'] = 0x0643;
- t['afii57444'] = 0x0644;
- t['afii57445'] = 0x0645;
- t['afii57446'] = 0x0646;
- t['afii57448'] = 0x0648;
- t['afii57449'] = 0x0649;
- t['afii57450'] = 0x064A;
- t['afii57451'] = 0x064B;
- t['afii57452'] = 0x064C;
- t['afii57453'] = 0x064D;
- t['afii57454'] = 0x064E;
- t['afii57455'] = 0x064F;
- t['afii57456'] = 0x0650;
- t['afii57457'] = 0x0651;
- t['afii57458'] = 0x0652;
- t['afii57470'] = 0x0647;
- t['afii57505'] = 0x06A4;
- t['afii57506'] = 0x067E;
- t['afii57507'] = 0x0686;
- t['afii57508'] = 0x0698;
- t['afii57509'] = 0x06AF;
- t['afii57511'] = 0x0679;
- t['afii57512'] = 0x0688;
- t['afii57513'] = 0x0691;
- t['afii57514'] = 0x06BA;
- t['afii57519'] = 0x06D2;
- t['afii57534'] = 0x06D5;
- t['afii57636'] = 0x20AA;
- t['afii57645'] = 0x05BE;
- t['afii57658'] = 0x05C3;
- t['afii57664'] = 0x05D0;
- t['afii57665'] = 0x05D1;
- t['afii57666'] = 0x05D2;
- t['afii57667'] = 0x05D3;
- t['afii57668'] = 0x05D4;
- t['afii57669'] = 0x05D5;
- t['afii57670'] = 0x05D6;
- t['afii57671'] = 0x05D7;
- t['afii57672'] = 0x05D8;
- t['afii57673'] = 0x05D9;
- t['afii57674'] = 0x05DA;
- t['afii57675'] = 0x05DB;
- t['afii57676'] = 0x05DC;
- t['afii57677'] = 0x05DD;
- t['afii57678'] = 0x05DE;
- t['afii57679'] = 0x05DF;
- t['afii57680'] = 0x05E0;
- t['afii57681'] = 0x05E1;
- t['afii57682'] = 0x05E2;
- t['afii57683'] = 0x05E3;
- t['afii57684'] = 0x05E4;
- t['afii57685'] = 0x05E5;
- t['afii57686'] = 0x05E6;
- t['afii57687'] = 0x05E7;
- t['afii57688'] = 0x05E8;
- t['afii57689'] = 0x05E9;
- t['afii57690'] = 0x05EA;
- t['afii57694'] = 0xFB2A;
- t['afii57695'] = 0xFB2B;
- t['afii57700'] = 0xFB4B;
- t['afii57705'] = 0xFB1F;
- t['afii57716'] = 0x05F0;
- t['afii57717'] = 0x05F1;
- t['afii57718'] = 0x05F2;
- t['afii57723'] = 0xFB35;
- t['afii57793'] = 0x05B4;
- t['afii57794'] = 0x05B5;
- t['afii57795'] = 0x05B6;
- t['afii57796'] = 0x05BB;
- t['afii57797'] = 0x05B8;
- t['afii57798'] = 0x05B7;
- t['afii57799'] = 0x05B0;
- t['afii57800'] = 0x05B2;
- t['afii57801'] = 0x05B1;
- t['afii57802'] = 0x05B3;
- t['afii57803'] = 0x05C2;
- t['afii57804'] = 0x05C1;
- t['afii57806'] = 0x05B9;
- t['afii57807'] = 0x05BC;
- t['afii57839'] = 0x05BD;
- t['afii57841'] = 0x05BF;
- t['afii57842'] = 0x05C0;
- t['afii57929'] = 0x02BC;
- t['afii61248'] = 0x2105;
- t['afii61289'] = 0x2113;
- t['afii61352'] = 0x2116;
- t['afii61573'] = 0x202C;
- t['afii61574'] = 0x202D;
- t['afii61575'] = 0x202E;
- t['afii61664'] = 0x200C;
- t['afii63167'] = 0x066D;
- t['afii64937'] = 0x02BD;
- t['agrave'] = 0x00E0;
- t['agujarati'] = 0x0A85;
- t['agurmukhi'] = 0x0A05;
- t['ahiragana'] = 0x3042;
- t['ahookabove'] = 0x1EA3;
- t['aibengali'] = 0x0990;
- t['aibopomofo'] = 0x311E;
- t['aideva'] = 0x0910;
- t['aiecyrillic'] = 0x04D5;
- t['aigujarati'] = 0x0A90;
- t['aigurmukhi'] = 0x0A10;
- t['aimatragurmukhi'] = 0x0A48;
- t['ainarabic'] = 0x0639;
- t['ainfinalarabic'] = 0xFECA;
- t['aininitialarabic'] = 0xFECB;
- t['ainmedialarabic'] = 0xFECC;
- t['ainvertedbreve'] = 0x0203;
- t['aivowelsignbengali'] = 0x09C8;
- t['aivowelsigndeva'] = 0x0948;
- t['aivowelsigngujarati'] = 0x0AC8;
- t['akatakana'] = 0x30A2;
- t['akatakanahalfwidth'] = 0xFF71;
- t['akorean'] = 0x314F;
- t['alef'] = 0x05D0;
- t['alefarabic'] = 0x0627;
- t['alefdageshhebrew'] = 0xFB30;
- t['aleffinalarabic'] = 0xFE8E;
- t['alefhamzaabovearabic'] = 0x0623;
- t['alefhamzaabovefinalarabic'] = 0xFE84;
- t['alefhamzabelowarabic'] = 0x0625;
- t['alefhamzabelowfinalarabic'] = 0xFE88;
- t['alefhebrew'] = 0x05D0;
- t['aleflamedhebrew'] = 0xFB4F;
- t['alefmaddaabovearabic'] = 0x0622;
- t['alefmaddaabovefinalarabic'] = 0xFE82;
- t['alefmaksuraarabic'] = 0x0649;
- t['alefmaksurafinalarabic'] = 0xFEF0;
- t['alefmaksurainitialarabic'] = 0xFEF3;
- t['alefmaksuramedialarabic'] = 0xFEF4;
- t['alefpatahhebrew'] = 0xFB2E;
- t['alefqamatshebrew'] = 0xFB2F;
- t['aleph'] = 0x2135;
- t['allequal'] = 0x224C;
- t['alpha'] = 0x03B1;
- t['alphatonos'] = 0x03AC;
- t['amacron'] = 0x0101;
- t['amonospace'] = 0xFF41;
- t['ampersand'] = 0x0026;
- t['ampersandmonospace'] = 0xFF06;
- t['ampersandsmall'] = 0xF726;
- t['amsquare'] = 0x33C2;
- t['anbopomofo'] = 0x3122;
- t['angbopomofo'] = 0x3124;
- t['angbracketleft'] = 0x3008;
- t['angbracketright'] = 0x3009;
- t['angkhankhuthai'] = 0x0E5A;
- t['angle'] = 0x2220;
- t['anglebracketleft'] = 0x3008;
- t['anglebracketleftvertical'] = 0xFE3F;
- t['anglebracketright'] = 0x3009;
- t['anglebracketrightvertical'] = 0xFE40;
- t['angleleft'] = 0x2329;
- t['angleright'] = 0x232A;
- t['angstrom'] = 0x212B;
- t['anoteleia'] = 0x0387;
- t['anudattadeva'] = 0x0952;
- t['anusvarabengali'] = 0x0982;
- t['anusvaradeva'] = 0x0902;
- t['anusvaragujarati'] = 0x0A82;
- t['aogonek'] = 0x0105;
- t['apaatosquare'] = 0x3300;
- t['aparen'] = 0x249C;
- t['apostrophearmenian'] = 0x055A;
- t['apostrophemod'] = 0x02BC;
- t['apple'] = 0xF8FF;
- t['approaches'] = 0x2250;
- t['approxequal'] = 0x2248;
- t['approxequalorimage'] = 0x2252;
- t['approximatelyequal'] = 0x2245;
- t['araeaekorean'] = 0x318E;
- t['araeakorean'] = 0x318D;
- t['arc'] = 0x2312;
- t['arighthalfring'] = 0x1E9A;
- t['aring'] = 0x00E5;
- t['aringacute'] = 0x01FB;
- t['aringbelow'] = 0x1E01;
- t['arrowboth'] = 0x2194;
- t['arrowdashdown'] = 0x21E3;
- t['arrowdashleft'] = 0x21E0;
- t['arrowdashright'] = 0x21E2;
- t['arrowdashup'] = 0x21E1;
- t['arrowdblboth'] = 0x21D4;
- t['arrowdbldown'] = 0x21D3;
- t['arrowdblleft'] = 0x21D0;
- t['arrowdblright'] = 0x21D2;
- t['arrowdblup'] = 0x21D1;
- t['arrowdown'] = 0x2193;
- t['arrowdownleft'] = 0x2199;
- t['arrowdownright'] = 0x2198;
- t['arrowdownwhite'] = 0x21E9;
- t['arrowheaddownmod'] = 0x02C5;
- t['arrowheadleftmod'] = 0x02C2;
- t['arrowheadrightmod'] = 0x02C3;
- t['arrowheadupmod'] = 0x02C4;
- t['arrowhorizex'] = 0xF8E7;
- t['arrowleft'] = 0x2190;
- t['arrowleftdbl'] = 0x21D0;
- t['arrowleftdblstroke'] = 0x21CD;
- t['arrowleftoverright'] = 0x21C6;
- t['arrowleftwhite'] = 0x21E6;
- t['arrowright'] = 0x2192;
- t['arrowrightdblstroke'] = 0x21CF;
- t['arrowrightheavy'] = 0x279E;
- t['arrowrightoverleft'] = 0x21C4;
- t['arrowrightwhite'] = 0x21E8;
- t['arrowtableft'] = 0x21E4;
- t['arrowtabright'] = 0x21E5;
- t['arrowup'] = 0x2191;
- t['arrowupdn'] = 0x2195;
- t['arrowupdnbse'] = 0x21A8;
- t['arrowupdownbase'] = 0x21A8;
- t['arrowupleft'] = 0x2196;
- t['arrowupleftofdown'] = 0x21C5;
- t['arrowupright'] = 0x2197;
- t['arrowupwhite'] = 0x21E7;
- t['arrowvertex'] = 0xF8E6;
- t['asciicircum'] = 0x005E;
- t['asciicircummonospace'] = 0xFF3E;
- t['asciitilde'] = 0x007E;
- t['asciitildemonospace'] = 0xFF5E;
- t['ascript'] = 0x0251;
- t['ascriptturned'] = 0x0252;
- t['asmallhiragana'] = 0x3041;
- t['asmallkatakana'] = 0x30A1;
- t['asmallkatakanahalfwidth'] = 0xFF67;
- t['asterisk'] = 0x002A;
- t['asteriskaltonearabic'] = 0x066D;
- t['asteriskarabic'] = 0x066D;
- t['asteriskmath'] = 0x2217;
- t['asteriskmonospace'] = 0xFF0A;
- t['asterisksmall'] = 0xFE61;
- t['asterism'] = 0x2042;
- t['asuperior'] = 0xF6E9;
- t['asymptoticallyequal'] = 0x2243;
- t['at'] = 0x0040;
- t['atilde'] = 0x00E3;
- t['atmonospace'] = 0xFF20;
- t['atsmall'] = 0xFE6B;
- t['aturned'] = 0x0250;
- t['aubengali'] = 0x0994;
- t['aubopomofo'] = 0x3120;
- t['audeva'] = 0x0914;
- t['augujarati'] = 0x0A94;
- t['augurmukhi'] = 0x0A14;
- t['aulengthmarkbengali'] = 0x09D7;
- t['aumatragurmukhi'] = 0x0A4C;
- t['auvowelsignbengali'] = 0x09CC;
- t['auvowelsigndeva'] = 0x094C;
- t['auvowelsigngujarati'] = 0x0ACC;
- t['avagrahadeva'] = 0x093D;
- t['aybarmenian'] = 0x0561;
- t['ayin'] = 0x05E2;
- t['ayinaltonehebrew'] = 0xFB20;
- t['ayinhebrew'] = 0x05E2;
- t['b'] = 0x0062;
- t['babengali'] = 0x09AC;
- t['backslash'] = 0x005C;
- t['backslashmonospace'] = 0xFF3C;
- t['badeva'] = 0x092C;
- t['bagujarati'] = 0x0AAC;
- t['bagurmukhi'] = 0x0A2C;
- t['bahiragana'] = 0x3070;
- t['bahtthai'] = 0x0E3F;
- t['bakatakana'] = 0x30D0;
- t['bar'] = 0x007C;
- t['barmonospace'] = 0xFF5C;
- t['bbopomofo'] = 0x3105;
- t['bcircle'] = 0x24D1;
- t['bdotaccent'] = 0x1E03;
- t['bdotbelow'] = 0x1E05;
- t['beamedsixteenthnotes'] = 0x266C;
- t['because'] = 0x2235;
- t['becyrillic'] = 0x0431;
- t['beharabic'] = 0x0628;
- t['behfinalarabic'] = 0xFE90;
- t['behinitialarabic'] = 0xFE91;
- t['behiragana'] = 0x3079;
- t['behmedialarabic'] = 0xFE92;
- t['behmeeminitialarabic'] = 0xFC9F;
- t['behmeemisolatedarabic'] = 0xFC08;
- t['behnoonfinalarabic'] = 0xFC6D;
- t['bekatakana'] = 0x30D9;
- t['benarmenian'] = 0x0562;
- t['bet'] = 0x05D1;
- t['beta'] = 0x03B2;
- t['betasymbolgreek'] = 0x03D0;
- t['betdagesh'] = 0xFB31;
- t['betdageshhebrew'] = 0xFB31;
- t['bethebrew'] = 0x05D1;
- t['betrafehebrew'] = 0xFB4C;
- t['bhabengali'] = 0x09AD;
- t['bhadeva'] = 0x092D;
- t['bhagujarati'] = 0x0AAD;
- t['bhagurmukhi'] = 0x0A2D;
- t['bhook'] = 0x0253;
- t['bihiragana'] = 0x3073;
- t['bikatakana'] = 0x30D3;
- t['bilabialclick'] = 0x0298;
- t['bindigurmukhi'] = 0x0A02;
- t['birusquare'] = 0x3331;
- t['blackcircle'] = 0x25CF;
- t['blackdiamond'] = 0x25C6;
- t['blackdownpointingtriangle'] = 0x25BC;
- t['blackleftpointingpointer'] = 0x25C4;
- t['blackleftpointingtriangle'] = 0x25C0;
- t['blacklenticularbracketleft'] = 0x3010;
- t['blacklenticularbracketleftvertical'] = 0xFE3B;
- t['blacklenticularbracketright'] = 0x3011;
- t['blacklenticularbracketrightvertical'] = 0xFE3C;
- t['blacklowerlefttriangle'] = 0x25E3;
- t['blacklowerrighttriangle'] = 0x25E2;
- t['blackrectangle'] = 0x25AC;
- t['blackrightpointingpointer'] = 0x25BA;
- t['blackrightpointingtriangle'] = 0x25B6;
- t['blacksmallsquare'] = 0x25AA;
- t['blacksmilingface'] = 0x263B;
- t['blacksquare'] = 0x25A0;
- t['blackstar'] = 0x2605;
- t['blackupperlefttriangle'] = 0x25E4;
- t['blackupperrighttriangle'] = 0x25E5;
- t['blackuppointingsmalltriangle'] = 0x25B4;
- t['blackuppointingtriangle'] = 0x25B2;
- t['blank'] = 0x2423;
- t['blinebelow'] = 0x1E07;
- t['block'] = 0x2588;
- t['bmonospace'] = 0xFF42;
- t['bobaimaithai'] = 0x0E1A;
- t['bohiragana'] = 0x307C;
- t['bokatakana'] = 0x30DC;
- t['bparen'] = 0x249D;
- t['bqsquare'] = 0x33C3;
- t['braceex'] = 0xF8F4;
- t['braceleft'] = 0x007B;
- t['braceleftbt'] = 0xF8F3;
- t['braceleftmid'] = 0xF8F2;
- t['braceleftmonospace'] = 0xFF5B;
- t['braceleftsmall'] = 0xFE5B;
- t['bracelefttp'] = 0xF8F1;
- t['braceleftvertical'] = 0xFE37;
- t['braceright'] = 0x007D;
- t['bracerightbt'] = 0xF8FE;
- t['bracerightmid'] = 0xF8FD;
- t['bracerightmonospace'] = 0xFF5D;
- t['bracerightsmall'] = 0xFE5C;
- t['bracerighttp'] = 0xF8FC;
- t['bracerightvertical'] = 0xFE38;
- t['bracketleft'] = 0x005B;
- t['bracketleftbt'] = 0xF8F0;
- t['bracketleftex'] = 0xF8EF;
- t['bracketleftmonospace'] = 0xFF3B;
- t['bracketlefttp'] = 0xF8EE;
- t['bracketright'] = 0x005D;
- t['bracketrightbt'] = 0xF8FB;
- t['bracketrightex'] = 0xF8FA;
- t['bracketrightmonospace'] = 0xFF3D;
- t['bracketrighttp'] = 0xF8F9;
- t['breve'] = 0x02D8;
- t['brevebelowcmb'] = 0x032E;
- t['brevecmb'] = 0x0306;
- t['breveinvertedbelowcmb'] = 0x032F;
- t['breveinvertedcmb'] = 0x0311;
- t['breveinverteddoublecmb'] = 0x0361;
- t['bridgebelowcmb'] = 0x032A;
- t['bridgeinvertedbelowcmb'] = 0x033A;
- t['brokenbar'] = 0x00A6;
- t['bstroke'] = 0x0180;
- t['bsuperior'] = 0xF6EA;
- t['btopbar'] = 0x0183;
- t['buhiragana'] = 0x3076;
- t['bukatakana'] = 0x30D6;
- t['bullet'] = 0x2022;
- t['bulletinverse'] = 0x25D8;
- t['bulletoperator'] = 0x2219;
- t['bullseye'] = 0x25CE;
- t['c'] = 0x0063;
- t['caarmenian'] = 0x056E;
- t['cabengali'] = 0x099A;
- t['cacute'] = 0x0107;
- t['cadeva'] = 0x091A;
- t['cagujarati'] = 0x0A9A;
- t['cagurmukhi'] = 0x0A1A;
- t['calsquare'] = 0x3388;
- t['candrabindubengali'] = 0x0981;
- t['candrabinducmb'] = 0x0310;
- t['candrabindudeva'] = 0x0901;
- t['candrabindugujarati'] = 0x0A81;
- t['capslock'] = 0x21EA;
- t['careof'] = 0x2105;
- t['caron'] = 0x02C7;
- t['caronbelowcmb'] = 0x032C;
- t['caroncmb'] = 0x030C;
- t['carriagereturn'] = 0x21B5;
- t['cbopomofo'] = 0x3118;
- t['ccaron'] = 0x010D;
- t['ccedilla'] = 0x00E7;
- t['ccedillaacute'] = 0x1E09;
- t['ccircle'] = 0x24D2;
- t['ccircumflex'] = 0x0109;
- t['ccurl'] = 0x0255;
- t['cdot'] = 0x010B;
- t['cdotaccent'] = 0x010B;
- t['cdsquare'] = 0x33C5;
- t['cedilla'] = 0x00B8;
- t['cedillacmb'] = 0x0327;
- t['cent'] = 0x00A2;
- t['centigrade'] = 0x2103;
- t['centinferior'] = 0xF6DF;
- t['centmonospace'] = 0xFFE0;
- t['centoldstyle'] = 0xF7A2;
- t['centsuperior'] = 0xF6E0;
- t['chaarmenian'] = 0x0579;
- t['chabengali'] = 0x099B;
- t['chadeva'] = 0x091B;
- t['chagujarati'] = 0x0A9B;
- t['chagurmukhi'] = 0x0A1B;
- t['chbopomofo'] = 0x3114;
- t['cheabkhasiancyrillic'] = 0x04BD;
- t['checkmark'] = 0x2713;
- t['checyrillic'] = 0x0447;
- t['chedescenderabkhasiancyrillic'] = 0x04BF;
- t['chedescendercyrillic'] = 0x04B7;
- t['chedieresiscyrillic'] = 0x04F5;
- t['cheharmenian'] = 0x0573;
- t['chekhakassiancyrillic'] = 0x04CC;
- t['cheverticalstrokecyrillic'] = 0x04B9;
- t['chi'] = 0x03C7;
- t['chieuchacirclekorean'] = 0x3277;
- t['chieuchaparenkorean'] = 0x3217;
- t['chieuchcirclekorean'] = 0x3269;
- t['chieuchkorean'] = 0x314A;
- t['chieuchparenkorean'] = 0x3209;
- t['chochangthai'] = 0x0E0A;
- t['chochanthai'] = 0x0E08;
- t['chochingthai'] = 0x0E09;
- t['chochoethai'] = 0x0E0C;
- t['chook'] = 0x0188;
- t['cieucacirclekorean'] = 0x3276;
- t['cieucaparenkorean'] = 0x3216;
- t['cieuccirclekorean'] = 0x3268;
- t['cieuckorean'] = 0x3148;
- t['cieucparenkorean'] = 0x3208;
- t['cieucuparenkorean'] = 0x321C;
- t['circle'] = 0x25CB;
- t['circlecopyrt'] = 0x00A9;
- t['circlemultiply'] = 0x2297;
- t['circleot'] = 0x2299;
- t['circleplus'] = 0x2295;
- t['circlepostalmark'] = 0x3036;
- t['circlewithlefthalfblack'] = 0x25D0;
- t['circlewithrighthalfblack'] = 0x25D1;
- t['circumflex'] = 0x02C6;
- t['circumflexbelowcmb'] = 0x032D;
- t['circumflexcmb'] = 0x0302;
- t['clear'] = 0x2327;
- t['clickalveolar'] = 0x01C2;
- t['clickdental'] = 0x01C0;
- t['clicklateral'] = 0x01C1;
- t['clickretroflex'] = 0x01C3;
- t['club'] = 0x2663;
- t['clubsuitblack'] = 0x2663;
- t['clubsuitwhite'] = 0x2667;
- t['cmcubedsquare'] = 0x33A4;
- t['cmonospace'] = 0xFF43;
- t['cmsquaredsquare'] = 0x33A0;
- t['coarmenian'] = 0x0581;
- t['colon'] = 0x003A;
- t['colonmonetary'] = 0x20A1;
- t['colonmonospace'] = 0xFF1A;
- t['colonsign'] = 0x20A1;
- t['colonsmall'] = 0xFE55;
- t['colontriangularhalfmod'] = 0x02D1;
- t['colontriangularmod'] = 0x02D0;
- t['comma'] = 0x002C;
- t['commaabovecmb'] = 0x0313;
- t['commaaboverightcmb'] = 0x0315;
- t['commaaccent'] = 0xF6C3;
- t['commaarabic'] = 0x060C;
- t['commaarmenian'] = 0x055D;
- t['commainferior'] = 0xF6E1;
- t['commamonospace'] = 0xFF0C;
- t['commareversedabovecmb'] = 0x0314;
- t['commareversedmod'] = 0x02BD;
- t['commasmall'] = 0xFE50;
- t['commasuperior'] = 0xF6E2;
- t['commaturnedabovecmb'] = 0x0312;
- t['commaturnedmod'] = 0x02BB;
- t['compass'] = 0x263C;
- t['congruent'] = 0x2245;
- t['contourintegral'] = 0x222E;
- t['control'] = 0x2303;
- t['controlACK'] = 0x0006;
- t['controlBEL'] = 0x0007;
- t['controlBS'] = 0x0008;
- t['controlCAN'] = 0x0018;
- t['controlCR'] = 0x000D;
- t['controlDC1'] = 0x0011;
- t['controlDC2'] = 0x0012;
- t['controlDC3'] = 0x0013;
- t['controlDC4'] = 0x0014;
- t['controlDEL'] = 0x007F;
- t['controlDLE'] = 0x0010;
- t['controlEM'] = 0x0019;
- t['controlENQ'] = 0x0005;
- t['controlEOT'] = 0x0004;
- t['controlESC'] = 0x001B;
- t['controlETB'] = 0x0017;
- t['controlETX'] = 0x0003;
- t['controlFF'] = 0x000C;
- t['controlFS'] = 0x001C;
- t['controlGS'] = 0x001D;
- t['controlHT'] = 0x0009;
- t['controlLF'] = 0x000A;
- t['controlNAK'] = 0x0015;
- t['controlNULL'] = 0x0000;
- t['controlRS'] = 0x001E;
- t['controlSI'] = 0x000F;
- t['controlSO'] = 0x000E;
- t['controlSOT'] = 0x0002;
- t['controlSTX'] = 0x0001;
- t['controlSUB'] = 0x001A;
- t['controlSYN'] = 0x0016;
- t['controlUS'] = 0x001F;
- t['controlVT'] = 0x000B;
- t['copyright'] = 0x00A9;
- t['copyrightsans'] = 0xF8E9;
- t['copyrightserif'] = 0xF6D9;
- t['cornerbracketleft'] = 0x300C;
- t['cornerbracketlefthalfwidth'] = 0xFF62;
- t['cornerbracketleftvertical'] = 0xFE41;
- t['cornerbracketright'] = 0x300D;
- t['cornerbracketrighthalfwidth'] = 0xFF63;
- t['cornerbracketrightvertical'] = 0xFE42;
- t['corporationsquare'] = 0x337F;
- t['cosquare'] = 0x33C7;
- t['coverkgsquare'] = 0x33C6;
- t['cparen'] = 0x249E;
- t['cruzeiro'] = 0x20A2;
- t['cstretched'] = 0x0297;
- t['curlyand'] = 0x22CF;
- t['curlyor'] = 0x22CE;
- t['currency'] = 0x00A4;
- t['cyrBreve'] = 0xF6D1;
- t['cyrFlex'] = 0xF6D2;
- t['cyrbreve'] = 0xF6D4;
- t['cyrflex'] = 0xF6D5;
- t['d'] = 0x0064;
- t['daarmenian'] = 0x0564;
- t['dabengali'] = 0x09A6;
- t['dadarabic'] = 0x0636;
- t['dadeva'] = 0x0926;
- t['dadfinalarabic'] = 0xFEBE;
- t['dadinitialarabic'] = 0xFEBF;
- t['dadmedialarabic'] = 0xFEC0;
- t['dagesh'] = 0x05BC;
- t['dageshhebrew'] = 0x05BC;
- t['dagger'] = 0x2020;
- t['daggerdbl'] = 0x2021;
- t['dagujarati'] = 0x0AA6;
- t['dagurmukhi'] = 0x0A26;
- t['dahiragana'] = 0x3060;
- t['dakatakana'] = 0x30C0;
- t['dalarabic'] = 0x062F;
- t['dalet'] = 0x05D3;
- t['daletdagesh'] = 0xFB33;
- t['daletdageshhebrew'] = 0xFB33;
- t['dalethebrew'] = 0x05D3;
- t['dalfinalarabic'] = 0xFEAA;
- t['dammaarabic'] = 0x064F;
- t['dammalowarabic'] = 0x064F;
- t['dammatanaltonearabic'] = 0x064C;
- t['dammatanarabic'] = 0x064C;
- t['danda'] = 0x0964;
- t['dargahebrew'] = 0x05A7;
- t['dargalefthebrew'] = 0x05A7;
- t['dasiapneumatacyrilliccmb'] = 0x0485;
- t['dblGrave'] = 0xF6D3;
- t['dblanglebracketleft'] = 0x300A;
- t['dblanglebracketleftvertical'] = 0xFE3D;
- t['dblanglebracketright'] = 0x300B;
- t['dblanglebracketrightvertical'] = 0xFE3E;
- t['dblarchinvertedbelowcmb'] = 0x032B;
- t['dblarrowleft'] = 0x21D4;
- t['dblarrowright'] = 0x21D2;
- t['dbldanda'] = 0x0965;
- t['dblgrave'] = 0xF6D6;
- t['dblgravecmb'] = 0x030F;
- t['dblintegral'] = 0x222C;
- t['dbllowline'] = 0x2017;
- t['dbllowlinecmb'] = 0x0333;
- t['dbloverlinecmb'] = 0x033F;
- t['dblprimemod'] = 0x02BA;
- t['dblverticalbar'] = 0x2016;
- t['dblverticallineabovecmb'] = 0x030E;
- t['dbopomofo'] = 0x3109;
- t['dbsquare'] = 0x33C8;
- t['dcaron'] = 0x010F;
- t['dcedilla'] = 0x1E11;
- t['dcircle'] = 0x24D3;
- t['dcircumflexbelow'] = 0x1E13;
- t['dcroat'] = 0x0111;
- t['ddabengali'] = 0x09A1;
- t['ddadeva'] = 0x0921;
- t['ddagujarati'] = 0x0AA1;
- t['ddagurmukhi'] = 0x0A21;
- t['ddalarabic'] = 0x0688;
- t['ddalfinalarabic'] = 0xFB89;
- t['dddhadeva'] = 0x095C;
- t['ddhabengali'] = 0x09A2;
- t['ddhadeva'] = 0x0922;
- t['ddhagujarati'] = 0x0AA2;
- t['ddhagurmukhi'] = 0x0A22;
- t['ddotaccent'] = 0x1E0B;
- t['ddotbelow'] = 0x1E0D;
- t['decimalseparatorarabic'] = 0x066B;
- t['decimalseparatorpersian'] = 0x066B;
- t['decyrillic'] = 0x0434;
- t['degree'] = 0x00B0;
- t['dehihebrew'] = 0x05AD;
- t['dehiragana'] = 0x3067;
- t['deicoptic'] = 0x03EF;
- t['dekatakana'] = 0x30C7;
- t['deleteleft'] = 0x232B;
- t['deleteright'] = 0x2326;
- t['delta'] = 0x03B4;
- t['deltaturned'] = 0x018D;
- t['denominatorminusonenumeratorbengali'] = 0x09F8;
- t['dezh'] = 0x02A4;
- t['dhabengali'] = 0x09A7;
- t['dhadeva'] = 0x0927;
- t['dhagujarati'] = 0x0AA7;
- t['dhagurmukhi'] = 0x0A27;
- t['dhook'] = 0x0257;
- t['dialytikatonos'] = 0x0385;
- t['dialytikatonoscmb'] = 0x0344;
- t['diamond'] = 0x2666;
- t['diamondsuitwhite'] = 0x2662;
- t['dieresis'] = 0x00A8;
- t['dieresisacute'] = 0xF6D7;
- t['dieresisbelowcmb'] = 0x0324;
- t['dieresiscmb'] = 0x0308;
- t['dieresisgrave'] = 0xF6D8;
- t['dieresistonos'] = 0x0385;
- t['dihiragana'] = 0x3062;
- t['dikatakana'] = 0x30C2;
- t['dittomark'] = 0x3003;
- t['divide'] = 0x00F7;
- t['divides'] = 0x2223;
- t['divisionslash'] = 0x2215;
- t['djecyrillic'] = 0x0452;
- t['dkshade'] = 0x2593;
- t['dlinebelow'] = 0x1E0F;
- t['dlsquare'] = 0x3397;
- t['dmacron'] = 0x0111;
- t['dmonospace'] = 0xFF44;
- t['dnblock'] = 0x2584;
- t['dochadathai'] = 0x0E0E;
- t['dodekthai'] = 0x0E14;
- t['dohiragana'] = 0x3069;
- t['dokatakana'] = 0x30C9;
- t['dollar'] = 0x0024;
- t['dollarinferior'] = 0xF6E3;
- t['dollarmonospace'] = 0xFF04;
- t['dollaroldstyle'] = 0xF724;
- t['dollarsmall'] = 0xFE69;
- t['dollarsuperior'] = 0xF6E4;
- t['dong'] = 0x20AB;
- t['dorusquare'] = 0x3326;
- t['dotaccent'] = 0x02D9;
- t['dotaccentcmb'] = 0x0307;
- t['dotbelowcmb'] = 0x0323;
- t['dotbelowcomb'] = 0x0323;
- t['dotkatakana'] = 0x30FB;
- t['dotlessi'] = 0x0131;
- t['dotlessj'] = 0xF6BE;
- t['dotlessjstrokehook'] = 0x0284;
- t['dotmath'] = 0x22C5;
- t['dottedcircle'] = 0x25CC;
- t['doubleyodpatah'] = 0xFB1F;
- t['doubleyodpatahhebrew'] = 0xFB1F;
- t['downtackbelowcmb'] = 0x031E;
- t['downtackmod'] = 0x02D5;
- t['dparen'] = 0x249F;
- t['dsuperior'] = 0xF6EB;
- t['dtail'] = 0x0256;
- t['dtopbar'] = 0x018C;
- t['duhiragana'] = 0x3065;
- t['dukatakana'] = 0x30C5;
- t['dz'] = 0x01F3;
- t['dzaltone'] = 0x02A3;
- t['dzcaron'] = 0x01C6;
- t['dzcurl'] = 0x02A5;
- t['dzeabkhasiancyrillic'] = 0x04E1;
- t['dzecyrillic'] = 0x0455;
- t['dzhecyrillic'] = 0x045F;
- t['e'] = 0x0065;
- t['eacute'] = 0x00E9;
- t['earth'] = 0x2641;
- t['ebengali'] = 0x098F;
- t['ebopomofo'] = 0x311C;
- t['ebreve'] = 0x0115;
- t['ecandradeva'] = 0x090D;
- t['ecandragujarati'] = 0x0A8D;
- t['ecandravowelsigndeva'] = 0x0945;
- t['ecandravowelsigngujarati'] = 0x0AC5;
- t['ecaron'] = 0x011B;
- t['ecedillabreve'] = 0x1E1D;
- t['echarmenian'] = 0x0565;
- t['echyiwnarmenian'] = 0x0587;
- t['ecircle'] = 0x24D4;
- t['ecircumflex'] = 0x00EA;
- t['ecircumflexacute'] = 0x1EBF;
- t['ecircumflexbelow'] = 0x1E19;
- t['ecircumflexdotbelow'] = 0x1EC7;
- t['ecircumflexgrave'] = 0x1EC1;
- t['ecircumflexhookabove'] = 0x1EC3;
- t['ecircumflextilde'] = 0x1EC5;
- t['ecyrillic'] = 0x0454;
- t['edblgrave'] = 0x0205;
- t['edeva'] = 0x090F;
- t['edieresis'] = 0x00EB;
- t['edot'] = 0x0117;
- t['edotaccent'] = 0x0117;
- t['edotbelow'] = 0x1EB9;
- t['eegurmukhi'] = 0x0A0F;
- t['eematragurmukhi'] = 0x0A47;
- t['efcyrillic'] = 0x0444;
- t['egrave'] = 0x00E8;
- t['egujarati'] = 0x0A8F;
- t['eharmenian'] = 0x0567;
- t['ehbopomofo'] = 0x311D;
- t['ehiragana'] = 0x3048;
- t['ehookabove'] = 0x1EBB;
- t['eibopomofo'] = 0x311F;
- t['eight'] = 0x0038;
- t['eightarabic'] = 0x0668;
- t['eightbengali'] = 0x09EE;
- t['eightcircle'] = 0x2467;
- t['eightcircleinversesansserif'] = 0x2791;
- t['eightdeva'] = 0x096E;
- t['eighteencircle'] = 0x2471;
- t['eighteenparen'] = 0x2485;
- t['eighteenperiod'] = 0x2499;
- t['eightgujarati'] = 0x0AEE;
- t['eightgurmukhi'] = 0x0A6E;
- t['eighthackarabic'] = 0x0668;
- t['eighthangzhou'] = 0x3028;
- t['eighthnotebeamed'] = 0x266B;
- t['eightideographicparen'] = 0x3227;
- t['eightinferior'] = 0x2088;
- t['eightmonospace'] = 0xFF18;
- t['eightoldstyle'] = 0xF738;
- t['eightparen'] = 0x247B;
- t['eightperiod'] = 0x248F;
- t['eightpersian'] = 0x06F8;
- t['eightroman'] = 0x2177;
- t['eightsuperior'] = 0x2078;
- t['eightthai'] = 0x0E58;
- t['einvertedbreve'] = 0x0207;
- t['eiotifiedcyrillic'] = 0x0465;
- t['ekatakana'] = 0x30A8;
- t['ekatakanahalfwidth'] = 0xFF74;
- t['ekonkargurmukhi'] = 0x0A74;
- t['ekorean'] = 0x3154;
- t['elcyrillic'] = 0x043B;
- t['element'] = 0x2208;
- t['elevencircle'] = 0x246A;
- t['elevenparen'] = 0x247E;
- t['elevenperiod'] = 0x2492;
- t['elevenroman'] = 0x217A;
- t['ellipsis'] = 0x2026;
- t['ellipsisvertical'] = 0x22EE;
- t['emacron'] = 0x0113;
- t['emacronacute'] = 0x1E17;
- t['emacrongrave'] = 0x1E15;
- t['emcyrillic'] = 0x043C;
- t['emdash'] = 0x2014;
- t['emdashvertical'] = 0xFE31;
- t['emonospace'] = 0xFF45;
- t['emphasismarkarmenian'] = 0x055B;
- t['emptyset'] = 0x2205;
- t['enbopomofo'] = 0x3123;
- t['encyrillic'] = 0x043D;
- t['endash'] = 0x2013;
- t['endashvertical'] = 0xFE32;
- t['endescendercyrillic'] = 0x04A3;
- t['eng'] = 0x014B;
- t['engbopomofo'] = 0x3125;
- t['enghecyrillic'] = 0x04A5;
- t['enhookcyrillic'] = 0x04C8;
- t['enspace'] = 0x2002;
- t['eogonek'] = 0x0119;
- t['eokorean'] = 0x3153;
- t['eopen'] = 0x025B;
- t['eopenclosed'] = 0x029A;
- t['eopenreversed'] = 0x025C;
- t['eopenreversedclosed'] = 0x025E;
- t['eopenreversedhook'] = 0x025D;
- t['eparen'] = 0x24A0;
- t['epsilon'] = 0x03B5;
- t['epsilontonos'] = 0x03AD;
- t['equal'] = 0x003D;
- t['equalmonospace'] = 0xFF1D;
- t['equalsmall'] = 0xFE66;
- t['equalsuperior'] = 0x207C;
- t['equivalence'] = 0x2261;
- t['erbopomofo'] = 0x3126;
- t['ercyrillic'] = 0x0440;
- t['ereversed'] = 0x0258;
- t['ereversedcyrillic'] = 0x044D;
- t['escyrillic'] = 0x0441;
- t['esdescendercyrillic'] = 0x04AB;
- t['esh'] = 0x0283;
- t['eshcurl'] = 0x0286;
- t['eshortdeva'] = 0x090E;
- t['eshortvowelsigndeva'] = 0x0946;
- t['eshreversedloop'] = 0x01AA;
- t['eshsquatreversed'] = 0x0285;
- t['esmallhiragana'] = 0x3047;
- t['esmallkatakana'] = 0x30A7;
- t['esmallkatakanahalfwidth'] = 0xFF6A;
- t['estimated'] = 0x212E;
- t['esuperior'] = 0xF6EC;
- t['eta'] = 0x03B7;
- t['etarmenian'] = 0x0568;
- t['etatonos'] = 0x03AE;
- t['eth'] = 0x00F0;
- t['etilde'] = 0x1EBD;
- t['etildebelow'] = 0x1E1B;
- t['etnahtafoukhhebrew'] = 0x0591;
- t['etnahtafoukhlefthebrew'] = 0x0591;
- t['etnahtahebrew'] = 0x0591;
- t['etnahtalefthebrew'] = 0x0591;
- t['eturned'] = 0x01DD;
- t['eukorean'] = 0x3161;
- t['euro'] = 0x20AC;
- t['evowelsignbengali'] = 0x09C7;
- t['evowelsigndeva'] = 0x0947;
- t['evowelsigngujarati'] = 0x0AC7;
- t['exclam'] = 0x0021;
- t['exclamarmenian'] = 0x055C;
- t['exclamdbl'] = 0x203C;
- t['exclamdown'] = 0x00A1;
- t['exclamdownsmall'] = 0xF7A1;
- t['exclammonospace'] = 0xFF01;
- t['exclamsmall'] = 0xF721;
- t['existential'] = 0x2203;
- t['ezh'] = 0x0292;
- t['ezhcaron'] = 0x01EF;
- t['ezhcurl'] = 0x0293;
- t['ezhreversed'] = 0x01B9;
- t['ezhtail'] = 0x01BA;
- t['f'] = 0x0066;
- t['fadeva'] = 0x095E;
- t['fagurmukhi'] = 0x0A5E;
- t['fahrenheit'] = 0x2109;
- t['fathaarabic'] = 0x064E;
- t['fathalowarabic'] = 0x064E;
- t['fathatanarabic'] = 0x064B;
- t['fbopomofo'] = 0x3108;
- t['fcircle'] = 0x24D5;
- t['fdotaccent'] = 0x1E1F;
- t['feharabic'] = 0x0641;
- t['feharmenian'] = 0x0586;
- t['fehfinalarabic'] = 0xFED2;
- t['fehinitialarabic'] = 0xFED3;
- t['fehmedialarabic'] = 0xFED4;
- t['feicoptic'] = 0x03E5;
- t['female'] = 0x2640;
- t['ff'] = 0xFB00;
- t['ffi'] = 0xFB03;
- t['ffl'] = 0xFB04;
- t['fi'] = 0xFB01;
- t['fifteencircle'] = 0x246E;
- t['fifteenparen'] = 0x2482;
- t['fifteenperiod'] = 0x2496;
- t['figuredash'] = 0x2012;
- t['filledbox'] = 0x25A0;
- t['filledrect'] = 0x25AC;
- t['finalkaf'] = 0x05DA;
- t['finalkafdagesh'] = 0xFB3A;
- t['finalkafdageshhebrew'] = 0xFB3A;
- t['finalkafhebrew'] = 0x05DA;
- t['finalmem'] = 0x05DD;
- t['finalmemhebrew'] = 0x05DD;
- t['finalnun'] = 0x05DF;
- t['finalnunhebrew'] = 0x05DF;
- t['finalpe'] = 0x05E3;
- t['finalpehebrew'] = 0x05E3;
- t['finaltsadi'] = 0x05E5;
- t['finaltsadihebrew'] = 0x05E5;
- t['firsttonechinese'] = 0x02C9;
- t['fisheye'] = 0x25C9;
- t['fitacyrillic'] = 0x0473;
- t['five'] = 0x0035;
- t['fivearabic'] = 0x0665;
- t['fivebengali'] = 0x09EB;
- t['fivecircle'] = 0x2464;
- t['fivecircleinversesansserif'] = 0x278E;
- t['fivedeva'] = 0x096B;
- t['fiveeighths'] = 0x215D;
- t['fivegujarati'] = 0x0AEB;
- t['fivegurmukhi'] = 0x0A6B;
- t['fivehackarabic'] = 0x0665;
- t['fivehangzhou'] = 0x3025;
- t['fiveideographicparen'] = 0x3224;
- t['fiveinferior'] = 0x2085;
- t['fivemonospace'] = 0xFF15;
- t['fiveoldstyle'] = 0xF735;
- t['fiveparen'] = 0x2478;
- t['fiveperiod'] = 0x248C;
- t['fivepersian'] = 0x06F5;
- t['fiveroman'] = 0x2174;
- t['fivesuperior'] = 0x2075;
- t['fivethai'] = 0x0E55;
- t['fl'] = 0xFB02;
- t['florin'] = 0x0192;
- t['fmonospace'] = 0xFF46;
- t['fmsquare'] = 0x3399;
- t['fofanthai'] = 0x0E1F;
- t['fofathai'] = 0x0E1D;
- t['fongmanthai'] = 0x0E4F;
- t['forall'] = 0x2200;
- t['four'] = 0x0034;
- t['fourarabic'] = 0x0664;
- t['fourbengali'] = 0x09EA;
- t['fourcircle'] = 0x2463;
- t['fourcircleinversesansserif'] = 0x278D;
- t['fourdeva'] = 0x096A;
- t['fourgujarati'] = 0x0AEA;
- t['fourgurmukhi'] = 0x0A6A;
- t['fourhackarabic'] = 0x0664;
- t['fourhangzhou'] = 0x3024;
- t['fourideographicparen'] = 0x3223;
- t['fourinferior'] = 0x2084;
- t['fourmonospace'] = 0xFF14;
- t['fournumeratorbengali'] = 0x09F7;
- t['fouroldstyle'] = 0xF734;
- t['fourparen'] = 0x2477;
- t['fourperiod'] = 0x248B;
- t['fourpersian'] = 0x06F4;
- t['fourroman'] = 0x2173;
- t['foursuperior'] = 0x2074;
- t['fourteencircle'] = 0x246D;
- t['fourteenparen'] = 0x2481;
- t['fourteenperiod'] = 0x2495;
- t['fourthai'] = 0x0E54;
- t['fourthtonechinese'] = 0x02CB;
- t['fparen'] = 0x24A1;
- t['fraction'] = 0x2044;
- t['franc'] = 0x20A3;
- t['g'] = 0x0067;
- t['gabengali'] = 0x0997;
- t['gacute'] = 0x01F5;
- t['gadeva'] = 0x0917;
- t['gafarabic'] = 0x06AF;
- t['gaffinalarabic'] = 0xFB93;
- t['gafinitialarabic'] = 0xFB94;
- t['gafmedialarabic'] = 0xFB95;
- t['gagujarati'] = 0x0A97;
- t['gagurmukhi'] = 0x0A17;
- t['gahiragana'] = 0x304C;
- t['gakatakana'] = 0x30AC;
- t['gamma'] = 0x03B3;
- t['gammalatinsmall'] = 0x0263;
- t['gammasuperior'] = 0x02E0;
- t['gangiacoptic'] = 0x03EB;
- t['gbopomofo'] = 0x310D;
- t['gbreve'] = 0x011F;
- t['gcaron'] = 0x01E7;
- t['gcedilla'] = 0x0123;
- t['gcircle'] = 0x24D6;
- t['gcircumflex'] = 0x011D;
- t['gcommaaccent'] = 0x0123;
- t['gdot'] = 0x0121;
- t['gdotaccent'] = 0x0121;
- t['gecyrillic'] = 0x0433;
- t['gehiragana'] = 0x3052;
- t['gekatakana'] = 0x30B2;
- t['geometricallyequal'] = 0x2251;
- t['gereshaccenthebrew'] = 0x059C;
- t['gereshhebrew'] = 0x05F3;
- t['gereshmuqdamhebrew'] = 0x059D;
- t['germandbls'] = 0x00DF;
- t['gershayimaccenthebrew'] = 0x059E;
- t['gershayimhebrew'] = 0x05F4;
- t['getamark'] = 0x3013;
- t['ghabengali'] = 0x0998;
- t['ghadarmenian'] = 0x0572;
- t['ghadeva'] = 0x0918;
- t['ghagujarati'] = 0x0A98;
- t['ghagurmukhi'] = 0x0A18;
- t['ghainarabic'] = 0x063A;
- t['ghainfinalarabic'] = 0xFECE;
- t['ghaininitialarabic'] = 0xFECF;
- t['ghainmedialarabic'] = 0xFED0;
- t['ghemiddlehookcyrillic'] = 0x0495;
- t['ghestrokecyrillic'] = 0x0493;
- t['gheupturncyrillic'] = 0x0491;
- t['ghhadeva'] = 0x095A;
- t['ghhagurmukhi'] = 0x0A5A;
- t['ghook'] = 0x0260;
- t['ghzsquare'] = 0x3393;
- t['gihiragana'] = 0x304E;
- t['gikatakana'] = 0x30AE;
- t['gimarmenian'] = 0x0563;
- t['gimel'] = 0x05D2;
- t['gimeldagesh'] = 0xFB32;
- t['gimeldageshhebrew'] = 0xFB32;
- t['gimelhebrew'] = 0x05D2;
- t['gjecyrillic'] = 0x0453;
- t['glottalinvertedstroke'] = 0x01BE;
- t['glottalstop'] = 0x0294;
- t['glottalstopinverted'] = 0x0296;
- t['glottalstopmod'] = 0x02C0;
- t['glottalstopreversed'] = 0x0295;
- t['glottalstopreversedmod'] = 0x02C1;
- t['glottalstopreversedsuperior'] = 0x02E4;
- t['glottalstopstroke'] = 0x02A1;
- t['glottalstopstrokereversed'] = 0x02A2;
- t['gmacron'] = 0x1E21;
- t['gmonospace'] = 0xFF47;
- t['gohiragana'] = 0x3054;
- t['gokatakana'] = 0x30B4;
- t['gparen'] = 0x24A2;
- t['gpasquare'] = 0x33AC;
- t['gradient'] = 0x2207;
- t['grave'] = 0x0060;
- t['gravebelowcmb'] = 0x0316;
- t['gravecmb'] = 0x0300;
- t['gravecomb'] = 0x0300;
- t['gravedeva'] = 0x0953;
- t['gravelowmod'] = 0x02CE;
- t['gravemonospace'] = 0xFF40;
- t['gravetonecmb'] = 0x0340;
- t['greater'] = 0x003E;
- t['greaterequal'] = 0x2265;
- t['greaterequalorless'] = 0x22DB;
- t['greatermonospace'] = 0xFF1E;
- t['greaterorequivalent'] = 0x2273;
- t['greaterorless'] = 0x2277;
- t['greateroverequal'] = 0x2267;
- t['greatersmall'] = 0xFE65;
- t['gscript'] = 0x0261;
- t['gstroke'] = 0x01E5;
- t['guhiragana'] = 0x3050;
- t['guillemotleft'] = 0x00AB;
- t['guillemotright'] = 0x00BB;
- t['guilsinglleft'] = 0x2039;
- t['guilsinglright'] = 0x203A;
- t['gukatakana'] = 0x30B0;
- t['guramusquare'] = 0x3318;
- t['gysquare'] = 0x33C9;
- t['h'] = 0x0068;
- t['haabkhasiancyrillic'] = 0x04A9;
- t['haaltonearabic'] = 0x06C1;
- t['habengali'] = 0x09B9;
- t['hadescendercyrillic'] = 0x04B3;
- t['hadeva'] = 0x0939;
- t['hagujarati'] = 0x0AB9;
- t['hagurmukhi'] = 0x0A39;
- t['haharabic'] = 0x062D;
- t['hahfinalarabic'] = 0xFEA2;
- t['hahinitialarabic'] = 0xFEA3;
- t['hahiragana'] = 0x306F;
- t['hahmedialarabic'] = 0xFEA4;
- t['haitusquare'] = 0x332A;
- t['hakatakana'] = 0x30CF;
- t['hakatakanahalfwidth'] = 0xFF8A;
- t['halantgurmukhi'] = 0x0A4D;
- t['hamzaarabic'] = 0x0621;
- t['hamzalowarabic'] = 0x0621;
- t['hangulfiller'] = 0x3164;
- t['hardsigncyrillic'] = 0x044A;
- t['harpoonleftbarbup'] = 0x21BC;
- t['harpoonrightbarbup'] = 0x21C0;
- t['hasquare'] = 0x33CA;
- t['hatafpatah'] = 0x05B2;
- t['hatafpatah16'] = 0x05B2;
- t['hatafpatah23'] = 0x05B2;
- t['hatafpatah2f'] = 0x05B2;
- t['hatafpatahhebrew'] = 0x05B2;
- t['hatafpatahnarrowhebrew'] = 0x05B2;
- t['hatafpatahquarterhebrew'] = 0x05B2;
- t['hatafpatahwidehebrew'] = 0x05B2;
- t['hatafqamats'] = 0x05B3;
- t['hatafqamats1b'] = 0x05B3;
- t['hatafqamats28'] = 0x05B3;
- t['hatafqamats34'] = 0x05B3;
- t['hatafqamatshebrew'] = 0x05B3;
- t['hatafqamatsnarrowhebrew'] = 0x05B3;
- t['hatafqamatsquarterhebrew'] = 0x05B3;
- t['hatafqamatswidehebrew'] = 0x05B3;
- t['hatafsegol'] = 0x05B1;
- t['hatafsegol17'] = 0x05B1;
- t['hatafsegol24'] = 0x05B1;
- t['hatafsegol30'] = 0x05B1;
- t['hatafsegolhebrew'] = 0x05B1;
- t['hatafsegolnarrowhebrew'] = 0x05B1;
- t['hatafsegolquarterhebrew'] = 0x05B1;
- t['hatafsegolwidehebrew'] = 0x05B1;
- t['hbar'] = 0x0127;
- t['hbopomofo'] = 0x310F;
- t['hbrevebelow'] = 0x1E2B;
- t['hcedilla'] = 0x1E29;
- t['hcircle'] = 0x24D7;
- t['hcircumflex'] = 0x0125;
- t['hdieresis'] = 0x1E27;
- t['hdotaccent'] = 0x1E23;
- t['hdotbelow'] = 0x1E25;
- t['he'] = 0x05D4;
- t['heart'] = 0x2665;
- t['heartsuitblack'] = 0x2665;
- t['heartsuitwhite'] = 0x2661;
- t['hedagesh'] = 0xFB34;
- t['hedageshhebrew'] = 0xFB34;
- t['hehaltonearabic'] = 0x06C1;
- t['heharabic'] = 0x0647;
- t['hehebrew'] = 0x05D4;
- t['hehfinalaltonearabic'] = 0xFBA7;
- t['hehfinalalttwoarabic'] = 0xFEEA;
- t['hehfinalarabic'] = 0xFEEA;
- t['hehhamzaabovefinalarabic'] = 0xFBA5;
- t['hehhamzaaboveisolatedarabic'] = 0xFBA4;
- t['hehinitialaltonearabic'] = 0xFBA8;
- t['hehinitialarabic'] = 0xFEEB;
- t['hehiragana'] = 0x3078;
- t['hehmedialaltonearabic'] = 0xFBA9;
- t['hehmedialarabic'] = 0xFEEC;
- t['heiseierasquare'] = 0x337B;
- t['hekatakana'] = 0x30D8;
- t['hekatakanahalfwidth'] = 0xFF8D;
- t['hekutaarusquare'] = 0x3336;
- t['henghook'] = 0x0267;
- t['herutusquare'] = 0x3339;
- t['het'] = 0x05D7;
- t['hethebrew'] = 0x05D7;
- t['hhook'] = 0x0266;
- t['hhooksuperior'] = 0x02B1;
- t['hieuhacirclekorean'] = 0x327B;
- t['hieuhaparenkorean'] = 0x321B;
- t['hieuhcirclekorean'] = 0x326D;
- t['hieuhkorean'] = 0x314E;
- t['hieuhparenkorean'] = 0x320D;
- t['hihiragana'] = 0x3072;
- t['hikatakana'] = 0x30D2;
- t['hikatakanahalfwidth'] = 0xFF8B;
- t['hiriq'] = 0x05B4;
- t['hiriq14'] = 0x05B4;
- t['hiriq21'] = 0x05B4;
- t['hiriq2d'] = 0x05B4;
- t['hiriqhebrew'] = 0x05B4;
- t['hiriqnarrowhebrew'] = 0x05B4;
- t['hiriqquarterhebrew'] = 0x05B4;
- t['hiriqwidehebrew'] = 0x05B4;
- t['hlinebelow'] = 0x1E96;
- t['hmonospace'] = 0xFF48;
- t['hoarmenian'] = 0x0570;
- t['hohipthai'] = 0x0E2B;
- t['hohiragana'] = 0x307B;
- t['hokatakana'] = 0x30DB;
- t['hokatakanahalfwidth'] = 0xFF8E;
- t['holam'] = 0x05B9;
- t['holam19'] = 0x05B9;
- t['holam26'] = 0x05B9;
- t['holam32'] = 0x05B9;
- t['holamhebrew'] = 0x05B9;
- t['holamnarrowhebrew'] = 0x05B9;
- t['holamquarterhebrew'] = 0x05B9;
- t['holamwidehebrew'] = 0x05B9;
- t['honokhukthai'] = 0x0E2E;
- t['hookabovecomb'] = 0x0309;
- t['hookcmb'] = 0x0309;
- t['hookpalatalizedbelowcmb'] = 0x0321;
- t['hookretroflexbelowcmb'] = 0x0322;
- t['hoonsquare'] = 0x3342;
- t['horicoptic'] = 0x03E9;
- t['horizontalbar'] = 0x2015;
- t['horncmb'] = 0x031B;
- t['hotsprings'] = 0x2668;
- t['house'] = 0x2302;
- t['hparen'] = 0x24A3;
- t['hsuperior'] = 0x02B0;
- t['hturned'] = 0x0265;
- t['huhiragana'] = 0x3075;
- t['huiitosquare'] = 0x3333;
- t['hukatakana'] = 0x30D5;
- t['hukatakanahalfwidth'] = 0xFF8C;
- t['hungarumlaut'] = 0x02DD;
- t['hungarumlautcmb'] = 0x030B;
- t['hv'] = 0x0195;
- t['hyphen'] = 0x002D;
- t['hypheninferior'] = 0xF6E5;
- t['hyphenmonospace'] = 0xFF0D;
- t['hyphensmall'] = 0xFE63;
- t['hyphensuperior'] = 0xF6E6;
- t['hyphentwo'] = 0x2010;
- t['i'] = 0x0069;
- t['iacute'] = 0x00ED;
- t['iacyrillic'] = 0x044F;
- t['ibengali'] = 0x0987;
- t['ibopomofo'] = 0x3127;
- t['ibreve'] = 0x012D;
- t['icaron'] = 0x01D0;
- t['icircle'] = 0x24D8;
- t['icircumflex'] = 0x00EE;
- t['icyrillic'] = 0x0456;
- t['idblgrave'] = 0x0209;
- t['ideographearthcircle'] = 0x328F;
- t['ideographfirecircle'] = 0x328B;
- t['ideographicallianceparen'] = 0x323F;
- t['ideographiccallparen'] = 0x323A;
- t['ideographiccentrecircle'] = 0x32A5;
- t['ideographicclose'] = 0x3006;
- t['ideographiccomma'] = 0x3001;
- t['ideographiccommaleft'] = 0xFF64;
- t['ideographiccongratulationparen'] = 0x3237;
- t['ideographiccorrectcircle'] = 0x32A3;
- t['ideographicearthparen'] = 0x322F;
- t['ideographicenterpriseparen'] = 0x323D;
- t['ideographicexcellentcircle'] = 0x329D;
- t['ideographicfestivalparen'] = 0x3240;
- t['ideographicfinancialcircle'] = 0x3296;
- t['ideographicfinancialparen'] = 0x3236;
- t['ideographicfireparen'] = 0x322B;
- t['ideographichaveparen'] = 0x3232;
- t['ideographichighcircle'] = 0x32A4;
- t['ideographiciterationmark'] = 0x3005;
- t['ideographiclaborcircle'] = 0x3298;
- t['ideographiclaborparen'] = 0x3238;
- t['ideographicleftcircle'] = 0x32A7;
- t['ideographiclowcircle'] = 0x32A6;
- t['ideographicmedicinecircle'] = 0x32A9;
- t['ideographicmetalparen'] = 0x322E;
- t['ideographicmoonparen'] = 0x322A;
- t['ideographicnameparen'] = 0x3234;
- t['ideographicperiod'] = 0x3002;
- t['ideographicprintcircle'] = 0x329E;
- t['ideographicreachparen'] = 0x3243;
- t['ideographicrepresentparen'] = 0x3239;
- t['ideographicresourceparen'] = 0x323E;
- t['ideographicrightcircle'] = 0x32A8;
- t['ideographicsecretcircle'] = 0x3299;
- t['ideographicselfparen'] = 0x3242;
- t['ideographicsocietyparen'] = 0x3233;
- t['ideographicspace'] = 0x3000;
- t['ideographicspecialparen'] = 0x3235;
- t['ideographicstockparen'] = 0x3231;
- t['ideographicstudyparen'] = 0x323B;
- t['ideographicsunparen'] = 0x3230;
- t['ideographicsuperviseparen'] = 0x323C;
- t['ideographicwaterparen'] = 0x322C;
- t['ideographicwoodparen'] = 0x322D;
- t['ideographiczero'] = 0x3007;
- t['ideographmetalcircle'] = 0x328E;
- t['ideographmooncircle'] = 0x328A;
- t['ideographnamecircle'] = 0x3294;
- t['ideographsuncircle'] = 0x3290;
- t['ideographwatercircle'] = 0x328C;
- t['ideographwoodcircle'] = 0x328D;
- t['ideva'] = 0x0907;
- t['idieresis'] = 0x00EF;
- t['idieresisacute'] = 0x1E2F;
- t['idieresiscyrillic'] = 0x04E5;
- t['idotbelow'] = 0x1ECB;
- t['iebrevecyrillic'] = 0x04D7;
- t['iecyrillic'] = 0x0435;
- t['ieungacirclekorean'] = 0x3275;
- t['ieungaparenkorean'] = 0x3215;
- t['ieungcirclekorean'] = 0x3267;
- t['ieungkorean'] = 0x3147;
- t['ieungparenkorean'] = 0x3207;
- t['igrave'] = 0x00EC;
- t['igujarati'] = 0x0A87;
- t['igurmukhi'] = 0x0A07;
- t['ihiragana'] = 0x3044;
- t['ihookabove'] = 0x1EC9;
- t['iibengali'] = 0x0988;
- t['iicyrillic'] = 0x0438;
- t['iideva'] = 0x0908;
- t['iigujarati'] = 0x0A88;
- t['iigurmukhi'] = 0x0A08;
- t['iimatragurmukhi'] = 0x0A40;
- t['iinvertedbreve'] = 0x020B;
- t['iishortcyrillic'] = 0x0439;
- t['iivowelsignbengali'] = 0x09C0;
- t['iivowelsigndeva'] = 0x0940;
- t['iivowelsigngujarati'] = 0x0AC0;
- t['ij'] = 0x0133;
- t['ikatakana'] = 0x30A4;
- t['ikatakanahalfwidth'] = 0xFF72;
- t['ikorean'] = 0x3163;
- t['ilde'] = 0x02DC;
- t['iluyhebrew'] = 0x05AC;
- t['imacron'] = 0x012B;
- t['imacroncyrillic'] = 0x04E3;
- t['imageorapproximatelyequal'] = 0x2253;
- t['imatragurmukhi'] = 0x0A3F;
- t['imonospace'] = 0xFF49;
- t['increment'] = 0x2206;
- t['infinity'] = 0x221E;
- t['iniarmenian'] = 0x056B;
- t['integral'] = 0x222B;
- t['integralbottom'] = 0x2321;
- t['integralbt'] = 0x2321;
- t['integralex'] = 0xF8F5;
- t['integraltop'] = 0x2320;
- t['integraltp'] = 0x2320;
- t['intersection'] = 0x2229;
- t['intisquare'] = 0x3305;
- t['invbullet'] = 0x25D8;
- t['invcircle'] = 0x25D9;
- t['invsmileface'] = 0x263B;
- t['iocyrillic'] = 0x0451;
- t['iogonek'] = 0x012F;
- t['iota'] = 0x03B9;
- t['iotadieresis'] = 0x03CA;
- t['iotadieresistonos'] = 0x0390;
- t['iotalatin'] = 0x0269;
- t['iotatonos'] = 0x03AF;
- t['iparen'] = 0x24A4;
- t['irigurmukhi'] = 0x0A72;
- t['ismallhiragana'] = 0x3043;
- t['ismallkatakana'] = 0x30A3;
- t['ismallkatakanahalfwidth'] = 0xFF68;
- t['issharbengali'] = 0x09FA;
- t['istroke'] = 0x0268;
- t['isuperior'] = 0xF6ED;
- t['iterationhiragana'] = 0x309D;
- t['iterationkatakana'] = 0x30FD;
- t['itilde'] = 0x0129;
- t['itildebelow'] = 0x1E2D;
- t['iubopomofo'] = 0x3129;
- t['iucyrillic'] = 0x044E;
- t['ivowelsignbengali'] = 0x09BF;
- t['ivowelsigndeva'] = 0x093F;
- t['ivowelsigngujarati'] = 0x0ABF;
- t['izhitsacyrillic'] = 0x0475;
- t['izhitsadblgravecyrillic'] = 0x0477;
- t['j'] = 0x006A;
- t['jaarmenian'] = 0x0571;
- t['jabengali'] = 0x099C;
- t['jadeva'] = 0x091C;
- t['jagujarati'] = 0x0A9C;
- t['jagurmukhi'] = 0x0A1C;
- t['jbopomofo'] = 0x3110;
- t['jcaron'] = 0x01F0;
- t['jcircle'] = 0x24D9;
- t['jcircumflex'] = 0x0135;
- t['jcrossedtail'] = 0x029D;
- t['jdotlessstroke'] = 0x025F;
- t['jecyrillic'] = 0x0458;
- t['jeemarabic'] = 0x062C;
- t['jeemfinalarabic'] = 0xFE9E;
- t['jeeminitialarabic'] = 0xFE9F;
- t['jeemmedialarabic'] = 0xFEA0;
- t['jeharabic'] = 0x0698;
- t['jehfinalarabic'] = 0xFB8B;
- t['jhabengali'] = 0x099D;
- t['jhadeva'] = 0x091D;
- t['jhagujarati'] = 0x0A9D;
- t['jhagurmukhi'] = 0x0A1D;
- t['jheharmenian'] = 0x057B;
- t['jis'] = 0x3004;
- t['jmonospace'] = 0xFF4A;
- t['jparen'] = 0x24A5;
- t['jsuperior'] = 0x02B2;
- t['k'] = 0x006B;
- t['kabashkircyrillic'] = 0x04A1;
- t['kabengali'] = 0x0995;
- t['kacute'] = 0x1E31;
- t['kacyrillic'] = 0x043A;
- t['kadescendercyrillic'] = 0x049B;
- t['kadeva'] = 0x0915;
- t['kaf'] = 0x05DB;
- t['kafarabic'] = 0x0643;
- t['kafdagesh'] = 0xFB3B;
- t['kafdageshhebrew'] = 0xFB3B;
- t['kaffinalarabic'] = 0xFEDA;
- t['kafhebrew'] = 0x05DB;
- t['kafinitialarabic'] = 0xFEDB;
- t['kafmedialarabic'] = 0xFEDC;
- t['kafrafehebrew'] = 0xFB4D;
- t['kagujarati'] = 0x0A95;
- t['kagurmukhi'] = 0x0A15;
- t['kahiragana'] = 0x304B;
- t['kahookcyrillic'] = 0x04C4;
- t['kakatakana'] = 0x30AB;
- t['kakatakanahalfwidth'] = 0xFF76;
- t['kappa'] = 0x03BA;
- t['kappasymbolgreek'] = 0x03F0;
- t['kapyeounmieumkorean'] = 0x3171;
- t['kapyeounphieuphkorean'] = 0x3184;
- t['kapyeounpieupkorean'] = 0x3178;
- t['kapyeounssangpieupkorean'] = 0x3179;
- t['karoriisquare'] = 0x330D;
- t['kashidaautoarabic'] = 0x0640;
- t['kashidaautonosidebearingarabic'] = 0x0640;
- t['kasmallkatakana'] = 0x30F5;
- t['kasquare'] = 0x3384;
- t['kasraarabic'] = 0x0650;
- t['kasratanarabic'] = 0x064D;
- t['kastrokecyrillic'] = 0x049F;
- t['katahiraprolongmarkhalfwidth'] = 0xFF70;
- t['kaverticalstrokecyrillic'] = 0x049D;
- t['kbopomofo'] = 0x310E;
- t['kcalsquare'] = 0x3389;
- t['kcaron'] = 0x01E9;
- t['kcedilla'] = 0x0137;
- t['kcircle'] = 0x24DA;
- t['kcommaaccent'] = 0x0137;
- t['kdotbelow'] = 0x1E33;
- t['keharmenian'] = 0x0584;
- t['kehiragana'] = 0x3051;
- t['kekatakana'] = 0x30B1;
- t['kekatakanahalfwidth'] = 0xFF79;
- t['kenarmenian'] = 0x056F;
- t['kesmallkatakana'] = 0x30F6;
- t['kgreenlandic'] = 0x0138;
- t['khabengali'] = 0x0996;
- t['khacyrillic'] = 0x0445;
- t['khadeva'] = 0x0916;
- t['khagujarati'] = 0x0A96;
- t['khagurmukhi'] = 0x0A16;
- t['khaharabic'] = 0x062E;
- t['khahfinalarabic'] = 0xFEA6;
- t['khahinitialarabic'] = 0xFEA7;
- t['khahmedialarabic'] = 0xFEA8;
- t['kheicoptic'] = 0x03E7;
- t['khhadeva'] = 0x0959;
- t['khhagurmukhi'] = 0x0A59;
- t['khieukhacirclekorean'] = 0x3278;
- t['khieukhaparenkorean'] = 0x3218;
- t['khieukhcirclekorean'] = 0x326A;
- t['khieukhkorean'] = 0x314B;
- t['khieukhparenkorean'] = 0x320A;
- t['khokhaithai'] = 0x0E02;
- t['khokhonthai'] = 0x0E05;
- t['khokhuatthai'] = 0x0E03;
- t['khokhwaithai'] = 0x0E04;
- t['khomutthai'] = 0x0E5B;
- t['khook'] = 0x0199;
- t['khorakhangthai'] = 0x0E06;
- t['khzsquare'] = 0x3391;
- t['kihiragana'] = 0x304D;
- t['kikatakana'] = 0x30AD;
- t['kikatakanahalfwidth'] = 0xFF77;
- t['kiroguramusquare'] = 0x3315;
- t['kiromeetorusquare'] = 0x3316;
- t['kirosquare'] = 0x3314;
- t['kiyeokacirclekorean'] = 0x326E;
- t['kiyeokaparenkorean'] = 0x320E;
- t['kiyeokcirclekorean'] = 0x3260;
- t['kiyeokkorean'] = 0x3131;
- t['kiyeokparenkorean'] = 0x3200;
- t['kiyeoksioskorean'] = 0x3133;
- t['kjecyrillic'] = 0x045C;
- t['klinebelow'] = 0x1E35;
- t['klsquare'] = 0x3398;
- t['kmcubedsquare'] = 0x33A6;
- t['kmonospace'] = 0xFF4B;
- t['kmsquaredsquare'] = 0x33A2;
- t['kohiragana'] = 0x3053;
- t['kohmsquare'] = 0x33C0;
- t['kokaithai'] = 0x0E01;
- t['kokatakana'] = 0x30B3;
- t['kokatakanahalfwidth'] = 0xFF7A;
- t['kooposquare'] = 0x331E;
- t['koppacyrillic'] = 0x0481;
- t['koreanstandardsymbol'] = 0x327F;
- t['koroniscmb'] = 0x0343;
- t['kparen'] = 0x24A6;
- t['kpasquare'] = 0x33AA;
- t['ksicyrillic'] = 0x046F;
- t['ktsquare'] = 0x33CF;
- t['kturned'] = 0x029E;
- t['kuhiragana'] = 0x304F;
- t['kukatakana'] = 0x30AF;
- t['kukatakanahalfwidth'] = 0xFF78;
- t['kvsquare'] = 0x33B8;
- t['kwsquare'] = 0x33BE;
- t['l'] = 0x006C;
- t['labengali'] = 0x09B2;
- t['lacute'] = 0x013A;
- t['ladeva'] = 0x0932;
- t['lagujarati'] = 0x0AB2;
- t['lagurmukhi'] = 0x0A32;
- t['lakkhangyaothai'] = 0x0E45;
- t['lamaleffinalarabic'] = 0xFEFC;
- t['lamalefhamzaabovefinalarabic'] = 0xFEF8;
- t['lamalefhamzaaboveisolatedarabic'] = 0xFEF7;
- t['lamalefhamzabelowfinalarabic'] = 0xFEFA;
- t['lamalefhamzabelowisolatedarabic'] = 0xFEF9;
- t['lamalefisolatedarabic'] = 0xFEFB;
- t['lamalefmaddaabovefinalarabic'] = 0xFEF6;
- t['lamalefmaddaaboveisolatedarabic'] = 0xFEF5;
- t['lamarabic'] = 0x0644;
- t['lambda'] = 0x03BB;
- t['lambdastroke'] = 0x019B;
- t['lamed'] = 0x05DC;
- t['lameddagesh'] = 0xFB3C;
- t['lameddageshhebrew'] = 0xFB3C;
- t['lamedhebrew'] = 0x05DC;
- t['lamfinalarabic'] = 0xFEDE;
- t['lamhahinitialarabic'] = 0xFCCA;
- t['laminitialarabic'] = 0xFEDF;
- t['lamjeeminitialarabic'] = 0xFCC9;
- t['lamkhahinitialarabic'] = 0xFCCB;
- t['lamlamhehisolatedarabic'] = 0xFDF2;
- t['lammedialarabic'] = 0xFEE0;
- t['lammeemhahinitialarabic'] = 0xFD88;
- t['lammeeminitialarabic'] = 0xFCCC;
- t['largecircle'] = 0x25EF;
- t['lbar'] = 0x019A;
- t['lbelt'] = 0x026C;
- t['lbopomofo'] = 0x310C;
- t['lcaron'] = 0x013E;
- t['lcedilla'] = 0x013C;
- t['lcircle'] = 0x24DB;
- t['lcircumflexbelow'] = 0x1E3D;
- t['lcommaaccent'] = 0x013C;
- t['ldot'] = 0x0140;
- t['ldotaccent'] = 0x0140;
- t['ldotbelow'] = 0x1E37;
- t['ldotbelowmacron'] = 0x1E39;
- t['leftangleabovecmb'] = 0x031A;
- t['lefttackbelowcmb'] = 0x0318;
- t['less'] = 0x003C;
- t['lessequal'] = 0x2264;
- t['lessequalorgreater'] = 0x22DA;
- t['lessmonospace'] = 0xFF1C;
- t['lessorequivalent'] = 0x2272;
- t['lessorgreater'] = 0x2276;
- t['lessoverequal'] = 0x2266;
- t['lesssmall'] = 0xFE64;
- t['lezh'] = 0x026E;
- t['lfblock'] = 0x258C;
- t['lhookretroflex'] = 0x026D;
- t['lira'] = 0x20A4;
- t['liwnarmenian'] = 0x056C;
- t['lj'] = 0x01C9;
- t['ljecyrillic'] = 0x0459;
- t['ll'] = 0xF6C0;
- t['lladeva'] = 0x0933;
- t['llagujarati'] = 0x0AB3;
- t['llinebelow'] = 0x1E3B;
- t['llladeva'] = 0x0934;
- t['llvocalicbengali'] = 0x09E1;
- t['llvocalicdeva'] = 0x0961;
- t['llvocalicvowelsignbengali'] = 0x09E3;
- t['llvocalicvowelsigndeva'] = 0x0963;
- t['lmiddletilde'] = 0x026B;
- t['lmonospace'] = 0xFF4C;
- t['lmsquare'] = 0x33D0;
- t['lochulathai'] = 0x0E2C;
- t['logicaland'] = 0x2227;
- t['logicalnot'] = 0x00AC;
- t['logicalnotreversed'] = 0x2310;
- t['logicalor'] = 0x2228;
- t['lolingthai'] = 0x0E25;
- t['longs'] = 0x017F;
- t['lowlinecenterline'] = 0xFE4E;
- t['lowlinecmb'] = 0x0332;
- t['lowlinedashed'] = 0xFE4D;
- t['lozenge'] = 0x25CA;
- t['lparen'] = 0x24A7;
- t['lslash'] = 0x0142;
- t['lsquare'] = 0x2113;
- t['lsuperior'] = 0xF6EE;
- t['ltshade'] = 0x2591;
- t['luthai'] = 0x0E26;
- t['lvocalicbengali'] = 0x098C;
- t['lvocalicdeva'] = 0x090C;
- t['lvocalicvowelsignbengali'] = 0x09E2;
- t['lvocalicvowelsigndeva'] = 0x0962;
- t['lxsquare'] = 0x33D3;
- t['m'] = 0x006D;
- t['mabengali'] = 0x09AE;
- t['macron'] = 0x00AF;
- t['macronbelowcmb'] = 0x0331;
- t['macroncmb'] = 0x0304;
- t['macronlowmod'] = 0x02CD;
- t['macronmonospace'] = 0xFFE3;
- t['macute'] = 0x1E3F;
- t['madeva'] = 0x092E;
- t['magujarati'] = 0x0AAE;
- t['magurmukhi'] = 0x0A2E;
- t['mahapakhhebrew'] = 0x05A4;
- t['mahapakhlefthebrew'] = 0x05A4;
- t['mahiragana'] = 0x307E;
- t['maichattawalowleftthai'] = 0xF895;
- t['maichattawalowrightthai'] = 0xF894;
- t['maichattawathai'] = 0x0E4B;
- t['maichattawaupperleftthai'] = 0xF893;
- t['maieklowleftthai'] = 0xF88C;
- t['maieklowrightthai'] = 0xF88B;
- t['maiekthai'] = 0x0E48;
- t['maiekupperleftthai'] = 0xF88A;
- t['maihanakatleftthai'] = 0xF884;
- t['maihanakatthai'] = 0x0E31;
- t['maitaikhuleftthai'] = 0xF889;
- t['maitaikhuthai'] = 0x0E47;
- t['maitholowleftthai'] = 0xF88F;
- t['maitholowrightthai'] = 0xF88E;
- t['maithothai'] = 0x0E49;
- t['maithoupperleftthai'] = 0xF88D;
- t['maitrilowleftthai'] = 0xF892;
- t['maitrilowrightthai'] = 0xF891;
- t['maitrithai'] = 0x0E4A;
- t['maitriupperleftthai'] = 0xF890;
- t['maiyamokthai'] = 0x0E46;
- t['makatakana'] = 0x30DE;
- t['makatakanahalfwidth'] = 0xFF8F;
- t['male'] = 0x2642;
- t['mansyonsquare'] = 0x3347;
- t['maqafhebrew'] = 0x05BE;
- t['mars'] = 0x2642;
- t['masoracirclehebrew'] = 0x05AF;
- t['masquare'] = 0x3383;
- t['mbopomofo'] = 0x3107;
- t['mbsquare'] = 0x33D4;
- t['mcircle'] = 0x24DC;
- t['mcubedsquare'] = 0x33A5;
- t['mdotaccent'] = 0x1E41;
- t['mdotbelow'] = 0x1E43;
- t['meemarabic'] = 0x0645;
- t['meemfinalarabic'] = 0xFEE2;
- t['meeminitialarabic'] = 0xFEE3;
- t['meemmedialarabic'] = 0xFEE4;
- t['meemmeeminitialarabic'] = 0xFCD1;
- t['meemmeemisolatedarabic'] = 0xFC48;
- t['meetorusquare'] = 0x334D;
- t['mehiragana'] = 0x3081;
- t['meizierasquare'] = 0x337E;
- t['mekatakana'] = 0x30E1;
- t['mekatakanahalfwidth'] = 0xFF92;
- t['mem'] = 0x05DE;
- t['memdagesh'] = 0xFB3E;
- t['memdageshhebrew'] = 0xFB3E;
- t['memhebrew'] = 0x05DE;
- t['menarmenian'] = 0x0574;
- t['merkhahebrew'] = 0x05A5;
- t['merkhakefulahebrew'] = 0x05A6;
- t['merkhakefulalefthebrew'] = 0x05A6;
- t['merkhalefthebrew'] = 0x05A5;
- t['mhook'] = 0x0271;
- t['mhzsquare'] = 0x3392;
- t['middledotkatakanahalfwidth'] = 0xFF65;
- t['middot'] = 0x00B7;
- t['mieumacirclekorean'] = 0x3272;
- t['mieumaparenkorean'] = 0x3212;
- t['mieumcirclekorean'] = 0x3264;
- t['mieumkorean'] = 0x3141;
- t['mieumpansioskorean'] = 0x3170;
- t['mieumparenkorean'] = 0x3204;
- t['mieumpieupkorean'] = 0x316E;
- t['mieumsioskorean'] = 0x316F;
- t['mihiragana'] = 0x307F;
- t['mikatakana'] = 0x30DF;
- t['mikatakanahalfwidth'] = 0xFF90;
- t['minus'] = 0x2212;
- t['minusbelowcmb'] = 0x0320;
- t['minuscircle'] = 0x2296;
- t['minusmod'] = 0x02D7;
- t['minusplus'] = 0x2213;
- t['minute'] = 0x2032;
- t['miribaarusquare'] = 0x334A;
- t['mirisquare'] = 0x3349;
- t['mlonglegturned'] = 0x0270;
- t['mlsquare'] = 0x3396;
- t['mmcubedsquare'] = 0x33A3;
- t['mmonospace'] = 0xFF4D;
- t['mmsquaredsquare'] = 0x339F;
- t['mohiragana'] = 0x3082;
- t['mohmsquare'] = 0x33C1;
- t['mokatakana'] = 0x30E2;
- t['mokatakanahalfwidth'] = 0xFF93;
- t['molsquare'] = 0x33D6;
- t['momathai'] = 0x0E21;
- t['moverssquare'] = 0x33A7;
- t['moverssquaredsquare'] = 0x33A8;
- t['mparen'] = 0x24A8;
- t['mpasquare'] = 0x33AB;
- t['mssquare'] = 0x33B3;
- t['msuperior'] = 0xF6EF;
- t['mturned'] = 0x026F;
- t['mu'] = 0x00B5;
- t['mu1'] = 0x00B5;
- t['muasquare'] = 0x3382;
- t['muchgreater'] = 0x226B;
- t['muchless'] = 0x226A;
- t['mufsquare'] = 0x338C;
- t['mugreek'] = 0x03BC;
- t['mugsquare'] = 0x338D;
- t['muhiragana'] = 0x3080;
- t['mukatakana'] = 0x30E0;
- t['mukatakanahalfwidth'] = 0xFF91;
- t['mulsquare'] = 0x3395;
- t['multiply'] = 0x00D7;
- t['mumsquare'] = 0x339B;
- t['munahhebrew'] = 0x05A3;
- t['munahlefthebrew'] = 0x05A3;
- t['musicalnote'] = 0x266A;
- t['musicalnotedbl'] = 0x266B;
- t['musicflatsign'] = 0x266D;
- t['musicsharpsign'] = 0x266F;
- t['mussquare'] = 0x33B2;
- t['muvsquare'] = 0x33B6;
- t['muwsquare'] = 0x33BC;
- t['mvmegasquare'] = 0x33B9;
- t['mvsquare'] = 0x33B7;
- t['mwmegasquare'] = 0x33BF;
- t['mwsquare'] = 0x33BD;
- t['n'] = 0x006E;
- t['nabengali'] = 0x09A8;
- t['nabla'] = 0x2207;
- t['nacute'] = 0x0144;
- t['nadeva'] = 0x0928;
- t['nagujarati'] = 0x0AA8;
- t['nagurmukhi'] = 0x0A28;
- t['nahiragana'] = 0x306A;
- t['nakatakana'] = 0x30CA;
- t['nakatakanahalfwidth'] = 0xFF85;
- t['napostrophe'] = 0x0149;
- t['nasquare'] = 0x3381;
- t['nbopomofo'] = 0x310B;
- t['nbspace'] = 0x00A0;
- t['ncaron'] = 0x0148;
- t['ncedilla'] = 0x0146;
- t['ncircle'] = 0x24DD;
- t['ncircumflexbelow'] = 0x1E4B;
- t['ncommaaccent'] = 0x0146;
- t['ndotaccent'] = 0x1E45;
- t['ndotbelow'] = 0x1E47;
- t['nehiragana'] = 0x306D;
- t['nekatakana'] = 0x30CD;
- t['nekatakanahalfwidth'] = 0xFF88;
- t['newsheqelsign'] = 0x20AA;
- t['nfsquare'] = 0x338B;
- t['ngabengali'] = 0x0999;
- t['ngadeva'] = 0x0919;
- t['ngagujarati'] = 0x0A99;
- t['ngagurmukhi'] = 0x0A19;
- t['ngonguthai'] = 0x0E07;
- t['nhiragana'] = 0x3093;
- t['nhookleft'] = 0x0272;
- t['nhookretroflex'] = 0x0273;
- t['nieunacirclekorean'] = 0x326F;
- t['nieunaparenkorean'] = 0x320F;
- t['nieuncieuckorean'] = 0x3135;
- t['nieuncirclekorean'] = 0x3261;
- t['nieunhieuhkorean'] = 0x3136;
- t['nieunkorean'] = 0x3134;
- t['nieunpansioskorean'] = 0x3168;
- t['nieunparenkorean'] = 0x3201;
- t['nieunsioskorean'] = 0x3167;
- t['nieuntikeutkorean'] = 0x3166;
- t['nihiragana'] = 0x306B;
- t['nikatakana'] = 0x30CB;
- t['nikatakanahalfwidth'] = 0xFF86;
- t['nikhahitleftthai'] = 0xF899;
- t['nikhahitthai'] = 0x0E4D;
- t['nine'] = 0x0039;
- t['ninearabic'] = 0x0669;
- t['ninebengali'] = 0x09EF;
- t['ninecircle'] = 0x2468;
- t['ninecircleinversesansserif'] = 0x2792;
- t['ninedeva'] = 0x096F;
- t['ninegujarati'] = 0x0AEF;
- t['ninegurmukhi'] = 0x0A6F;
- t['ninehackarabic'] = 0x0669;
- t['ninehangzhou'] = 0x3029;
- t['nineideographicparen'] = 0x3228;
- t['nineinferior'] = 0x2089;
- t['ninemonospace'] = 0xFF19;
- t['nineoldstyle'] = 0xF739;
- t['nineparen'] = 0x247C;
- t['nineperiod'] = 0x2490;
- t['ninepersian'] = 0x06F9;
- t['nineroman'] = 0x2178;
- t['ninesuperior'] = 0x2079;
- t['nineteencircle'] = 0x2472;
- t['nineteenparen'] = 0x2486;
- t['nineteenperiod'] = 0x249A;
- t['ninethai'] = 0x0E59;
- t['nj'] = 0x01CC;
- t['njecyrillic'] = 0x045A;
- t['nkatakana'] = 0x30F3;
- t['nkatakanahalfwidth'] = 0xFF9D;
- t['nlegrightlong'] = 0x019E;
- t['nlinebelow'] = 0x1E49;
- t['nmonospace'] = 0xFF4E;
- t['nmsquare'] = 0x339A;
- t['nnabengali'] = 0x09A3;
- t['nnadeva'] = 0x0923;
- t['nnagujarati'] = 0x0AA3;
- t['nnagurmukhi'] = 0x0A23;
- t['nnnadeva'] = 0x0929;
- t['nohiragana'] = 0x306E;
- t['nokatakana'] = 0x30CE;
- t['nokatakanahalfwidth'] = 0xFF89;
- t['nonbreakingspace'] = 0x00A0;
- t['nonenthai'] = 0x0E13;
- t['nonuthai'] = 0x0E19;
- t['noonarabic'] = 0x0646;
- t['noonfinalarabic'] = 0xFEE6;
- t['noonghunnaarabic'] = 0x06BA;
- t['noonghunnafinalarabic'] = 0xFB9F;
- t['nooninitialarabic'] = 0xFEE7;
- t['noonjeeminitialarabic'] = 0xFCD2;
- t['noonjeemisolatedarabic'] = 0xFC4B;
- t['noonmedialarabic'] = 0xFEE8;
- t['noonmeeminitialarabic'] = 0xFCD5;
- t['noonmeemisolatedarabic'] = 0xFC4E;
- t['noonnoonfinalarabic'] = 0xFC8D;
- t['notcontains'] = 0x220C;
- t['notelement'] = 0x2209;
- t['notelementof'] = 0x2209;
- t['notequal'] = 0x2260;
- t['notgreater'] = 0x226F;
- t['notgreaternorequal'] = 0x2271;
- t['notgreaternorless'] = 0x2279;
- t['notidentical'] = 0x2262;
- t['notless'] = 0x226E;
- t['notlessnorequal'] = 0x2270;
- t['notparallel'] = 0x2226;
- t['notprecedes'] = 0x2280;
- t['notsubset'] = 0x2284;
- t['notsucceeds'] = 0x2281;
- t['notsuperset'] = 0x2285;
- t['nowarmenian'] = 0x0576;
- t['nparen'] = 0x24A9;
- t['nssquare'] = 0x33B1;
- t['nsuperior'] = 0x207F;
- t['ntilde'] = 0x00F1;
- t['nu'] = 0x03BD;
- t['nuhiragana'] = 0x306C;
- t['nukatakana'] = 0x30CC;
- t['nukatakanahalfwidth'] = 0xFF87;
- t['nuktabengali'] = 0x09BC;
- t['nuktadeva'] = 0x093C;
- t['nuktagujarati'] = 0x0ABC;
- t['nuktagurmukhi'] = 0x0A3C;
- t['numbersign'] = 0x0023;
- t['numbersignmonospace'] = 0xFF03;
- t['numbersignsmall'] = 0xFE5F;
- t['numeralsigngreek'] = 0x0374;
- t['numeralsignlowergreek'] = 0x0375;
- t['numero'] = 0x2116;
- t['nun'] = 0x05E0;
- t['nundagesh'] = 0xFB40;
- t['nundageshhebrew'] = 0xFB40;
- t['nunhebrew'] = 0x05E0;
- t['nvsquare'] = 0x33B5;
- t['nwsquare'] = 0x33BB;
- t['nyabengali'] = 0x099E;
- t['nyadeva'] = 0x091E;
- t['nyagujarati'] = 0x0A9E;
- t['nyagurmukhi'] = 0x0A1E;
- t['o'] = 0x006F;
- t['oacute'] = 0x00F3;
- t['oangthai'] = 0x0E2D;
- t['obarred'] = 0x0275;
- t['obarredcyrillic'] = 0x04E9;
- t['obarreddieresiscyrillic'] = 0x04EB;
- t['obengali'] = 0x0993;
- t['obopomofo'] = 0x311B;
- t['obreve'] = 0x014F;
- t['ocandradeva'] = 0x0911;
- t['ocandragujarati'] = 0x0A91;
- t['ocandravowelsigndeva'] = 0x0949;
- t['ocandravowelsigngujarati'] = 0x0AC9;
- t['ocaron'] = 0x01D2;
- t['ocircle'] = 0x24DE;
- t['ocircumflex'] = 0x00F4;
- t['ocircumflexacute'] = 0x1ED1;
- t['ocircumflexdotbelow'] = 0x1ED9;
- t['ocircumflexgrave'] = 0x1ED3;
- t['ocircumflexhookabove'] = 0x1ED5;
- t['ocircumflextilde'] = 0x1ED7;
- t['ocyrillic'] = 0x043E;
- t['odblacute'] = 0x0151;
- t['odblgrave'] = 0x020D;
- t['odeva'] = 0x0913;
- t['odieresis'] = 0x00F6;
- t['odieresiscyrillic'] = 0x04E7;
- t['odotbelow'] = 0x1ECD;
- t['oe'] = 0x0153;
- t['oekorean'] = 0x315A;
- t['ogonek'] = 0x02DB;
- t['ogonekcmb'] = 0x0328;
- t['ograve'] = 0x00F2;
- t['ogujarati'] = 0x0A93;
- t['oharmenian'] = 0x0585;
- t['ohiragana'] = 0x304A;
- t['ohookabove'] = 0x1ECF;
- t['ohorn'] = 0x01A1;
- t['ohornacute'] = 0x1EDB;
- t['ohorndotbelow'] = 0x1EE3;
- t['ohorngrave'] = 0x1EDD;
- t['ohornhookabove'] = 0x1EDF;
- t['ohorntilde'] = 0x1EE1;
- t['ohungarumlaut'] = 0x0151;
- t['oi'] = 0x01A3;
- t['oinvertedbreve'] = 0x020F;
- t['okatakana'] = 0x30AA;
- t['okatakanahalfwidth'] = 0xFF75;
- t['okorean'] = 0x3157;
- t['olehebrew'] = 0x05AB;
- t['omacron'] = 0x014D;
- t['omacronacute'] = 0x1E53;
- t['omacrongrave'] = 0x1E51;
- t['omdeva'] = 0x0950;
- t['omega'] = 0x03C9;
- t['omega1'] = 0x03D6;
- t['omegacyrillic'] = 0x0461;
- t['omegalatinclosed'] = 0x0277;
- t['omegaroundcyrillic'] = 0x047B;
- t['omegatitlocyrillic'] = 0x047D;
- t['omegatonos'] = 0x03CE;
- t['omgujarati'] = 0x0AD0;
- t['omicron'] = 0x03BF;
- t['omicrontonos'] = 0x03CC;
- t['omonospace'] = 0xFF4F;
- t['one'] = 0x0031;
- t['onearabic'] = 0x0661;
- t['onebengali'] = 0x09E7;
- t['onecircle'] = 0x2460;
- t['onecircleinversesansserif'] = 0x278A;
- t['onedeva'] = 0x0967;
- t['onedotenleader'] = 0x2024;
- t['oneeighth'] = 0x215B;
- t['onefitted'] = 0xF6DC;
- t['onegujarati'] = 0x0AE7;
- t['onegurmukhi'] = 0x0A67;
- t['onehackarabic'] = 0x0661;
- t['onehalf'] = 0x00BD;
- t['onehangzhou'] = 0x3021;
- t['oneideographicparen'] = 0x3220;
- t['oneinferior'] = 0x2081;
- t['onemonospace'] = 0xFF11;
- t['onenumeratorbengali'] = 0x09F4;
- t['oneoldstyle'] = 0xF731;
- t['oneparen'] = 0x2474;
- t['oneperiod'] = 0x2488;
- t['onepersian'] = 0x06F1;
- t['onequarter'] = 0x00BC;
- t['oneroman'] = 0x2170;
- t['onesuperior'] = 0x00B9;
- t['onethai'] = 0x0E51;
- t['onethird'] = 0x2153;
- t['oogonek'] = 0x01EB;
- t['oogonekmacron'] = 0x01ED;
- t['oogurmukhi'] = 0x0A13;
- t['oomatragurmukhi'] = 0x0A4B;
- t['oopen'] = 0x0254;
- t['oparen'] = 0x24AA;
- t['openbullet'] = 0x25E6;
- t['option'] = 0x2325;
- t['ordfeminine'] = 0x00AA;
- t['ordmasculine'] = 0x00BA;
- t['orthogonal'] = 0x221F;
- t['oshortdeva'] = 0x0912;
- t['oshortvowelsigndeva'] = 0x094A;
- t['oslash'] = 0x00F8;
- t['oslashacute'] = 0x01FF;
- t['osmallhiragana'] = 0x3049;
- t['osmallkatakana'] = 0x30A9;
- t['osmallkatakanahalfwidth'] = 0xFF6B;
- t['ostrokeacute'] = 0x01FF;
- t['osuperior'] = 0xF6F0;
- t['otcyrillic'] = 0x047F;
- t['otilde'] = 0x00F5;
- t['otildeacute'] = 0x1E4D;
- t['otildedieresis'] = 0x1E4F;
- t['oubopomofo'] = 0x3121;
- t['overline'] = 0x203E;
- t['overlinecenterline'] = 0xFE4A;
- t['overlinecmb'] = 0x0305;
- t['overlinedashed'] = 0xFE49;
- t['overlinedblwavy'] = 0xFE4C;
- t['overlinewavy'] = 0xFE4B;
- t['overscore'] = 0x00AF;
- t['ovowelsignbengali'] = 0x09CB;
- t['ovowelsigndeva'] = 0x094B;
- t['ovowelsigngujarati'] = 0x0ACB;
- t['p'] = 0x0070;
- t['paampssquare'] = 0x3380;
- t['paasentosquare'] = 0x332B;
- t['pabengali'] = 0x09AA;
- t['pacute'] = 0x1E55;
- t['padeva'] = 0x092A;
- t['pagedown'] = 0x21DF;
- t['pageup'] = 0x21DE;
- t['pagujarati'] = 0x0AAA;
- t['pagurmukhi'] = 0x0A2A;
- t['pahiragana'] = 0x3071;
- t['paiyannoithai'] = 0x0E2F;
- t['pakatakana'] = 0x30D1;
- t['palatalizationcyrilliccmb'] = 0x0484;
- t['palochkacyrillic'] = 0x04C0;
- t['pansioskorean'] = 0x317F;
- t['paragraph'] = 0x00B6;
- t['parallel'] = 0x2225;
- t['parenleft'] = 0x0028;
- t['parenleftaltonearabic'] = 0xFD3E;
- t['parenleftbt'] = 0xF8ED;
- t['parenleftex'] = 0xF8EC;
- t['parenleftinferior'] = 0x208D;
- t['parenleftmonospace'] = 0xFF08;
- t['parenleftsmall'] = 0xFE59;
- t['parenleftsuperior'] = 0x207D;
- t['parenlefttp'] = 0xF8EB;
- t['parenleftvertical'] = 0xFE35;
- t['parenright'] = 0x0029;
- t['parenrightaltonearabic'] = 0xFD3F;
- t['parenrightbt'] = 0xF8F8;
- t['parenrightex'] = 0xF8F7;
- t['parenrightinferior'] = 0x208E;
- t['parenrightmonospace'] = 0xFF09;
- t['parenrightsmall'] = 0xFE5A;
- t['parenrightsuperior'] = 0x207E;
- t['parenrighttp'] = 0xF8F6;
- t['parenrightvertical'] = 0xFE36;
- t['partialdiff'] = 0x2202;
- t['paseqhebrew'] = 0x05C0;
- t['pashtahebrew'] = 0x0599;
- t['pasquare'] = 0x33A9;
- t['patah'] = 0x05B7;
- t['patah11'] = 0x05B7;
- t['patah1d'] = 0x05B7;
- t['patah2a'] = 0x05B7;
- t['patahhebrew'] = 0x05B7;
- t['patahnarrowhebrew'] = 0x05B7;
- t['patahquarterhebrew'] = 0x05B7;
- t['patahwidehebrew'] = 0x05B7;
- t['pazerhebrew'] = 0x05A1;
- t['pbopomofo'] = 0x3106;
- t['pcircle'] = 0x24DF;
- t['pdotaccent'] = 0x1E57;
- t['pe'] = 0x05E4;
- t['pecyrillic'] = 0x043F;
- t['pedagesh'] = 0xFB44;
- t['pedageshhebrew'] = 0xFB44;
- t['peezisquare'] = 0x333B;
- t['pefinaldageshhebrew'] = 0xFB43;
- t['peharabic'] = 0x067E;
- t['peharmenian'] = 0x057A;
- t['pehebrew'] = 0x05E4;
- t['pehfinalarabic'] = 0xFB57;
- t['pehinitialarabic'] = 0xFB58;
- t['pehiragana'] = 0x307A;
- t['pehmedialarabic'] = 0xFB59;
- t['pekatakana'] = 0x30DA;
- t['pemiddlehookcyrillic'] = 0x04A7;
- t['perafehebrew'] = 0xFB4E;
- t['percent'] = 0x0025;
- t['percentarabic'] = 0x066A;
- t['percentmonospace'] = 0xFF05;
- t['percentsmall'] = 0xFE6A;
- t['period'] = 0x002E;
- t['periodarmenian'] = 0x0589;
- t['periodcentered'] = 0x00B7;
- t['periodhalfwidth'] = 0xFF61;
- t['periodinferior'] = 0xF6E7;
- t['periodmonospace'] = 0xFF0E;
- t['periodsmall'] = 0xFE52;
- t['periodsuperior'] = 0xF6E8;
- t['perispomenigreekcmb'] = 0x0342;
- t['perpendicular'] = 0x22A5;
- t['perthousand'] = 0x2030;
- t['peseta'] = 0x20A7;
- t['pfsquare'] = 0x338A;
- t['phabengali'] = 0x09AB;
- t['phadeva'] = 0x092B;
- t['phagujarati'] = 0x0AAB;
- t['phagurmukhi'] = 0x0A2B;
- t['phi'] = 0x03C6;
- t['phi1'] = 0x03D5;
- t['phieuphacirclekorean'] = 0x327A;
- t['phieuphaparenkorean'] = 0x321A;
- t['phieuphcirclekorean'] = 0x326C;
- t['phieuphkorean'] = 0x314D;
- t['phieuphparenkorean'] = 0x320C;
- t['philatin'] = 0x0278;
- t['phinthuthai'] = 0x0E3A;
- t['phisymbolgreek'] = 0x03D5;
- t['phook'] = 0x01A5;
- t['phophanthai'] = 0x0E1E;
- t['phophungthai'] = 0x0E1C;
- t['phosamphaothai'] = 0x0E20;
- t['pi'] = 0x03C0;
- t['pieupacirclekorean'] = 0x3273;
- t['pieupaparenkorean'] = 0x3213;
- t['pieupcieuckorean'] = 0x3176;
- t['pieupcirclekorean'] = 0x3265;
- t['pieupkiyeokkorean'] = 0x3172;
- t['pieupkorean'] = 0x3142;
- t['pieupparenkorean'] = 0x3205;
- t['pieupsioskiyeokkorean'] = 0x3174;
- t['pieupsioskorean'] = 0x3144;
- t['pieupsiostikeutkorean'] = 0x3175;
- t['pieupthieuthkorean'] = 0x3177;
- t['pieuptikeutkorean'] = 0x3173;
- t['pihiragana'] = 0x3074;
- t['pikatakana'] = 0x30D4;
- t['pisymbolgreek'] = 0x03D6;
- t['piwrarmenian'] = 0x0583;
- t['plus'] = 0x002B;
- t['plusbelowcmb'] = 0x031F;
- t['pluscircle'] = 0x2295;
- t['plusminus'] = 0x00B1;
- t['plusmod'] = 0x02D6;
- t['plusmonospace'] = 0xFF0B;
- t['plussmall'] = 0xFE62;
- t['plussuperior'] = 0x207A;
- t['pmonospace'] = 0xFF50;
- t['pmsquare'] = 0x33D8;
- t['pohiragana'] = 0x307D;
- t['pointingindexdownwhite'] = 0x261F;
- t['pointingindexleftwhite'] = 0x261C;
- t['pointingindexrightwhite'] = 0x261E;
- t['pointingindexupwhite'] = 0x261D;
- t['pokatakana'] = 0x30DD;
- t['poplathai'] = 0x0E1B;
- t['postalmark'] = 0x3012;
- t['postalmarkface'] = 0x3020;
- t['pparen'] = 0x24AB;
- t['precedes'] = 0x227A;
- t['prescription'] = 0x211E;
- t['primemod'] = 0x02B9;
- t['primereversed'] = 0x2035;
- t['product'] = 0x220F;
- t['projective'] = 0x2305;
- t['prolongedkana'] = 0x30FC;
- t['propellor'] = 0x2318;
- t['propersubset'] = 0x2282;
- t['propersuperset'] = 0x2283;
- t['proportion'] = 0x2237;
- t['proportional'] = 0x221D;
- t['psi'] = 0x03C8;
- t['psicyrillic'] = 0x0471;
- t['psilipneumatacyrilliccmb'] = 0x0486;
- t['pssquare'] = 0x33B0;
- t['puhiragana'] = 0x3077;
- t['pukatakana'] = 0x30D7;
- t['pvsquare'] = 0x33B4;
- t['pwsquare'] = 0x33BA;
- t['q'] = 0x0071;
- t['qadeva'] = 0x0958;
- t['qadmahebrew'] = 0x05A8;
- t['qafarabic'] = 0x0642;
- t['qaffinalarabic'] = 0xFED6;
- t['qafinitialarabic'] = 0xFED7;
- t['qafmedialarabic'] = 0xFED8;
- t['qamats'] = 0x05B8;
- t['qamats10'] = 0x05B8;
- t['qamats1a'] = 0x05B8;
- t['qamats1c'] = 0x05B8;
- t['qamats27'] = 0x05B8;
- t['qamats29'] = 0x05B8;
- t['qamats33'] = 0x05B8;
- t['qamatsde'] = 0x05B8;
- t['qamatshebrew'] = 0x05B8;
- t['qamatsnarrowhebrew'] = 0x05B8;
- t['qamatsqatanhebrew'] = 0x05B8;
- t['qamatsqatannarrowhebrew'] = 0x05B8;
- t['qamatsqatanquarterhebrew'] = 0x05B8;
- t['qamatsqatanwidehebrew'] = 0x05B8;
- t['qamatsquarterhebrew'] = 0x05B8;
- t['qamatswidehebrew'] = 0x05B8;
- t['qarneyparahebrew'] = 0x059F;
- t['qbopomofo'] = 0x3111;
- t['qcircle'] = 0x24E0;
- t['qhook'] = 0x02A0;
- t['qmonospace'] = 0xFF51;
- t['qof'] = 0x05E7;
- t['qofdagesh'] = 0xFB47;
- t['qofdageshhebrew'] = 0xFB47;
- t['qofhebrew'] = 0x05E7;
- t['qparen'] = 0x24AC;
- t['quarternote'] = 0x2669;
- t['qubuts'] = 0x05BB;
- t['qubuts18'] = 0x05BB;
- t['qubuts25'] = 0x05BB;
- t['qubuts31'] = 0x05BB;
- t['qubutshebrew'] = 0x05BB;
- t['qubutsnarrowhebrew'] = 0x05BB;
- t['qubutsquarterhebrew'] = 0x05BB;
- t['qubutswidehebrew'] = 0x05BB;
- t['question'] = 0x003F;
- t['questionarabic'] = 0x061F;
- t['questionarmenian'] = 0x055E;
- t['questiondown'] = 0x00BF;
- t['questiondownsmall'] = 0xF7BF;
- t['questiongreek'] = 0x037E;
- t['questionmonospace'] = 0xFF1F;
- t['questionsmall'] = 0xF73F;
- t['quotedbl'] = 0x0022;
- t['quotedblbase'] = 0x201E;
- t['quotedblleft'] = 0x201C;
- t['quotedblmonospace'] = 0xFF02;
- t['quotedblprime'] = 0x301E;
- t['quotedblprimereversed'] = 0x301D;
- t['quotedblright'] = 0x201D;
- t['quoteleft'] = 0x2018;
- t['quoteleftreversed'] = 0x201B;
- t['quotereversed'] = 0x201B;
- t['quoteright'] = 0x2019;
- t['quoterightn'] = 0x0149;
- t['quotesinglbase'] = 0x201A;
- t['quotesingle'] = 0x0027;
- t['quotesinglemonospace'] = 0xFF07;
- t['r'] = 0x0072;
- t['raarmenian'] = 0x057C;
- t['rabengali'] = 0x09B0;
- t['racute'] = 0x0155;
- t['radeva'] = 0x0930;
- t['radical'] = 0x221A;
- t['radicalex'] = 0xF8E5;
- t['radoverssquare'] = 0x33AE;
- t['radoverssquaredsquare'] = 0x33AF;
- t['radsquare'] = 0x33AD;
- t['rafe'] = 0x05BF;
- t['rafehebrew'] = 0x05BF;
- t['ragujarati'] = 0x0AB0;
- t['ragurmukhi'] = 0x0A30;
- t['rahiragana'] = 0x3089;
- t['rakatakana'] = 0x30E9;
- t['rakatakanahalfwidth'] = 0xFF97;
- t['ralowerdiagonalbengali'] = 0x09F1;
- t['ramiddlediagonalbengali'] = 0x09F0;
- t['ramshorn'] = 0x0264;
- t['ratio'] = 0x2236;
- t['rbopomofo'] = 0x3116;
- t['rcaron'] = 0x0159;
- t['rcedilla'] = 0x0157;
- t['rcircle'] = 0x24E1;
- t['rcommaaccent'] = 0x0157;
- t['rdblgrave'] = 0x0211;
- t['rdotaccent'] = 0x1E59;
- t['rdotbelow'] = 0x1E5B;
- t['rdotbelowmacron'] = 0x1E5D;
- t['referencemark'] = 0x203B;
- t['reflexsubset'] = 0x2286;
- t['reflexsuperset'] = 0x2287;
- t['registered'] = 0x00AE;
- t['registersans'] = 0xF8E8;
- t['registerserif'] = 0xF6DA;
- t['reharabic'] = 0x0631;
- t['reharmenian'] = 0x0580;
- t['rehfinalarabic'] = 0xFEAE;
- t['rehiragana'] = 0x308C;
- t['rekatakana'] = 0x30EC;
- t['rekatakanahalfwidth'] = 0xFF9A;
- t['resh'] = 0x05E8;
- t['reshdageshhebrew'] = 0xFB48;
- t['reshhebrew'] = 0x05E8;
- t['reversedtilde'] = 0x223D;
- t['reviahebrew'] = 0x0597;
- t['reviamugrashhebrew'] = 0x0597;
- t['revlogicalnot'] = 0x2310;
- t['rfishhook'] = 0x027E;
- t['rfishhookreversed'] = 0x027F;
- t['rhabengali'] = 0x09DD;
- t['rhadeva'] = 0x095D;
- t['rho'] = 0x03C1;
- t['rhook'] = 0x027D;
- t['rhookturned'] = 0x027B;
- t['rhookturnedsuperior'] = 0x02B5;
- t['rhosymbolgreek'] = 0x03F1;
- t['rhotichookmod'] = 0x02DE;
- t['rieulacirclekorean'] = 0x3271;
- t['rieulaparenkorean'] = 0x3211;
- t['rieulcirclekorean'] = 0x3263;
- t['rieulhieuhkorean'] = 0x3140;
- t['rieulkiyeokkorean'] = 0x313A;
- t['rieulkiyeoksioskorean'] = 0x3169;
- t['rieulkorean'] = 0x3139;
- t['rieulmieumkorean'] = 0x313B;
- t['rieulpansioskorean'] = 0x316C;
- t['rieulparenkorean'] = 0x3203;
- t['rieulphieuphkorean'] = 0x313F;
- t['rieulpieupkorean'] = 0x313C;
- t['rieulpieupsioskorean'] = 0x316B;
- t['rieulsioskorean'] = 0x313D;
- t['rieulthieuthkorean'] = 0x313E;
- t['rieultikeutkorean'] = 0x316A;
- t['rieulyeorinhieuhkorean'] = 0x316D;
- t['rightangle'] = 0x221F;
- t['righttackbelowcmb'] = 0x0319;
- t['righttriangle'] = 0x22BF;
- t['rihiragana'] = 0x308A;
- t['rikatakana'] = 0x30EA;
- t['rikatakanahalfwidth'] = 0xFF98;
- t['ring'] = 0x02DA;
- t['ringbelowcmb'] = 0x0325;
- t['ringcmb'] = 0x030A;
- t['ringhalfleft'] = 0x02BF;
- t['ringhalfleftarmenian'] = 0x0559;
- t['ringhalfleftbelowcmb'] = 0x031C;
- t['ringhalfleftcentered'] = 0x02D3;
- t['ringhalfright'] = 0x02BE;
- t['ringhalfrightbelowcmb'] = 0x0339;
- t['ringhalfrightcentered'] = 0x02D2;
- t['rinvertedbreve'] = 0x0213;
- t['rittorusquare'] = 0x3351;
- t['rlinebelow'] = 0x1E5F;
- t['rlongleg'] = 0x027C;
- t['rlonglegturned'] = 0x027A;
- t['rmonospace'] = 0xFF52;
- t['rohiragana'] = 0x308D;
- t['rokatakana'] = 0x30ED;
- t['rokatakanahalfwidth'] = 0xFF9B;
- t['roruathai'] = 0x0E23;
- t['rparen'] = 0x24AD;
- t['rrabengali'] = 0x09DC;
- t['rradeva'] = 0x0931;
- t['rragurmukhi'] = 0x0A5C;
- t['rreharabic'] = 0x0691;
- t['rrehfinalarabic'] = 0xFB8D;
- t['rrvocalicbengali'] = 0x09E0;
- t['rrvocalicdeva'] = 0x0960;
- t['rrvocalicgujarati'] = 0x0AE0;
- t['rrvocalicvowelsignbengali'] = 0x09C4;
- t['rrvocalicvowelsigndeva'] = 0x0944;
- t['rrvocalicvowelsigngujarati'] = 0x0AC4;
- t['rsuperior'] = 0xF6F1;
- t['rtblock'] = 0x2590;
- t['rturned'] = 0x0279;
- t['rturnedsuperior'] = 0x02B4;
- t['ruhiragana'] = 0x308B;
- t['rukatakana'] = 0x30EB;
- t['rukatakanahalfwidth'] = 0xFF99;
- t['rupeemarkbengali'] = 0x09F2;
- t['rupeesignbengali'] = 0x09F3;
- t['rupiah'] = 0xF6DD;
- t['ruthai'] = 0x0E24;
- t['rvocalicbengali'] = 0x098B;
- t['rvocalicdeva'] = 0x090B;
- t['rvocalicgujarati'] = 0x0A8B;
- t['rvocalicvowelsignbengali'] = 0x09C3;
- t['rvocalicvowelsigndeva'] = 0x0943;
- t['rvocalicvowelsigngujarati'] = 0x0AC3;
- t['s'] = 0x0073;
- t['sabengali'] = 0x09B8;
- t['sacute'] = 0x015B;
- t['sacutedotaccent'] = 0x1E65;
- t['sadarabic'] = 0x0635;
- t['sadeva'] = 0x0938;
- t['sadfinalarabic'] = 0xFEBA;
- t['sadinitialarabic'] = 0xFEBB;
- t['sadmedialarabic'] = 0xFEBC;
- t['sagujarati'] = 0x0AB8;
- t['sagurmukhi'] = 0x0A38;
- t['sahiragana'] = 0x3055;
- t['sakatakana'] = 0x30B5;
- t['sakatakanahalfwidth'] = 0xFF7B;
- t['sallallahoualayhewasallamarabic'] = 0xFDFA;
- t['samekh'] = 0x05E1;
- t['samekhdagesh'] = 0xFB41;
- t['samekhdageshhebrew'] = 0xFB41;
- t['samekhhebrew'] = 0x05E1;
- t['saraaathai'] = 0x0E32;
- t['saraaethai'] = 0x0E41;
- t['saraaimaimalaithai'] = 0x0E44;
- t['saraaimaimuanthai'] = 0x0E43;
- t['saraamthai'] = 0x0E33;
- t['saraathai'] = 0x0E30;
- t['saraethai'] = 0x0E40;
- t['saraiileftthai'] = 0xF886;
- t['saraiithai'] = 0x0E35;
- t['saraileftthai'] = 0xF885;
- t['saraithai'] = 0x0E34;
- t['saraothai'] = 0x0E42;
- t['saraueeleftthai'] = 0xF888;
- t['saraueethai'] = 0x0E37;
- t['saraueleftthai'] = 0xF887;
- t['sarauethai'] = 0x0E36;
- t['sarauthai'] = 0x0E38;
- t['sarauuthai'] = 0x0E39;
- t['sbopomofo'] = 0x3119;
- t['scaron'] = 0x0161;
- t['scarondotaccent'] = 0x1E67;
- t['scedilla'] = 0x015F;
- t['schwa'] = 0x0259;
- t['schwacyrillic'] = 0x04D9;
- t['schwadieresiscyrillic'] = 0x04DB;
- t['schwahook'] = 0x025A;
- t['scircle'] = 0x24E2;
- t['scircumflex'] = 0x015D;
- t['scommaaccent'] = 0x0219;
- t['sdotaccent'] = 0x1E61;
- t['sdotbelow'] = 0x1E63;
- t['sdotbelowdotaccent'] = 0x1E69;
- t['seagullbelowcmb'] = 0x033C;
- t['second'] = 0x2033;
- t['secondtonechinese'] = 0x02CA;
- t['section'] = 0x00A7;
- t['seenarabic'] = 0x0633;
- t['seenfinalarabic'] = 0xFEB2;
- t['seeninitialarabic'] = 0xFEB3;
- t['seenmedialarabic'] = 0xFEB4;
- t['segol'] = 0x05B6;
- t['segol13'] = 0x05B6;
- t['segol1f'] = 0x05B6;
- t['segol2c'] = 0x05B6;
- t['segolhebrew'] = 0x05B6;
- t['segolnarrowhebrew'] = 0x05B6;
- t['segolquarterhebrew'] = 0x05B6;
- t['segoltahebrew'] = 0x0592;
- t['segolwidehebrew'] = 0x05B6;
- t['seharmenian'] = 0x057D;
- t['sehiragana'] = 0x305B;
- t['sekatakana'] = 0x30BB;
- t['sekatakanahalfwidth'] = 0xFF7E;
- t['semicolon'] = 0x003B;
- t['semicolonarabic'] = 0x061B;
- t['semicolonmonospace'] = 0xFF1B;
- t['semicolonsmall'] = 0xFE54;
- t['semivoicedmarkkana'] = 0x309C;
- t['semivoicedmarkkanahalfwidth'] = 0xFF9F;
- t['sentisquare'] = 0x3322;
- t['sentosquare'] = 0x3323;
- t['seven'] = 0x0037;
- t['sevenarabic'] = 0x0667;
- t['sevenbengali'] = 0x09ED;
- t['sevencircle'] = 0x2466;
- t['sevencircleinversesansserif'] = 0x2790;
- t['sevendeva'] = 0x096D;
- t['seveneighths'] = 0x215E;
- t['sevengujarati'] = 0x0AED;
- t['sevengurmukhi'] = 0x0A6D;
- t['sevenhackarabic'] = 0x0667;
- t['sevenhangzhou'] = 0x3027;
- t['sevenideographicparen'] = 0x3226;
- t['seveninferior'] = 0x2087;
- t['sevenmonospace'] = 0xFF17;
- t['sevenoldstyle'] = 0xF737;
- t['sevenparen'] = 0x247A;
- t['sevenperiod'] = 0x248E;
- t['sevenpersian'] = 0x06F7;
- t['sevenroman'] = 0x2176;
- t['sevensuperior'] = 0x2077;
- t['seventeencircle'] = 0x2470;
- t['seventeenparen'] = 0x2484;
- t['seventeenperiod'] = 0x2498;
- t['seventhai'] = 0x0E57;
- t['sfthyphen'] = 0x00AD;
- t['shaarmenian'] = 0x0577;
- t['shabengali'] = 0x09B6;
- t['shacyrillic'] = 0x0448;
- t['shaddaarabic'] = 0x0651;
- t['shaddadammaarabic'] = 0xFC61;
- t['shaddadammatanarabic'] = 0xFC5E;
- t['shaddafathaarabic'] = 0xFC60;
- t['shaddakasraarabic'] = 0xFC62;
- t['shaddakasratanarabic'] = 0xFC5F;
- t['shade'] = 0x2592;
- t['shadedark'] = 0x2593;
- t['shadelight'] = 0x2591;
- t['shademedium'] = 0x2592;
- t['shadeva'] = 0x0936;
- t['shagujarati'] = 0x0AB6;
- t['shagurmukhi'] = 0x0A36;
- t['shalshelethebrew'] = 0x0593;
- t['shbopomofo'] = 0x3115;
- t['shchacyrillic'] = 0x0449;
- t['sheenarabic'] = 0x0634;
- t['sheenfinalarabic'] = 0xFEB6;
- t['sheeninitialarabic'] = 0xFEB7;
- t['sheenmedialarabic'] = 0xFEB8;
- t['sheicoptic'] = 0x03E3;
- t['sheqel'] = 0x20AA;
- t['sheqelhebrew'] = 0x20AA;
- t['sheva'] = 0x05B0;
- t['sheva115'] = 0x05B0;
- t['sheva15'] = 0x05B0;
- t['sheva22'] = 0x05B0;
- t['sheva2e'] = 0x05B0;
- t['shevahebrew'] = 0x05B0;
- t['shevanarrowhebrew'] = 0x05B0;
- t['shevaquarterhebrew'] = 0x05B0;
- t['shevawidehebrew'] = 0x05B0;
- t['shhacyrillic'] = 0x04BB;
- t['shimacoptic'] = 0x03ED;
- t['shin'] = 0x05E9;
- t['shindagesh'] = 0xFB49;
- t['shindageshhebrew'] = 0xFB49;
- t['shindageshshindot'] = 0xFB2C;
- t['shindageshshindothebrew'] = 0xFB2C;
- t['shindageshsindot'] = 0xFB2D;
- t['shindageshsindothebrew'] = 0xFB2D;
- t['shindothebrew'] = 0x05C1;
- t['shinhebrew'] = 0x05E9;
- t['shinshindot'] = 0xFB2A;
- t['shinshindothebrew'] = 0xFB2A;
- t['shinsindot'] = 0xFB2B;
- t['shinsindothebrew'] = 0xFB2B;
- t['shook'] = 0x0282;
- t['sigma'] = 0x03C3;
- t['sigma1'] = 0x03C2;
- t['sigmafinal'] = 0x03C2;
- t['sigmalunatesymbolgreek'] = 0x03F2;
- t['sihiragana'] = 0x3057;
- t['sikatakana'] = 0x30B7;
- t['sikatakanahalfwidth'] = 0xFF7C;
- t['siluqhebrew'] = 0x05BD;
- t['siluqlefthebrew'] = 0x05BD;
- t['similar'] = 0x223C;
- t['sindothebrew'] = 0x05C2;
- t['siosacirclekorean'] = 0x3274;
- t['siosaparenkorean'] = 0x3214;
- t['sioscieuckorean'] = 0x317E;
- t['sioscirclekorean'] = 0x3266;
- t['sioskiyeokkorean'] = 0x317A;
- t['sioskorean'] = 0x3145;
- t['siosnieunkorean'] = 0x317B;
- t['siosparenkorean'] = 0x3206;
- t['siospieupkorean'] = 0x317D;
- t['siostikeutkorean'] = 0x317C;
- t['six'] = 0x0036;
- t['sixarabic'] = 0x0666;
- t['sixbengali'] = 0x09EC;
- t['sixcircle'] = 0x2465;
- t['sixcircleinversesansserif'] = 0x278F;
- t['sixdeva'] = 0x096C;
- t['sixgujarati'] = 0x0AEC;
- t['sixgurmukhi'] = 0x0A6C;
- t['sixhackarabic'] = 0x0666;
- t['sixhangzhou'] = 0x3026;
- t['sixideographicparen'] = 0x3225;
- t['sixinferior'] = 0x2086;
- t['sixmonospace'] = 0xFF16;
- t['sixoldstyle'] = 0xF736;
- t['sixparen'] = 0x2479;
- t['sixperiod'] = 0x248D;
- t['sixpersian'] = 0x06F6;
- t['sixroman'] = 0x2175;
- t['sixsuperior'] = 0x2076;
- t['sixteencircle'] = 0x246F;
- t['sixteencurrencydenominatorbengali'] = 0x09F9;
- t['sixteenparen'] = 0x2483;
- t['sixteenperiod'] = 0x2497;
- t['sixthai'] = 0x0E56;
- t['slash'] = 0x002F;
- t['slashmonospace'] = 0xFF0F;
- t['slong'] = 0x017F;
- t['slongdotaccent'] = 0x1E9B;
- t['smileface'] = 0x263A;
- t['smonospace'] = 0xFF53;
- t['sofpasuqhebrew'] = 0x05C3;
- t['softhyphen'] = 0x00AD;
- t['softsigncyrillic'] = 0x044C;
- t['sohiragana'] = 0x305D;
- t['sokatakana'] = 0x30BD;
- t['sokatakanahalfwidth'] = 0xFF7F;
- t['soliduslongoverlaycmb'] = 0x0338;
- t['solidusshortoverlaycmb'] = 0x0337;
- t['sorusithai'] = 0x0E29;
- t['sosalathai'] = 0x0E28;
- t['sosothai'] = 0x0E0B;
- t['sosuathai'] = 0x0E2A;
- t['space'] = 0x0020;
- t['spacehackarabic'] = 0x0020;
- t['spade'] = 0x2660;
- t['spadesuitblack'] = 0x2660;
- t['spadesuitwhite'] = 0x2664;
- t['sparen'] = 0x24AE;
- t['squarebelowcmb'] = 0x033B;
- t['squarecc'] = 0x33C4;
- t['squarecm'] = 0x339D;
- t['squarediagonalcrosshatchfill'] = 0x25A9;
- t['squarehorizontalfill'] = 0x25A4;
- t['squarekg'] = 0x338F;
- t['squarekm'] = 0x339E;
- t['squarekmcapital'] = 0x33CE;
- t['squareln'] = 0x33D1;
- t['squarelog'] = 0x33D2;
- t['squaremg'] = 0x338E;
- t['squaremil'] = 0x33D5;
- t['squaremm'] = 0x339C;
- t['squaremsquared'] = 0x33A1;
- t['squareorthogonalcrosshatchfill'] = 0x25A6;
- t['squareupperlefttolowerrightfill'] = 0x25A7;
- t['squareupperrighttolowerleftfill'] = 0x25A8;
- t['squareverticalfill'] = 0x25A5;
- t['squarewhitewithsmallblack'] = 0x25A3;
- t['srsquare'] = 0x33DB;
- t['ssabengali'] = 0x09B7;
- t['ssadeva'] = 0x0937;
- t['ssagujarati'] = 0x0AB7;
- t['ssangcieuckorean'] = 0x3149;
- t['ssanghieuhkorean'] = 0x3185;
- t['ssangieungkorean'] = 0x3180;
- t['ssangkiyeokkorean'] = 0x3132;
- t['ssangnieunkorean'] = 0x3165;
- t['ssangpieupkorean'] = 0x3143;
- t['ssangsioskorean'] = 0x3146;
- t['ssangtikeutkorean'] = 0x3138;
- t['ssuperior'] = 0xF6F2;
- t['sterling'] = 0x00A3;
- t['sterlingmonospace'] = 0xFFE1;
- t['strokelongoverlaycmb'] = 0x0336;
- t['strokeshortoverlaycmb'] = 0x0335;
- t['subset'] = 0x2282;
- t['subsetnotequal'] = 0x228A;
- t['subsetorequal'] = 0x2286;
- t['succeeds'] = 0x227B;
- t['suchthat'] = 0x220B;
- t['suhiragana'] = 0x3059;
- t['sukatakana'] = 0x30B9;
- t['sukatakanahalfwidth'] = 0xFF7D;
- t['sukunarabic'] = 0x0652;
- t['summation'] = 0x2211;
- t['sun'] = 0x263C;
- t['superset'] = 0x2283;
- t['supersetnotequal'] = 0x228B;
- t['supersetorequal'] = 0x2287;
- t['svsquare'] = 0x33DC;
- t['syouwaerasquare'] = 0x337C;
- t['t'] = 0x0074;
- t['tabengali'] = 0x09A4;
- t['tackdown'] = 0x22A4;
- t['tackleft'] = 0x22A3;
- t['tadeva'] = 0x0924;
- t['tagujarati'] = 0x0AA4;
- t['tagurmukhi'] = 0x0A24;
- t['taharabic'] = 0x0637;
- t['tahfinalarabic'] = 0xFEC2;
- t['tahinitialarabic'] = 0xFEC3;
- t['tahiragana'] = 0x305F;
- t['tahmedialarabic'] = 0xFEC4;
- t['taisyouerasquare'] = 0x337D;
- t['takatakana'] = 0x30BF;
- t['takatakanahalfwidth'] = 0xFF80;
- t['tatweelarabic'] = 0x0640;
- t['tau'] = 0x03C4;
- t['tav'] = 0x05EA;
- t['tavdages'] = 0xFB4A;
- t['tavdagesh'] = 0xFB4A;
- t['tavdageshhebrew'] = 0xFB4A;
- t['tavhebrew'] = 0x05EA;
- t['tbar'] = 0x0167;
- t['tbopomofo'] = 0x310A;
- t['tcaron'] = 0x0165;
- t['tccurl'] = 0x02A8;
- t['tcedilla'] = 0x0163;
- t['tcheharabic'] = 0x0686;
- t['tchehfinalarabic'] = 0xFB7B;
- t['tchehinitialarabic'] = 0xFB7C;
- t['tchehmedialarabic'] = 0xFB7D;
- t['tcircle'] = 0x24E3;
- t['tcircumflexbelow'] = 0x1E71;
- t['tcommaaccent'] = 0x0163;
- t['tdieresis'] = 0x1E97;
- t['tdotaccent'] = 0x1E6B;
- t['tdotbelow'] = 0x1E6D;
- t['tecyrillic'] = 0x0442;
- t['tedescendercyrillic'] = 0x04AD;
- t['teharabic'] = 0x062A;
- t['tehfinalarabic'] = 0xFE96;
- t['tehhahinitialarabic'] = 0xFCA2;
- t['tehhahisolatedarabic'] = 0xFC0C;
- t['tehinitialarabic'] = 0xFE97;
- t['tehiragana'] = 0x3066;
- t['tehjeeminitialarabic'] = 0xFCA1;
- t['tehjeemisolatedarabic'] = 0xFC0B;
- t['tehmarbutaarabic'] = 0x0629;
- t['tehmarbutafinalarabic'] = 0xFE94;
- t['tehmedialarabic'] = 0xFE98;
- t['tehmeeminitialarabic'] = 0xFCA4;
- t['tehmeemisolatedarabic'] = 0xFC0E;
- t['tehnoonfinalarabic'] = 0xFC73;
- t['tekatakana'] = 0x30C6;
- t['tekatakanahalfwidth'] = 0xFF83;
- t['telephone'] = 0x2121;
- t['telephoneblack'] = 0x260E;
- t['telishagedolahebrew'] = 0x05A0;
- t['telishaqetanahebrew'] = 0x05A9;
- t['tencircle'] = 0x2469;
- t['tenideographicparen'] = 0x3229;
- t['tenparen'] = 0x247D;
- t['tenperiod'] = 0x2491;
- t['tenroman'] = 0x2179;
- t['tesh'] = 0x02A7;
- t['tet'] = 0x05D8;
- t['tetdagesh'] = 0xFB38;
- t['tetdageshhebrew'] = 0xFB38;
- t['tethebrew'] = 0x05D8;
- t['tetsecyrillic'] = 0x04B5;
- t['tevirhebrew'] = 0x059B;
- t['tevirlefthebrew'] = 0x059B;
- t['thabengali'] = 0x09A5;
- t['thadeva'] = 0x0925;
- t['thagujarati'] = 0x0AA5;
- t['thagurmukhi'] = 0x0A25;
- t['thalarabic'] = 0x0630;
- t['thalfinalarabic'] = 0xFEAC;
- t['thanthakhatlowleftthai'] = 0xF898;
- t['thanthakhatlowrightthai'] = 0xF897;
- t['thanthakhatthai'] = 0x0E4C;
- t['thanthakhatupperleftthai'] = 0xF896;
- t['theharabic'] = 0x062B;
- t['thehfinalarabic'] = 0xFE9A;
- t['thehinitialarabic'] = 0xFE9B;
- t['thehmedialarabic'] = 0xFE9C;
- t['thereexists'] = 0x2203;
- t['therefore'] = 0x2234;
- t['theta'] = 0x03B8;
- t['theta1'] = 0x03D1;
- t['thetasymbolgreek'] = 0x03D1;
- t['thieuthacirclekorean'] = 0x3279;
- t['thieuthaparenkorean'] = 0x3219;
- t['thieuthcirclekorean'] = 0x326B;
- t['thieuthkorean'] = 0x314C;
- t['thieuthparenkorean'] = 0x320B;
- t['thirteencircle'] = 0x246C;
- t['thirteenparen'] = 0x2480;
- t['thirteenperiod'] = 0x2494;
- t['thonangmonthothai'] = 0x0E11;
- t['thook'] = 0x01AD;
- t['thophuthaothai'] = 0x0E12;
- t['thorn'] = 0x00FE;
- t['thothahanthai'] = 0x0E17;
- t['thothanthai'] = 0x0E10;
- t['thothongthai'] = 0x0E18;
- t['thothungthai'] = 0x0E16;
- t['thousandcyrillic'] = 0x0482;
- t['thousandsseparatorarabic'] = 0x066C;
- t['thousandsseparatorpersian'] = 0x066C;
- t['three'] = 0x0033;
- t['threearabic'] = 0x0663;
- t['threebengali'] = 0x09E9;
- t['threecircle'] = 0x2462;
- t['threecircleinversesansserif'] = 0x278C;
- t['threedeva'] = 0x0969;
- t['threeeighths'] = 0x215C;
- t['threegujarati'] = 0x0AE9;
- t['threegurmukhi'] = 0x0A69;
- t['threehackarabic'] = 0x0663;
- t['threehangzhou'] = 0x3023;
- t['threeideographicparen'] = 0x3222;
- t['threeinferior'] = 0x2083;
- t['threemonospace'] = 0xFF13;
- t['threenumeratorbengali'] = 0x09F6;
- t['threeoldstyle'] = 0xF733;
- t['threeparen'] = 0x2476;
- t['threeperiod'] = 0x248A;
- t['threepersian'] = 0x06F3;
- t['threequarters'] = 0x00BE;
- t['threequartersemdash'] = 0xF6DE;
- t['threeroman'] = 0x2172;
- t['threesuperior'] = 0x00B3;
- t['threethai'] = 0x0E53;
- t['thzsquare'] = 0x3394;
- t['tihiragana'] = 0x3061;
- t['tikatakana'] = 0x30C1;
- t['tikatakanahalfwidth'] = 0xFF81;
- t['tikeutacirclekorean'] = 0x3270;
- t['tikeutaparenkorean'] = 0x3210;
- t['tikeutcirclekorean'] = 0x3262;
- t['tikeutkorean'] = 0x3137;
- t['tikeutparenkorean'] = 0x3202;
- t['tilde'] = 0x02DC;
- t['tildebelowcmb'] = 0x0330;
- t['tildecmb'] = 0x0303;
- t['tildecomb'] = 0x0303;
- t['tildedoublecmb'] = 0x0360;
- t['tildeoperator'] = 0x223C;
- t['tildeoverlaycmb'] = 0x0334;
- t['tildeverticalcmb'] = 0x033E;
- t['timescircle'] = 0x2297;
- t['tipehahebrew'] = 0x0596;
- t['tipehalefthebrew'] = 0x0596;
- t['tippigurmukhi'] = 0x0A70;
- t['titlocyrilliccmb'] = 0x0483;
- t['tiwnarmenian'] = 0x057F;
- t['tlinebelow'] = 0x1E6F;
- t['tmonospace'] = 0xFF54;
- t['toarmenian'] = 0x0569;
- t['tohiragana'] = 0x3068;
- t['tokatakana'] = 0x30C8;
- t['tokatakanahalfwidth'] = 0xFF84;
- t['tonebarextrahighmod'] = 0x02E5;
- t['tonebarextralowmod'] = 0x02E9;
- t['tonebarhighmod'] = 0x02E6;
- t['tonebarlowmod'] = 0x02E8;
- t['tonebarmidmod'] = 0x02E7;
- t['tonefive'] = 0x01BD;
- t['tonesix'] = 0x0185;
- t['tonetwo'] = 0x01A8;
- t['tonos'] = 0x0384;
- t['tonsquare'] = 0x3327;
- t['topatakthai'] = 0x0E0F;
- t['tortoiseshellbracketleft'] = 0x3014;
- t['tortoiseshellbracketleftsmall'] = 0xFE5D;
- t['tortoiseshellbracketleftvertical'] = 0xFE39;
- t['tortoiseshellbracketright'] = 0x3015;
- t['tortoiseshellbracketrightsmall'] = 0xFE5E;
- t['tortoiseshellbracketrightvertical'] = 0xFE3A;
- t['totaothai'] = 0x0E15;
- t['tpalatalhook'] = 0x01AB;
- t['tparen'] = 0x24AF;
- t['trademark'] = 0x2122;
- t['trademarksans'] = 0xF8EA;
- t['trademarkserif'] = 0xF6DB;
- t['tretroflexhook'] = 0x0288;
- t['triagdn'] = 0x25BC;
- t['triaglf'] = 0x25C4;
- t['triagrt'] = 0x25BA;
- t['triagup'] = 0x25B2;
- t['ts'] = 0x02A6;
- t['tsadi'] = 0x05E6;
- t['tsadidagesh'] = 0xFB46;
- t['tsadidageshhebrew'] = 0xFB46;
- t['tsadihebrew'] = 0x05E6;
- t['tsecyrillic'] = 0x0446;
- t['tsere'] = 0x05B5;
- t['tsere12'] = 0x05B5;
- t['tsere1e'] = 0x05B5;
- t['tsere2b'] = 0x05B5;
- t['tserehebrew'] = 0x05B5;
- t['tserenarrowhebrew'] = 0x05B5;
- t['tserequarterhebrew'] = 0x05B5;
- t['tserewidehebrew'] = 0x05B5;
- t['tshecyrillic'] = 0x045B;
- t['tsuperior'] = 0xF6F3;
- t['ttabengali'] = 0x099F;
- t['ttadeva'] = 0x091F;
- t['ttagujarati'] = 0x0A9F;
- t['ttagurmukhi'] = 0x0A1F;
- t['tteharabic'] = 0x0679;
- t['ttehfinalarabic'] = 0xFB67;
- t['ttehinitialarabic'] = 0xFB68;
- t['ttehmedialarabic'] = 0xFB69;
- t['tthabengali'] = 0x09A0;
- t['tthadeva'] = 0x0920;
- t['tthagujarati'] = 0x0AA0;
- t['tthagurmukhi'] = 0x0A20;
- t['tturned'] = 0x0287;
- t['tuhiragana'] = 0x3064;
- t['tukatakana'] = 0x30C4;
- t['tukatakanahalfwidth'] = 0xFF82;
- t['tusmallhiragana'] = 0x3063;
- t['tusmallkatakana'] = 0x30C3;
- t['tusmallkatakanahalfwidth'] = 0xFF6F;
- t['twelvecircle'] = 0x246B;
- t['twelveparen'] = 0x247F;
- t['twelveperiod'] = 0x2493;
- t['twelveroman'] = 0x217B;
- t['twentycircle'] = 0x2473;
- t['twentyhangzhou'] = 0x5344;
- t['twentyparen'] = 0x2487;
- t['twentyperiod'] = 0x249B;
- t['two'] = 0x0032;
- t['twoarabic'] = 0x0662;
- t['twobengali'] = 0x09E8;
- t['twocircle'] = 0x2461;
- t['twocircleinversesansserif'] = 0x278B;
- t['twodeva'] = 0x0968;
- t['twodotenleader'] = 0x2025;
- t['twodotleader'] = 0x2025;
- t['twodotleadervertical'] = 0xFE30;
- t['twogujarati'] = 0x0AE8;
- t['twogurmukhi'] = 0x0A68;
- t['twohackarabic'] = 0x0662;
- t['twohangzhou'] = 0x3022;
- t['twoideographicparen'] = 0x3221;
- t['twoinferior'] = 0x2082;
- t['twomonospace'] = 0xFF12;
- t['twonumeratorbengali'] = 0x09F5;
- t['twooldstyle'] = 0xF732;
- t['twoparen'] = 0x2475;
- t['twoperiod'] = 0x2489;
- t['twopersian'] = 0x06F2;
- t['tworoman'] = 0x2171;
- t['twostroke'] = 0x01BB;
- t['twosuperior'] = 0x00B2;
- t['twothai'] = 0x0E52;
- t['twothirds'] = 0x2154;
- t['u'] = 0x0075;
- t['uacute'] = 0x00FA;
- t['ubar'] = 0x0289;
- t['ubengali'] = 0x0989;
- t['ubopomofo'] = 0x3128;
- t['ubreve'] = 0x016D;
- t['ucaron'] = 0x01D4;
- t['ucircle'] = 0x24E4;
- t['ucircumflex'] = 0x00FB;
- t['ucircumflexbelow'] = 0x1E77;
- t['ucyrillic'] = 0x0443;
- t['udattadeva'] = 0x0951;
- t['udblacute'] = 0x0171;
- t['udblgrave'] = 0x0215;
- t['udeva'] = 0x0909;
- t['udieresis'] = 0x00FC;
- t['udieresisacute'] = 0x01D8;
- t['udieresisbelow'] = 0x1E73;
- t['udieresiscaron'] = 0x01DA;
- t['udieresiscyrillic'] = 0x04F1;
- t['udieresisgrave'] = 0x01DC;
- t['udieresismacron'] = 0x01D6;
- t['udotbelow'] = 0x1EE5;
- t['ugrave'] = 0x00F9;
- t['ugujarati'] = 0x0A89;
- t['ugurmukhi'] = 0x0A09;
- t['uhiragana'] = 0x3046;
- t['uhookabove'] = 0x1EE7;
- t['uhorn'] = 0x01B0;
- t['uhornacute'] = 0x1EE9;
- t['uhorndotbelow'] = 0x1EF1;
- t['uhorngrave'] = 0x1EEB;
- t['uhornhookabove'] = 0x1EED;
- t['uhorntilde'] = 0x1EEF;
- t['uhungarumlaut'] = 0x0171;
- t['uhungarumlautcyrillic'] = 0x04F3;
- t['uinvertedbreve'] = 0x0217;
- t['ukatakana'] = 0x30A6;
- t['ukatakanahalfwidth'] = 0xFF73;
- t['ukcyrillic'] = 0x0479;
- t['ukorean'] = 0x315C;
- t['umacron'] = 0x016B;
- t['umacroncyrillic'] = 0x04EF;
- t['umacrondieresis'] = 0x1E7B;
- t['umatragurmukhi'] = 0x0A41;
- t['umonospace'] = 0xFF55;
- t['underscore'] = 0x005F;
- t['underscoredbl'] = 0x2017;
- t['underscoremonospace'] = 0xFF3F;
- t['underscorevertical'] = 0xFE33;
- t['underscorewavy'] = 0xFE4F;
- t['union'] = 0x222A;
- t['universal'] = 0x2200;
- t['uogonek'] = 0x0173;
- t['uparen'] = 0x24B0;
- t['upblock'] = 0x2580;
- t['upperdothebrew'] = 0x05C4;
- t['upsilon'] = 0x03C5;
- t['upsilondieresis'] = 0x03CB;
- t['upsilondieresistonos'] = 0x03B0;
- t['upsilonlatin'] = 0x028A;
- t['upsilontonos'] = 0x03CD;
- t['uptackbelowcmb'] = 0x031D;
- t['uptackmod'] = 0x02D4;
- t['uragurmukhi'] = 0x0A73;
- t['uring'] = 0x016F;
- t['ushortcyrillic'] = 0x045E;
- t['usmallhiragana'] = 0x3045;
- t['usmallkatakana'] = 0x30A5;
- t['usmallkatakanahalfwidth'] = 0xFF69;
- t['ustraightcyrillic'] = 0x04AF;
- t['ustraightstrokecyrillic'] = 0x04B1;
- t['utilde'] = 0x0169;
- t['utildeacute'] = 0x1E79;
- t['utildebelow'] = 0x1E75;
- t['uubengali'] = 0x098A;
- t['uudeva'] = 0x090A;
- t['uugujarati'] = 0x0A8A;
- t['uugurmukhi'] = 0x0A0A;
- t['uumatragurmukhi'] = 0x0A42;
- t['uuvowelsignbengali'] = 0x09C2;
- t['uuvowelsigndeva'] = 0x0942;
- t['uuvowelsigngujarati'] = 0x0AC2;
- t['uvowelsignbengali'] = 0x09C1;
- t['uvowelsigndeva'] = 0x0941;
- t['uvowelsigngujarati'] = 0x0AC1;
- t['v'] = 0x0076;
- t['vadeva'] = 0x0935;
- t['vagujarati'] = 0x0AB5;
- t['vagurmukhi'] = 0x0A35;
- t['vakatakana'] = 0x30F7;
- t['vav'] = 0x05D5;
- t['vavdagesh'] = 0xFB35;
- t['vavdagesh65'] = 0xFB35;
- t['vavdageshhebrew'] = 0xFB35;
- t['vavhebrew'] = 0x05D5;
- t['vavholam'] = 0xFB4B;
- t['vavholamhebrew'] = 0xFB4B;
- t['vavvavhebrew'] = 0x05F0;
- t['vavyodhebrew'] = 0x05F1;
- t['vcircle'] = 0x24E5;
- t['vdotbelow'] = 0x1E7F;
- t['vecyrillic'] = 0x0432;
- t['veharabic'] = 0x06A4;
- t['vehfinalarabic'] = 0xFB6B;
- t['vehinitialarabic'] = 0xFB6C;
- t['vehmedialarabic'] = 0xFB6D;
- t['vekatakana'] = 0x30F9;
- t['venus'] = 0x2640;
- t['verticalbar'] = 0x007C;
- t['verticallineabovecmb'] = 0x030D;
- t['verticallinebelowcmb'] = 0x0329;
- t['verticallinelowmod'] = 0x02CC;
- t['verticallinemod'] = 0x02C8;
- t['vewarmenian'] = 0x057E;
- t['vhook'] = 0x028B;
- t['vikatakana'] = 0x30F8;
- t['viramabengali'] = 0x09CD;
- t['viramadeva'] = 0x094D;
- t['viramagujarati'] = 0x0ACD;
- t['visargabengali'] = 0x0983;
- t['visargadeva'] = 0x0903;
- t['visargagujarati'] = 0x0A83;
- t['vmonospace'] = 0xFF56;
- t['voarmenian'] = 0x0578;
- t['voicediterationhiragana'] = 0x309E;
- t['voicediterationkatakana'] = 0x30FE;
- t['voicedmarkkana'] = 0x309B;
- t['voicedmarkkanahalfwidth'] = 0xFF9E;
- t['vokatakana'] = 0x30FA;
- t['vparen'] = 0x24B1;
- t['vtilde'] = 0x1E7D;
- t['vturned'] = 0x028C;
- t['vuhiragana'] = 0x3094;
- t['vukatakana'] = 0x30F4;
- t['w'] = 0x0077;
- t['wacute'] = 0x1E83;
- t['waekorean'] = 0x3159;
- t['wahiragana'] = 0x308F;
- t['wakatakana'] = 0x30EF;
- t['wakatakanahalfwidth'] = 0xFF9C;
- t['wakorean'] = 0x3158;
- t['wasmallhiragana'] = 0x308E;
- t['wasmallkatakana'] = 0x30EE;
- t['wattosquare'] = 0x3357;
- t['wavedash'] = 0x301C;
- t['wavyunderscorevertical'] = 0xFE34;
- t['wawarabic'] = 0x0648;
- t['wawfinalarabic'] = 0xFEEE;
- t['wawhamzaabovearabic'] = 0x0624;
- t['wawhamzaabovefinalarabic'] = 0xFE86;
- t['wbsquare'] = 0x33DD;
- t['wcircle'] = 0x24E6;
- t['wcircumflex'] = 0x0175;
- t['wdieresis'] = 0x1E85;
- t['wdotaccent'] = 0x1E87;
- t['wdotbelow'] = 0x1E89;
- t['wehiragana'] = 0x3091;
- t['weierstrass'] = 0x2118;
- t['wekatakana'] = 0x30F1;
- t['wekorean'] = 0x315E;
- t['weokorean'] = 0x315D;
- t['wgrave'] = 0x1E81;
- t['whitebullet'] = 0x25E6;
- t['whitecircle'] = 0x25CB;
- t['whitecircleinverse'] = 0x25D9;
- t['whitecornerbracketleft'] = 0x300E;
- t['whitecornerbracketleftvertical'] = 0xFE43;
- t['whitecornerbracketright'] = 0x300F;
- t['whitecornerbracketrightvertical'] = 0xFE44;
- t['whitediamond'] = 0x25C7;
- t['whitediamondcontainingblacksmalldiamond'] = 0x25C8;
- t['whitedownpointingsmalltriangle'] = 0x25BF;
- t['whitedownpointingtriangle'] = 0x25BD;
- t['whiteleftpointingsmalltriangle'] = 0x25C3;
- t['whiteleftpointingtriangle'] = 0x25C1;
- t['whitelenticularbracketleft'] = 0x3016;
- t['whitelenticularbracketright'] = 0x3017;
- t['whiterightpointingsmalltriangle'] = 0x25B9;
- t['whiterightpointingtriangle'] = 0x25B7;
- t['whitesmallsquare'] = 0x25AB;
- t['whitesmilingface'] = 0x263A;
- t['whitesquare'] = 0x25A1;
- t['whitestar'] = 0x2606;
- t['whitetelephone'] = 0x260F;
- t['whitetortoiseshellbracketleft'] = 0x3018;
- t['whitetortoiseshellbracketright'] = 0x3019;
- t['whiteuppointingsmalltriangle'] = 0x25B5;
- t['whiteuppointingtriangle'] = 0x25B3;
- t['wihiragana'] = 0x3090;
- t['wikatakana'] = 0x30F0;
- t['wikorean'] = 0x315F;
- t['wmonospace'] = 0xFF57;
- t['wohiragana'] = 0x3092;
- t['wokatakana'] = 0x30F2;
- t['wokatakanahalfwidth'] = 0xFF66;
- t['won'] = 0x20A9;
- t['wonmonospace'] = 0xFFE6;
- t['wowaenthai'] = 0x0E27;
- t['wparen'] = 0x24B2;
- t['wring'] = 0x1E98;
- t['wsuperior'] = 0x02B7;
- t['wturned'] = 0x028D;
- t['wynn'] = 0x01BF;
- t['x'] = 0x0078;
- t['xabovecmb'] = 0x033D;
- t['xbopomofo'] = 0x3112;
- t['xcircle'] = 0x24E7;
- t['xdieresis'] = 0x1E8D;
- t['xdotaccent'] = 0x1E8B;
- t['xeharmenian'] = 0x056D;
- t['xi'] = 0x03BE;
- t['xmonospace'] = 0xFF58;
- t['xparen'] = 0x24B3;
- t['xsuperior'] = 0x02E3;
- t['y'] = 0x0079;
- t['yaadosquare'] = 0x334E;
- t['yabengali'] = 0x09AF;
- t['yacute'] = 0x00FD;
- t['yadeva'] = 0x092F;
- t['yaekorean'] = 0x3152;
- t['yagujarati'] = 0x0AAF;
- t['yagurmukhi'] = 0x0A2F;
- t['yahiragana'] = 0x3084;
- t['yakatakana'] = 0x30E4;
- t['yakatakanahalfwidth'] = 0xFF94;
- t['yakorean'] = 0x3151;
- t['yamakkanthai'] = 0x0E4E;
- t['yasmallhiragana'] = 0x3083;
- t['yasmallkatakana'] = 0x30E3;
- t['yasmallkatakanahalfwidth'] = 0xFF6C;
- t['yatcyrillic'] = 0x0463;
- t['ycircle'] = 0x24E8;
- t['ycircumflex'] = 0x0177;
- t['ydieresis'] = 0x00FF;
- t['ydotaccent'] = 0x1E8F;
- t['ydotbelow'] = 0x1EF5;
- t['yeharabic'] = 0x064A;
- t['yehbarreearabic'] = 0x06D2;
- t['yehbarreefinalarabic'] = 0xFBAF;
- t['yehfinalarabic'] = 0xFEF2;
- t['yehhamzaabovearabic'] = 0x0626;
- t['yehhamzaabovefinalarabic'] = 0xFE8A;
- t['yehhamzaaboveinitialarabic'] = 0xFE8B;
- t['yehhamzaabovemedialarabic'] = 0xFE8C;
- t['yehinitialarabic'] = 0xFEF3;
- t['yehmedialarabic'] = 0xFEF4;
- t['yehmeeminitialarabic'] = 0xFCDD;
- t['yehmeemisolatedarabic'] = 0xFC58;
- t['yehnoonfinalarabic'] = 0xFC94;
- t['yehthreedotsbelowarabic'] = 0x06D1;
- t['yekorean'] = 0x3156;
- t['yen'] = 0x00A5;
- t['yenmonospace'] = 0xFFE5;
- t['yeokorean'] = 0x3155;
- t['yeorinhieuhkorean'] = 0x3186;
- t['yerahbenyomohebrew'] = 0x05AA;
- t['yerahbenyomolefthebrew'] = 0x05AA;
- t['yericyrillic'] = 0x044B;
- t['yerudieresiscyrillic'] = 0x04F9;
- t['yesieungkorean'] = 0x3181;
- t['yesieungpansioskorean'] = 0x3183;
- t['yesieungsioskorean'] = 0x3182;
- t['yetivhebrew'] = 0x059A;
- t['ygrave'] = 0x1EF3;
- t['yhook'] = 0x01B4;
- t['yhookabove'] = 0x1EF7;
- t['yiarmenian'] = 0x0575;
- t['yicyrillic'] = 0x0457;
- t['yikorean'] = 0x3162;
- t['yinyang'] = 0x262F;
- t['yiwnarmenian'] = 0x0582;
- t['ymonospace'] = 0xFF59;
- t['yod'] = 0x05D9;
- t['yoddagesh'] = 0xFB39;
- t['yoddageshhebrew'] = 0xFB39;
- t['yodhebrew'] = 0x05D9;
- t['yodyodhebrew'] = 0x05F2;
- t['yodyodpatahhebrew'] = 0xFB1F;
- t['yohiragana'] = 0x3088;
- t['yoikorean'] = 0x3189;
- t['yokatakana'] = 0x30E8;
- t['yokatakanahalfwidth'] = 0xFF96;
- t['yokorean'] = 0x315B;
- t['yosmallhiragana'] = 0x3087;
- t['yosmallkatakana'] = 0x30E7;
- t['yosmallkatakanahalfwidth'] = 0xFF6E;
- t['yotgreek'] = 0x03F3;
- t['yoyaekorean'] = 0x3188;
- t['yoyakorean'] = 0x3187;
- t['yoyakthai'] = 0x0E22;
- t['yoyingthai'] = 0x0E0D;
- t['yparen'] = 0x24B4;
- t['ypogegrammeni'] = 0x037A;
- t['ypogegrammenigreekcmb'] = 0x0345;
- t['yr'] = 0x01A6;
- t['yring'] = 0x1E99;
- t['ysuperior'] = 0x02B8;
- t['ytilde'] = 0x1EF9;
- t['yturned'] = 0x028E;
- t['yuhiragana'] = 0x3086;
- t['yuikorean'] = 0x318C;
- t['yukatakana'] = 0x30E6;
- t['yukatakanahalfwidth'] = 0xFF95;
- t['yukorean'] = 0x3160;
- t['yusbigcyrillic'] = 0x046B;
- t['yusbigiotifiedcyrillic'] = 0x046D;
- t['yuslittlecyrillic'] = 0x0467;
- t['yuslittleiotifiedcyrillic'] = 0x0469;
- t['yusmallhiragana'] = 0x3085;
- t['yusmallkatakana'] = 0x30E5;
- t['yusmallkatakanahalfwidth'] = 0xFF6D;
- t['yuyekorean'] = 0x318B;
- t['yuyeokorean'] = 0x318A;
- t['yyabengali'] = 0x09DF;
- t['yyadeva'] = 0x095F;
- t['z'] = 0x007A;
- t['zaarmenian'] = 0x0566;
- t['zacute'] = 0x017A;
- t['zadeva'] = 0x095B;
- t['zagurmukhi'] = 0x0A5B;
- t['zaharabic'] = 0x0638;
- t['zahfinalarabic'] = 0xFEC6;
- t['zahinitialarabic'] = 0xFEC7;
- t['zahiragana'] = 0x3056;
- t['zahmedialarabic'] = 0xFEC8;
- t['zainarabic'] = 0x0632;
- t['zainfinalarabic'] = 0xFEB0;
- t['zakatakana'] = 0x30B6;
- t['zaqefgadolhebrew'] = 0x0595;
- t['zaqefqatanhebrew'] = 0x0594;
- t['zarqahebrew'] = 0x0598;
- t['zayin'] = 0x05D6;
- t['zayindagesh'] = 0xFB36;
- t['zayindageshhebrew'] = 0xFB36;
- t['zayinhebrew'] = 0x05D6;
- t['zbopomofo'] = 0x3117;
- t['zcaron'] = 0x017E;
- t['zcircle'] = 0x24E9;
- t['zcircumflex'] = 0x1E91;
- t['zcurl'] = 0x0291;
- t['zdot'] = 0x017C;
- t['zdotaccent'] = 0x017C;
- t['zdotbelow'] = 0x1E93;
- t['zecyrillic'] = 0x0437;
- t['zedescendercyrillic'] = 0x0499;
- t['zedieresiscyrillic'] = 0x04DF;
- t['zehiragana'] = 0x305C;
- t['zekatakana'] = 0x30BC;
- t['zero'] = 0x0030;
- t['zeroarabic'] = 0x0660;
- t['zerobengali'] = 0x09E6;
- t['zerodeva'] = 0x0966;
- t['zerogujarati'] = 0x0AE6;
- t['zerogurmukhi'] = 0x0A66;
- t['zerohackarabic'] = 0x0660;
- t['zeroinferior'] = 0x2080;
- t['zeromonospace'] = 0xFF10;
- t['zerooldstyle'] = 0xF730;
- t['zeropersian'] = 0x06F0;
- t['zerosuperior'] = 0x2070;
- t['zerothai'] = 0x0E50;
- t['zerowidthjoiner'] = 0xFEFF;
- t['zerowidthnonjoiner'] = 0x200C;
- t['zerowidthspace'] = 0x200B;
- t['zeta'] = 0x03B6;
- t['zhbopomofo'] = 0x3113;
- t['zhearmenian'] = 0x056A;
- t['zhebrevecyrillic'] = 0x04C2;
- t['zhecyrillic'] = 0x0436;
- t['zhedescendercyrillic'] = 0x0497;
- t['zhedieresiscyrillic'] = 0x04DD;
- t['zihiragana'] = 0x3058;
- t['zikatakana'] = 0x30B8;
- t['zinorhebrew'] = 0x05AE;
- t['zlinebelow'] = 0x1E95;
- t['zmonospace'] = 0xFF5A;
- t['zohiragana'] = 0x305E;
- t['zokatakana'] = 0x30BE;
- t['zparen'] = 0x24B5;
- t['zretroflexhook'] = 0x0290;
- t['zstroke'] = 0x01B6;
- t['zuhiragana'] = 0x305A;
- t['zukatakana'] = 0x30BA;
- t['.notdef'] = 0x0000;
- t['angbracketleftbig'] = 0x2329;
- t['angbracketleftBig'] = 0x2329;
- t['angbracketleftbigg'] = 0x2329;
- t['angbracketleftBigg'] = 0x2329;
- t['angbracketrightBig'] = 0x232A;
- t['angbracketrightbig'] = 0x232A;
- t['angbracketrightBigg'] = 0x232A;
- t['angbracketrightbigg'] = 0x232A;
- t['arrowhookleft'] = 0x21AA;
- t['arrowhookright'] = 0x21A9;
- t['arrowlefttophalf'] = 0x21BC;
- t['arrowleftbothalf'] = 0x21BD;
- t['arrownortheast'] = 0x2197;
- t['arrownorthwest'] = 0x2196;
- t['arrowrighttophalf'] = 0x21C0;
- t['arrowrightbothalf'] = 0x21C1;
- t['arrowsoutheast'] = 0x2198;
- t['arrowsouthwest'] = 0x2199;
- t['backslashbig'] = 0x2216;
- t['backslashBig'] = 0x2216;
- t['backslashBigg'] = 0x2216;
- t['backslashbigg'] = 0x2216;
- t['bardbl'] = 0x2016;
- t['bracehtipdownleft'] = 0xFE37;
- t['bracehtipdownright'] = 0xFE37;
- t['bracehtipupleft'] = 0xFE38;
- t['bracehtipupright'] = 0xFE38;
- t['braceleftBig'] = 0x007B;
- t['braceleftbig'] = 0x007B;
- t['braceleftbigg'] = 0x007B;
- t['braceleftBigg'] = 0x007B;
- t['bracerightBig'] = 0x007D;
- t['bracerightbig'] = 0x007D;
- t['bracerightbigg'] = 0x007D;
- t['bracerightBigg'] = 0x007D;
- t['bracketleftbig'] = 0x005B;
- t['bracketleftBig'] = 0x005B;
- t['bracketleftbigg'] = 0x005B;
- t['bracketleftBigg'] = 0x005B;
- t['bracketrightBig'] = 0x005D;
- t['bracketrightbig'] = 0x005D;
- t['bracketrightbigg'] = 0x005D;
- t['bracketrightBigg'] = 0x005D;
- t['ceilingleftbig'] = 0x2308;
- t['ceilingleftBig'] = 0x2308;
- t['ceilingleftBigg'] = 0x2308;
- t['ceilingleftbigg'] = 0x2308;
- t['ceilingrightbig'] = 0x2309;
- t['ceilingrightBig'] = 0x2309;
- t['ceilingrightbigg'] = 0x2309;
- t['ceilingrightBigg'] = 0x2309;
- t['circledotdisplay'] = 0x2299;
- t['circledottext'] = 0x2299;
- t['circlemultiplydisplay'] = 0x2297;
- t['circlemultiplytext'] = 0x2297;
- t['circleplusdisplay'] = 0x2295;
- t['circleplustext'] = 0x2295;
- t['contintegraldisplay'] = 0x222E;
- t['contintegraltext'] = 0x222E;
- t['coproductdisplay'] = 0x2210;
- t['coproducttext'] = 0x2210;
- t['floorleftBig'] = 0x230A;
- t['floorleftbig'] = 0x230A;
- t['floorleftbigg'] = 0x230A;
- t['floorleftBigg'] = 0x230A;
- t['floorrightbig'] = 0x230B;
- t['floorrightBig'] = 0x230B;
- t['floorrightBigg'] = 0x230B;
- t['floorrightbigg'] = 0x230B;
- t['hatwide'] = 0x0302;
- t['hatwider'] = 0x0302;
- t['hatwidest'] = 0x0302;
- t['intercal'] = 0x1D40;
- t['integraldisplay'] = 0x222B;
- t['integraltext'] = 0x222B;
- t['intersectiondisplay'] = 0x22C2;
- t['intersectiontext'] = 0x22C2;
- t['logicalanddisplay'] = 0x2227;
- t['logicalandtext'] = 0x2227;
- t['logicalordisplay'] = 0x2228;
- t['logicalortext'] = 0x2228;
- t['parenleftBig'] = 0x0028;
- t['parenleftbig'] = 0x0028;
- t['parenleftBigg'] = 0x0028;
- t['parenleftbigg'] = 0x0028;
- t['parenrightBig'] = 0x0029;
- t['parenrightbig'] = 0x0029;
- t['parenrightBigg'] = 0x0029;
- t['parenrightbigg'] = 0x0029;
- t['prime'] = 0x2032;
- t['productdisplay'] = 0x220F;
- t['producttext'] = 0x220F;
- t['radicalbig'] = 0x221A;
- t['radicalBig'] = 0x221A;
- t['radicalBigg'] = 0x221A;
- t['radicalbigg'] = 0x221A;
- t['radicalbt'] = 0x221A;
- t['radicaltp'] = 0x221A;
- t['radicalvertex'] = 0x221A;
- t['slashbig'] = 0x002F;
- t['slashBig'] = 0x002F;
- t['slashBigg'] = 0x002F;
- t['slashbigg'] = 0x002F;
- t['summationdisplay'] = 0x2211;
- t['summationtext'] = 0x2211;
- t['tildewide'] = 0x02DC;
- t['tildewider'] = 0x02DC;
- t['tildewidest'] = 0x02DC;
- t['uniondisplay'] = 0x22C3;
- t['unionmultidisplay'] = 0x228E;
- t['unionmultitext'] = 0x228E;
- t['unionsqdisplay'] = 0x2294;
- t['unionsqtext'] = 0x2294;
- t['uniontext'] = 0x22C3;
- t['vextenddouble'] = 0x2225;
- t['vextendsingle'] = 0x2223;
+  t['A'] = 0x0041;
+  t['AE'] = 0x00C6;
+  t['AEacute'] = 0x01FC;
+  t['AEmacron'] = 0x01E2;
+  t['AEsmall'] = 0xF7E6;
+  t['Aacute'] = 0x00C1;
+  t['Aacutesmall'] = 0xF7E1;
+  t['Abreve'] = 0x0102;
+  t['Abreveacute'] = 0x1EAE;
+  t['Abrevecyrillic'] = 0x04D0;
+  t['Abrevedotbelow'] = 0x1EB6;
+  t['Abrevegrave'] = 0x1EB0;
+  t['Abrevehookabove'] = 0x1EB2;
+  t['Abrevetilde'] = 0x1EB4;
+  t['Acaron'] = 0x01CD;
+  t['Acircle'] = 0x24B6;
+  t['Acircumflex'] = 0x00C2;
+  t['Acircumflexacute'] = 0x1EA4;
+  t['Acircumflexdotbelow'] = 0x1EAC;
+  t['Acircumflexgrave'] = 0x1EA6;
+  t['Acircumflexhookabove'] = 0x1EA8;
+  t['Acircumflexsmall'] = 0xF7E2;
+  t['Acircumflextilde'] = 0x1EAA;
+  t['Acute'] = 0xF6C9;
+  t['Acutesmall'] = 0xF7B4;
+  t['Acyrillic'] = 0x0410;
+  t['Adblgrave'] = 0x0200;
+  t['Adieresis'] = 0x00C4;
+  t['Adieresiscyrillic'] = 0x04D2;
+  t['Adieresismacron'] = 0x01DE;
+  t['Adieresissmall'] = 0xF7E4;
+  t['Adotbelow'] = 0x1EA0;
+  t['Adotmacron'] = 0x01E0;
+  t['Agrave'] = 0x00C0;
+  t['Agravesmall'] = 0xF7E0;
+  t['Ahookabove'] = 0x1EA2;
+  t['Aiecyrillic'] = 0x04D4;
+  t['Ainvertedbreve'] = 0x0202;
+  t['Alpha'] = 0x0391;
+  t['Alphatonos'] = 0x0386;
+  t['Amacron'] = 0x0100;
+  t['Amonospace'] = 0xFF21;
+  t['Aogonek'] = 0x0104;
+  t['Aring'] = 0x00C5;
+  t['Aringacute'] = 0x01FA;
+  t['Aringbelow'] = 0x1E00;
+  t['Aringsmall'] = 0xF7E5;
+  t['Asmall'] = 0xF761;
+  t['Atilde'] = 0x00C3;
+  t['Atildesmall'] = 0xF7E3;
+  t['Aybarmenian'] = 0x0531;
+  t['B'] = 0x0042;
+  t['Bcircle'] = 0x24B7;
+  t['Bdotaccent'] = 0x1E02;
+  t['Bdotbelow'] = 0x1E04;
+  t['Becyrillic'] = 0x0411;
+  t['Benarmenian'] = 0x0532;
+  t['Beta'] = 0x0392;
+  t['Bhook'] = 0x0181;
+  t['Blinebelow'] = 0x1E06;
+  t['Bmonospace'] = 0xFF22;
+  t['Brevesmall'] = 0xF6F4;
+  t['Bsmall'] = 0xF762;
+  t['Btopbar'] = 0x0182;
+  t['C'] = 0x0043;
+  t['Caarmenian'] = 0x053E;
+  t['Cacute'] = 0x0106;
+  t['Caron'] = 0xF6CA;
+  t['Caronsmall'] = 0xF6F5;
+  t['Ccaron'] = 0x010C;
+  t['Ccedilla'] = 0x00C7;
+  t['Ccedillaacute'] = 0x1E08;
+  t['Ccedillasmall'] = 0xF7E7;
+  t['Ccircle'] = 0x24B8;
+  t['Ccircumflex'] = 0x0108;
+  t['Cdot'] = 0x010A;
+  t['Cdotaccent'] = 0x010A;
+  t['Cedillasmall'] = 0xF7B8;
+  t['Chaarmenian'] = 0x0549;
+  t['Cheabkhasiancyrillic'] = 0x04BC;
+  t['Checyrillic'] = 0x0427;
+  t['Chedescenderabkhasiancyrillic'] = 0x04BE;
+  t['Chedescendercyrillic'] = 0x04B6;
+  t['Chedieresiscyrillic'] = 0x04F4;
+  t['Cheharmenian'] = 0x0543;
+  t['Chekhakassiancyrillic'] = 0x04CB;
+  t['Cheverticalstrokecyrillic'] = 0x04B8;
+  t['Chi'] = 0x03A7;
+  t['Chook'] = 0x0187;
+  t['Circumflexsmall'] = 0xF6F6;
+  t['Cmonospace'] = 0xFF23;
+  t['Coarmenian'] = 0x0551;
+  t['Csmall'] = 0xF763;
+  t['D'] = 0x0044;
+  t['DZ'] = 0x01F1;
+  t['DZcaron'] = 0x01C4;
+  t['Daarmenian'] = 0x0534;
+  t['Dafrican'] = 0x0189;
+  t['Dcaron'] = 0x010E;
+  t['Dcedilla'] = 0x1E10;
+  t['Dcircle'] = 0x24B9;
+  t['Dcircumflexbelow'] = 0x1E12;
+  t['Dcroat'] = 0x0110;
+  t['Ddotaccent'] = 0x1E0A;
+  t['Ddotbelow'] = 0x1E0C;
+  t['Decyrillic'] = 0x0414;
+  t['Deicoptic'] = 0x03EE;
+  t['Delta'] = 0x2206;
+  t['Deltagreek'] = 0x0394;
+  t['Dhook'] = 0x018A;
+  t['Dieresis'] = 0xF6CB;
+  t['DieresisAcute'] = 0xF6CC;
+  t['DieresisGrave'] = 0xF6CD;
+  t['Dieresissmall'] = 0xF7A8;
+  t['Digammagreek'] = 0x03DC;
+  t['Djecyrillic'] = 0x0402;
+  t['Dlinebelow'] = 0x1E0E;
+  t['Dmonospace'] = 0xFF24;
+  t['Dotaccentsmall'] = 0xF6F7;
+  t['Dslash'] = 0x0110;
+  t['Dsmall'] = 0xF764;
+  t['Dtopbar'] = 0x018B;
+  t['Dz'] = 0x01F2;
+  t['Dzcaron'] = 0x01C5;
+  t['Dzeabkhasiancyrillic'] = 0x04E0;
+  t['Dzecyrillic'] = 0x0405;
+  t['Dzhecyrillic'] = 0x040F;
+  t['E'] = 0x0045;
+  t['Eacute'] = 0x00C9;
+  t['Eacutesmall'] = 0xF7E9;
+  t['Ebreve'] = 0x0114;
+  t['Ecaron'] = 0x011A;
+  t['Ecedillabreve'] = 0x1E1C;
+  t['Echarmenian'] = 0x0535;
+  t['Ecircle'] = 0x24BA;
+  t['Ecircumflex'] = 0x00CA;
+  t['Ecircumflexacute'] = 0x1EBE;
+  t['Ecircumflexbelow'] = 0x1E18;
+  t['Ecircumflexdotbelow'] = 0x1EC6;
+  t['Ecircumflexgrave'] = 0x1EC0;
+  t['Ecircumflexhookabove'] = 0x1EC2;
+  t['Ecircumflexsmall'] = 0xF7EA;
+  t['Ecircumflextilde'] = 0x1EC4;
+  t['Ecyrillic'] = 0x0404;
+  t['Edblgrave'] = 0x0204;
+  t['Edieresis'] = 0x00CB;
+  t['Edieresissmall'] = 0xF7EB;
+  t['Edot'] = 0x0116;
+  t['Edotaccent'] = 0x0116;
+  t['Edotbelow'] = 0x1EB8;
+  t['Efcyrillic'] = 0x0424;
+  t['Egrave'] = 0x00C8;
+  t['Egravesmall'] = 0xF7E8;
+  t['Eharmenian'] = 0x0537;
+  t['Ehookabove'] = 0x1EBA;
+  t['Eightroman'] = 0x2167;
+  t['Einvertedbreve'] = 0x0206;
+  t['Eiotifiedcyrillic'] = 0x0464;
+  t['Elcyrillic'] = 0x041B;
+  t['Elevenroman'] = 0x216A;
+  t['Emacron'] = 0x0112;
+  t['Emacronacute'] = 0x1E16;
+  t['Emacrongrave'] = 0x1E14;
+  t['Emcyrillic'] = 0x041C;
+  t['Emonospace'] = 0xFF25;
+  t['Encyrillic'] = 0x041D;
+  t['Endescendercyrillic'] = 0x04A2;
+  t['Eng'] = 0x014A;
+  t['Enghecyrillic'] = 0x04A4;
+  t['Enhookcyrillic'] = 0x04C7;
+  t['Eogonek'] = 0x0118;
+  t['Eopen'] = 0x0190;
+  t['Epsilon'] = 0x0395;
+  t['Epsilontonos'] = 0x0388;
+  t['Ercyrillic'] = 0x0420;
+  t['Ereversed'] = 0x018E;
+  t['Ereversedcyrillic'] = 0x042D;
+  t['Escyrillic'] = 0x0421;
+  t['Esdescendercyrillic'] = 0x04AA;
+  t['Esh'] = 0x01A9;
+  t['Esmall'] = 0xF765;
+  t['Eta'] = 0x0397;
+  t['Etarmenian'] = 0x0538;
+  t['Etatonos'] = 0x0389;
+  t['Eth'] = 0x00D0;
+  t['Ethsmall'] = 0xF7F0;
+  t['Etilde'] = 0x1EBC;
+  t['Etildebelow'] = 0x1E1A;
+  t['Euro'] = 0x20AC;
+  t['Ezh'] = 0x01B7;
+  t['Ezhcaron'] = 0x01EE;
+  t['Ezhreversed'] = 0x01B8;
+  t['F'] = 0x0046;
+  t['Fcircle'] = 0x24BB;
+  t['Fdotaccent'] = 0x1E1E;
+  t['Feharmenian'] = 0x0556;
+  t['Feicoptic'] = 0x03E4;
+  t['Fhook'] = 0x0191;
+  t['Fitacyrillic'] = 0x0472;
+  t['Fiveroman'] = 0x2164;
+  t['Fmonospace'] = 0xFF26;
+  t['Fourroman'] = 0x2163;
+  t['Fsmall'] = 0xF766;
+  t['G'] = 0x0047;
+  t['GBsquare'] = 0x3387;
+  t['Gacute'] = 0x01F4;
+  t['Gamma'] = 0x0393;
+  t['Gammaafrican'] = 0x0194;
+  t['Gangiacoptic'] = 0x03EA;
+  t['Gbreve'] = 0x011E;
+  t['Gcaron'] = 0x01E6;
+  t['Gcedilla'] = 0x0122;
+  t['Gcircle'] = 0x24BC;
+  t['Gcircumflex'] = 0x011C;
+  t['Gcommaaccent'] = 0x0122;
+  t['Gdot'] = 0x0120;
+  t['Gdotaccent'] = 0x0120;
+  t['Gecyrillic'] = 0x0413;
+  t['Ghadarmenian'] = 0x0542;
+  t['Ghemiddlehookcyrillic'] = 0x0494;
+  t['Ghestrokecyrillic'] = 0x0492;
+  t['Gheupturncyrillic'] = 0x0490;
+  t['Ghook'] = 0x0193;
+  t['Gimarmenian'] = 0x0533;
+  t['Gjecyrillic'] = 0x0403;
+  t['Gmacron'] = 0x1E20;
+  t['Gmonospace'] = 0xFF27;
+  t['Grave'] = 0xF6CE;
+  t['Gravesmall'] = 0xF760;
+  t['Gsmall'] = 0xF767;
+  t['Gsmallhook'] = 0x029B;
+  t['Gstroke'] = 0x01E4;
+  t['H'] = 0x0048;
+  t['H18533'] = 0x25CF;
+  t['H18543'] = 0x25AA;
+  t['H18551'] = 0x25AB;
+  t['H22073'] = 0x25A1;
+  t['HPsquare'] = 0x33CB;
+  t['Haabkhasiancyrillic'] = 0x04A8;
+  t['Hadescendercyrillic'] = 0x04B2;
+  t['Hardsigncyrillic'] = 0x042A;
+  t['Hbar'] = 0x0126;
+  t['Hbrevebelow'] = 0x1E2A;
+  t['Hcedilla'] = 0x1E28;
+  t['Hcircle'] = 0x24BD;
+  t['Hcircumflex'] = 0x0124;
+  t['Hdieresis'] = 0x1E26;
+  t['Hdotaccent'] = 0x1E22;
+  t['Hdotbelow'] = 0x1E24;
+  t['Hmonospace'] = 0xFF28;
+  t['Hoarmenian'] = 0x0540;
+  t['Horicoptic'] = 0x03E8;
+  t['Hsmall'] = 0xF768;
+  t['Hungarumlaut'] = 0xF6CF;
+  t['Hungarumlautsmall'] = 0xF6F8;
+  t['Hzsquare'] = 0x3390;
+  t['I'] = 0x0049;
+  t['IAcyrillic'] = 0x042F;
+  t['IJ'] = 0x0132;
+  t['IUcyrillic'] = 0x042E;
+  t['Iacute'] = 0x00CD;
+  t['Iacutesmall'] = 0xF7ED;
+  t['Ibreve'] = 0x012C;
+  t['Icaron'] = 0x01CF;
+  t['Icircle'] = 0x24BE;
+  t['Icircumflex'] = 0x00CE;
+  t['Icircumflexsmall'] = 0xF7EE;
+  t['Icyrillic'] = 0x0406;
+  t['Idblgrave'] = 0x0208;
+  t['Idieresis'] = 0x00CF;
+  t['Idieresisacute'] = 0x1E2E;
+  t['Idieresiscyrillic'] = 0x04E4;
+  t['Idieresissmall'] = 0xF7EF;
+  t['Idot'] = 0x0130;
+  t['Idotaccent'] = 0x0130;
+  t['Idotbelow'] = 0x1ECA;
+  t['Iebrevecyrillic'] = 0x04D6;
+  t['Iecyrillic'] = 0x0415;
+  t['Ifraktur'] = 0x2111;
+  t['Igrave'] = 0x00CC;
+  t['Igravesmall'] = 0xF7EC;
+  t['Ihookabove'] = 0x1EC8;
+  t['Iicyrillic'] = 0x0418;
+  t['Iinvertedbreve'] = 0x020A;
+  t['Iishortcyrillic'] = 0x0419;
+  t['Imacron'] = 0x012A;
+  t['Imacroncyrillic'] = 0x04E2;
+  t['Imonospace'] = 0xFF29;
+  t['Iniarmenian'] = 0x053B;
+  t['Iocyrillic'] = 0x0401;
+  t['Iogonek'] = 0x012E;
+  t['Iota'] = 0x0399;
+  t['Iotaafrican'] = 0x0196;
+  t['Iotadieresis'] = 0x03AA;
+  t['Iotatonos'] = 0x038A;
+  t['Ismall'] = 0xF769;
+  t['Istroke'] = 0x0197;
+  t['Itilde'] = 0x0128;
+  t['Itildebelow'] = 0x1E2C;
+  t['Izhitsacyrillic'] = 0x0474;
+  t['Izhitsadblgravecyrillic'] = 0x0476;
+  t['J'] = 0x004A;
+  t['Jaarmenian'] = 0x0541;
+  t['Jcircle'] = 0x24BF;
+  t['Jcircumflex'] = 0x0134;
+  t['Jecyrillic'] = 0x0408;
+  t['Jheharmenian'] = 0x054B;
+  t['Jmonospace'] = 0xFF2A;
+  t['Jsmall'] = 0xF76A;
+  t['K'] = 0x004B;
+  t['KBsquare'] = 0x3385;
+  t['KKsquare'] = 0x33CD;
+  t['Kabashkircyrillic'] = 0x04A0;
+  t['Kacute'] = 0x1E30;
+  t['Kacyrillic'] = 0x041A;
+  t['Kadescendercyrillic'] = 0x049A;
+  t['Kahookcyrillic'] = 0x04C3;
+  t['Kappa'] = 0x039A;
+  t['Kastrokecyrillic'] = 0x049E;
+  t['Kaverticalstrokecyrillic'] = 0x049C;
+  t['Kcaron'] = 0x01E8;
+  t['Kcedilla'] = 0x0136;
+  t['Kcircle'] = 0x24C0;
+  t['Kcommaaccent'] = 0x0136;
+  t['Kdotbelow'] = 0x1E32;
+  t['Keharmenian'] = 0x0554;
+  t['Kenarmenian'] = 0x053F;
+  t['Khacyrillic'] = 0x0425;
+  t['Kheicoptic'] = 0x03E6;
+  t['Khook'] = 0x0198;
+  t['Kjecyrillic'] = 0x040C;
+  t['Klinebelow'] = 0x1E34;
+  t['Kmonospace'] = 0xFF2B;
+  t['Koppacyrillic'] = 0x0480;
+  t['Koppagreek'] = 0x03DE;
+  t['Ksicyrillic'] = 0x046E;
+  t['Ksmall'] = 0xF76B;
+  t['L'] = 0x004C;
+  t['LJ'] = 0x01C7;
+  t['LL'] = 0xF6BF;
+  t['Lacute'] = 0x0139;
+  t['Lambda'] = 0x039B;
+  t['Lcaron'] = 0x013D;
+  t['Lcedilla'] = 0x013B;
+  t['Lcircle'] = 0x24C1;
+  t['Lcircumflexbelow'] = 0x1E3C;
+  t['Lcommaaccent'] = 0x013B;
+  t['Ldot'] = 0x013F;
+  t['Ldotaccent'] = 0x013F;
+  t['Ldotbelow'] = 0x1E36;
+  t['Ldotbelowmacron'] = 0x1E38;
+  t['Liwnarmenian'] = 0x053C;
+  t['Lj'] = 0x01C8;
+  t['Ljecyrillic'] = 0x0409;
+  t['Llinebelow'] = 0x1E3A;
+  t['Lmonospace'] = 0xFF2C;
+  t['Lslash'] = 0x0141;
+  t['Lslashsmall'] = 0xF6F9;
+  t['Lsmall'] = 0xF76C;
+  t['M'] = 0x004D;
+  t['MBsquare'] = 0x3386;
+  t['Macron'] = 0xF6D0;
+  t['Macronsmall'] = 0xF7AF;
+  t['Macute'] = 0x1E3E;
+  t['Mcircle'] = 0x24C2;
+  t['Mdotaccent'] = 0x1E40;
+  t['Mdotbelow'] = 0x1E42;
+  t['Menarmenian'] = 0x0544;
+  t['Mmonospace'] = 0xFF2D;
+  t['Msmall'] = 0xF76D;
+  t['Mturned'] = 0x019C;
+  t['Mu'] = 0x039C;
+  t['N'] = 0x004E;
+  t['NJ'] = 0x01CA;
+  t['Nacute'] = 0x0143;
+  t['Ncaron'] = 0x0147;
+  t['Ncedilla'] = 0x0145;
+  t['Ncircle'] = 0x24C3;
+  t['Ncircumflexbelow'] = 0x1E4A;
+  t['Ncommaaccent'] = 0x0145;
+  t['Ndotaccent'] = 0x1E44;
+  t['Ndotbelow'] = 0x1E46;
+  t['Nhookleft'] = 0x019D;
+  t['Nineroman'] = 0x2168;
+  t['Nj'] = 0x01CB;
+  t['Njecyrillic'] = 0x040A;
+  t['Nlinebelow'] = 0x1E48;
+  t['Nmonospace'] = 0xFF2E;
+  t['Nowarmenian'] = 0x0546;
+  t['Nsmall'] = 0xF76E;
+  t['Ntilde'] = 0x00D1;
+  t['Ntildesmall'] = 0xF7F1;
+  t['Nu'] = 0x039D;
+  t['O'] = 0x004F;
+  t['OE'] = 0x0152;
+  t['OEsmall'] = 0xF6FA;
+  t['Oacute'] = 0x00D3;
+  t['Oacutesmall'] = 0xF7F3;
+  t['Obarredcyrillic'] = 0x04E8;
+  t['Obarreddieresiscyrillic'] = 0x04EA;
+  t['Obreve'] = 0x014E;
+  t['Ocaron'] = 0x01D1;
+  t['Ocenteredtilde'] = 0x019F;
+  t['Ocircle'] = 0x24C4;
+  t['Ocircumflex'] = 0x00D4;
+  t['Ocircumflexacute'] = 0x1ED0;
+  t['Ocircumflexdotbelow'] = 0x1ED8;
+  t['Ocircumflexgrave'] = 0x1ED2;
+  t['Ocircumflexhookabove'] = 0x1ED4;
+  t['Ocircumflexsmall'] = 0xF7F4;
+  t['Ocircumflextilde'] = 0x1ED6;
+  t['Ocyrillic'] = 0x041E;
+  t['Odblacute'] = 0x0150;
+  t['Odblgrave'] = 0x020C;
+  t['Odieresis'] = 0x00D6;
+  t['Odieresiscyrillic'] = 0x04E6;
+  t['Odieresissmall'] = 0xF7F6;
+  t['Odotbelow'] = 0x1ECC;
+  t['Ogoneksmall'] = 0xF6FB;
+  t['Ograve'] = 0x00D2;
+  t['Ogravesmall'] = 0xF7F2;
+  t['Oharmenian'] = 0x0555;
+  t['Ohm'] = 0x2126;
+  t['Ohookabove'] = 0x1ECE;
+  t['Ohorn'] = 0x01A0;
+  t['Ohornacute'] = 0x1EDA;
+  t['Ohorndotbelow'] = 0x1EE2;
+  t['Ohorngrave'] = 0x1EDC;
+  t['Ohornhookabove'] = 0x1EDE;
+  t['Ohorntilde'] = 0x1EE0;
+  t['Ohungarumlaut'] = 0x0150;
+  t['Oi'] = 0x01A2;
+  t['Oinvertedbreve'] = 0x020E;
+  t['Omacron'] = 0x014C;
+  t['Omacronacute'] = 0x1E52;
+  t['Omacrongrave'] = 0x1E50;
+  t['Omega'] = 0x2126;
+  t['Omegacyrillic'] = 0x0460;
+  t['Omegagreek'] = 0x03A9;
+  t['Omegaroundcyrillic'] = 0x047A;
+  t['Omegatitlocyrillic'] = 0x047C;
+  t['Omegatonos'] = 0x038F;
+  t['Omicron'] = 0x039F;
+  t['Omicrontonos'] = 0x038C;
+  t['Omonospace'] = 0xFF2F;
+  t['Oneroman'] = 0x2160;
+  t['Oogonek'] = 0x01EA;
+  t['Oogonekmacron'] = 0x01EC;
+  t['Oopen'] = 0x0186;
+  t['Oslash'] = 0x00D8;
+  t['Oslashacute'] = 0x01FE;
+  t['Oslashsmall'] = 0xF7F8;
+  t['Osmall'] = 0xF76F;
+  t['Ostrokeacute'] = 0x01FE;
+  t['Otcyrillic'] = 0x047E;
+  t['Otilde'] = 0x00D5;
+  t['Otildeacute'] = 0x1E4C;
+  t['Otildedieresis'] = 0x1E4E;
+  t['Otildesmall'] = 0xF7F5;
+  t['P'] = 0x0050;
+  t['Pacute'] = 0x1E54;
+  t['Pcircle'] = 0x24C5;
+  t['Pdotaccent'] = 0x1E56;
+  t['Pecyrillic'] = 0x041F;
+  t['Peharmenian'] = 0x054A;
+  t['Pemiddlehookcyrillic'] = 0x04A6;
+  t['Phi'] = 0x03A6;
+  t['Phook'] = 0x01A4;
+  t['Pi'] = 0x03A0;
+  t['Piwrarmenian'] = 0x0553;
+  t['Pmonospace'] = 0xFF30;
+  t['Psi'] = 0x03A8;
+  t['Psicyrillic'] = 0x0470;
+  t['Psmall'] = 0xF770;
+  t['Q'] = 0x0051;
+  t['Qcircle'] = 0x24C6;
+  t['Qmonospace'] = 0xFF31;
+  t['Qsmall'] = 0xF771;
+  t['R'] = 0x0052;
+  t['Raarmenian'] = 0x054C;
+  t['Racute'] = 0x0154;
+  t['Rcaron'] = 0x0158;
+  t['Rcedilla'] = 0x0156;
+  t['Rcircle'] = 0x24C7;
+  t['Rcommaaccent'] = 0x0156;
+  t['Rdblgrave'] = 0x0210;
+  t['Rdotaccent'] = 0x1E58;
+  t['Rdotbelow'] = 0x1E5A;
+  t['Rdotbelowmacron'] = 0x1E5C;
+  t['Reharmenian'] = 0x0550;
+  t['Rfraktur'] = 0x211C;
+  t['Rho'] = 0x03A1;
+  t['Ringsmall'] = 0xF6FC;
+  t['Rinvertedbreve'] = 0x0212;
+  t['Rlinebelow'] = 0x1E5E;
+  t['Rmonospace'] = 0xFF32;
+  t['Rsmall'] = 0xF772;
+  t['Rsmallinverted'] = 0x0281;
+  t['Rsmallinvertedsuperior'] = 0x02B6;
+  t['S'] = 0x0053;
+  t['SF010000'] = 0x250C;
+  t['SF020000'] = 0x2514;
+  t['SF030000'] = 0x2510;
+  t['SF040000'] = 0x2518;
+  t['SF050000'] = 0x253C;
+  t['SF060000'] = 0x252C;
+  t['SF070000'] = 0x2534;
+  t['SF080000'] = 0x251C;
+  t['SF090000'] = 0x2524;
+  t['SF100000'] = 0x2500;
+  t['SF110000'] = 0x2502;
+  t['SF190000'] = 0x2561;
+  t['SF200000'] = 0x2562;
+  t['SF210000'] = 0x2556;
+  t['SF220000'] = 0x2555;
+  t['SF230000'] = 0x2563;
+  t['SF240000'] = 0x2551;
+  t['SF250000'] = 0x2557;
+  t['SF260000'] = 0x255D;
+  t['SF270000'] = 0x255C;
+  t['SF280000'] = 0x255B;
+  t['SF360000'] = 0x255E;
+  t['SF370000'] = 0x255F;
+  t['SF380000'] = 0x255A;
+  t['SF390000'] = 0x2554;
+  t['SF400000'] = 0x2569;
+  t['SF410000'] = 0x2566;
+  t['SF420000'] = 0x2560;
+  t['SF430000'] = 0x2550;
+  t['SF440000'] = 0x256C;
+  t['SF450000'] = 0x2567;
+  t['SF460000'] = 0x2568;
+  t['SF470000'] = 0x2564;
+  t['SF480000'] = 0x2565;
+  t['SF490000'] = 0x2559;
+  t['SF500000'] = 0x2558;
+  t['SF510000'] = 0x2552;
+  t['SF520000'] = 0x2553;
+  t['SF530000'] = 0x256B;
+  t['SF540000'] = 0x256A;
+  t['Sacute'] = 0x015A;
+  t['Sacutedotaccent'] = 0x1E64;
+  t['Sampigreek'] = 0x03E0;
+  t['Scaron'] = 0x0160;
+  t['Scarondotaccent'] = 0x1E66;
+  t['Scaronsmall'] = 0xF6FD;
+  t['Scedilla'] = 0x015E;
+  t['Schwa'] = 0x018F;
+  t['Schwacyrillic'] = 0x04D8;
+  t['Schwadieresiscyrillic'] = 0x04DA;
+  t['Scircle'] = 0x24C8;
+  t['Scircumflex'] = 0x015C;
+  t['Scommaaccent'] = 0x0218;
+  t['Sdotaccent'] = 0x1E60;
+  t['Sdotbelow'] = 0x1E62;
+  t['Sdotbelowdotaccent'] = 0x1E68;
+  t['Seharmenian'] = 0x054D;
+  t['Sevenroman'] = 0x2166;
+  t['Shaarmenian'] = 0x0547;
+  t['Shacyrillic'] = 0x0428;
+  t['Shchacyrillic'] = 0x0429;
+  t['Sheicoptic'] = 0x03E2;
+  t['Shhacyrillic'] = 0x04BA;
+  t['Shimacoptic'] = 0x03EC;
+  t['Sigma'] = 0x03A3;
+  t['Sixroman'] = 0x2165;
+  t['Smonospace'] = 0xFF33;
+  t['Softsigncyrillic'] = 0x042C;
+  t['Ssmall'] = 0xF773;
+  t['Stigmagreek'] = 0x03DA;
+  t['T'] = 0x0054;
+  t['Tau'] = 0x03A4;
+  t['Tbar'] = 0x0166;
+  t['Tcaron'] = 0x0164;
+  t['Tcedilla'] = 0x0162;
+  t['Tcircle'] = 0x24C9;
+  t['Tcircumflexbelow'] = 0x1E70;
+  t['Tcommaaccent'] = 0x0162;
+  t['Tdotaccent'] = 0x1E6A;
+  t['Tdotbelow'] = 0x1E6C;
+  t['Tecyrillic'] = 0x0422;
+  t['Tedescendercyrillic'] = 0x04AC;
+  t['Tenroman'] = 0x2169;
+  t['Tetsecyrillic'] = 0x04B4;
+  t['Theta'] = 0x0398;
+  t['Thook'] = 0x01AC;
+  t['Thorn'] = 0x00DE;
+  t['Thornsmall'] = 0xF7FE;
+  t['Threeroman'] = 0x2162;
+  t['Tildesmall'] = 0xF6FE;
+  t['Tiwnarmenian'] = 0x054F;
+  t['Tlinebelow'] = 0x1E6E;
+  t['Tmonospace'] = 0xFF34;
+  t['Toarmenian'] = 0x0539;
+  t['Tonefive'] = 0x01BC;
+  t['Tonesix'] = 0x0184;
+  t['Tonetwo'] = 0x01A7;
+  t['Tretroflexhook'] = 0x01AE;
+  t['Tsecyrillic'] = 0x0426;
+  t['Tshecyrillic'] = 0x040B;
+  t['Tsmall'] = 0xF774;
+  t['Twelveroman'] = 0x216B;
+  t['Tworoman'] = 0x2161;
+  t['U'] = 0x0055;
+  t['Uacute'] = 0x00DA;
+  t['Uacutesmall'] = 0xF7FA;
+  t['Ubreve'] = 0x016C;
+  t['Ucaron'] = 0x01D3;
+  t['Ucircle'] = 0x24CA;
+  t['Ucircumflex'] = 0x00DB;
+  t['Ucircumflexbelow'] = 0x1E76;
+  t['Ucircumflexsmall'] = 0xF7FB;
+  t['Ucyrillic'] = 0x0423;
+  t['Udblacute'] = 0x0170;
+  t['Udblgrave'] = 0x0214;
+  t['Udieresis'] = 0x00DC;
+  t['Udieresisacute'] = 0x01D7;
+  t['Udieresisbelow'] = 0x1E72;
+  t['Udieresiscaron'] = 0x01D9;
+  t['Udieresiscyrillic'] = 0x04F0;
+  t['Udieresisgrave'] = 0x01DB;
+  t['Udieresismacron'] = 0x01D5;
+  t['Udieresissmall'] = 0xF7FC;
+  t['Udotbelow'] = 0x1EE4;
+  t['Ugrave'] = 0x00D9;
+  t['Ugravesmall'] = 0xF7F9;
+  t['Uhookabove'] = 0x1EE6;
+  t['Uhorn'] = 0x01AF;
+  t['Uhornacute'] = 0x1EE8;
+  t['Uhorndotbelow'] = 0x1EF0;
+  t['Uhorngrave'] = 0x1EEA;
+  t['Uhornhookabove'] = 0x1EEC;
+  t['Uhorntilde'] = 0x1EEE;
+  t['Uhungarumlaut'] = 0x0170;
+  t['Uhungarumlautcyrillic'] = 0x04F2;
+  t['Uinvertedbreve'] = 0x0216;
+  t['Ukcyrillic'] = 0x0478;
+  t['Umacron'] = 0x016A;
+  t['Umacroncyrillic'] = 0x04EE;
+  t['Umacrondieresis'] = 0x1E7A;
+  t['Umonospace'] = 0xFF35;
+  t['Uogonek'] = 0x0172;
+  t['Upsilon'] = 0x03A5;
+  t['Upsilon1'] = 0x03D2;
+  t['Upsilonacutehooksymbolgreek'] = 0x03D3;
+  t['Upsilonafrican'] = 0x01B1;
+  t['Upsilondieresis'] = 0x03AB;
+  t['Upsilondieresishooksymbolgreek'] = 0x03D4;
+  t['Upsilonhooksymbol'] = 0x03D2;
+  t['Upsilontonos'] = 0x038E;
+  t['Uring'] = 0x016E;
+  t['Ushortcyrillic'] = 0x040E;
+  t['Usmall'] = 0xF775;
+  t['Ustraightcyrillic'] = 0x04AE;
+  t['Ustraightstrokecyrillic'] = 0x04B0;
+  t['Utilde'] = 0x0168;
+  t['Utildeacute'] = 0x1E78;
+  t['Utildebelow'] = 0x1E74;
+  t['V'] = 0x0056;
+  t['Vcircle'] = 0x24CB;
+  t['Vdotbelow'] = 0x1E7E;
+  t['Vecyrillic'] = 0x0412;
+  t['Vewarmenian'] = 0x054E;
+  t['Vhook'] = 0x01B2;
+  t['Vmonospace'] = 0xFF36;
+  t['Voarmenian'] = 0x0548;
+  t['Vsmall'] = 0xF776;
+  t['Vtilde'] = 0x1E7C;
+  t['W'] = 0x0057;
+  t['Wacute'] = 0x1E82;
+  t['Wcircle'] = 0x24CC;
+  t['Wcircumflex'] = 0x0174;
+  t['Wdieresis'] = 0x1E84;
+  t['Wdotaccent'] = 0x1E86;
+  t['Wdotbelow'] = 0x1E88;
+  t['Wgrave'] = 0x1E80;
+  t['Wmonospace'] = 0xFF37;
+  t['Wsmall'] = 0xF777;
+  t['X'] = 0x0058;
+  t['Xcircle'] = 0x24CD;
+  t['Xdieresis'] = 0x1E8C;
+  t['Xdotaccent'] = 0x1E8A;
+  t['Xeharmenian'] = 0x053D;
+  t['Xi'] = 0x039E;
+  t['Xmonospace'] = 0xFF38;
+  t['Xsmall'] = 0xF778;
+  t['Y'] = 0x0059;
+  t['Yacute'] = 0x00DD;
+  t['Yacutesmall'] = 0xF7FD;
+  t['Yatcyrillic'] = 0x0462;
+  t['Ycircle'] = 0x24CE;
+  t['Ycircumflex'] = 0x0176;
+  t['Ydieresis'] = 0x0178;
+  t['Ydieresissmall'] = 0xF7FF;
+  t['Ydotaccent'] = 0x1E8E;
+  t['Ydotbelow'] = 0x1EF4;
+  t['Yericyrillic'] = 0x042B;
+  t['Yerudieresiscyrillic'] = 0x04F8;
+  t['Ygrave'] = 0x1EF2;
+  t['Yhook'] = 0x01B3;
+  t['Yhookabove'] = 0x1EF6;
+  t['Yiarmenian'] = 0x0545;
+  t['Yicyrillic'] = 0x0407;
+  t['Yiwnarmenian'] = 0x0552;
+  t['Ymonospace'] = 0xFF39;
+  t['Ysmall'] = 0xF779;
+  t['Ytilde'] = 0x1EF8;
+  t['Yusbigcyrillic'] = 0x046A;
+  t['Yusbigiotifiedcyrillic'] = 0x046C;
+  t['Yuslittlecyrillic'] = 0x0466;
+  t['Yuslittleiotifiedcyrillic'] = 0x0468;
+  t['Z'] = 0x005A;
+  t['Zaarmenian'] = 0x0536;
+  t['Zacute'] = 0x0179;
+  t['Zcaron'] = 0x017D;
+  t['Zcaronsmall'] = 0xF6FF;
+  t['Zcircle'] = 0x24CF;
+  t['Zcircumflex'] = 0x1E90;
+  t['Zdot'] = 0x017B;
+  t['Zdotaccent'] = 0x017B;
+  t['Zdotbelow'] = 0x1E92;
+  t['Zecyrillic'] = 0x0417;
+  t['Zedescendercyrillic'] = 0x0498;
+  t['Zedieresiscyrillic'] = 0x04DE;
+  t['Zeta'] = 0x0396;
+  t['Zhearmenian'] = 0x053A;
+  t['Zhebrevecyrillic'] = 0x04C1;
+  t['Zhecyrillic'] = 0x0416;
+  t['Zhedescendercyrillic'] = 0x0496;
+  t['Zhedieresiscyrillic'] = 0x04DC;
+  t['Zlinebelow'] = 0x1E94;
+  t['Zmonospace'] = 0xFF3A;
+  t['Zsmall'] = 0xF77A;
+  t['Zstroke'] = 0x01B5;
+  t['a'] = 0x0061;
+  t['aabengali'] = 0x0986;
+  t['aacute'] = 0x00E1;
+  t['aadeva'] = 0x0906;
+  t['aagujarati'] = 0x0A86;
+  t['aagurmukhi'] = 0x0A06;
+  t['aamatragurmukhi'] = 0x0A3E;
+  t['aarusquare'] = 0x3303;
+  t['aavowelsignbengali'] = 0x09BE;
+  t['aavowelsigndeva'] = 0x093E;
+  t['aavowelsigngujarati'] = 0x0ABE;
+  t['abbreviationmarkarmenian'] = 0x055F;
+  t['abbreviationsigndeva'] = 0x0970;
+  t['abengali'] = 0x0985;
+  t['abopomofo'] = 0x311A;
+  t['abreve'] = 0x0103;
+  t['abreveacute'] = 0x1EAF;
+  t['abrevecyrillic'] = 0x04D1;
+  t['abrevedotbelow'] = 0x1EB7;
+  t['abrevegrave'] = 0x1EB1;
+  t['abrevehookabove'] = 0x1EB3;
+  t['abrevetilde'] = 0x1EB5;
+  t['acaron'] = 0x01CE;
+  t['acircle'] = 0x24D0;
+  t['acircumflex'] = 0x00E2;
+  t['acircumflexacute'] = 0x1EA5;
+  t['acircumflexdotbelow'] = 0x1EAD;
+  t['acircumflexgrave'] = 0x1EA7;
+  t['acircumflexhookabove'] = 0x1EA9;
+  t['acircumflextilde'] = 0x1EAB;
+  t['acute'] = 0x00B4;
+  t['acutebelowcmb'] = 0x0317;
+  t['acutecmb'] = 0x0301;
+  t['acutecomb'] = 0x0301;
+  t['acutedeva'] = 0x0954;
+  t['acutelowmod'] = 0x02CF;
+  t['acutetonecmb'] = 0x0341;
+  t['acyrillic'] = 0x0430;
+  t['adblgrave'] = 0x0201;
+  t['addakgurmukhi'] = 0x0A71;
+  t['adeva'] = 0x0905;
+  t['adieresis'] = 0x00E4;
+  t['adieresiscyrillic'] = 0x04D3;
+  t['adieresismacron'] = 0x01DF;
+  t['adotbelow'] = 0x1EA1;
+  t['adotmacron'] = 0x01E1;
+  t['ae'] = 0x00E6;
+  t['aeacute'] = 0x01FD;
+  t['aekorean'] = 0x3150;
+  t['aemacron'] = 0x01E3;
+  t['afii00208'] = 0x2015;
+  t['afii08941'] = 0x20A4;
+  t['afii10017'] = 0x0410;
+  t['afii10018'] = 0x0411;
+  t['afii10019'] = 0x0412;
+  t['afii10020'] = 0x0413;
+  t['afii10021'] = 0x0414;
+  t['afii10022'] = 0x0415;
+  t['afii10023'] = 0x0401;
+  t['afii10024'] = 0x0416;
+  t['afii10025'] = 0x0417;
+  t['afii10026'] = 0x0418;
+  t['afii10027'] = 0x0419;
+  t['afii10028'] = 0x041A;
+  t['afii10029'] = 0x041B;
+  t['afii10030'] = 0x041C;
+  t['afii10031'] = 0x041D;
+  t['afii10032'] = 0x041E;
+  t['afii10033'] = 0x041F;
+  t['afii10034'] = 0x0420;
+  t['afii10035'] = 0x0421;
+  t['afii10036'] = 0x0422;
+  t['afii10037'] = 0x0423;
+  t['afii10038'] = 0x0424;
+  t['afii10039'] = 0x0425;
+  t['afii10040'] = 0x0426;
+  t['afii10041'] = 0x0427;
+  t['afii10042'] = 0x0428;
+  t['afii10043'] = 0x0429;
+  t['afii10044'] = 0x042A;
+  t['afii10045'] = 0x042B;
+  t['afii10046'] = 0x042C;
+  t['afii10047'] = 0x042D;
+  t['afii10048'] = 0x042E;
+  t['afii10049'] = 0x042F;
+  t['afii10050'] = 0x0490;
+  t['afii10051'] = 0x0402;
+  t['afii10052'] = 0x0403;
+  t['afii10053'] = 0x0404;
+  t['afii10054'] = 0x0405;
+  t['afii10055'] = 0x0406;
+  t['afii10056'] = 0x0407;
+  t['afii10057'] = 0x0408;
+  t['afii10058'] = 0x0409;
+  t['afii10059'] = 0x040A;
+  t['afii10060'] = 0x040B;
+  t['afii10061'] = 0x040C;
+  t['afii10062'] = 0x040E;
+  t['afii10063'] = 0xF6C4;
+  t['afii10064'] = 0xF6C5;
+  t['afii10065'] = 0x0430;
+  t['afii10066'] = 0x0431;
+  t['afii10067'] = 0x0432;
+  t['afii10068'] = 0x0433;
+  t['afii10069'] = 0x0434;
+  t['afii10070'] = 0x0435;
+  t['afii10071'] = 0x0451;
+  t['afii10072'] = 0x0436;
+  t['afii10073'] = 0x0437;
+  t['afii10074'] = 0x0438;
+  t['afii10075'] = 0x0439;
+  t['afii10076'] = 0x043A;
+  t['afii10077'] = 0x043B;
+  t['afii10078'] = 0x043C;
+  t['afii10079'] = 0x043D;
+  t['afii10080'] = 0x043E;
+  t['afii10081'] = 0x043F;
+  t['afii10082'] = 0x0440;
+  t['afii10083'] = 0x0441;
+  t['afii10084'] = 0x0442;
+  t['afii10085'] = 0x0443;
+  t['afii10086'] = 0x0444;
+  t['afii10087'] = 0x0445;
+  t['afii10088'] = 0x0446;
+  t['afii10089'] = 0x0447;
+  t['afii10090'] = 0x0448;
+  t['afii10091'] = 0x0449;
+  t['afii10092'] = 0x044A;
+  t['afii10093'] = 0x044B;
+  t['afii10094'] = 0x044C;
+  t['afii10095'] = 0x044D;
+  t['afii10096'] = 0x044E;
+  t['afii10097'] = 0x044F;
+  t['afii10098'] = 0x0491;
+  t['afii10099'] = 0x0452;
+  t['afii10100'] = 0x0453;
+  t['afii10101'] = 0x0454;
+  t['afii10102'] = 0x0455;
+  t['afii10103'] = 0x0456;
+  t['afii10104'] = 0x0457;
+  t['afii10105'] = 0x0458;
+  t['afii10106'] = 0x0459;
+  t['afii10107'] = 0x045A;
+  t['afii10108'] = 0x045B;
+  t['afii10109'] = 0x045C;
+  t['afii10110'] = 0x045E;
+  t['afii10145'] = 0x040F;
+  t['afii10146'] = 0x0462;
+  t['afii10147'] = 0x0472;
+  t['afii10148'] = 0x0474;
+  t['afii10192'] = 0xF6C6;
+  t['afii10193'] = 0x045F;
+  t['afii10194'] = 0x0463;
+  t['afii10195'] = 0x0473;
+  t['afii10196'] = 0x0475;
+  t['afii10831'] = 0xF6C7;
+  t['afii10832'] = 0xF6C8;
+  t['afii10846'] = 0x04D9;
+  t['afii299'] = 0x200E;
+  t['afii300'] = 0x200F;
+  t['afii301'] = 0x200D;
+  t['afii57381'] = 0x066A;
+  t['afii57388'] = 0x060C;
+  t['afii57392'] = 0x0660;
+  t['afii57393'] = 0x0661;
+  t['afii57394'] = 0x0662;
+  t['afii57395'] = 0x0663;
+  t['afii57396'] = 0x0664;
+  t['afii57397'] = 0x0665;
+  t['afii57398'] = 0x0666;
+  t['afii57399'] = 0x0667;
+  t['afii57400'] = 0x0668;
+  t['afii57401'] = 0x0669;
+  t['afii57403'] = 0x061B;
+  t['afii57407'] = 0x061F;
+  t['afii57409'] = 0x0621;
+  t['afii57410'] = 0x0622;
+  t['afii57411'] = 0x0623;
+  t['afii57412'] = 0x0624;
+  t['afii57413'] = 0x0625;
+  t['afii57414'] = 0x0626;
+  t['afii57415'] = 0x0627;
+  t['afii57416'] = 0x0628;
+  t['afii57417'] = 0x0629;
+  t['afii57418'] = 0x062A;
+  t['afii57419'] = 0x062B;
+  t['afii57420'] = 0x062C;
+  t['afii57421'] = 0x062D;
+  t['afii57422'] = 0x062E;
+  t['afii57423'] = 0x062F;
+  t['afii57424'] = 0x0630;
+  t['afii57425'] = 0x0631;
+  t['afii57426'] = 0x0632;
+  t['afii57427'] = 0x0633;
+  t['afii57428'] = 0x0634;
+  t['afii57429'] = 0x0635;
+  t['afii57430'] = 0x0636;
+  t['afii57431'] = 0x0637;
+  t['afii57432'] = 0x0638;
+  t['afii57433'] = 0x0639;
+  t['afii57434'] = 0x063A;
+  t['afii57440'] = 0x0640;
+  t['afii57441'] = 0x0641;
+  t['afii57442'] = 0x0642;
+  t['afii57443'] = 0x0643;
+  t['afii57444'] = 0x0644;
+  t['afii57445'] = 0x0645;
+  t['afii57446'] = 0x0646;
+  t['afii57448'] = 0x0648;
+  t['afii57449'] = 0x0649;
+  t['afii57450'] = 0x064A;
+  t['afii57451'] = 0x064B;
+  t['afii57452'] = 0x064C;
+  t['afii57453'] = 0x064D;
+  t['afii57454'] = 0x064E;
+  t['afii57455'] = 0x064F;
+  t['afii57456'] = 0x0650;
+  t['afii57457'] = 0x0651;
+  t['afii57458'] = 0x0652;
+  t['afii57470'] = 0x0647;
+  t['afii57505'] = 0x06A4;
+  t['afii57506'] = 0x067E;
+  t['afii57507'] = 0x0686;
+  t['afii57508'] = 0x0698;
+  t['afii57509'] = 0x06AF;
+  t['afii57511'] = 0x0679;
+  t['afii57512'] = 0x0688;
+  t['afii57513'] = 0x0691;
+  t['afii57514'] = 0x06BA;
+  t['afii57519'] = 0x06D2;
+  t['afii57534'] = 0x06D5;
+  t['afii57636'] = 0x20AA;
+  t['afii57645'] = 0x05BE;
+  t['afii57658'] = 0x05C3;
+  t['afii57664'] = 0x05D0;
+  t['afii57665'] = 0x05D1;
+  t['afii57666'] = 0x05D2;
+  t['afii57667'] = 0x05D3;
+  t['afii57668'] = 0x05D4;
+  t['afii57669'] = 0x05D5;
+  t['afii57670'] = 0x05D6;
+  t['afii57671'] = 0x05D7;
+  t['afii57672'] = 0x05D8;
+  t['afii57673'] = 0x05D9;
+  t['afii57674'] = 0x05DA;
+  t['afii57675'] = 0x05DB;
+  t['afii57676'] = 0x05DC;
+  t['afii57677'] = 0x05DD;
+  t['afii57678'] = 0x05DE;
+  t['afii57679'] = 0x05DF;
+  t['afii57680'] = 0x05E0;
+  t['afii57681'] = 0x05E1;
+  t['afii57682'] = 0x05E2;
+  t['afii57683'] = 0x05E3;
+  t['afii57684'] = 0x05E4;
+  t['afii57685'] = 0x05E5;
+  t['afii57686'] = 0x05E6;
+  t['afii57687'] = 0x05E7;
+  t['afii57688'] = 0x05E8;
+  t['afii57689'] = 0x05E9;
+  t['afii57690'] = 0x05EA;
+  t['afii57694'] = 0xFB2A;
+  t['afii57695'] = 0xFB2B;
+  t['afii57700'] = 0xFB4B;
+  t['afii57705'] = 0xFB1F;
+  t['afii57716'] = 0x05F0;
+  t['afii57717'] = 0x05F1;
+  t['afii57718'] = 0x05F2;
+  t['afii57723'] = 0xFB35;
+  t['afii57793'] = 0x05B4;
+  t['afii57794'] = 0x05B5;
+  t['afii57795'] = 0x05B6;
+  t['afii57796'] = 0x05BB;
+  t['afii57797'] = 0x05B8;
+  t['afii57798'] = 0x05B7;
+  t['afii57799'] = 0x05B0;
+  t['afii57800'] = 0x05B2;
+  t['afii57801'] = 0x05B1;
+  t['afii57802'] = 0x05B3;
+  t['afii57803'] = 0x05C2;
+  t['afii57804'] = 0x05C1;
+  t['afii57806'] = 0x05B9;
+  t['afii57807'] = 0x05BC;
+  t['afii57839'] = 0x05BD;
+  t['afii57841'] = 0x05BF;
+  t['afii57842'] = 0x05C0;
+  t['afii57929'] = 0x02BC;
+  t['afii61248'] = 0x2105;
+  t['afii61289'] = 0x2113;
+  t['afii61352'] = 0x2116;
+  t['afii61573'] = 0x202C;
+  t['afii61574'] = 0x202D;
+  t['afii61575'] = 0x202E;
+  t['afii61664'] = 0x200C;
+  t['afii63167'] = 0x066D;
+  t['afii64937'] = 0x02BD;
+  t['agrave'] = 0x00E0;
+  t['agujarati'] = 0x0A85;
+  t['agurmukhi'] = 0x0A05;
+  t['ahiragana'] = 0x3042;
+  t['ahookabove'] = 0x1EA3;
+  t['aibengali'] = 0x0990;
+  t['aibopomofo'] = 0x311E;
+  t['aideva'] = 0x0910;
+  t['aiecyrillic'] = 0x04D5;
+  t['aigujarati'] = 0x0A90;
+  t['aigurmukhi'] = 0x0A10;
+  t['aimatragurmukhi'] = 0x0A48;
+  t['ainarabic'] = 0x0639;
+  t['ainfinalarabic'] = 0xFECA;
+  t['aininitialarabic'] = 0xFECB;
+  t['ainmedialarabic'] = 0xFECC;
+  t['ainvertedbreve'] = 0x0203;
+  t['aivowelsignbengali'] = 0x09C8;
+  t['aivowelsigndeva'] = 0x0948;
+  t['aivowelsigngujarati'] = 0x0AC8;
+  t['akatakana'] = 0x30A2;
+  t['akatakanahalfwidth'] = 0xFF71;
+  t['akorean'] = 0x314F;
+  t['alef'] = 0x05D0;
+  t['alefarabic'] = 0x0627;
+  t['alefdageshhebrew'] = 0xFB30;
+  t['aleffinalarabic'] = 0xFE8E;
+  t['alefhamzaabovearabic'] = 0x0623;
+  t['alefhamzaabovefinalarabic'] = 0xFE84;
+  t['alefhamzabelowarabic'] = 0x0625;
+  t['alefhamzabelowfinalarabic'] = 0xFE88;
+  t['alefhebrew'] = 0x05D0;
+  t['aleflamedhebrew'] = 0xFB4F;
+  t['alefmaddaabovearabic'] = 0x0622;
+  t['alefmaddaabovefinalarabic'] = 0xFE82;
+  t['alefmaksuraarabic'] = 0x0649;
+  t['alefmaksurafinalarabic'] = 0xFEF0;
+  t['alefmaksurainitialarabic'] = 0xFEF3;
+  t['alefmaksuramedialarabic'] = 0xFEF4;
+  t['alefpatahhebrew'] = 0xFB2E;
+  t['alefqamatshebrew'] = 0xFB2F;
+  t['aleph'] = 0x2135;
+  t['allequal'] = 0x224C;
+  t['alpha'] = 0x03B1;
+  t['alphatonos'] = 0x03AC;
+  t['amacron'] = 0x0101;
+  t['amonospace'] = 0xFF41;
+  t['ampersand'] = 0x0026;
+  t['ampersandmonospace'] = 0xFF06;
+  t['ampersandsmall'] = 0xF726;
+  t['amsquare'] = 0x33C2;
+  t['anbopomofo'] = 0x3122;
+  t['angbopomofo'] = 0x3124;
+  t['angbracketleft'] = 0x3008;
+  t['angbracketright'] = 0x3009;
+  t['angkhankhuthai'] = 0x0E5A;
+  t['angle'] = 0x2220;
+  t['anglebracketleft'] = 0x3008;
+  t['anglebracketleftvertical'] = 0xFE3F;
+  t['anglebracketright'] = 0x3009;
+  t['anglebracketrightvertical'] = 0xFE40;
+  t['angleleft'] = 0x2329;
+  t['angleright'] = 0x232A;
+  t['angstrom'] = 0x212B;
+  t['anoteleia'] = 0x0387;
+  t['anudattadeva'] = 0x0952;
+  t['anusvarabengali'] = 0x0982;
+  t['anusvaradeva'] = 0x0902;
+  t['anusvaragujarati'] = 0x0A82;
+  t['aogonek'] = 0x0105;
+  t['apaatosquare'] = 0x3300;
+  t['aparen'] = 0x249C;
+  t['apostrophearmenian'] = 0x055A;
+  t['apostrophemod'] = 0x02BC;
+  t['apple'] = 0xF8FF;
+  t['approaches'] = 0x2250;
+  t['approxequal'] = 0x2248;
+  t['approxequalorimage'] = 0x2252;
+  t['approximatelyequal'] = 0x2245;
+  t['araeaekorean'] = 0x318E;
+  t['araeakorean'] = 0x318D;
+  t['arc'] = 0x2312;
+  t['arighthalfring'] = 0x1E9A;
+  t['aring'] = 0x00E5;
+  t['aringacute'] = 0x01FB;
+  t['aringbelow'] = 0x1E01;
+  t['arrowboth'] = 0x2194;
+  t['arrowdashdown'] = 0x21E3;
+  t['arrowdashleft'] = 0x21E0;
+  t['arrowdashright'] = 0x21E2;
+  t['arrowdashup'] = 0x21E1;
+  t['arrowdblboth'] = 0x21D4;
+  t['arrowdbldown'] = 0x21D3;
+  t['arrowdblleft'] = 0x21D0;
+  t['arrowdblright'] = 0x21D2;
+  t['arrowdblup'] = 0x21D1;
+  t['arrowdown'] = 0x2193;
+  t['arrowdownleft'] = 0x2199;
+  t['arrowdownright'] = 0x2198;
+  t['arrowdownwhite'] = 0x21E9;
+  t['arrowheaddownmod'] = 0x02C5;
+  t['arrowheadleftmod'] = 0x02C2;
+  t['arrowheadrightmod'] = 0x02C3;
+  t['arrowheadupmod'] = 0x02C4;
+  t['arrowhorizex'] = 0xF8E7;
+  t['arrowleft'] = 0x2190;
+  t['arrowleftdbl'] = 0x21D0;
+  t['arrowleftdblstroke'] = 0x21CD;
+  t['arrowleftoverright'] = 0x21C6;
+  t['arrowleftwhite'] = 0x21E6;
+  t['arrowright'] = 0x2192;
+  t['arrowrightdblstroke'] = 0x21CF;
+  t['arrowrightheavy'] = 0x279E;
+  t['arrowrightoverleft'] = 0x21C4;
+  t['arrowrightwhite'] = 0x21E8;
+  t['arrowtableft'] = 0x21E4;
+  t['arrowtabright'] = 0x21E5;
+  t['arrowup'] = 0x2191;
+  t['arrowupdn'] = 0x2195;
+  t['arrowupdnbse'] = 0x21A8;
+  t['arrowupdownbase'] = 0x21A8;
+  t['arrowupleft'] = 0x2196;
+  t['arrowupleftofdown'] = 0x21C5;
+  t['arrowupright'] = 0x2197;
+  t['arrowupwhite'] = 0x21E7;
+  t['arrowvertex'] = 0xF8E6;
+  t['asciicircum'] = 0x005E;
+  t['asciicircummonospace'] = 0xFF3E;
+  t['asciitilde'] = 0x007E;
+  t['asciitildemonospace'] = 0xFF5E;
+  t['ascript'] = 0x0251;
+  t['ascriptturned'] = 0x0252;
+  t['asmallhiragana'] = 0x3041;
+  t['asmallkatakana'] = 0x30A1;
+  t['asmallkatakanahalfwidth'] = 0xFF67;
+  t['asterisk'] = 0x002A;
+  t['asteriskaltonearabic'] = 0x066D;
+  t['asteriskarabic'] = 0x066D;
+  t['asteriskmath'] = 0x2217;
+  t['asteriskmonospace'] = 0xFF0A;
+  t['asterisksmall'] = 0xFE61;
+  t['asterism'] = 0x2042;
+  t['asuperior'] = 0xF6E9;
+  t['asymptoticallyequal'] = 0x2243;
+  t['at'] = 0x0040;
+  t['atilde'] = 0x00E3;
+  t['atmonospace'] = 0xFF20;
+  t['atsmall'] = 0xFE6B;
+  t['aturned'] = 0x0250;
+  t['aubengali'] = 0x0994;
+  t['aubopomofo'] = 0x3120;
+  t['audeva'] = 0x0914;
+  t['augujarati'] = 0x0A94;
+  t['augurmukhi'] = 0x0A14;
+  t['aulengthmarkbengali'] = 0x09D7;
+  t['aumatragurmukhi'] = 0x0A4C;
+  t['auvowelsignbengali'] = 0x09CC;
+  t['auvowelsigndeva'] = 0x094C;
+  t['auvowelsigngujarati'] = 0x0ACC;
+  t['avagrahadeva'] = 0x093D;
+  t['aybarmenian'] = 0x0561;
+  t['ayin'] = 0x05E2;
+  t['ayinaltonehebrew'] = 0xFB20;
+  t['ayinhebrew'] = 0x05E2;
+  t['b'] = 0x0062;
+  t['babengali'] = 0x09AC;
+  t['backslash'] = 0x005C;
+  t['backslashmonospace'] = 0xFF3C;
+  t['badeva'] = 0x092C;
+  t['bagujarati'] = 0x0AAC;
+  t['bagurmukhi'] = 0x0A2C;
+  t['bahiragana'] = 0x3070;
+  t['bahtthai'] = 0x0E3F;
+  t['bakatakana'] = 0x30D0;
+  t['bar'] = 0x007C;
+  t['barmonospace'] = 0xFF5C;
+  t['bbopomofo'] = 0x3105;
+  t['bcircle'] = 0x24D1;
+  t['bdotaccent'] = 0x1E03;
+  t['bdotbelow'] = 0x1E05;
+  t['beamedsixteenthnotes'] = 0x266C;
+  t['because'] = 0x2235;
+  t['becyrillic'] = 0x0431;
+  t['beharabic'] = 0x0628;
+  t['behfinalarabic'] = 0xFE90;
+  t['behinitialarabic'] = 0xFE91;
+  t['behiragana'] = 0x3079;
+  t['behmedialarabic'] = 0xFE92;
+  t['behmeeminitialarabic'] = 0xFC9F;
+  t['behmeemisolatedarabic'] = 0xFC08;
+  t['behnoonfinalarabic'] = 0xFC6D;
+  t['bekatakana'] = 0x30D9;
+  t['benarmenian'] = 0x0562;
+  t['bet'] = 0x05D1;
+  t['beta'] = 0x03B2;
+  t['betasymbolgreek'] = 0x03D0;
+  t['betdagesh'] = 0xFB31;
+  t['betdageshhebrew'] = 0xFB31;
+  t['bethebrew'] = 0x05D1;
+  t['betrafehebrew'] = 0xFB4C;
+  t['bhabengali'] = 0x09AD;
+  t['bhadeva'] = 0x092D;
+  t['bhagujarati'] = 0x0AAD;
+  t['bhagurmukhi'] = 0x0A2D;
+  t['bhook'] = 0x0253;
+  t['bihiragana'] = 0x3073;
+  t['bikatakana'] = 0x30D3;
+  t['bilabialclick'] = 0x0298;
+  t['bindigurmukhi'] = 0x0A02;
+  t['birusquare'] = 0x3331;
+  t['blackcircle'] = 0x25CF;
+  t['blackdiamond'] = 0x25C6;
+  t['blackdownpointingtriangle'] = 0x25BC;
+  t['blackleftpointingpointer'] = 0x25C4;
+  t['blackleftpointingtriangle'] = 0x25C0;
+  t['blacklenticularbracketleft'] = 0x3010;
+  t['blacklenticularbracketleftvertical'] = 0xFE3B;
+  t['blacklenticularbracketright'] = 0x3011;
+  t['blacklenticularbracketrightvertical'] = 0xFE3C;
+  t['blacklowerlefttriangle'] = 0x25E3;
+  t['blacklowerrighttriangle'] = 0x25E2;
+  t['blackrectangle'] = 0x25AC;
+  t['blackrightpointingpointer'] = 0x25BA;
+  t['blackrightpointingtriangle'] = 0x25B6;
+  t['blacksmallsquare'] = 0x25AA;
+  t['blacksmilingface'] = 0x263B;
+  t['blacksquare'] = 0x25A0;
+  t['blackstar'] = 0x2605;
+  t['blackupperlefttriangle'] = 0x25E4;
+  t['blackupperrighttriangle'] = 0x25E5;
+  t['blackuppointingsmalltriangle'] = 0x25B4;
+  t['blackuppointingtriangle'] = 0x25B2;
+  t['blank'] = 0x2423;
+  t['blinebelow'] = 0x1E07;
+  t['block'] = 0x2588;
+  t['bmonospace'] = 0xFF42;
+  t['bobaimaithai'] = 0x0E1A;
+  t['bohiragana'] = 0x307C;
+  t['bokatakana'] = 0x30DC;
+  t['bparen'] = 0x249D;
+  t['bqsquare'] = 0x33C3;
+  t['braceex'] = 0xF8F4;
+  t['braceleft'] = 0x007B;
+  t['braceleftbt'] = 0xF8F3;
+  t['braceleftmid'] = 0xF8F2;
+  t['braceleftmonospace'] = 0xFF5B;
+  t['braceleftsmall'] = 0xFE5B;
+  t['bracelefttp'] = 0xF8F1;
+  t['braceleftvertical'] = 0xFE37;
+  t['braceright'] = 0x007D;
+  t['bracerightbt'] = 0xF8FE;
+  t['bracerightmid'] = 0xF8FD;
+  t['bracerightmonospace'] = 0xFF5D;
+  t['bracerightsmall'] = 0xFE5C;
+  t['bracerighttp'] = 0xF8FC;
+  t['bracerightvertical'] = 0xFE38;
+  t['bracketleft'] = 0x005B;
+  t['bracketleftbt'] = 0xF8F0;
+  t['bracketleftex'] = 0xF8EF;
+  t['bracketleftmonospace'] = 0xFF3B;
+  t['bracketlefttp'] = 0xF8EE;
+  t['bracketright'] = 0x005D;
+  t['bracketrightbt'] = 0xF8FB;
+  t['bracketrightex'] = 0xF8FA;
+  t['bracketrightmonospace'] = 0xFF3D;
+  t['bracketrighttp'] = 0xF8F9;
+  t['breve'] = 0x02D8;
+  t['brevebelowcmb'] = 0x032E;
+  t['brevecmb'] = 0x0306;
+  t['breveinvertedbelowcmb'] = 0x032F;
+  t['breveinvertedcmb'] = 0x0311;
+  t['breveinverteddoublecmb'] = 0x0361;
+  t['bridgebelowcmb'] = 0x032A;
+  t['bridgeinvertedbelowcmb'] = 0x033A;
+  t['brokenbar'] = 0x00A6;
+  t['bstroke'] = 0x0180;
+  t['bsuperior'] = 0xF6EA;
+  t['btopbar'] = 0x0183;
+  t['buhiragana'] = 0x3076;
+  t['bukatakana'] = 0x30D6;
+  t['bullet'] = 0x2022;
+  t['bulletinverse'] = 0x25D8;
+  t['bulletoperator'] = 0x2219;
+  t['bullseye'] = 0x25CE;
+  t['c'] = 0x0063;
+  t['caarmenian'] = 0x056E;
+  t['cabengali'] = 0x099A;
+  t['cacute'] = 0x0107;
+  t['cadeva'] = 0x091A;
+  t['cagujarati'] = 0x0A9A;
+  t['cagurmukhi'] = 0x0A1A;
+  t['calsquare'] = 0x3388;
+  t['candrabindubengali'] = 0x0981;
+  t['candrabinducmb'] = 0x0310;
+  t['candrabindudeva'] = 0x0901;
+  t['candrabindugujarati'] = 0x0A81;
+  t['capslock'] = 0x21EA;
+  t['careof'] = 0x2105;
+  t['caron'] = 0x02C7;
+  t['caronbelowcmb'] = 0x032C;
+  t['caroncmb'] = 0x030C;
+  t['carriagereturn'] = 0x21B5;
+  t['cbopomofo'] = 0x3118;
+  t['ccaron'] = 0x010D;
+  t['ccedilla'] = 0x00E7;
+  t['ccedillaacute'] = 0x1E09;
+  t['ccircle'] = 0x24D2;
+  t['ccircumflex'] = 0x0109;
+  t['ccurl'] = 0x0255;
+  t['cdot'] = 0x010B;
+  t['cdotaccent'] = 0x010B;
+  t['cdsquare'] = 0x33C5;
+  t['cedilla'] = 0x00B8;
+  t['cedillacmb'] = 0x0327;
+  t['cent'] = 0x00A2;
+  t['centigrade'] = 0x2103;
+  t['centinferior'] = 0xF6DF;
+  t['centmonospace'] = 0xFFE0;
+  t['centoldstyle'] = 0xF7A2;
+  t['centsuperior'] = 0xF6E0;
+  t['chaarmenian'] = 0x0579;
+  t['chabengali'] = 0x099B;
+  t['chadeva'] = 0x091B;
+  t['chagujarati'] = 0x0A9B;
+  t['chagurmukhi'] = 0x0A1B;
+  t['chbopomofo'] = 0x3114;
+  t['cheabkhasiancyrillic'] = 0x04BD;
+  t['checkmark'] = 0x2713;
+  t['checyrillic'] = 0x0447;
+  t['chedescenderabkhasiancyrillic'] = 0x04BF;
+  t['chedescendercyrillic'] = 0x04B7;
+  t['chedieresiscyrillic'] = 0x04F5;
+  t['cheharmenian'] = 0x0573;
+  t['chekhakassiancyrillic'] = 0x04CC;
+  t['cheverticalstrokecyrillic'] = 0x04B9;
+  t['chi'] = 0x03C7;
+  t['chieuchacirclekorean'] = 0x3277;
+  t['chieuchaparenkorean'] = 0x3217;
+  t['chieuchcirclekorean'] = 0x3269;
+  t['chieuchkorean'] = 0x314A;
+  t['chieuchparenkorean'] = 0x3209;
+  t['chochangthai'] = 0x0E0A;
+  t['chochanthai'] = 0x0E08;
+  t['chochingthai'] = 0x0E09;
+  t['chochoethai'] = 0x0E0C;
+  t['chook'] = 0x0188;
+  t['cieucacirclekorean'] = 0x3276;
+  t['cieucaparenkorean'] = 0x3216;
+  t['cieuccirclekorean'] = 0x3268;
+  t['cieuckorean'] = 0x3148;
+  t['cieucparenkorean'] = 0x3208;
+  t['cieucuparenkorean'] = 0x321C;
+  t['circle'] = 0x25CB;
+  t['circlecopyrt'] = 0x00A9;
+  t['circlemultiply'] = 0x2297;
+  t['circleot'] = 0x2299;
+  t['circleplus'] = 0x2295;
+  t['circlepostalmark'] = 0x3036;
+  t['circlewithlefthalfblack'] = 0x25D0;
+  t['circlewithrighthalfblack'] = 0x25D1;
+  t['circumflex'] = 0x02C6;
+  t['circumflexbelowcmb'] = 0x032D;
+  t['circumflexcmb'] = 0x0302;
+  t['clear'] = 0x2327;
+  t['clickalveolar'] = 0x01C2;
+  t['clickdental'] = 0x01C0;
+  t['clicklateral'] = 0x01C1;
+  t['clickretroflex'] = 0x01C3;
+  t['club'] = 0x2663;
+  t['clubsuitblack'] = 0x2663;
+  t['clubsuitwhite'] = 0x2667;
+  t['cmcubedsquare'] = 0x33A4;
+  t['cmonospace'] = 0xFF43;
+  t['cmsquaredsquare'] = 0x33A0;
+  t['coarmenian'] = 0x0581;
+  t['colon'] = 0x003A;
+  t['colonmonetary'] = 0x20A1;
+  t['colonmonospace'] = 0xFF1A;
+  t['colonsign'] = 0x20A1;
+  t['colonsmall'] = 0xFE55;
+  t['colontriangularhalfmod'] = 0x02D1;
+  t['colontriangularmod'] = 0x02D0;
+  t['comma'] = 0x002C;
+  t['commaabovecmb'] = 0x0313;
+  t['commaaboverightcmb'] = 0x0315;
+  t['commaaccent'] = 0xF6C3;
+  t['commaarabic'] = 0x060C;
+  t['commaarmenian'] = 0x055D;
+  t['commainferior'] = 0xF6E1;
+  t['commamonospace'] = 0xFF0C;
+  t['commareversedabovecmb'] = 0x0314;
+  t['commareversedmod'] = 0x02BD;
+  t['commasmall'] = 0xFE50;
+  t['commasuperior'] = 0xF6E2;
+  t['commaturnedabovecmb'] = 0x0312;
+  t['commaturnedmod'] = 0x02BB;
+  t['compass'] = 0x263C;
+  t['congruent'] = 0x2245;
+  t['contourintegral'] = 0x222E;
+  t['control'] = 0x2303;
+  t['controlACK'] = 0x0006;
+  t['controlBEL'] = 0x0007;
+  t['controlBS'] = 0x0008;
+  t['controlCAN'] = 0x0018;
+  t['controlCR'] = 0x000D;
+  t['controlDC1'] = 0x0011;
+  t['controlDC2'] = 0x0012;
+  t['controlDC3'] = 0x0013;
+  t['controlDC4'] = 0x0014;
+  t['controlDEL'] = 0x007F;
+  t['controlDLE'] = 0x0010;
+  t['controlEM'] = 0x0019;
+  t['controlENQ'] = 0x0005;
+  t['controlEOT'] = 0x0004;
+  t['controlESC'] = 0x001B;
+  t['controlETB'] = 0x0017;
+  t['controlETX'] = 0x0003;
+  t['controlFF'] = 0x000C;
+  t['controlFS'] = 0x001C;
+  t['controlGS'] = 0x001D;
+  t['controlHT'] = 0x0009;
+  t['controlLF'] = 0x000A;
+  t['controlNAK'] = 0x0015;
+  t['controlNULL'] = 0x0000;
+  t['controlRS'] = 0x001E;
+  t['controlSI'] = 0x000F;
+  t['controlSO'] = 0x000E;
+  t['controlSOT'] = 0x0002;
+  t['controlSTX'] = 0x0001;
+  t['controlSUB'] = 0x001A;
+  t['controlSYN'] = 0x0016;
+  t['controlUS'] = 0x001F;
+  t['controlVT'] = 0x000B;
+  t['copyright'] = 0x00A9;
+  t['copyrightsans'] = 0xF8E9;
+  t['copyrightserif'] = 0xF6D9;
+  t['cornerbracketleft'] = 0x300C;
+  t['cornerbracketlefthalfwidth'] = 0xFF62;
+  t['cornerbracketleftvertical'] = 0xFE41;
+  t['cornerbracketright'] = 0x300D;
+  t['cornerbracketrighthalfwidth'] = 0xFF63;
+  t['cornerbracketrightvertical'] = 0xFE42;
+  t['corporationsquare'] = 0x337F;
+  t['cosquare'] = 0x33C7;
+  t['coverkgsquare'] = 0x33C6;
+  t['cparen'] = 0x249E;
+  t['cruzeiro'] = 0x20A2;
+  t['cstretched'] = 0x0297;
+  t['curlyand'] = 0x22CF;
+  t['curlyor'] = 0x22CE;
+  t['currency'] = 0x00A4;
+  t['cyrBreve'] = 0xF6D1;
+  t['cyrFlex'] = 0xF6D2;
+  t['cyrbreve'] = 0xF6D4;
+  t['cyrflex'] = 0xF6D5;
+  t['d'] = 0x0064;
+  t['daarmenian'] = 0x0564;
+  t['dabengali'] = 0x09A6;
+  t['dadarabic'] = 0x0636;
+  t['dadeva'] = 0x0926;
+  t['dadfinalarabic'] = 0xFEBE;
+  t['dadinitialarabic'] = 0xFEBF;
+  t['dadmedialarabic'] = 0xFEC0;
+  t['dagesh'] = 0x05BC;
+  t['dageshhebrew'] = 0x05BC;
+  t['dagger'] = 0x2020;
+  t['daggerdbl'] = 0x2021;
+  t['dagujarati'] = 0x0AA6;
+  t['dagurmukhi'] = 0x0A26;
+  t['dahiragana'] = 0x3060;
+  t['dakatakana'] = 0x30C0;
+  t['dalarabic'] = 0x062F;
+  t['dalet'] = 0x05D3;
+  t['daletdagesh'] = 0xFB33;
+  t['daletdageshhebrew'] = 0xFB33;
+  t['dalethebrew'] = 0x05D3;
+  t['dalfinalarabic'] = 0xFEAA;
+  t['dammaarabic'] = 0x064F;
+  t['dammalowarabic'] = 0x064F;
+  t['dammatanaltonearabic'] = 0x064C;
+  t['dammatanarabic'] = 0x064C;
+  t['danda'] = 0x0964;
+  t['dargahebrew'] = 0x05A7;
+  t['dargalefthebrew'] = 0x05A7;
+  t['dasiapneumatacyrilliccmb'] = 0x0485;
+  t['dblGrave'] = 0xF6D3;
+  t['dblanglebracketleft'] = 0x300A;
+  t['dblanglebracketleftvertical'] = 0xFE3D;
+  t['dblanglebracketright'] = 0x300B;
+  t['dblanglebracketrightvertical'] = 0xFE3E;
+  t['dblarchinvertedbelowcmb'] = 0x032B;
+  t['dblarrowleft'] = 0x21D4;
+  t['dblarrowright'] = 0x21D2;
+  t['dbldanda'] = 0x0965;
+  t['dblgrave'] = 0xF6D6;
+  t['dblgravecmb'] = 0x030F;
+  t['dblintegral'] = 0x222C;
+  t['dbllowline'] = 0x2017;
+  t['dbllowlinecmb'] = 0x0333;
+  t['dbloverlinecmb'] = 0x033F;
+  t['dblprimemod'] = 0x02BA;
+  t['dblverticalbar'] = 0x2016;
+  t['dblverticallineabovecmb'] = 0x030E;
+  t['dbopomofo'] = 0x3109;
+  t['dbsquare'] = 0x33C8;
+  t['dcaron'] = 0x010F;
+  t['dcedilla'] = 0x1E11;
+  t['dcircle'] = 0x24D3;
+  t['dcircumflexbelow'] = 0x1E13;
+  t['dcroat'] = 0x0111;
+  t['ddabengali'] = 0x09A1;
+  t['ddadeva'] = 0x0921;
+  t['ddagujarati'] = 0x0AA1;
+  t['ddagurmukhi'] = 0x0A21;
+  t['ddalarabic'] = 0x0688;
+  t['ddalfinalarabic'] = 0xFB89;
+  t['dddhadeva'] = 0x095C;
+  t['ddhabengali'] = 0x09A2;
+  t['ddhadeva'] = 0x0922;
+  t['ddhagujarati'] = 0x0AA2;
+  t['ddhagurmukhi'] = 0x0A22;
+  t['ddotaccent'] = 0x1E0B;
+  t['ddotbelow'] = 0x1E0D;
+  t['decimalseparatorarabic'] = 0x066B;
+  t['decimalseparatorpersian'] = 0x066B;
+  t['decyrillic'] = 0x0434;
+  t['degree'] = 0x00B0;
+  t['dehihebrew'] = 0x05AD;
+  t['dehiragana'] = 0x3067;
+  t['deicoptic'] = 0x03EF;
+  t['dekatakana'] = 0x30C7;
+  t['deleteleft'] = 0x232B;
+  t['deleteright'] = 0x2326;
+  t['delta'] = 0x03B4;
+  t['deltaturned'] = 0x018D;
+  t['denominatorminusonenumeratorbengali'] = 0x09F8;
+  t['dezh'] = 0x02A4;
+  t['dhabengali'] = 0x09A7;
+  t['dhadeva'] = 0x0927;
+  t['dhagujarati'] = 0x0AA7;
+  t['dhagurmukhi'] = 0x0A27;
+  t['dhook'] = 0x0257;
+  t['dialytikatonos'] = 0x0385;
+  t['dialytikatonoscmb'] = 0x0344;
+  t['diamond'] = 0x2666;
+  t['diamondsuitwhite'] = 0x2662;
+  t['dieresis'] = 0x00A8;
+  t['dieresisacute'] = 0xF6D7;
+  t['dieresisbelowcmb'] = 0x0324;
+  t['dieresiscmb'] = 0x0308;
+  t['dieresisgrave'] = 0xF6D8;
+  t['dieresistonos'] = 0x0385;
+  t['dihiragana'] = 0x3062;
+  t['dikatakana'] = 0x30C2;
+  t['dittomark'] = 0x3003;
+  t['divide'] = 0x00F7;
+  t['divides'] = 0x2223;
+  t['divisionslash'] = 0x2215;
+  t['djecyrillic'] = 0x0452;
+  t['dkshade'] = 0x2593;
+  t['dlinebelow'] = 0x1E0F;
+  t['dlsquare'] = 0x3397;
+  t['dmacron'] = 0x0111;
+  t['dmonospace'] = 0xFF44;
+  t['dnblock'] = 0x2584;
+  t['dochadathai'] = 0x0E0E;
+  t['dodekthai'] = 0x0E14;
+  t['dohiragana'] = 0x3069;
+  t['dokatakana'] = 0x30C9;
+  t['dollar'] = 0x0024;
+  t['dollarinferior'] = 0xF6E3;
+  t['dollarmonospace'] = 0xFF04;
+  t['dollaroldstyle'] = 0xF724;
+  t['dollarsmall'] = 0xFE69;
+  t['dollarsuperior'] = 0xF6E4;
+  t['dong'] = 0x20AB;
+  t['dorusquare'] = 0x3326;
+  t['dotaccent'] = 0x02D9;
+  t['dotaccentcmb'] = 0x0307;
+  t['dotbelowcmb'] = 0x0323;
+  t['dotbelowcomb'] = 0x0323;
+  t['dotkatakana'] = 0x30FB;
+  t['dotlessi'] = 0x0131;
+  t['dotlessj'] = 0xF6BE;
+  t['dotlessjstrokehook'] = 0x0284;
+  t['dotmath'] = 0x22C5;
+  t['dottedcircle'] = 0x25CC;
+  t['doubleyodpatah'] = 0xFB1F;
+  t['doubleyodpatahhebrew'] = 0xFB1F;
+  t['downtackbelowcmb'] = 0x031E;
+  t['downtackmod'] = 0x02D5;
+  t['dparen'] = 0x249F;
+  t['dsuperior'] = 0xF6EB;
+  t['dtail'] = 0x0256;
+  t['dtopbar'] = 0x018C;
+  t['duhiragana'] = 0x3065;
+  t['dukatakana'] = 0x30C5;
+  t['dz'] = 0x01F3;
+  t['dzaltone'] = 0x02A3;
+  t['dzcaron'] = 0x01C6;
+  t['dzcurl'] = 0x02A5;
+  t['dzeabkhasiancyrillic'] = 0x04E1;
+  t['dzecyrillic'] = 0x0455;
+  t['dzhecyrillic'] = 0x045F;
+  t['e'] = 0x0065;
+  t['eacute'] = 0x00E9;
+  t['earth'] = 0x2641;
+  t['ebengali'] = 0x098F;
+  t['ebopomofo'] = 0x311C;
+  t['ebreve'] = 0x0115;
+  t['ecandradeva'] = 0x090D;
+  t['ecandragujarati'] = 0x0A8D;
+  t['ecandravowelsigndeva'] = 0x0945;
+  t['ecandravowelsigngujarati'] = 0x0AC5;
+  t['ecaron'] = 0x011B;
+  t['ecedillabreve'] = 0x1E1D;
+  t['echarmenian'] = 0x0565;
+  t['echyiwnarmenian'] = 0x0587;
+  t['ecircle'] = 0x24D4;
+  t['ecircumflex'] = 0x00EA;
+  t['ecircumflexacute'] = 0x1EBF;
+  t['ecircumflexbelow'] = 0x1E19;
+  t['ecircumflexdotbelow'] = 0x1EC7;
+  t['ecircumflexgrave'] = 0x1EC1;
+  t['ecircumflexhookabove'] = 0x1EC3;
+  t['ecircumflextilde'] = 0x1EC5;
+  t['ecyrillic'] = 0x0454;
+  t['edblgrave'] = 0x0205;
+  t['edeva'] = 0x090F;
+  t['edieresis'] = 0x00EB;
+  t['edot'] = 0x0117;
+  t['edotaccent'] = 0x0117;
+  t['edotbelow'] = 0x1EB9;
+  t['eegurmukhi'] = 0x0A0F;
+  t['eematragurmukhi'] = 0x0A47;
+  t['efcyrillic'] = 0x0444;
+  t['egrave'] = 0x00E8;
+  t['egujarati'] = 0x0A8F;
+  t['eharmenian'] = 0x0567;
+  t['ehbopomofo'] = 0x311D;
+  t['ehiragana'] = 0x3048;
+  t['ehookabove'] = 0x1EBB;
+  t['eibopomofo'] = 0x311F;
+  t['eight'] = 0x0038;
+  t['eightarabic'] = 0x0668;
+  t['eightbengali'] = 0x09EE;
+  t['eightcircle'] = 0x2467;
+  t['eightcircleinversesansserif'] = 0x2791;
+  t['eightdeva'] = 0x096E;
+  t['eighteencircle'] = 0x2471;
+  t['eighteenparen'] = 0x2485;
+  t['eighteenperiod'] = 0x2499;
+  t['eightgujarati'] = 0x0AEE;
+  t['eightgurmukhi'] = 0x0A6E;
+  t['eighthackarabic'] = 0x0668;
+  t['eighthangzhou'] = 0x3028;
+  t['eighthnotebeamed'] = 0x266B;
+  t['eightideographicparen'] = 0x3227;
+  t['eightinferior'] = 0x2088;
+  t['eightmonospace'] = 0xFF18;
+  t['eightoldstyle'] = 0xF738;
+  t['eightparen'] = 0x247B;
+  t['eightperiod'] = 0x248F;
+  t['eightpersian'] = 0x06F8;
+  t['eightroman'] = 0x2177;
+  t['eightsuperior'] = 0x2078;
+  t['eightthai'] = 0x0E58;
+  t['einvertedbreve'] = 0x0207;
+  t['eiotifiedcyrillic'] = 0x0465;
+  t['ekatakana'] = 0x30A8;
+  t['ekatakanahalfwidth'] = 0xFF74;
+  t['ekonkargurmukhi'] = 0x0A74;
+  t['ekorean'] = 0x3154;
+  t['elcyrillic'] = 0x043B;
+  t['element'] = 0x2208;
+  t['elevencircle'] = 0x246A;
+  t['elevenparen'] = 0x247E;
+  t['elevenperiod'] = 0x2492;
+  t['elevenroman'] = 0x217A;
+  t['ellipsis'] = 0x2026;
+  t['ellipsisvertical'] = 0x22EE;
+  t['emacron'] = 0x0113;
+  t['emacronacute'] = 0x1E17;
+  t['emacrongrave'] = 0x1E15;
+  t['emcyrillic'] = 0x043C;
+  t['emdash'] = 0x2014;
+  t['emdashvertical'] = 0xFE31;
+  t['emonospace'] = 0xFF45;
+  t['emphasismarkarmenian'] = 0x055B;
+  t['emptyset'] = 0x2205;
+  t['enbopomofo'] = 0x3123;
+  t['encyrillic'] = 0x043D;
+  t['endash'] = 0x2013;
+  t['endashvertical'] = 0xFE32;
+  t['endescendercyrillic'] = 0x04A3;
+  t['eng'] = 0x014B;
+  t['engbopomofo'] = 0x3125;
+  t['enghecyrillic'] = 0x04A5;
+  t['enhookcyrillic'] = 0x04C8;
+  t['enspace'] = 0x2002;
+  t['eogonek'] = 0x0119;
+  t['eokorean'] = 0x3153;
+  t['eopen'] = 0x025B;
+  t['eopenclosed'] = 0x029A;
+  t['eopenreversed'] = 0x025C;
+  t['eopenreversedclosed'] = 0x025E;
+  t['eopenreversedhook'] = 0x025D;
+  t['eparen'] = 0x24A0;
+  t['epsilon'] = 0x03B5;
+  t['epsilontonos'] = 0x03AD;
+  t['equal'] = 0x003D;
+  t['equalmonospace'] = 0xFF1D;
+  t['equalsmall'] = 0xFE66;
+  t['equalsuperior'] = 0x207C;
+  t['equivalence'] = 0x2261;
+  t['erbopomofo'] = 0x3126;
+  t['ercyrillic'] = 0x0440;
+  t['ereversed'] = 0x0258;
+  t['ereversedcyrillic'] = 0x044D;
+  t['escyrillic'] = 0x0441;
+  t['esdescendercyrillic'] = 0x04AB;
+  t['esh'] = 0x0283;
+  t['eshcurl'] = 0x0286;
+  t['eshortdeva'] = 0x090E;
+  t['eshortvowelsigndeva'] = 0x0946;
+  t['eshreversedloop'] = 0x01AA;
+  t['eshsquatreversed'] = 0x0285;
+  t['esmallhiragana'] = 0x3047;
+  t['esmallkatakana'] = 0x30A7;
+  t['esmallkatakanahalfwidth'] = 0xFF6A;
+  t['estimated'] = 0x212E;
+  t['esuperior'] = 0xF6EC;
+  t['eta'] = 0x03B7;
+  t['etarmenian'] = 0x0568;
+  t['etatonos'] = 0x03AE;
+  t['eth'] = 0x00F0;
+  t['etilde'] = 0x1EBD;
+  t['etildebelow'] = 0x1E1B;
+  t['etnahtafoukhhebrew'] = 0x0591;
+  t['etnahtafoukhlefthebrew'] = 0x0591;
+  t['etnahtahebrew'] = 0x0591;
+  t['etnahtalefthebrew'] = 0x0591;
+  t['eturned'] = 0x01DD;
+  t['eukorean'] = 0x3161;
+  t['euro'] = 0x20AC;
+  t['evowelsignbengali'] = 0x09C7;
+  t['evowelsigndeva'] = 0x0947;
+  t['evowelsigngujarati'] = 0x0AC7;
+  t['exclam'] = 0x0021;
+  t['exclamarmenian'] = 0x055C;
+  t['exclamdbl'] = 0x203C;
+  t['exclamdown'] = 0x00A1;
+  t['exclamdownsmall'] = 0xF7A1;
+  t['exclammonospace'] = 0xFF01;
+  t['exclamsmall'] = 0xF721;
+  t['existential'] = 0x2203;
+  t['ezh'] = 0x0292;
+  t['ezhcaron'] = 0x01EF;
+  t['ezhcurl'] = 0x0293;
+  t['ezhreversed'] = 0x01B9;
+  t['ezhtail'] = 0x01BA;
+  t['f'] = 0x0066;
+  t['fadeva'] = 0x095E;
+  t['fagurmukhi'] = 0x0A5E;
+  t['fahrenheit'] = 0x2109;
+  t['fathaarabic'] = 0x064E;
+  t['fathalowarabic'] = 0x064E;
+  t['fathatanarabic'] = 0x064B;
+  t['fbopomofo'] = 0x3108;
+  t['fcircle'] = 0x24D5;
+  t['fdotaccent'] = 0x1E1F;
+  t['feharabic'] = 0x0641;
+  t['feharmenian'] = 0x0586;
+  t['fehfinalarabic'] = 0xFED2;
+  t['fehinitialarabic'] = 0xFED3;
+  t['fehmedialarabic'] = 0xFED4;
+  t['feicoptic'] = 0x03E5;
+  t['female'] = 0x2640;
+  t['ff'] = 0xFB00;
+  t['ffi'] = 0xFB03;
+  t['ffl'] = 0xFB04;
+  t['fi'] = 0xFB01;
+  t['fifteencircle'] = 0x246E;
+  t['fifteenparen'] = 0x2482;
+  t['fifteenperiod'] = 0x2496;
+  t['figuredash'] = 0x2012;
+  t['filledbox'] = 0x25A0;
+  t['filledrect'] = 0x25AC;
+  t['finalkaf'] = 0x05DA;
+  t['finalkafdagesh'] = 0xFB3A;
+  t['finalkafdageshhebrew'] = 0xFB3A;
+  t['finalkafhebrew'] = 0x05DA;
+  t['finalmem'] = 0x05DD;
+  t['finalmemhebrew'] = 0x05DD;
+  t['finalnun'] = 0x05DF;
+  t['finalnunhebrew'] = 0x05DF;
+  t['finalpe'] = 0x05E3;
+  t['finalpehebrew'] = 0x05E3;
+  t['finaltsadi'] = 0x05E5;
+  t['finaltsadihebrew'] = 0x05E5;
+  t['firsttonechinese'] = 0x02C9;
+  t['fisheye'] = 0x25C9;
+  t['fitacyrillic'] = 0x0473;
+  t['five'] = 0x0035;
+  t['fivearabic'] = 0x0665;
+  t['fivebengali'] = 0x09EB;
+  t['fivecircle'] = 0x2464;
+  t['fivecircleinversesansserif'] = 0x278E;
+  t['fivedeva'] = 0x096B;
+  t['fiveeighths'] = 0x215D;
+  t['fivegujarati'] = 0x0AEB;
+  t['fivegurmukhi'] = 0x0A6B;
+  t['fivehackarabic'] = 0x0665;
+  t['fivehangzhou'] = 0x3025;
+  t['fiveideographicparen'] = 0x3224;
+  t['fiveinferior'] = 0x2085;
+  t['fivemonospace'] = 0xFF15;
+  t['fiveoldstyle'] = 0xF735;
+  t['fiveparen'] = 0x2478;
+  t['fiveperiod'] = 0x248C;
+  t['fivepersian'] = 0x06F5;
+  t['fiveroman'] = 0x2174;
+  t['fivesuperior'] = 0x2075;
+  t['fivethai'] = 0x0E55;
+  t['fl'] = 0xFB02;
+  t['florin'] = 0x0192;
+  t['fmonospace'] = 0xFF46;
+  t['fmsquare'] = 0x3399;
+  t['fofanthai'] = 0x0E1F;
+  t['fofathai'] = 0x0E1D;
+  t['fongmanthai'] = 0x0E4F;
+  t['forall'] = 0x2200;
+  t['four'] = 0x0034;
+  t['fourarabic'] = 0x0664;
+  t['fourbengali'] = 0x09EA;
+  t['fourcircle'] = 0x2463;
+  t['fourcircleinversesansserif'] = 0x278D;
+  t['fourdeva'] = 0x096A;
+  t['fourgujarati'] = 0x0AEA;
+  t['fourgurmukhi'] = 0x0A6A;
+  t['fourhackarabic'] = 0x0664;
+  t['fourhangzhou'] = 0x3024;
+  t['fourideographicparen'] = 0x3223;
+  t['fourinferior'] = 0x2084;
+  t['fourmonospace'] = 0xFF14;
+  t['fournumeratorbengali'] = 0x09F7;
+  t['fouroldstyle'] = 0xF734;
+  t['fourparen'] = 0x2477;
+  t['fourperiod'] = 0x248B;
+  t['fourpersian'] = 0x06F4;
+  t['fourroman'] = 0x2173;
+  t['foursuperior'] = 0x2074;
+  t['fourteencircle'] = 0x246D;
+  t['fourteenparen'] = 0x2481;
+  t['fourteenperiod'] = 0x2495;
+  t['fourthai'] = 0x0E54;
+  t['fourthtonechinese'] = 0x02CB;
+  t['fparen'] = 0x24A1;
+  t['fraction'] = 0x2044;
+  t['franc'] = 0x20A3;
+  t['g'] = 0x0067;
+  t['gabengali'] = 0x0997;
+  t['gacute'] = 0x01F5;
+  t['gadeva'] = 0x0917;
+  t['gafarabic'] = 0x06AF;
+  t['gaffinalarabic'] = 0xFB93;
+  t['gafinitialarabic'] = 0xFB94;
+  t['gafmedialarabic'] = 0xFB95;
+  t['gagujarati'] = 0x0A97;
+  t['gagurmukhi'] = 0x0A17;
+  t['gahiragana'] = 0x304C;
+  t['gakatakana'] = 0x30AC;
+  t['gamma'] = 0x03B3;
+  t['gammalatinsmall'] = 0x0263;
+  t['gammasuperior'] = 0x02E0;
+  t['gangiacoptic'] = 0x03EB;
+  t['gbopomofo'] = 0x310D;
+  t['gbreve'] = 0x011F;
+  t['gcaron'] = 0x01E7;
+  t['gcedilla'] = 0x0123;
+  t['gcircle'] = 0x24D6;
+  t['gcircumflex'] = 0x011D;
+  t['gcommaaccent'] = 0x0123;
+  t['gdot'] = 0x0121;
+  t['gdotaccent'] = 0x0121;
+  t['gecyrillic'] = 0x0433;
+  t['gehiragana'] = 0x3052;
+  t['gekatakana'] = 0x30B2;
+  t['geometricallyequal'] = 0x2251;
+  t['gereshaccenthebrew'] = 0x059C;
+  t['gereshhebrew'] = 0x05F3;
+  t['gereshmuqdamhebrew'] = 0x059D;
+  t['germandbls'] = 0x00DF;
+  t['gershayimaccenthebrew'] = 0x059E;
+  t['gershayimhebrew'] = 0x05F4;
+  t['getamark'] = 0x3013;
+  t['ghabengali'] = 0x0998;
+  t['ghadarmenian'] = 0x0572;
+  t['ghadeva'] = 0x0918;
+  t['ghagujarati'] = 0x0A98;
+  t['ghagurmukhi'] = 0x0A18;
+  t['ghainarabic'] = 0x063A;
+  t['ghainfinalarabic'] = 0xFECE;
+  t['ghaininitialarabic'] = 0xFECF;
+  t['ghainmedialarabic'] = 0xFED0;
+  t['ghemiddlehookcyrillic'] = 0x0495;
+  t['ghestrokecyrillic'] = 0x0493;
+  t['gheupturncyrillic'] = 0x0491;
+  t['ghhadeva'] = 0x095A;
+  t['ghhagurmukhi'] = 0x0A5A;
+  t['ghook'] = 0x0260;
+  t['ghzsquare'] = 0x3393;
+  t['gihiragana'] = 0x304E;
+  t['gikatakana'] = 0x30AE;
+  t['gimarmenian'] = 0x0563;
+  t['gimel'] = 0x05D2;
+  t['gimeldagesh'] = 0xFB32;
+  t['gimeldageshhebrew'] = 0xFB32;
+  t['gimelhebrew'] = 0x05D2;
+  t['gjecyrillic'] = 0x0453;
+  t['glottalinvertedstroke'] = 0x01BE;
+  t['glottalstop'] = 0x0294;
+  t['glottalstopinverted'] = 0x0296;
+  t['glottalstopmod'] = 0x02C0;
+  t['glottalstopreversed'] = 0x0295;
+  t['glottalstopreversedmod'] = 0x02C1;
+  t['glottalstopreversedsuperior'] = 0x02E4;
+  t['glottalstopstroke'] = 0x02A1;
+  t['glottalstopstrokereversed'] = 0x02A2;
+  t['gmacron'] = 0x1E21;
+  t['gmonospace'] = 0xFF47;
+  t['gohiragana'] = 0x3054;
+  t['gokatakana'] = 0x30B4;
+  t['gparen'] = 0x24A2;
+  t['gpasquare'] = 0x33AC;
+  t['gradient'] = 0x2207;
+  t['grave'] = 0x0060;
+  t['gravebelowcmb'] = 0x0316;
+  t['gravecmb'] = 0x0300;
+  t['gravecomb'] = 0x0300;
+  t['gravedeva'] = 0x0953;
+  t['gravelowmod'] = 0x02CE;
+  t['gravemonospace'] = 0xFF40;
+  t['gravetonecmb'] = 0x0340;
+  t['greater'] = 0x003E;
+  t['greaterequal'] = 0x2265;
+  t['greaterequalorless'] = 0x22DB;
+  t['greatermonospace'] = 0xFF1E;
+  t['greaterorequivalent'] = 0x2273;
+  t['greaterorless'] = 0x2277;
+  t['greateroverequal'] = 0x2267;
+  t['greatersmall'] = 0xFE65;
+  t['gscript'] = 0x0261;
+  t['gstroke'] = 0x01E5;
+  t['guhiragana'] = 0x3050;
+  t['guillemotleft'] = 0x00AB;
+  t['guillemotright'] = 0x00BB;
+  t['guilsinglleft'] = 0x2039;
+  t['guilsinglright'] = 0x203A;
+  t['gukatakana'] = 0x30B0;
+  t['guramusquare'] = 0x3318;
+  t['gysquare'] = 0x33C9;
+  t['h'] = 0x0068;
+  t['haabkhasiancyrillic'] = 0x04A9;
+  t['haaltonearabic'] = 0x06C1;
+  t['habengali'] = 0x09B9;
+  t['hadescendercyrillic'] = 0x04B3;
+  t['hadeva'] = 0x0939;
+  t['hagujarati'] = 0x0AB9;
+  t['hagurmukhi'] = 0x0A39;
+  t['haharabic'] = 0x062D;
+  t['hahfinalarabic'] = 0xFEA2;
+  t['hahinitialarabic'] = 0xFEA3;
+  t['hahiragana'] = 0x306F;
+  t['hahmedialarabic'] = 0xFEA4;
+  t['haitusquare'] = 0x332A;
+  t['hakatakana'] = 0x30CF;
+  t['hakatakanahalfwidth'] = 0xFF8A;
+  t['halantgurmukhi'] = 0x0A4D;
+  t['hamzaarabic'] = 0x0621;
+  t['hamzalowarabic'] = 0x0621;
+  t['hangulfiller'] = 0x3164;
+  t['hardsigncyrillic'] = 0x044A;
+  t['harpoonleftbarbup'] = 0x21BC;
+  t['harpoonrightbarbup'] = 0x21C0;
+  t['hasquare'] = 0x33CA;
+  t['hatafpatah'] = 0x05B2;
+  t['hatafpatah16'] = 0x05B2;
+  t['hatafpatah23'] = 0x05B2;
+  t['hatafpatah2f'] = 0x05B2;
+  t['hatafpatahhebrew'] = 0x05B2;
+  t['hatafpatahnarrowhebrew'] = 0x05B2;
+  t['hatafpatahquarterhebrew'] = 0x05B2;
+  t['hatafpatahwidehebrew'] = 0x05B2;
+  t['hatafqamats'] = 0x05B3;
+  t['hatafqamats1b'] = 0x05B3;
+  t['hatafqamats28'] = 0x05B3;
+  t['hatafqamats34'] = 0x05B3;
+  t['hatafqamatshebrew'] = 0x05B3;
+  t['hatafqamatsnarrowhebrew'] = 0x05B3;
+  t['hatafqamatsquarterhebrew'] = 0x05B3;
+  t['hatafqamatswidehebrew'] = 0x05B3;
+  t['hatafsegol'] = 0x05B1;
+  t['hatafsegol17'] = 0x05B1;
+  t['hatafsegol24'] = 0x05B1;
+  t['hatafsegol30'] = 0x05B1;
+  t['hatafsegolhebrew'] = 0x05B1;
+  t['hatafsegolnarrowhebrew'] = 0x05B1;
+  t['hatafsegolquarterhebrew'] = 0x05B1;
+  t['hatafsegolwidehebrew'] = 0x05B1;
+  t['hbar'] = 0x0127;
+  t['hbopomofo'] = 0x310F;
+  t['hbrevebelow'] = 0x1E2B;
+  t['hcedilla'] = 0x1E29;
+  t['hcircle'] = 0x24D7;
+  t['hcircumflex'] = 0x0125;
+  t['hdieresis'] = 0x1E27;
+  t['hdotaccent'] = 0x1E23;
+  t['hdotbelow'] = 0x1E25;
+  t['he'] = 0x05D4;
+  t['heart'] = 0x2665;
+  t['heartsuitblack'] = 0x2665;
+  t['heartsuitwhite'] = 0x2661;
+  t['hedagesh'] = 0xFB34;
+  t['hedageshhebrew'] = 0xFB34;
+  t['hehaltonearabic'] = 0x06C1;
+  t['heharabic'] = 0x0647;
+  t['hehebrew'] = 0x05D4;
+  t['hehfinalaltonearabic'] = 0xFBA7;
+  t['hehfinalalttwoarabic'] = 0xFEEA;
+  t['hehfinalarabic'] = 0xFEEA;
+  t['hehhamzaabovefinalarabic'] = 0xFBA5;
+  t['hehhamzaaboveisolatedarabic'] = 0xFBA4;
+  t['hehinitialaltonearabic'] = 0xFBA8;
+  t['hehinitialarabic'] = 0xFEEB;
+  t['hehiragana'] = 0x3078;
+  t['hehmedialaltonearabic'] = 0xFBA9;
+  t['hehmedialarabic'] = 0xFEEC;
+  t['heiseierasquare'] = 0x337B;
+  t['hekatakana'] = 0x30D8;
+  t['hekatakanahalfwidth'] = 0xFF8D;
+  t['hekutaarusquare'] = 0x3336;
+  t['henghook'] = 0x0267;
+  t['herutusquare'] = 0x3339;
+  t['het'] = 0x05D7;
+  t['hethebrew'] = 0x05D7;
+  t['hhook'] = 0x0266;
+  t['hhooksuperior'] = 0x02B1;
+  t['hieuhacirclekorean'] = 0x327B;
+  t['hieuhaparenkorean'] = 0x321B;
+  t['hieuhcirclekorean'] = 0x326D;
+  t['hieuhkorean'] = 0x314E;
+  t['hieuhparenkorean'] = 0x320D;
+  t['hihiragana'] = 0x3072;
+  t['hikatakana'] = 0x30D2;
+  t['hikatakanahalfwidth'] = 0xFF8B;
+  t['hiriq'] = 0x05B4;
+  t['hiriq14'] = 0x05B4;
+  t['hiriq21'] = 0x05B4;
+  t['hiriq2d'] = 0x05B4;
+  t['hiriqhebrew'] = 0x05B4;
+  t['hiriqnarrowhebrew'] = 0x05B4;
+  t['hiriqquarterhebrew'] = 0x05B4;
+  t['hiriqwidehebrew'] = 0x05B4;
+  t['hlinebelow'] = 0x1E96;
+  t['hmonospace'] = 0xFF48;
+  t['hoarmenian'] = 0x0570;
+  t['hohipthai'] = 0x0E2B;
+  t['hohiragana'] = 0x307B;
+  t['hokatakana'] = 0x30DB;
+  t['hokatakanahalfwidth'] = 0xFF8E;
+  t['holam'] = 0x05B9;
+  t['holam19'] = 0x05B9;
+  t['holam26'] = 0x05B9;
+  t['holam32'] = 0x05B9;
+  t['holamhebrew'] = 0x05B9;
+  t['holamnarrowhebrew'] = 0x05B9;
+  t['holamquarterhebrew'] = 0x05B9;
+  t['holamwidehebrew'] = 0x05B9;
+  t['honokhukthai'] = 0x0E2E;
+  t['hookabovecomb'] = 0x0309;
+  t['hookcmb'] = 0x0309;
+  t['hookpalatalizedbelowcmb'] = 0x0321;
+  t['hookretroflexbelowcmb'] = 0x0322;
+  t['hoonsquare'] = 0x3342;
+  t['horicoptic'] = 0x03E9;
+  t['horizontalbar'] = 0x2015;
+  t['horncmb'] = 0x031B;
+  t['hotsprings'] = 0x2668;
+  t['house'] = 0x2302;
+  t['hparen'] = 0x24A3;
+  t['hsuperior'] = 0x02B0;
+  t['hturned'] = 0x0265;
+  t['huhiragana'] = 0x3075;
+  t['huiitosquare'] = 0x3333;
+  t['hukatakana'] = 0x30D5;
+  t['hukatakanahalfwidth'] = 0xFF8C;
+  t['hungarumlaut'] = 0x02DD;
+  t['hungarumlautcmb'] = 0x030B;
+  t['hv'] = 0x0195;
+  t['hyphen'] = 0x002D;
+  t['hypheninferior'] = 0xF6E5;
+  t['hyphenmonospace'] = 0xFF0D;
+  t['hyphensmall'] = 0xFE63;
+  t['hyphensuperior'] = 0xF6E6;
+  t['hyphentwo'] = 0x2010;
+  t['i'] = 0x0069;
+  t['iacute'] = 0x00ED;
+  t['iacyrillic'] = 0x044F;
+  t['ibengali'] = 0x0987;
+  t['ibopomofo'] = 0x3127;
+  t['ibreve'] = 0x012D;
+  t['icaron'] = 0x01D0;
+  t['icircle'] = 0x24D8;
+  t['icircumflex'] = 0x00EE;
+  t['icyrillic'] = 0x0456;
+  t['idblgrave'] = 0x0209;
+  t['ideographearthcircle'] = 0x328F;
+  t['ideographfirecircle'] = 0x328B;
+  t['ideographicallianceparen'] = 0x323F;
+  t['ideographiccallparen'] = 0x323A;
+  t['ideographiccentrecircle'] = 0x32A5;
+  t['ideographicclose'] = 0x3006;
+  t['ideographiccomma'] = 0x3001;
+  t['ideographiccommaleft'] = 0xFF64;
+  t['ideographiccongratulationparen'] = 0x3237;
+  t['ideographiccorrectcircle'] = 0x32A3;
+  t['ideographicearthparen'] = 0x322F;
+  t['ideographicenterpriseparen'] = 0x323D;
+  t['ideographicexcellentcircle'] = 0x329D;
+  t['ideographicfestivalparen'] = 0x3240;
+  t['ideographicfinancialcircle'] = 0x3296;
+  t['ideographicfinancialparen'] = 0x3236;
+  t['ideographicfireparen'] = 0x322B;
+  t['ideographichaveparen'] = 0x3232;
+  t['ideographichighcircle'] = 0x32A4;
+  t['ideographiciterationmark'] = 0x3005;
+  t['ideographiclaborcircle'] = 0x3298;
+  t['ideographiclaborparen'] = 0x3238;
+  t['ideographicleftcircle'] = 0x32A7;
+  t['ideographiclowcircle'] = 0x32A6;
+  t['ideographicmedicinecircle'] = 0x32A9;
+  t['ideographicmetalparen'] = 0x322E;
+  t['ideographicmoonparen'] = 0x322A;
+  t['ideographicnameparen'] = 0x3234;
+  t['ideographicperiod'] = 0x3002;
+  t['ideographicprintcircle'] = 0x329E;
+  t['ideographicreachparen'] = 0x3243;
+  t['ideographicrepresentparen'] = 0x3239;
+  t['ideographicresourceparen'] = 0x323E;
+  t['ideographicrightcircle'] = 0x32A8;
+  t['ideographicsecretcircle'] = 0x3299;
+  t['ideographicselfparen'] = 0x3242;
+  t['ideographicsocietyparen'] = 0x3233;
+  t['ideographicspace'] = 0x3000;
+  t['ideographicspecialparen'] = 0x3235;
+  t['ideographicstockparen'] = 0x3231;
+  t['ideographicstudyparen'] = 0x323B;
+  t['ideographicsunparen'] = 0x3230;
+  t['ideographicsuperviseparen'] = 0x323C;
+  t['ideographicwaterparen'] = 0x322C;
+  t['ideographicwoodparen'] = 0x322D;
+  t['ideographiczero'] = 0x3007;
+  t['ideographmetalcircle'] = 0x328E;
+  t['ideographmooncircle'] = 0x328A;
+  t['ideographnamecircle'] = 0x3294;
+  t['ideographsuncircle'] = 0x3290;
+  t['ideographwatercircle'] = 0x328C;
+  t['ideographwoodcircle'] = 0x328D;
+  t['ideva'] = 0x0907;
+  t['idieresis'] = 0x00EF;
+  t['idieresisacute'] = 0x1E2F;
+  t['idieresiscyrillic'] = 0x04E5;
+  t['idotbelow'] = 0x1ECB;
+  t['iebrevecyrillic'] = 0x04D7;
+  t['iecyrillic'] = 0x0435;
+  t['ieungacirclekorean'] = 0x3275;
+  t['ieungaparenkorean'] = 0x3215;
+  t['ieungcirclekorean'] = 0x3267;
+  t['ieungkorean'] = 0x3147;
+  t['ieungparenkorean'] = 0x3207;
+  t['igrave'] = 0x00EC;
+  t['igujarati'] = 0x0A87;
+  t['igurmukhi'] = 0x0A07;
+  t['ihiragana'] = 0x3044;
+  t['ihookabove'] = 0x1EC9;
+  t['iibengali'] = 0x0988;
+  t['iicyrillic'] = 0x0438;
+  t['iideva'] = 0x0908;
+  t['iigujarati'] = 0x0A88;
+  t['iigurmukhi'] = 0x0A08;
+  t['iimatragurmukhi'] = 0x0A40;
+  t['iinvertedbreve'] = 0x020B;
+  t['iishortcyrillic'] = 0x0439;
+  t['iivowelsignbengali'] = 0x09C0;
+  t['iivowelsigndeva'] = 0x0940;
+  t['iivowelsigngujarati'] = 0x0AC0;
+  t['ij'] = 0x0133;
+  t['ikatakana'] = 0x30A4;
+  t['ikatakanahalfwidth'] = 0xFF72;
+  t['ikorean'] = 0x3163;
+  t['ilde'] = 0x02DC;
+  t['iluyhebrew'] = 0x05AC;
+  t['imacron'] = 0x012B;
+  t['imacroncyrillic'] = 0x04E3;
+  t['imageorapproximatelyequal'] = 0x2253;
+  t['imatragurmukhi'] = 0x0A3F;
+  t['imonospace'] = 0xFF49;
+  t['increment'] = 0x2206;
+  t['infinity'] = 0x221E;
+  t['iniarmenian'] = 0x056B;
+  t['integral'] = 0x222B;
+  t['integralbottom'] = 0x2321;
+  t['integralbt'] = 0x2321;
+  t['integralex'] = 0xF8F5;
+  t['integraltop'] = 0x2320;
+  t['integraltp'] = 0x2320;
+  t['intersection'] = 0x2229;
+  t['intisquare'] = 0x3305;
+  t['invbullet'] = 0x25D8;
+  t['invcircle'] = 0x25D9;
+  t['invsmileface'] = 0x263B;
+  t['iocyrillic'] = 0x0451;
+  t['iogonek'] = 0x012F;
+  t['iota'] = 0x03B9;
+  t['iotadieresis'] = 0x03CA;
+  t['iotadieresistonos'] = 0x0390;
+  t['iotalatin'] = 0x0269;
+  t['iotatonos'] = 0x03AF;
+  t['iparen'] = 0x24A4;
+  t['irigurmukhi'] = 0x0A72;
+  t['ismallhiragana'] = 0x3043;
+  t['ismallkatakana'] = 0x30A3;
+  t['ismallkatakanahalfwidth'] = 0xFF68;
+  t['issharbengali'] = 0x09FA;
+  t['istroke'] = 0x0268;
+  t['isuperior'] = 0xF6ED;
+  t['iterationhiragana'] = 0x309D;
+  t['iterationkatakana'] = 0x30FD;
+  t['itilde'] = 0x0129;
+  t['itildebelow'] = 0x1E2D;
+  t['iubopomofo'] = 0x3129;
+  t['iucyrillic'] = 0x044E;
+  t['ivowelsignbengali'] = 0x09BF;
+  t['ivowelsigndeva'] = 0x093F;
+  t['ivowelsigngujarati'] = 0x0ABF;
+  t['izhitsacyrillic'] = 0x0475;
+  t['izhitsadblgravecyrillic'] = 0x0477;
+  t['j'] = 0x006A;
+  t['jaarmenian'] = 0x0571;
+  t['jabengali'] = 0x099C;
+  t['jadeva'] = 0x091C;
+  t['jagujarati'] = 0x0A9C;
+  t['jagurmukhi'] = 0x0A1C;
+  t['jbopomofo'] = 0x3110;
+  t['jcaron'] = 0x01F0;
+  t['jcircle'] = 0x24D9;
+  t['jcircumflex'] = 0x0135;
+  t['jcrossedtail'] = 0x029D;
+  t['jdotlessstroke'] = 0x025F;
+  t['jecyrillic'] = 0x0458;
+  t['jeemarabic'] = 0x062C;
+  t['jeemfinalarabic'] = 0xFE9E;
+  t['jeeminitialarabic'] = 0xFE9F;
+  t['jeemmedialarabic'] = 0xFEA0;
+  t['jeharabic'] = 0x0698;
+  t['jehfinalarabic'] = 0xFB8B;
+  t['jhabengali'] = 0x099D;
+  t['jhadeva'] = 0x091D;
+  t['jhagujarati'] = 0x0A9D;
+  t['jhagurmukhi'] = 0x0A1D;
+  t['jheharmenian'] = 0x057B;
+  t['jis'] = 0x3004;
+  t['jmonospace'] = 0xFF4A;
+  t['jparen'] = 0x24A5;
+  t['jsuperior'] = 0x02B2;
+  t['k'] = 0x006B;
+  t['kabashkircyrillic'] = 0x04A1;
+  t['kabengali'] = 0x0995;
+  t['kacute'] = 0x1E31;
+  t['kacyrillic'] = 0x043A;
+  t['kadescendercyrillic'] = 0x049B;
+  t['kadeva'] = 0x0915;
+  t['kaf'] = 0x05DB;
+  t['kafarabic'] = 0x0643;
+  t['kafdagesh'] = 0xFB3B;
+  t['kafdageshhebrew'] = 0xFB3B;
+  t['kaffinalarabic'] = 0xFEDA;
+  t['kafhebrew'] = 0x05DB;
+  t['kafinitialarabic'] = 0xFEDB;
+  t['kafmedialarabic'] = 0xFEDC;
+  t['kafrafehebrew'] = 0xFB4D;
+  t['kagujarati'] = 0x0A95;
+  t['kagurmukhi'] = 0x0A15;
+  t['kahiragana'] = 0x304B;
+  t['kahookcyrillic'] = 0x04C4;
+  t['kakatakana'] = 0x30AB;
+  t['kakatakanahalfwidth'] = 0xFF76;
+  t['kappa'] = 0x03BA;
+  t['kappasymbolgreek'] = 0x03F0;
+  t['kapyeounmieumkorean'] = 0x3171;
+  t['kapyeounphieuphkorean'] = 0x3184;
+  t['kapyeounpieupkorean'] = 0x3178;
+  t['kapyeounssangpieupkorean'] = 0x3179;
+  t['karoriisquare'] = 0x330D;
+  t['kashidaautoarabic'] = 0x0640;
+  t['kashidaautonosidebearingarabic'] = 0x0640;
+  t['kasmallkatakana'] = 0x30F5;
+  t['kasquare'] = 0x3384;
+  t['kasraarabic'] = 0x0650;
+  t['kasratanarabic'] = 0x064D;
+  t['kastrokecyrillic'] = 0x049F;
+  t['katahiraprolongmarkhalfwidth'] = 0xFF70;
+  t['kaverticalstrokecyrillic'] = 0x049D;
+  t['kbopomofo'] = 0x310E;
+  t['kcalsquare'] = 0x3389;
+  t['kcaron'] = 0x01E9;
+  t['kcedilla'] = 0x0137;
+  t['kcircle'] = 0x24DA;
+  t['kcommaaccent'] = 0x0137;
+  t['kdotbelow'] = 0x1E33;
+  t['keharmenian'] = 0x0584;
+  t['kehiragana'] = 0x3051;
+  t['kekatakana'] = 0x30B1;
+  t['kekatakanahalfwidth'] = 0xFF79;
+  t['kenarmenian'] = 0x056F;
+  t['kesmallkatakana'] = 0x30F6;
+  t['kgreenlandic'] = 0x0138;
+  t['khabengali'] = 0x0996;
+  t['khacyrillic'] = 0x0445;
+  t['khadeva'] = 0x0916;
+  t['khagujarati'] = 0x0A96;
+  t['khagurmukhi'] = 0x0A16;
+  t['khaharabic'] = 0x062E;
+  t['khahfinalarabic'] = 0xFEA6;
+  t['khahinitialarabic'] = 0xFEA7;
+  t['khahmedialarabic'] = 0xFEA8;
+  t['kheicoptic'] = 0x03E7;
+  t['khhadeva'] = 0x0959;
+  t['khhagurmukhi'] = 0x0A59;
+  t['khieukhacirclekorean'] = 0x3278;
+  t['khieukhaparenkorean'] = 0x3218;
+  t['khieukhcirclekorean'] = 0x326A;
+  t['khieukhkorean'] = 0x314B;
+  t['khieukhparenkorean'] = 0x320A;
+  t['khokhaithai'] = 0x0E02;
+  t['khokhonthai'] = 0x0E05;
+  t['khokhuatthai'] = 0x0E03;
+  t['khokhwaithai'] = 0x0E04;
+  t['khomutthai'] = 0x0E5B;
+  t['khook'] = 0x0199;
+  t['khorakhangthai'] = 0x0E06;
+  t['khzsquare'] = 0x3391;
+  t['kihiragana'] = 0x304D;
+  t['kikatakana'] = 0x30AD;
+  t['kikatakanahalfwidth'] = 0xFF77;
+  t['kiroguramusquare'] = 0x3315;
+  t['kiromeetorusquare'] = 0x3316;
+  t['kirosquare'] = 0x3314;
+  t['kiyeokacirclekorean'] = 0x326E;
+  t['kiyeokaparenkorean'] = 0x320E;
+  t['kiyeokcirclekorean'] = 0x3260;
+  t['kiyeokkorean'] = 0x3131;
+  t['kiyeokparenkorean'] = 0x3200;
+  t['kiyeoksioskorean'] = 0x3133;
+  t['kjecyrillic'] = 0x045C;
+  t['klinebelow'] = 0x1E35;
+  t['klsquare'] = 0x3398;
+  t['kmcubedsquare'] = 0x33A6;
+  t['kmonospace'] = 0xFF4B;
+  t['kmsquaredsquare'] = 0x33A2;
+  t['kohiragana'] = 0x3053;
+  t['kohmsquare'] = 0x33C0;
+  t['kokaithai'] = 0x0E01;
+  t['kokatakana'] = 0x30B3;
+  t['kokatakanahalfwidth'] = 0xFF7A;
+  t['kooposquare'] = 0x331E;
+  t['koppacyrillic'] = 0x0481;
+  t['koreanstandardsymbol'] = 0x327F;
+  t['koroniscmb'] = 0x0343;
+  t['kparen'] = 0x24A6;
+  t['kpasquare'] = 0x33AA;
+  t['ksicyrillic'] = 0x046F;
+  t['ktsquare'] = 0x33CF;
+  t['kturned'] = 0x029E;
+  t['kuhiragana'] = 0x304F;
+  t['kukatakana'] = 0x30AF;
+  t['kukatakanahalfwidth'] = 0xFF78;
+  t['kvsquare'] = 0x33B8;
+  t['kwsquare'] = 0x33BE;
+  t['l'] = 0x006C;
+  t['labengali'] = 0x09B2;
+  t['lacute'] = 0x013A;
+  t['ladeva'] = 0x0932;
+  t['lagujarati'] = 0x0AB2;
+  t['lagurmukhi'] = 0x0A32;
+  t['lakkhangyaothai'] = 0x0E45;
+  t['lamaleffinalarabic'] = 0xFEFC;
+  t['lamalefhamzaabovefinalarabic'] = 0xFEF8;
+  t['lamalefhamzaaboveisolatedarabic'] = 0xFEF7;
+  t['lamalefhamzabelowfinalarabic'] = 0xFEFA;
+  t['lamalefhamzabelowisolatedarabic'] = 0xFEF9;
+  t['lamalefisolatedarabic'] = 0xFEFB;
+  t['lamalefmaddaabovefinalarabic'] = 0xFEF6;
+  t['lamalefmaddaaboveisolatedarabic'] = 0xFEF5;
+  t['lamarabic'] = 0x0644;
+  t['lambda'] = 0x03BB;
+  t['lambdastroke'] = 0x019B;
+  t['lamed'] = 0x05DC;
+  t['lameddagesh'] = 0xFB3C;
+  t['lameddageshhebrew'] = 0xFB3C;
+  t['lamedhebrew'] = 0x05DC;
+  t['lamfinalarabic'] = 0xFEDE;
+  t['lamhahinitialarabic'] = 0xFCCA;
+  t['laminitialarabic'] = 0xFEDF;
+  t['lamjeeminitialarabic'] = 0xFCC9;
+  t['lamkhahinitialarabic'] = 0xFCCB;
+  t['lamlamhehisolatedarabic'] = 0xFDF2;
+  t['lammedialarabic'] = 0xFEE0;
+  t['lammeemhahinitialarabic'] = 0xFD88;
+  t['lammeeminitialarabic'] = 0xFCCC;
+  t['largecircle'] = 0x25EF;
+  t['lbar'] = 0x019A;
+  t['lbelt'] = 0x026C;
+  t['lbopomofo'] = 0x310C;
+  t['lcaron'] = 0x013E;
+  t['lcedilla'] = 0x013C;
+  t['lcircle'] = 0x24DB;
+  t['lcircumflexbelow'] = 0x1E3D;
+  t['lcommaaccent'] = 0x013C;
+  t['ldot'] = 0x0140;
+  t['ldotaccent'] = 0x0140;
+  t['ldotbelow'] = 0x1E37;
+  t['ldotbelowmacron'] = 0x1E39;
+  t['leftangleabovecmb'] = 0x031A;
+  t['lefttackbelowcmb'] = 0x0318;
+  t['less'] = 0x003C;
+  t['lessequal'] = 0x2264;
+  t['lessequalorgreater'] = 0x22DA;
+  t['lessmonospace'] = 0xFF1C;
+  t['lessorequivalent'] = 0x2272;
+  t['lessorgreater'] = 0x2276;
+  t['lessoverequal'] = 0x2266;
+  t['lesssmall'] = 0xFE64;
+  t['lezh'] = 0x026E;
+  t['lfblock'] = 0x258C;
+  t['lhookretroflex'] = 0x026D;
+  t['lira'] = 0x20A4;
+  t['liwnarmenian'] = 0x056C;
+  t['lj'] = 0x01C9;
+  t['ljecyrillic'] = 0x0459;
+  t['ll'] = 0xF6C0;
+  t['lladeva'] = 0x0933;
+  t['llagujarati'] = 0x0AB3;
+  t['llinebelow'] = 0x1E3B;
+  t['llladeva'] = 0x0934;
+  t['llvocalicbengali'] = 0x09E1;
+  t['llvocalicdeva'] = 0x0961;
+  t['llvocalicvowelsignbengali'] = 0x09E3;
+  t['llvocalicvowelsigndeva'] = 0x0963;
+  t['lmiddletilde'] = 0x026B;
+  t['lmonospace'] = 0xFF4C;
+  t['lmsquare'] = 0x33D0;
+  t['lochulathai'] = 0x0E2C;
+  t['logicaland'] = 0x2227;
+  t['logicalnot'] = 0x00AC;
+  t['logicalnotreversed'] = 0x2310;
+  t['logicalor'] = 0x2228;
+  t['lolingthai'] = 0x0E25;
+  t['longs'] = 0x017F;
+  t['lowlinecenterline'] = 0xFE4E;
+  t['lowlinecmb'] = 0x0332;
+  t['lowlinedashed'] = 0xFE4D;
+  t['lozenge'] = 0x25CA;
+  t['lparen'] = 0x24A7;
+  t['lslash'] = 0x0142;
+  t['lsquare'] = 0x2113;
+  t['lsuperior'] = 0xF6EE;
+  t['ltshade'] = 0x2591;
+  t['luthai'] = 0x0E26;
+  t['lvocalicbengali'] = 0x098C;
+  t['lvocalicdeva'] = 0x090C;
+  t['lvocalicvowelsignbengali'] = 0x09E2;
+  t['lvocalicvowelsigndeva'] = 0x0962;
+  t['lxsquare'] = 0x33D3;
+  t['m'] = 0x006D;
+  t['mabengali'] = 0x09AE;
+  t['macron'] = 0x00AF;
+  t['macronbelowcmb'] = 0x0331;
+  t['macroncmb'] = 0x0304;
+  t['macronlowmod'] = 0x02CD;
+  t['macronmonospace'] = 0xFFE3;
+  t['macute'] = 0x1E3F;
+  t['madeva'] = 0x092E;
+  t['magujarati'] = 0x0AAE;
+  t['magurmukhi'] = 0x0A2E;
+  t['mahapakhhebrew'] = 0x05A4;
+  t['mahapakhlefthebrew'] = 0x05A4;
+  t['mahiragana'] = 0x307E;
+  t['maichattawalowleftthai'] = 0xF895;
+  t['maichattawalowrightthai'] = 0xF894;
+  t['maichattawathai'] = 0x0E4B;
+  t['maichattawaupperleftthai'] = 0xF893;
+  t['maieklowleftthai'] = 0xF88C;
+  t['maieklowrightthai'] = 0xF88B;
+  t['maiekthai'] = 0x0E48;
+  t['maiekupperleftthai'] = 0xF88A;
+  t['maihanakatleftthai'] = 0xF884;
+  t['maihanakatthai'] = 0x0E31;
+  t['maitaikhuleftthai'] = 0xF889;
+  t['maitaikhuthai'] = 0x0E47;
+  t['maitholowleftthai'] = 0xF88F;
+  t['maitholowrightthai'] = 0xF88E;
+  t['maithothai'] = 0x0E49;
+  t['maithoupperleftthai'] = 0xF88D;
+  t['maitrilowleftthai'] = 0xF892;
+  t['maitrilowrightthai'] = 0xF891;
+  t['maitrithai'] = 0x0E4A;
+  t['maitriupperleftthai'] = 0xF890;
+  t['maiyamokthai'] = 0x0E46;
+  t['makatakana'] = 0x30DE;
+  t['makatakanahalfwidth'] = 0xFF8F;
+  t['male'] = 0x2642;
+  t['mansyonsquare'] = 0x3347;
+  t['maqafhebrew'] = 0x05BE;
+  t['mars'] = 0x2642;
+  t['masoracirclehebrew'] = 0x05AF;
+  t['masquare'] = 0x3383;
+  t['mbopomofo'] = 0x3107;
+  t['mbsquare'] = 0x33D4;
+  t['mcircle'] = 0x24DC;
+  t['mcubedsquare'] = 0x33A5;
+  t['mdotaccent'] = 0x1E41;
+  t['mdotbelow'] = 0x1E43;
+  t['meemarabic'] = 0x0645;
+  t['meemfinalarabic'] = 0xFEE2;
+  t['meeminitialarabic'] = 0xFEE3;
+  t['meemmedialarabic'] = 0xFEE4;
+  t['meemmeeminitialarabic'] = 0xFCD1;
+  t['meemmeemisolatedarabic'] = 0xFC48;
+  t['meetorusquare'] = 0x334D;
+  t['mehiragana'] = 0x3081;
+  t['meizierasquare'] = 0x337E;
+  t['mekatakana'] = 0x30E1;
+  t['mekatakanahalfwidth'] = 0xFF92;
+  t['mem'] = 0x05DE;
+  t['memdagesh'] = 0xFB3E;
+  t['memdageshhebrew'] = 0xFB3E;
+  t['memhebrew'] = 0x05DE;
+  t['menarmenian'] = 0x0574;
+  t['merkhahebrew'] = 0x05A5;
+  t['merkhakefulahebrew'] = 0x05A6;
+  t['merkhakefulalefthebrew'] = 0x05A6;
+  t['merkhalefthebrew'] = 0x05A5;
+  t['mhook'] = 0x0271;
+  t['mhzsquare'] = 0x3392;
+  t['middledotkatakanahalfwidth'] = 0xFF65;
+  t['middot'] = 0x00B7;
+  t['mieumacirclekorean'] = 0x3272;
+  t['mieumaparenkorean'] = 0x3212;
+  t['mieumcirclekorean'] = 0x3264;
+  t['mieumkorean'] = 0x3141;
+  t['mieumpansioskorean'] = 0x3170;
+  t['mieumparenkorean'] = 0x3204;
+  t['mieumpieupkorean'] = 0x316E;
+  t['mieumsioskorean'] = 0x316F;
+  t['mihiragana'] = 0x307F;
+  t['mikatakana'] = 0x30DF;
+  t['mikatakanahalfwidth'] = 0xFF90;
+  t['minus'] = 0x2212;
+  t['minusbelowcmb'] = 0x0320;
+  t['minuscircle'] = 0x2296;
+  t['minusmod'] = 0x02D7;
+  t['minusplus'] = 0x2213;
+  t['minute'] = 0x2032;
+  t['miribaarusquare'] = 0x334A;
+  t['mirisquare'] = 0x3349;
+  t['mlonglegturned'] = 0x0270;
+  t['mlsquare'] = 0x3396;
+  t['mmcubedsquare'] = 0x33A3;
+  t['mmonospace'] = 0xFF4D;
+  t['mmsquaredsquare'] = 0x339F;
+  t['mohiragana'] = 0x3082;
+  t['mohmsquare'] = 0x33C1;
+  t['mokatakana'] = 0x30E2;
+  t['mokatakanahalfwidth'] = 0xFF93;
+  t['molsquare'] = 0x33D6;
+  t['momathai'] = 0x0E21;
+  t['moverssquare'] = 0x33A7;
+  t['moverssquaredsquare'] = 0x33A8;
+  t['mparen'] = 0x24A8;
+  t['mpasquare'] = 0x33AB;
+  t['mssquare'] = 0x33B3;
+  t['msuperior'] = 0xF6EF;
+  t['mturned'] = 0x026F;
+  t['mu'] = 0x00B5;
+  t['mu1'] = 0x00B5;
+  t['muasquare'] = 0x3382;
+  t['muchgreater'] = 0x226B;
+  t['muchless'] = 0x226A;
+  t['mufsquare'] = 0x338C;
+  t['mugreek'] = 0x03BC;
+  t['mugsquare'] = 0x338D;
+  t['muhiragana'] = 0x3080;
+  t['mukatakana'] = 0x30E0;
+  t['mukatakanahalfwidth'] = 0xFF91;
+  t['mulsquare'] = 0x3395;
+  t['multiply'] = 0x00D7;
+  t['mumsquare'] = 0x339B;
+  t['munahhebrew'] = 0x05A3;
+  t['munahlefthebrew'] = 0x05A3;
+  t['musicalnote'] = 0x266A;
+  t['musicalnotedbl'] = 0x266B;
+  t['musicflatsign'] = 0x266D;
+  t['musicsharpsign'] = 0x266F;
+  t['mussquare'] = 0x33B2;
+  t['muvsquare'] = 0x33B6;
+  t['muwsquare'] = 0x33BC;
+  t['mvmegasquare'] = 0x33B9;
+  t['mvsquare'] = 0x33B7;
+  t['mwmegasquare'] = 0x33BF;
+  t['mwsquare'] = 0x33BD;
+  t['n'] = 0x006E;
+  t['nabengali'] = 0x09A8;
+  t['nabla'] = 0x2207;
+  t['nacute'] = 0x0144;
+  t['nadeva'] = 0x0928;
+  t['nagujarati'] = 0x0AA8;
+  t['nagurmukhi'] = 0x0A28;
+  t['nahiragana'] = 0x306A;
+  t['nakatakana'] = 0x30CA;
+  t['nakatakanahalfwidth'] = 0xFF85;
+  t['napostrophe'] = 0x0149;
+  t['nasquare'] = 0x3381;
+  t['nbopomofo'] = 0x310B;
+  t['nbspace'] = 0x00A0;
+  t['ncaron'] = 0x0148;
+  t['ncedilla'] = 0x0146;
+  t['ncircle'] = 0x24DD;
+  t['ncircumflexbelow'] = 0x1E4B;
+  t['ncommaaccent'] = 0x0146;
+  t['ndotaccent'] = 0x1E45;
+  t['ndotbelow'] = 0x1E47;
+  t['nehiragana'] = 0x306D;
+  t['nekatakana'] = 0x30CD;
+  t['nekatakanahalfwidth'] = 0xFF88;
+  t['newsheqelsign'] = 0x20AA;
+  t['nfsquare'] = 0x338B;
+  t['ngabengali'] = 0x0999;
+  t['ngadeva'] = 0x0919;
+  t['ngagujarati'] = 0x0A99;
+  t['ngagurmukhi'] = 0x0A19;
+  t['ngonguthai'] = 0x0E07;
+  t['nhiragana'] = 0x3093;
+  t['nhookleft'] = 0x0272;
+  t['nhookretroflex'] = 0x0273;
+  t['nieunacirclekorean'] = 0x326F;
+  t['nieunaparenkorean'] = 0x320F;
+  t['nieuncieuckorean'] = 0x3135;
+  t['nieuncirclekorean'] = 0x3261;
+  t['nieunhieuhkorean'] = 0x3136;
+  t['nieunkorean'] = 0x3134;
+  t['nieunpansioskorean'] = 0x3168;
+  t['nieunparenkorean'] = 0x3201;
+  t['nieunsioskorean'] = 0x3167;
+  t['nieuntikeutkorean'] = 0x3166;
+  t['nihiragana'] = 0x306B;
+  t['nikatakana'] = 0x30CB;
+  t['nikatakanahalfwidth'] = 0xFF86;
+  t['nikhahitleftthai'] = 0xF899;
+  t['nikhahitthai'] = 0x0E4D;
+  t['nine'] = 0x0039;
+  t['ninearabic'] = 0x0669;
+  t['ninebengali'] = 0x09EF;
+  t['ninecircle'] = 0x2468;
+  t['ninecircleinversesansserif'] = 0x2792;
+  t['ninedeva'] = 0x096F;
+  t['ninegujarati'] = 0x0AEF;
+  t['ninegurmukhi'] = 0x0A6F;
+  t['ninehackarabic'] = 0x0669;
+  t['ninehangzhou'] = 0x3029;
+  t['nineideographicparen'] = 0x3228;
+  t['nineinferior'] = 0x2089;
+  t['ninemonospace'] = 0xFF19;
+  t['nineoldstyle'] = 0xF739;
+  t['nineparen'] = 0x247C;
+  t['nineperiod'] = 0x2490;
+  t['ninepersian'] = 0x06F9;
+  t['nineroman'] = 0x2178;
+  t['ninesuperior'] = 0x2079;
+  t['nineteencircle'] = 0x2472;
+  t['nineteenparen'] = 0x2486;
+  t['nineteenperiod'] = 0x249A;
+  t['ninethai'] = 0x0E59;
+  t['nj'] = 0x01CC;
+  t['njecyrillic'] = 0x045A;
+  t['nkatakana'] = 0x30F3;
+  t['nkatakanahalfwidth'] = 0xFF9D;
+  t['nlegrightlong'] = 0x019E;
+  t['nlinebelow'] = 0x1E49;
+  t['nmonospace'] = 0xFF4E;
+  t['nmsquare'] = 0x339A;
+  t['nnabengali'] = 0x09A3;
+  t['nnadeva'] = 0x0923;
+  t['nnagujarati'] = 0x0AA3;
+  t['nnagurmukhi'] = 0x0A23;
+  t['nnnadeva'] = 0x0929;
+  t['nohiragana'] = 0x306E;
+  t['nokatakana'] = 0x30CE;
+  t['nokatakanahalfwidth'] = 0xFF89;
+  t['nonbreakingspace'] = 0x00A0;
+  t['nonenthai'] = 0x0E13;
+  t['nonuthai'] = 0x0E19;
+  t['noonarabic'] = 0x0646;
+  t['noonfinalarabic'] = 0xFEE6;
+  t['noonghunnaarabic'] = 0x06BA;
+  t['noonghunnafinalarabic'] = 0xFB9F;
+  t['nooninitialarabic'] = 0xFEE7;
+  t['noonjeeminitialarabic'] = 0xFCD2;
+  t['noonjeemisolatedarabic'] = 0xFC4B;
+  t['noonmedialarabic'] = 0xFEE8;
+  t['noonmeeminitialarabic'] = 0xFCD5;
+  t['noonmeemisolatedarabic'] = 0xFC4E;
+  t['noonnoonfinalarabic'] = 0xFC8D;
+  t['notcontains'] = 0x220C;
+  t['notelement'] = 0x2209;
+  t['notelementof'] = 0x2209;
+  t['notequal'] = 0x2260;
+  t['notgreater'] = 0x226F;
+  t['notgreaternorequal'] = 0x2271;
+  t['notgreaternorless'] = 0x2279;
+  t['notidentical'] = 0x2262;
+  t['notless'] = 0x226E;
+  t['notlessnorequal'] = 0x2270;
+  t['notparallel'] = 0x2226;
+  t['notprecedes'] = 0x2280;
+  t['notsubset'] = 0x2284;
+  t['notsucceeds'] = 0x2281;
+  t['notsuperset'] = 0x2285;
+  t['nowarmenian'] = 0x0576;
+  t['nparen'] = 0x24A9;
+  t['nssquare'] = 0x33B1;
+  t['nsuperior'] = 0x207F;
+  t['ntilde'] = 0x00F1;
+  t['nu'] = 0x03BD;
+  t['nuhiragana'] = 0x306C;
+  t['nukatakana'] = 0x30CC;
+  t['nukatakanahalfwidth'] = 0xFF87;
+  t['nuktabengali'] = 0x09BC;
+  t['nuktadeva'] = 0x093C;
+  t['nuktagujarati'] = 0x0ABC;
+  t['nuktagurmukhi'] = 0x0A3C;
+  t['numbersign'] = 0x0023;
+  t['numbersignmonospace'] = 0xFF03;
+  t['numbersignsmall'] = 0xFE5F;
+  t['numeralsigngreek'] = 0x0374;
+  t['numeralsignlowergreek'] = 0x0375;
+  t['numero'] = 0x2116;
+  t['nun'] = 0x05E0;
+  t['nundagesh'] = 0xFB40;
+  t['nundageshhebrew'] = 0xFB40;
+  t['nunhebrew'] = 0x05E0;
+  t['nvsquare'] = 0x33B5;
+  t['nwsquare'] = 0x33BB;
+  t['nyabengali'] = 0x099E;
+  t['nyadeva'] = 0x091E;
+  t['nyagujarati'] = 0x0A9E;
+  t['nyagurmukhi'] = 0x0A1E;
+  t['o'] = 0x006F;
+  t['oacute'] = 0x00F3;
+  t['oangthai'] = 0x0E2D;
+  t['obarred'] = 0x0275;
+  t['obarredcyrillic'] = 0x04E9;
+  t['obarreddieresiscyrillic'] = 0x04EB;
+  t['obengali'] = 0x0993;
+  t['obopomofo'] = 0x311B;
+  t['obreve'] = 0x014F;
+  t['ocandradeva'] = 0x0911;
+  t['ocandragujarati'] = 0x0A91;
+  t['ocandravowelsigndeva'] = 0x0949;
+  t['ocandravowelsigngujarati'] = 0x0AC9;
+  t['ocaron'] = 0x01D2;
+  t['ocircle'] = 0x24DE;
+  t['ocircumflex'] = 0x00F4;
+  t['ocircumflexacute'] = 0x1ED1;
+  t['ocircumflexdotbelow'] = 0x1ED9;
+  t['ocircumflexgrave'] = 0x1ED3;
+  t['ocircumflexhookabove'] = 0x1ED5;
+  t['ocircumflextilde'] = 0x1ED7;
+  t['ocyrillic'] = 0x043E;
+  t['odblacute'] = 0x0151;
+  t['odblgrave'] = 0x020D;
+  t['odeva'] = 0x0913;
+  t['odieresis'] = 0x00F6;
+  t['odieresiscyrillic'] = 0x04E7;
+  t['odotbelow'] = 0x1ECD;
+  t['oe'] = 0x0153;
+  t['oekorean'] = 0x315A;
+  t['ogonek'] = 0x02DB;
+  t['ogonekcmb'] = 0x0328;
+  t['ograve'] = 0x00F2;
+  t['ogujarati'] = 0x0A93;
+  t['oharmenian'] = 0x0585;
+  t['ohiragana'] = 0x304A;
+  t['ohookabove'] = 0x1ECF;
+  t['ohorn'] = 0x01A1;
+  t['ohornacute'] = 0x1EDB;
+  t['ohorndotbelow'] = 0x1EE3;
+  t['ohorngrave'] = 0x1EDD;
+  t['ohornhookabove'] = 0x1EDF;
+  t['ohorntilde'] = 0x1EE1;
+  t['ohungarumlaut'] = 0x0151;
+  t['oi'] = 0x01A3;
+  t['oinvertedbreve'] = 0x020F;
+  t['okatakana'] = 0x30AA;
+  t['okatakanahalfwidth'] = 0xFF75;
+  t['okorean'] = 0x3157;
+  t['olehebrew'] = 0x05AB;
+  t['omacron'] = 0x014D;
+  t['omacronacute'] = 0x1E53;
+  t['omacrongrave'] = 0x1E51;
+  t['omdeva'] = 0x0950;
+  t['omega'] = 0x03C9;
+  t['omega1'] = 0x03D6;
+  t['omegacyrillic'] = 0x0461;
+  t['omegalatinclosed'] = 0x0277;
+  t['omegaroundcyrillic'] = 0x047B;
+  t['omegatitlocyrillic'] = 0x047D;
+  t['omegatonos'] = 0x03CE;
+  t['omgujarati'] = 0x0AD0;
+  t['omicron'] = 0x03BF;
+  t['omicrontonos'] = 0x03CC;
+  t['omonospace'] = 0xFF4F;
+  t['one'] = 0x0031;
+  t['onearabic'] = 0x0661;
+  t['onebengali'] = 0x09E7;
+  t['onecircle'] = 0x2460;
+  t['onecircleinversesansserif'] = 0x278A;
+  t['onedeva'] = 0x0967;
+  t['onedotenleader'] = 0x2024;
+  t['oneeighth'] = 0x215B;
+  t['onefitted'] = 0xF6DC;
+  t['onegujarati'] = 0x0AE7;
+  t['onegurmukhi'] = 0x0A67;
+  t['onehackarabic'] = 0x0661;
+  t['onehalf'] = 0x00BD;
+  t['onehangzhou'] = 0x3021;
+  t['oneideographicparen'] = 0x3220;
+  t['oneinferior'] = 0x2081;
+  t['onemonospace'] = 0xFF11;
+  t['onenumeratorbengali'] = 0x09F4;
+  t['oneoldstyle'] = 0xF731;
+  t['oneparen'] = 0x2474;
+  t['oneperiod'] = 0x2488;
+  t['onepersian'] = 0x06F1;
+  t['onequarter'] = 0x00BC;
+  t['oneroman'] = 0x2170;
+  t['onesuperior'] = 0x00B9;
+  t['onethai'] = 0x0E51;
+  t['onethird'] = 0x2153;
+  t['oogonek'] = 0x01EB;
+  t['oogonekmacron'] = 0x01ED;
+  t['oogurmukhi'] = 0x0A13;
+  t['oomatragurmukhi'] = 0x0A4B;
+  t['oopen'] = 0x0254;
+  t['oparen'] = 0x24AA;
+  t['openbullet'] = 0x25E6;
+  t['option'] = 0x2325;
+  t['ordfeminine'] = 0x00AA;
+  t['ordmasculine'] = 0x00BA;
+  t['orthogonal'] = 0x221F;
+  t['oshortdeva'] = 0x0912;
+  t['oshortvowelsigndeva'] = 0x094A;
+  t['oslash'] = 0x00F8;
+  t['oslashacute'] = 0x01FF;
+  t['osmallhiragana'] = 0x3049;
+  t['osmallkatakana'] = 0x30A9;
+  t['osmallkatakanahalfwidth'] = 0xFF6B;
+  t['ostrokeacute'] = 0x01FF;
+  t['osuperior'] = 0xF6F0;
+  t['otcyrillic'] = 0x047F;
+  t['otilde'] = 0x00F5;
+  t['otildeacute'] = 0x1E4D;
+  t['otildedieresis'] = 0x1E4F;
+  t['oubopomofo'] = 0x3121;
+  t['overline'] = 0x203E;
+  t['overlinecenterline'] = 0xFE4A;
+  t['overlinecmb'] = 0x0305;
+  t['overlinedashed'] = 0xFE49;
+  t['overlinedblwavy'] = 0xFE4C;
+  t['overlinewavy'] = 0xFE4B;
+  t['overscore'] = 0x00AF;
+  t['ovowelsignbengali'] = 0x09CB;
+  t['ovowelsigndeva'] = 0x094B;
+  t['ovowelsigngujarati'] = 0x0ACB;
+  t['p'] = 0x0070;
+  t['paampssquare'] = 0x3380;
+  t['paasentosquare'] = 0x332B;
+  t['pabengali'] = 0x09AA;
+  t['pacute'] = 0x1E55;
+  t['padeva'] = 0x092A;
+  t['pagedown'] = 0x21DF;
+  t['pageup'] = 0x21DE;
+  t['pagujarati'] = 0x0AAA;
+  t['pagurmukhi'] = 0x0A2A;
+  t['pahiragana'] = 0x3071;
+  t['paiyannoithai'] = 0x0E2F;
+  t['pakatakana'] = 0x30D1;
+  t['palatalizationcyrilliccmb'] = 0x0484;
+  t['palochkacyrillic'] = 0x04C0;
+  t['pansioskorean'] = 0x317F;
+  t['paragraph'] = 0x00B6;
+  t['parallel'] = 0x2225;
+  t['parenleft'] = 0x0028;
+  t['parenleftaltonearabic'] = 0xFD3E;
+  t['parenleftbt'] = 0xF8ED;
+  t['parenleftex'] = 0xF8EC;
+  t['parenleftinferior'] = 0x208D;
+  t['parenleftmonospace'] = 0xFF08;
+  t['parenleftsmall'] = 0xFE59;
+  t['parenleftsuperior'] = 0x207D;
+  t['parenlefttp'] = 0xF8EB;
+  t['parenleftvertical'] = 0xFE35;
+  t['parenright'] = 0x0029;
+  t['parenrightaltonearabic'] = 0xFD3F;
+  t['parenrightbt'] = 0xF8F8;
+  t['parenrightex'] = 0xF8F7;
+  t['parenrightinferior'] = 0x208E;
+  t['parenrightmonospace'] = 0xFF09;
+  t['parenrightsmall'] = 0xFE5A;
+  t['parenrightsuperior'] = 0x207E;
+  t['parenrighttp'] = 0xF8F6;
+  t['parenrightvertical'] = 0xFE36;
+  t['partialdiff'] = 0x2202;
+  t['paseqhebrew'] = 0x05C0;
+  t['pashtahebrew'] = 0x0599;
+  t['pasquare'] = 0x33A9;
+  t['patah'] = 0x05B7;
+  t['patah11'] = 0x05B7;
+  t['patah1d'] = 0x05B7;
+  t['patah2a'] = 0x05B7;
+  t['patahhebrew'] = 0x05B7;
+  t['patahnarrowhebrew'] = 0x05B7;
+  t['patahquarterhebrew'] = 0x05B7;
+  t['patahwidehebrew'] = 0x05B7;
+  t['pazerhebrew'] = 0x05A1;
+  t['pbopomofo'] = 0x3106;
+  t['pcircle'] = 0x24DF;
+  t['pdotaccent'] = 0x1E57;
+  t['pe'] = 0x05E4;
+  t['pecyrillic'] = 0x043F;
+  t['pedagesh'] = 0xFB44;
+  t['pedageshhebrew'] = 0xFB44;
+  t['peezisquare'] = 0x333B;
+  t['pefinaldageshhebrew'] = 0xFB43;
+  t['peharabic'] = 0x067E;
+  t['peharmenian'] = 0x057A;
+  t['pehebrew'] = 0x05E4;
+  t['pehfinalarabic'] = 0xFB57;
+  t['pehinitialarabic'] = 0xFB58;
+  t['pehiragana'] = 0x307A;
+  t['pehmedialarabic'] = 0xFB59;
+  t['pekatakana'] = 0x30DA;
+  t['pemiddlehookcyrillic'] = 0x04A7;
+  t['perafehebrew'] = 0xFB4E;
+  t['percent'] = 0x0025;
+  t['percentarabic'] = 0x066A;
+  t['percentmonospace'] = 0xFF05;
+  t['percentsmall'] = 0xFE6A;
+  t['period'] = 0x002E;
+  t['periodarmenian'] = 0x0589;
+  t['periodcentered'] = 0x00B7;
+  t['periodhalfwidth'] = 0xFF61;
+  t['periodinferior'] = 0xF6E7;
+  t['periodmonospace'] = 0xFF0E;
+  t['periodsmall'] = 0xFE52;
+  t['periodsuperior'] = 0xF6E8;
+  t['perispomenigreekcmb'] = 0x0342;
+  t['perpendicular'] = 0x22A5;
+  t['perthousand'] = 0x2030;
+  t['peseta'] = 0x20A7;
+  t['pfsquare'] = 0x338A;
+  t['phabengali'] = 0x09AB;
+  t['phadeva'] = 0x092B;
+  t['phagujarati'] = 0x0AAB;
+  t['phagurmukhi'] = 0x0A2B;
+  t['phi'] = 0x03C6;
+  t['phi1'] = 0x03D5;
+  t['phieuphacirclekorean'] = 0x327A;
+  t['phieuphaparenkorean'] = 0x321A;
+  t['phieuphcirclekorean'] = 0x326C;
+  t['phieuphkorean'] = 0x314D;
+  t['phieuphparenkorean'] = 0x320C;
+  t['philatin'] = 0x0278;
+  t['phinthuthai'] = 0x0E3A;
+  t['phisymbolgreek'] = 0x03D5;
+  t['phook'] = 0x01A5;
+  t['phophanthai'] = 0x0E1E;
+  t['phophungthai'] = 0x0E1C;
+  t['phosamphaothai'] = 0x0E20;
+  t['pi'] = 0x03C0;
+  t['pieupacirclekorean'] = 0x3273;
+  t['pieupaparenkorean'] = 0x3213;
+  t['pieupcieuckorean'] = 0x3176;
+  t['pieupcirclekorean'] = 0x3265;
+  t['pieupkiyeokkorean'] = 0x3172;
+  t['pieupkorean'] = 0x3142;
+  t['pieupparenkorean'] = 0x3205;
+  t['pieupsioskiyeokkorean'] = 0x3174;
+  t['pieupsioskorean'] = 0x3144;
+  t['pieupsiostikeutkorean'] = 0x3175;
+  t['pieupthieuthkorean'] = 0x3177;
+  t['pieuptikeutkorean'] = 0x3173;
+  t['pihiragana'] = 0x3074;
+  t['pikatakana'] = 0x30D4;
+  t['pisymbolgreek'] = 0x03D6;
+  t['piwrarmenian'] = 0x0583;
+  t['plus'] = 0x002B;
+  t['plusbelowcmb'] = 0x031F;
+  t['pluscircle'] = 0x2295;
+  t['plusminus'] = 0x00B1;
+  t['plusmod'] = 0x02D6;
+  t['plusmonospace'] = 0xFF0B;
+  t['plussmall'] = 0xFE62;
+  t['plussuperior'] = 0x207A;
+  t['pmonospace'] = 0xFF50;
+  t['pmsquare'] = 0x33D8;
+  t['pohiragana'] = 0x307D;
+  t['pointingindexdownwhite'] = 0x261F;
+  t['pointingindexleftwhite'] = 0x261C;
+  t['pointingindexrightwhite'] = 0x261E;
+  t['pointingindexupwhite'] = 0x261D;
+  t['pokatakana'] = 0x30DD;
+  t['poplathai'] = 0x0E1B;
+  t['postalmark'] = 0x3012;
+  t['postalmarkface'] = 0x3020;
+  t['pparen'] = 0x24AB;
+  t['precedes'] = 0x227A;
+  t['prescription'] = 0x211E;
+  t['primemod'] = 0x02B9;
+  t['primereversed'] = 0x2035;
+  t['product'] = 0x220F;
+  t['projective'] = 0x2305;
+  t['prolongedkana'] = 0x30FC;
+  t['propellor'] = 0x2318;
+  t['propersubset'] = 0x2282;
+  t['propersuperset'] = 0x2283;
+  t['proportion'] = 0x2237;
+  t['proportional'] = 0x221D;
+  t['psi'] = 0x03C8;
+  t['psicyrillic'] = 0x0471;
+  t['psilipneumatacyrilliccmb'] = 0x0486;
+  t['pssquare'] = 0x33B0;
+  t['puhiragana'] = 0x3077;
+  t['pukatakana'] = 0x30D7;
+  t['pvsquare'] = 0x33B4;
+  t['pwsquare'] = 0x33BA;
+  t['q'] = 0x0071;
+  t['qadeva'] = 0x0958;
+  t['qadmahebrew'] = 0x05A8;
+  t['qafarabic'] = 0x0642;
+  t['qaffinalarabic'] = 0xFED6;
+  t['qafinitialarabic'] = 0xFED7;
+  t['qafmedialarabic'] = 0xFED8;
+  t['qamats'] = 0x05B8;
+  t['qamats10'] = 0x05B8;
+  t['qamats1a'] = 0x05B8;
+  t['qamats1c'] = 0x05B8;
+  t['qamats27'] = 0x05B8;
+  t['qamats29'] = 0x05B8;
+  t['qamats33'] = 0x05B8;
+  t['qamatsde'] = 0x05B8;
+  t['qamatshebrew'] = 0x05B8;
+  t['qamatsnarrowhebrew'] = 0x05B8;
+  t['qamatsqatanhebrew'] = 0x05B8;
+  t['qamatsqatannarrowhebrew'] = 0x05B8;
+  t['qamatsqatanquarterhebrew'] = 0x05B8;
+  t['qamatsqatanwidehebrew'] = 0x05B8;
+  t['qamatsquarterhebrew'] = 0x05B8;
+  t['qamatswidehebrew'] = 0x05B8;
+  t['qarneyparahebrew'] = 0x059F;
+  t['qbopomofo'] = 0x3111;
+  t['qcircle'] = 0x24E0;
+  t['qhook'] = 0x02A0;
+  t['qmonospace'] = 0xFF51;
+  t['qof'] = 0x05E7;
+  t['qofdagesh'] = 0xFB47;
+  t['qofdageshhebrew'] = 0xFB47;
+  t['qofhebrew'] = 0x05E7;
+  t['qparen'] = 0x24AC;
+  t['quarternote'] = 0x2669;
+  t['qubuts'] = 0x05BB;
+  t['qubuts18'] = 0x05BB;
+  t['qubuts25'] = 0x05BB;
+  t['qubuts31'] = 0x05BB;
+  t['qubutshebrew'] = 0x05BB;
+  t['qubutsnarrowhebrew'] = 0x05BB;
+  t['qubutsquarterhebrew'] = 0x05BB;
+  t['qubutswidehebrew'] = 0x05BB;
+  t['question'] = 0x003F;
+  t['questionarabic'] = 0x061F;
+  t['questionarmenian'] = 0x055E;
+  t['questiondown'] = 0x00BF;
+  t['questiondownsmall'] = 0xF7BF;
+  t['questiongreek'] = 0x037E;
+  t['questionmonospace'] = 0xFF1F;
+  t['questionsmall'] = 0xF73F;
+  t['quotedbl'] = 0x0022;
+  t['quotedblbase'] = 0x201E;
+  t['quotedblleft'] = 0x201C;
+  t['quotedblmonospace'] = 0xFF02;
+  t['quotedblprime'] = 0x301E;
+  t['quotedblprimereversed'] = 0x301D;
+  t['quotedblright'] = 0x201D;
+  t['quoteleft'] = 0x2018;
+  t['quoteleftreversed'] = 0x201B;
+  t['quotereversed'] = 0x201B;
+  t['quoteright'] = 0x2019;
+  t['quoterightn'] = 0x0149;
+  t['quotesinglbase'] = 0x201A;
+  t['quotesingle'] = 0x0027;
+  t['quotesinglemonospace'] = 0xFF07;
+  t['r'] = 0x0072;
+  t['raarmenian'] = 0x057C;
+  t['rabengali'] = 0x09B0;
+  t['racute'] = 0x0155;
+  t['radeva'] = 0x0930;
+  t['radical'] = 0x221A;
+  t['radicalex'] = 0xF8E5;
+  t['radoverssquare'] = 0x33AE;
+  t['radoverssquaredsquare'] = 0x33AF;
+  t['radsquare'] = 0x33AD;
+  t['rafe'] = 0x05BF;
+  t['rafehebrew'] = 0x05BF;
+  t['ragujarati'] = 0x0AB0;
+  t['ragurmukhi'] = 0x0A30;
+  t['rahiragana'] = 0x3089;
+  t['rakatakana'] = 0x30E9;
+  t['rakatakanahalfwidth'] = 0xFF97;
+  t['ralowerdiagonalbengali'] = 0x09F1;
+  t['ramiddlediagonalbengali'] = 0x09F0;
+  t['ramshorn'] = 0x0264;
+  t['ratio'] = 0x2236;
+  t['rbopomofo'] = 0x3116;
+  t['rcaron'] = 0x0159;
+  t['rcedilla'] = 0x0157;
+  t['rcircle'] = 0x24E1;
+  t['rcommaaccent'] = 0x0157;
+  t['rdblgrave'] = 0x0211;
+  t['rdotaccent'] = 0x1E59;
+  t['rdotbelow'] = 0x1E5B;
+  t['rdotbelowmacron'] = 0x1E5D;
+  t['referencemark'] = 0x203B;
+  t['reflexsubset'] = 0x2286;
+  t['reflexsuperset'] = 0x2287;
+  t['registered'] = 0x00AE;
+  t['registersans'] = 0xF8E8;
+  t['registerserif'] = 0xF6DA;
+  t['reharabic'] = 0x0631;
+  t['reharmenian'] = 0x0580;
+  t['rehfinalarabic'] = 0xFEAE;
+  t['rehiragana'] = 0x308C;
+  t['rekatakana'] = 0x30EC;
+  t['rekatakanahalfwidth'] = 0xFF9A;
+  t['resh'] = 0x05E8;
+  t['reshdageshhebrew'] = 0xFB48;
+  t['reshhebrew'] = 0x05E8;
+  t['reversedtilde'] = 0x223D;
+  t['reviahebrew'] = 0x0597;
+  t['reviamugrashhebrew'] = 0x0597;
+  t['revlogicalnot'] = 0x2310;
+  t['rfishhook'] = 0x027E;
+  t['rfishhookreversed'] = 0x027F;
+  t['rhabengali'] = 0x09DD;
+  t['rhadeva'] = 0x095D;
+  t['rho'] = 0x03C1;
+  t['rhook'] = 0x027D;
+  t['rhookturned'] = 0x027B;
+  t['rhookturnedsuperior'] = 0x02B5;
+  t['rhosymbolgreek'] = 0x03F1;
+  t['rhotichookmod'] = 0x02DE;
+  t['rieulacirclekorean'] = 0x3271;
+  t['rieulaparenkorean'] = 0x3211;
+  t['rieulcirclekorean'] = 0x3263;
+  t['rieulhieuhkorean'] = 0x3140;
+  t['rieulkiyeokkorean'] = 0x313A;
+  t['rieulkiyeoksioskorean'] = 0x3169;
+  t['rieulkorean'] = 0x3139;
+  t['rieulmieumkorean'] = 0x313B;
+  t['rieulpansioskorean'] = 0x316C;
+  t['rieulparenkorean'] = 0x3203;
+  t['rieulphieuphkorean'] = 0x313F;
+  t['rieulpieupkorean'] = 0x313C;
+  t['rieulpieupsioskorean'] = 0x316B;
+  t['rieulsioskorean'] = 0x313D;
+  t['rieulthieuthkorean'] = 0x313E;
+  t['rieultikeutkorean'] = 0x316A;
+  t['rieulyeorinhieuhkorean'] = 0x316D;
+  t['rightangle'] = 0x221F;
+  t['righttackbelowcmb'] = 0x0319;
+  t['righttriangle'] = 0x22BF;
+  t['rihiragana'] = 0x308A;
+  t['rikatakana'] = 0x30EA;
+  t['rikatakanahalfwidth'] = 0xFF98;
+  t['ring'] = 0x02DA;
+  t['ringbelowcmb'] = 0x0325;
+  t['ringcmb'] = 0x030A;
+  t['ringhalfleft'] = 0x02BF;
+  t['ringhalfleftarmenian'] = 0x0559;
+  t['ringhalfleftbelowcmb'] = 0x031C;
+  t['ringhalfleftcentered'] = 0x02D3;
+  t['ringhalfright'] = 0x02BE;
+  t['ringhalfrightbelowcmb'] = 0x0339;
+  t['ringhalfrightcentered'] = 0x02D2;
+  t['rinvertedbreve'] = 0x0213;
+  t['rittorusquare'] = 0x3351;
+  t['rlinebelow'] = 0x1E5F;
+  t['rlongleg'] = 0x027C;
+  t['rlonglegturned'] = 0x027A;
+  t['rmonospace'] = 0xFF52;
+  t['rohiragana'] = 0x308D;
+  t['rokatakana'] = 0x30ED;
+  t['rokatakanahalfwidth'] = 0xFF9B;
+  t['roruathai'] = 0x0E23;
+  t['rparen'] = 0x24AD;
+  t['rrabengali'] = 0x09DC;
+  t['rradeva'] = 0x0931;
+  t['rragurmukhi'] = 0x0A5C;
+  t['rreharabic'] = 0x0691;
+  t['rrehfinalarabic'] = 0xFB8D;
+  t['rrvocalicbengali'] = 0x09E0;
+  t['rrvocalicdeva'] = 0x0960;
+  t['rrvocalicgujarati'] = 0x0AE0;
+  t['rrvocalicvowelsignbengali'] = 0x09C4;
+  t['rrvocalicvowelsigndeva'] = 0x0944;
+  t['rrvocalicvowelsigngujarati'] = 0x0AC4;
+  t['rsuperior'] = 0xF6F1;
+  t['rtblock'] = 0x2590;
+  t['rturned'] = 0x0279;
+  t['rturnedsuperior'] = 0x02B4;
+  t['ruhiragana'] = 0x308B;
+  t['rukatakana'] = 0x30EB;
+  t['rukatakanahalfwidth'] = 0xFF99;
+  t['rupeemarkbengali'] = 0x09F2;
+  t['rupeesignbengali'] = 0x09F3;
+  t['rupiah'] = 0xF6DD;
+  t['ruthai'] = 0x0E24;
+  t['rvocalicbengali'] = 0x098B;
+  t['rvocalicdeva'] = 0x090B;
+  t['rvocalicgujarati'] = 0x0A8B;
+  t['rvocalicvowelsignbengali'] = 0x09C3;
+  t['rvocalicvowelsigndeva'] = 0x0943;
+  t['rvocalicvowelsigngujarati'] = 0x0AC3;
+  t['s'] = 0x0073;
+  t['sabengali'] = 0x09B8;
+  t['sacute'] = 0x015B;
+  t['sacutedotaccent'] = 0x1E65;
+  t['sadarabic'] = 0x0635;
+  t['sadeva'] = 0x0938;
+  t['sadfinalarabic'] = 0xFEBA;
+  t['sadinitialarabic'] = 0xFEBB;
+  t['sadmedialarabic'] = 0xFEBC;
+  t['sagujarati'] = 0x0AB8;
+  t['sagurmukhi'] = 0x0A38;
+  t['sahiragana'] = 0x3055;
+  t['sakatakana'] = 0x30B5;
+  t['sakatakanahalfwidth'] = 0xFF7B;
+  t['sallallahoualayhewasallamarabic'] = 0xFDFA;
+  t['samekh'] = 0x05E1;
+  t['samekhdagesh'] = 0xFB41;
+  t['samekhdageshhebrew'] = 0xFB41;
+  t['samekhhebrew'] = 0x05E1;
+  t['saraaathai'] = 0x0E32;
+  t['saraaethai'] = 0x0E41;
+  t['saraaimaimalaithai'] = 0x0E44;
+  t['saraaimaimuanthai'] = 0x0E43;
+  t['saraamthai'] = 0x0E33;
+  t['saraathai'] = 0x0E30;
+  t['saraethai'] = 0x0E40;
+  t['saraiileftthai'] = 0xF886;
+  t['saraiithai'] = 0x0E35;
+  t['saraileftthai'] = 0xF885;
+  t['saraithai'] = 0x0E34;
+  t['saraothai'] = 0x0E42;
+  t['saraueeleftthai'] = 0xF888;
+  t['saraueethai'] = 0x0E37;
+  t['saraueleftthai'] = 0xF887;
+  t['sarauethai'] = 0x0E36;
+  t['sarauthai'] = 0x0E38;
+  t['sarauuthai'] = 0x0E39;
+  t['sbopomofo'] = 0x3119;
+  t['scaron'] = 0x0161;
+  t['scarondotaccent'] = 0x1E67;
+  t['scedilla'] = 0x015F;
+  t['schwa'] = 0x0259;
+  t['schwacyrillic'] = 0x04D9;
+  t['schwadieresiscyrillic'] = 0x04DB;
+  t['schwahook'] = 0x025A;
+  t['scircle'] = 0x24E2;
+  t['scircumflex'] = 0x015D;
+  t['scommaaccent'] = 0x0219;
+  t['sdotaccent'] = 0x1E61;
+  t['sdotbelow'] = 0x1E63;
+  t['sdotbelowdotaccent'] = 0x1E69;
+  t['seagullbelowcmb'] = 0x033C;
+  t['second'] = 0x2033;
+  t['secondtonechinese'] = 0x02CA;
+  t['section'] = 0x00A7;
+  t['seenarabic'] = 0x0633;
+  t['seenfinalarabic'] = 0xFEB2;
+  t['seeninitialarabic'] = 0xFEB3;
+  t['seenmedialarabic'] = 0xFEB4;
+  t['segol'] = 0x05B6;
+  t['segol13'] = 0x05B6;
+  t['segol1f'] = 0x05B6;
+  t['segol2c'] = 0x05B6;
+  t['segolhebrew'] = 0x05B6;
+  t['segolnarrowhebrew'] = 0x05B6;
+  t['segolquarterhebrew'] = 0x05B6;
+  t['segoltahebrew'] = 0x0592;
+  t['segolwidehebrew'] = 0x05B6;
+  t['seharmenian'] = 0x057D;
+  t['sehiragana'] = 0x305B;
+  t['sekatakana'] = 0x30BB;
+  t['sekatakanahalfwidth'] = 0xFF7E;
+  t['semicolon'] = 0x003B;
+  t['semicolonarabic'] = 0x061B;
+  t['semicolonmonospace'] = 0xFF1B;
+  t['semicolonsmall'] = 0xFE54;
+  t['semivoicedmarkkana'] = 0x309C;
+  t['semivoicedmarkkanahalfwidth'] = 0xFF9F;
+  t['sentisquare'] = 0x3322;
+  t['sentosquare'] = 0x3323;
+  t['seven'] = 0x0037;
+  t['sevenarabic'] = 0x0667;
+  t['sevenbengali'] = 0x09ED;
+  t['sevencircle'] = 0x2466;
+  t['sevencircleinversesansserif'] = 0x2790;
+  t['sevendeva'] = 0x096D;
+  t['seveneighths'] = 0x215E;
+  t['sevengujarati'] = 0x0AED;
+  t['sevengurmukhi'] = 0x0A6D;
+  t['sevenhackarabic'] = 0x0667;
+  t['sevenhangzhou'] = 0x3027;
+  t['sevenideographicparen'] = 0x3226;
+  t['seveninferior'] = 0x2087;
+  t['sevenmonospace'] = 0xFF17;
+  t['sevenoldstyle'] = 0xF737;
+  t['sevenparen'] = 0x247A;
+  t['sevenperiod'] = 0x248E;
+  t['sevenpersian'] = 0x06F7;
+  t['sevenroman'] = 0x2176;
+  t['sevensuperior'] = 0x2077;
+  t['seventeencircle'] = 0x2470;
+  t['seventeenparen'] = 0x2484;
+  t['seventeenperiod'] = 0x2498;
+  t['seventhai'] = 0x0E57;
+  t['sfthyphen'] = 0x00AD;
+  t['shaarmenian'] = 0x0577;
+  t['shabengali'] = 0x09B6;
+  t['shacyrillic'] = 0x0448;
+  t['shaddaarabic'] = 0x0651;
+  t['shaddadammaarabic'] = 0xFC61;
+  t['shaddadammatanarabic'] = 0xFC5E;
+  t['shaddafathaarabic'] = 0xFC60;
+  t['shaddakasraarabic'] = 0xFC62;
+  t['shaddakasratanarabic'] = 0xFC5F;
+  t['shade'] = 0x2592;
+  t['shadedark'] = 0x2593;
+  t['shadelight'] = 0x2591;
+  t['shademedium'] = 0x2592;
+  t['shadeva'] = 0x0936;
+  t['shagujarati'] = 0x0AB6;
+  t['shagurmukhi'] = 0x0A36;
+  t['shalshelethebrew'] = 0x0593;
+  t['shbopomofo'] = 0x3115;
+  t['shchacyrillic'] = 0x0449;
+  t['sheenarabic'] = 0x0634;
+  t['sheenfinalarabic'] = 0xFEB6;
+  t['sheeninitialarabic'] = 0xFEB7;
+  t['sheenmedialarabic'] = 0xFEB8;
+  t['sheicoptic'] = 0x03E3;
+  t['sheqel'] = 0x20AA;
+  t['sheqelhebrew'] = 0x20AA;
+  t['sheva'] = 0x05B0;
+  t['sheva115'] = 0x05B0;
+  t['sheva15'] = 0x05B0;
+  t['sheva22'] = 0x05B0;
+  t['sheva2e'] = 0x05B0;
+  t['shevahebrew'] = 0x05B0;
+  t['shevanarrowhebrew'] = 0x05B0;
+  t['shevaquarterhebrew'] = 0x05B0;
+  t['shevawidehebrew'] = 0x05B0;
+  t['shhacyrillic'] = 0x04BB;
+  t['shimacoptic'] = 0x03ED;
+  t['shin'] = 0x05E9;
+  t['shindagesh'] = 0xFB49;
+  t['shindageshhebrew'] = 0xFB49;
+  t['shindageshshindot'] = 0xFB2C;
+  t['shindageshshindothebrew'] = 0xFB2C;
+  t['shindageshsindot'] = 0xFB2D;
+  t['shindageshsindothebrew'] = 0xFB2D;
+  t['shindothebrew'] = 0x05C1;
+  t['shinhebrew'] = 0x05E9;
+  t['shinshindot'] = 0xFB2A;
+  t['shinshindothebrew'] = 0xFB2A;
+  t['shinsindot'] = 0xFB2B;
+  t['shinsindothebrew'] = 0xFB2B;
+  t['shook'] = 0x0282;
+  t['sigma'] = 0x03C3;
+  t['sigma1'] = 0x03C2;
+  t['sigmafinal'] = 0x03C2;
+  t['sigmalunatesymbolgreek'] = 0x03F2;
+  t['sihiragana'] = 0x3057;
+  t['sikatakana'] = 0x30B7;
+  t['sikatakanahalfwidth'] = 0xFF7C;
+  t['siluqhebrew'] = 0x05BD;
+  t['siluqlefthebrew'] = 0x05BD;
+  t['similar'] = 0x223C;
+  t['sindothebrew'] = 0x05C2;
+  t['siosacirclekorean'] = 0x3274;
+  t['siosaparenkorean'] = 0x3214;
+  t['sioscieuckorean'] = 0x317E;
+  t['sioscirclekorean'] = 0x3266;
+  t['sioskiyeokkorean'] = 0x317A;
+  t['sioskorean'] = 0x3145;
+  t['siosnieunkorean'] = 0x317B;
+  t['siosparenkorean'] = 0x3206;
+  t['siospieupkorean'] = 0x317D;
+  t['siostikeutkorean'] = 0x317C;
+  t['six'] = 0x0036;
+  t['sixarabic'] = 0x0666;
+  t['sixbengali'] = 0x09EC;
+  t['sixcircle'] = 0x2465;
+  t['sixcircleinversesansserif'] = 0x278F;
+  t['sixdeva'] = 0x096C;
+  t['sixgujarati'] = 0x0AEC;
+  t['sixgurmukhi'] = 0x0A6C;
+  t['sixhackarabic'] = 0x0666;
+  t['sixhangzhou'] = 0x3026;
+  t['sixideographicparen'] = 0x3225;
+  t['sixinferior'] = 0x2086;
+  t['sixmonospace'] = 0xFF16;
+  t['sixoldstyle'] = 0xF736;
+  t['sixparen'] = 0x2479;
+  t['sixperiod'] = 0x248D;
+  t['sixpersian'] = 0x06F6;
+  t['sixroman'] = 0x2175;
+  t['sixsuperior'] = 0x2076;
+  t['sixteencircle'] = 0x246F;
+  t['sixteencurrencydenominatorbengali'] = 0x09F9;
+  t['sixteenparen'] = 0x2483;
+  t['sixteenperiod'] = 0x2497;
+  t['sixthai'] = 0x0E56;
+  t['slash'] = 0x002F;
+  t['slashmonospace'] = 0xFF0F;
+  t['slong'] = 0x017F;
+  t['slongdotaccent'] = 0x1E9B;
+  t['smileface'] = 0x263A;
+  t['smonospace'] = 0xFF53;
+  t['sofpasuqhebrew'] = 0x05C3;
+  t['softhyphen'] = 0x00AD;
+  t['softsigncyrillic'] = 0x044C;
+  t['sohiragana'] = 0x305D;
+  t['sokatakana'] = 0x30BD;
+  t['sokatakanahalfwidth'] = 0xFF7F;
+  t['soliduslongoverlaycmb'] = 0x0338;
+  t['solidusshortoverlaycmb'] = 0x0337;
+  t['sorusithai'] = 0x0E29;
+  t['sosalathai'] = 0x0E28;
+  t['sosothai'] = 0x0E0B;
+  t['sosuathai'] = 0x0E2A;
+  t['space'] = 0x0020;
+  t['spacehackarabic'] = 0x0020;
+  t['spade'] = 0x2660;
+  t['spadesuitblack'] = 0x2660;
+  t['spadesuitwhite'] = 0x2664;
+  t['sparen'] = 0x24AE;
+  t['squarebelowcmb'] = 0x033B;
+  t['squarecc'] = 0x33C4;
+  t['squarecm'] = 0x339D;
+  t['squarediagonalcrosshatchfill'] = 0x25A9;
+  t['squarehorizontalfill'] = 0x25A4;
+  t['squarekg'] = 0x338F;
+  t['squarekm'] = 0x339E;
+  t['squarekmcapital'] = 0x33CE;
+  t['squareln'] = 0x33D1;
+  t['squarelog'] = 0x33D2;
+  t['squaremg'] = 0x338E;
+  t['squaremil'] = 0x33D5;
+  t['squaremm'] = 0x339C;
+  t['squaremsquared'] = 0x33A1;
+  t['squareorthogonalcrosshatchfill'] = 0x25A6;
+  t['squareupperlefttolowerrightfill'] = 0x25A7;
+  t['squareupperrighttolowerleftfill'] = 0x25A8;
+  t['squareverticalfill'] = 0x25A5;
+  t['squarewhitewithsmallblack'] = 0x25A3;
+  t['srsquare'] = 0x33DB;
+  t['ssabengali'] = 0x09B7;
+  t['ssadeva'] = 0x0937;
+  t['ssagujarati'] = 0x0AB7;
+  t['ssangcieuckorean'] = 0x3149;
+  t['ssanghieuhkorean'] = 0x3185;
+  t['ssangieungkorean'] = 0x3180;
+  t['ssangkiyeokkorean'] = 0x3132;
+  t['ssangnieunkorean'] = 0x3165;
+  t['ssangpieupkorean'] = 0x3143;
+  t['ssangsioskorean'] = 0x3146;
+  t['ssangtikeutkorean'] = 0x3138;
+  t['ssuperior'] = 0xF6F2;
+  t['sterling'] = 0x00A3;
+  t['sterlingmonospace'] = 0xFFE1;
+  t['strokelongoverlaycmb'] = 0x0336;
+  t['strokeshortoverlaycmb'] = 0x0335;
+  t['subset'] = 0x2282;
+  t['subsetnotequal'] = 0x228A;
+  t['subsetorequal'] = 0x2286;
+  t['succeeds'] = 0x227B;
+  t['suchthat'] = 0x220B;
+  t['suhiragana'] = 0x3059;
+  t['sukatakana'] = 0x30B9;
+  t['sukatakanahalfwidth'] = 0xFF7D;
+  t['sukunarabic'] = 0x0652;
+  t['summation'] = 0x2211;
+  t['sun'] = 0x263C;
+  t['superset'] = 0x2283;
+  t['supersetnotequal'] = 0x228B;
+  t['supersetorequal'] = 0x2287;
+  t['svsquare'] = 0x33DC;
+  t['syouwaerasquare'] = 0x337C;
+  t['t'] = 0x0074;
+  t['tabengali'] = 0x09A4;
+  t['tackdown'] = 0x22A4;
+  t['tackleft'] = 0x22A3;
+  t['tadeva'] = 0x0924;
+  t['tagujarati'] = 0x0AA4;
+  t['tagurmukhi'] = 0x0A24;
+  t['taharabic'] = 0x0637;
+  t['tahfinalarabic'] = 0xFEC2;
+  t['tahinitialarabic'] = 0xFEC3;
+  t['tahiragana'] = 0x305F;
+  t['tahmedialarabic'] = 0xFEC4;
+  t['taisyouerasquare'] = 0x337D;
+  t['takatakana'] = 0x30BF;
+  t['takatakanahalfwidth'] = 0xFF80;
+  t['tatweelarabic'] = 0x0640;
+  t['tau'] = 0x03C4;
+  t['tav'] = 0x05EA;
+  t['tavdages'] = 0xFB4A;
+  t['tavdagesh'] = 0xFB4A;
+  t['tavdageshhebrew'] = 0xFB4A;
+  t['tavhebrew'] = 0x05EA;
+  t['tbar'] = 0x0167;
+  t['tbopomofo'] = 0x310A;
+  t['tcaron'] = 0x0165;
+  t['tccurl'] = 0x02A8;
+  t['tcedilla'] = 0x0163;
+  t['tcheharabic'] = 0x0686;
+  t['tchehfinalarabic'] = 0xFB7B;
+  t['tchehinitialarabic'] = 0xFB7C;
+  t['tchehmedialarabic'] = 0xFB7D;
+  t['tcircle'] = 0x24E3;
+  t['tcircumflexbelow'] = 0x1E71;
+  t['tcommaaccent'] = 0x0163;
+  t['tdieresis'] = 0x1E97;
+  t['tdotaccent'] = 0x1E6B;
+  t['tdotbelow'] = 0x1E6D;
+  t['tecyrillic'] = 0x0442;
+  t['tedescendercyrillic'] = 0x04AD;
+  t['teharabic'] = 0x062A;
+  t['tehfinalarabic'] = 0xFE96;
+  t['tehhahinitialarabic'] = 0xFCA2;
+  t['tehhahisolatedarabic'] = 0xFC0C;
+  t['tehinitialarabic'] = 0xFE97;
+  t['tehiragana'] = 0x3066;
+  t['tehjeeminitialarabic'] = 0xFCA1;
+  t['tehjeemisolatedarabic'] = 0xFC0B;
+  t['tehmarbutaarabic'] = 0x0629;
+  t['tehmarbutafinalarabic'] = 0xFE94;
+  t['tehmedialarabic'] = 0xFE98;
+  t['tehmeeminitialarabic'] = 0xFCA4;
+  t['tehmeemisolatedarabic'] = 0xFC0E;
+  t['tehnoonfinalarabic'] = 0xFC73;
+  t['tekatakana'] = 0x30C6;
+  t['tekatakanahalfwidth'] = 0xFF83;
+  t['telephone'] = 0x2121;
+  t['telephoneblack'] = 0x260E;
+  t['telishagedolahebrew'] = 0x05A0;
+  t['telishaqetanahebrew'] = 0x05A9;
+  t['tencircle'] = 0x2469;
+  t['tenideographicparen'] = 0x3229;
+  t['tenparen'] = 0x247D;
+  t['tenperiod'] = 0x2491;
+  t['tenroman'] = 0x2179;
+  t['tesh'] = 0x02A7;
+  t['tet'] = 0x05D8;
+  t['tetdagesh'] = 0xFB38;
+  t['tetdageshhebrew'] = 0xFB38;
+  t['tethebrew'] = 0x05D8;
+  t['tetsecyrillic'] = 0x04B5;
+  t['tevirhebrew'] = 0x059B;
+  t['tevirlefthebrew'] = 0x059B;
+  t['thabengali'] = 0x09A5;
+  t['thadeva'] = 0x0925;
+  t['thagujarati'] = 0x0AA5;
+  t['thagurmukhi'] = 0x0A25;
+  t['thalarabic'] = 0x0630;
+  t['thalfinalarabic'] = 0xFEAC;
+  t['thanthakhatlowleftthai'] = 0xF898;
+  t['thanthakhatlowrightthai'] = 0xF897;
+  t['thanthakhatthai'] = 0x0E4C;
+  t['thanthakhatupperleftthai'] = 0xF896;
+  t['theharabic'] = 0x062B;
+  t['thehfinalarabic'] = 0xFE9A;
+  t['thehinitialarabic'] = 0xFE9B;
+  t['thehmedialarabic'] = 0xFE9C;
+  t['thereexists'] = 0x2203;
+  t['therefore'] = 0x2234;
+  t['theta'] = 0x03B8;
+  t['theta1'] = 0x03D1;
+  t['thetasymbolgreek'] = 0x03D1;
+  t['thieuthacirclekorean'] = 0x3279;
+  t['thieuthaparenkorean'] = 0x3219;
+  t['thieuthcirclekorean'] = 0x326B;
+  t['thieuthkorean'] = 0x314C;
+  t['thieuthparenkorean'] = 0x320B;
+  t['thirteencircle'] = 0x246C;
+  t['thirteenparen'] = 0x2480;
+  t['thirteenperiod'] = 0x2494;
+  t['thonangmonthothai'] = 0x0E11;
+  t['thook'] = 0x01AD;
+  t['thophuthaothai'] = 0x0E12;
+  t['thorn'] = 0x00FE;
+  t['thothahanthai'] = 0x0E17;
+  t['thothanthai'] = 0x0E10;
+  t['thothongthai'] = 0x0E18;
+  t['thothungthai'] = 0x0E16;
+  t['thousandcyrillic'] = 0x0482;
+  t['thousandsseparatorarabic'] = 0x066C;
+  t['thousandsseparatorpersian'] = 0x066C;
+  t['three'] = 0x0033;
+  t['threearabic'] = 0x0663;
+  t['threebengali'] = 0x09E9;
+  t['threecircle'] = 0x2462;
+  t['threecircleinversesansserif'] = 0x278C;
+  t['threedeva'] = 0x0969;
+  t['threeeighths'] = 0x215C;
+  t['threegujarati'] = 0x0AE9;
+  t['threegurmukhi'] = 0x0A69;
+  t['threehackarabic'] = 0x0663;
+  t['threehangzhou'] = 0x3023;
+  t['threeideographicparen'] = 0x3222;
+  t['threeinferior'] = 0x2083;
+  t['threemonospace'] = 0xFF13;
+  t['threenumeratorbengali'] = 0x09F6;
+  t['threeoldstyle'] = 0xF733;
+  t['threeparen'] = 0x2476;
+  t['threeperiod'] = 0x248A;
+  t['threepersian'] = 0x06F3;
+  t['threequarters'] = 0x00BE;
+  t['threequartersemdash'] = 0xF6DE;
+  t['threeroman'] = 0x2172;
+  t['threesuperior'] = 0x00B3;
+  t['threethai'] = 0x0E53;
+  t['thzsquare'] = 0x3394;
+  t['tihiragana'] = 0x3061;
+  t['tikatakana'] = 0x30C1;
+  t['tikatakanahalfwidth'] = 0xFF81;
+  t['tikeutacirclekorean'] = 0x3270;
+  t['tikeutaparenkorean'] = 0x3210;
+  t['tikeutcirclekorean'] = 0x3262;
+  t['tikeutkorean'] = 0x3137;
+  t['tikeutparenkorean'] = 0x3202;
+  t['tilde'] = 0x02DC;
+  t['tildebelowcmb'] = 0x0330;
+  t['tildecmb'] = 0x0303;
+  t['tildecomb'] = 0x0303;
+  t['tildedoublecmb'] = 0x0360;
+  t['tildeoperator'] = 0x223C;
+  t['tildeoverlaycmb'] = 0x0334;
+  t['tildeverticalcmb'] = 0x033E;
+  t['timescircle'] = 0x2297;
+  t['tipehahebrew'] = 0x0596;
+  t['tipehalefthebrew'] = 0x0596;
+  t['tippigurmukhi'] = 0x0A70;
+  t['titlocyrilliccmb'] = 0x0483;
+  t['tiwnarmenian'] = 0x057F;
+  t['tlinebelow'] = 0x1E6F;
+  t['tmonospace'] = 0xFF54;
+  t['toarmenian'] = 0x0569;
+  t['tohiragana'] = 0x3068;
+  t['tokatakana'] = 0x30C8;
+  t['tokatakanahalfwidth'] = 0xFF84;
+  t['tonebarextrahighmod'] = 0x02E5;
+  t['tonebarextralowmod'] = 0x02E9;
+  t['tonebarhighmod'] = 0x02E6;
+  t['tonebarlowmod'] = 0x02E8;
+  t['tonebarmidmod'] = 0x02E7;
+  t['tonefive'] = 0x01BD;
+  t['tonesix'] = 0x0185;
+  t['tonetwo'] = 0x01A8;
+  t['tonos'] = 0x0384;
+  t['tonsquare'] = 0x3327;
+  t['topatakthai'] = 0x0E0F;
+  t['tortoiseshellbracketleft'] = 0x3014;
+  t['tortoiseshellbracketleftsmall'] = 0xFE5D;
+  t['tortoiseshellbracketleftvertical'] = 0xFE39;
+  t['tortoiseshellbracketright'] = 0x3015;
+  t['tortoiseshellbracketrightsmall'] = 0xFE5E;
+  t['tortoiseshellbracketrightvertical'] = 0xFE3A;
+  t['totaothai'] = 0x0E15;
+  t['tpalatalhook'] = 0x01AB;
+  t['tparen'] = 0x24AF;
+  t['trademark'] = 0x2122;
+  t['trademarksans'] = 0xF8EA;
+  t['trademarkserif'] = 0xF6DB;
+  t['tretroflexhook'] = 0x0288;
+  t['triagdn'] = 0x25BC;
+  t['triaglf'] = 0x25C4;
+  t['triagrt'] = 0x25BA;
+  t['triagup'] = 0x25B2;
+  t['ts'] = 0x02A6;
+  t['tsadi'] = 0x05E6;
+  t['tsadidagesh'] = 0xFB46;
+  t['tsadidageshhebrew'] = 0xFB46;
+  t['tsadihebrew'] = 0x05E6;
+  t['tsecyrillic'] = 0x0446;
+  t['tsere'] = 0x05B5;
+  t['tsere12'] = 0x05B5;
+  t['tsere1e'] = 0x05B5;
+  t['tsere2b'] = 0x05B5;
+  t['tserehebrew'] = 0x05B5;
+  t['tserenarrowhebrew'] = 0x05B5;
+  t['tserequarterhebrew'] = 0x05B5;
+  t['tserewidehebrew'] = 0x05B5;
+  t['tshecyrillic'] = 0x045B;
+  t['tsuperior'] = 0xF6F3;
+  t['ttabengali'] = 0x099F;
+  t['ttadeva'] = 0x091F;
+  t['ttagujarati'] = 0x0A9F;
+  t['ttagurmukhi'] = 0x0A1F;
+  t['tteharabic'] = 0x0679;
+  t['ttehfinalarabic'] = 0xFB67;
+  t['ttehinitialarabic'] = 0xFB68;
+  t['ttehmedialarabic'] = 0xFB69;
+  t['tthabengali'] = 0x09A0;
+  t['tthadeva'] = 0x0920;
+  t['tthagujarati'] = 0x0AA0;
+  t['tthagurmukhi'] = 0x0A20;
+  t['tturned'] = 0x0287;
+  t['tuhiragana'] = 0x3064;
+  t['tukatakana'] = 0x30C4;
+  t['tukatakanahalfwidth'] = 0xFF82;
+  t['tusmallhiragana'] = 0x3063;
+  t['tusmallkatakana'] = 0x30C3;
+  t['tusmallkatakanahalfwidth'] = 0xFF6F;
+  t['twelvecircle'] = 0x246B;
+  t['twelveparen'] = 0x247F;
+  t['twelveperiod'] = 0x2493;
+  t['twelveroman'] = 0x217B;
+  t['twentycircle'] = 0x2473;
+  t['twentyhangzhou'] = 0x5344;
+  t['twentyparen'] = 0x2487;
+  t['twentyperiod'] = 0x249B;
+  t['two'] = 0x0032;
+  t['twoarabic'] = 0x0662;
+  t['twobengali'] = 0x09E8;
+  t['twocircle'] = 0x2461;
+  t['twocircleinversesansserif'] = 0x278B;
+  t['twodeva'] = 0x0968;
+  t['twodotenleader'] = 0x2025;
+  t['twodotleader'] = 0x2025;
+  t['twodotleadervertical'] = 0xFE30;
+  t['twogujarati'] = 0x0AE8;
+  t['twogurmukhi'] = 0x0A68;
+  t['twohackarabic'] = 0x0662;
+  t['twohangzhou'] = 0x3022;
+  t['twoideographicparen'] = 0x3221;
+  t['twoinferior'] = 0x2082;
+  t['twomonospace'] = 0xFF12;
+  t['twonumeratorbengali'] = 0x09F5;
+  t['twooldstyle'] = 0xF732;
+  t['twoparen'] = 0x2475;
+  t['twoperiod'] = 0x2489;
+  t['twopersian'] = 0x06F2;
+  t['tworoman'] = 0x2171;
+  t['twostroke'] = 0x01BB;
+  t['twosuperior'] = 0x00B2;
+  t['twothai'] = 0x0E52;
+  t['twothirds'] = 0x2154;
+  t['u'] = 0x0075;
+  t['uacute'] = 0x00FA;
+  t['ubar'] = 0x0289;
+  t['ubengali'] = 0x0989;
+  t['ubopomofo'] = 0x3128;
+  t['ubreve'] = 0x016D;
+  t['ucaron'] = 0x01D4;
+  t['ucircle'] = 0x24E4;
+  t['ucircumflex'] = 0x00FB;
+  t['ucircumflexbelow'] = 0x1E77;
+  t['ucyrillic'] = 0x0443;
+  t['udattadeva'] = 0x0951;
+  t['udblacute'] = 0x0171;
+  t['udblgrave'] = 0x0215;
+  t['udeva'] = 0x0909;
+  t['udieresis'] = 0x00FC;
+  t['udieresisacute'] = 0x01D8;
+  t['udieresisbelow'] = 0x1E73;
+  t['udieresiscaron'] = 0x01DA;
+  t['udieresiscyrillic'] = 0x04F1;
+  t['udieresisgrave'] = 0x01DC;
+  t['udieresismacron'] = 0x01D6;
+  t['udotbelow'] = 0x1EE5;
+  t['ugrave'] = 0x00F9;
+  t['ugujarati'] = 0x0A89;
+  t['ugurmukhi'] = 0x0A09;
+  t['uhiragana'] = 0x3046;
+  t['uhookabove'] = 0x1EE7;
+  t['uhorn'] = 0x01B0;
+  t['uhornacute'] = 0x1EE9;
+  t['uhorndotbelow'] = 0x1EF1;
+  t['uhorngrave'] = 0x1EEB;
+  t['uhornhookabove'] = 0x1EED;
+  t['uhorntilde'] = 0x1EEF;
+  t['uhungarumlaut'] = 0x0171;
+  t['uhungarumlautcyrillic'] = 0x04F3;
+  t['uinvertedbreve'] = 0x0217;
+  t['ukatakana'] = 0x30A6;
+  t['ukatakanahalfwidth'] = 0xFF73;
+  t['ukcyrillic'] = 0x0479;
+  t['ukorean'] = 0x315C;
+  t['umacron'] = 0x016B;
+  t['umacroncyrillic'] = 0x04EF;
+  t['umacrondieresis'] = 0x1E7B;
+  t['umatragurmukhi'] = 0x0A41;
+  t['umonospace'] = 0xFF55;
+  t['underscore'] = 0x005F;
+  t['underscoredbl'] = 0x2017;
+  t['underscoremonospace'] = 0xFF3F;
+  t['underscorevertical'] = 0xFE33;
+  t['underscorewavy'] = 0xFE4F;
+  t['union'] = 0x222A;
+  t['universal'] = 0x2200;
+  t['uogonek'] = 0x0173;
+  t['uparen'] = 0x24B0;
+  t['upblock'] = 0x2580;
+  t['upperdothebrew'] = 0x05C4;
+  t['upsilon'] = 0x03C5;
+  t['upsilondieresis'] = 0x03CB;
+  t['upsilondieresistonos'] = 0x03B0;
+  t['upsilonlatin'] = 0x028A;
+  t['upsilontonos'] = 0x03CD;
+  t['uptackbelowcmb'] = 0x031D;
+  t['uptackmod'] = 0x02D4;
+  t['uragurmukhi'] = 0x0A73;
+  t['uring'] = 0x016F;
+  t['ushortcyrillic'] = 0x045E;
+  t['usmallhiragana'] = 0x3045;
+  t['usmallkatakana'] = 0x30A5;
+  t['usmallkatakanahalfwidth'] = 0xFF69;
+  t['ustraightcyrillic'] = 0x04AF;
+  t['ustraightstrokecyrillic'] = 0x04B1;
+  t['utilde'] = 0x0169;
+  t['utildeacute'] = 0x1E79;
+  t['utildebelow'] = 0x1E75;
+  t['uubengali'] = 0x098A;
+  t['uudeva'] = 0x090A;
+  t['uugujarati'] = 0x0A8A;
+  t['uugurmukhi'] = 0x0A0A;
+  t['uumatragurmukhi'] = 0x0A42;
+  t['uuvowelsignbengali'] = 0x09C2;
+  t['uuvowelsigndeva'] = 0x0942;
+  t['uuvowelsigngujarati'] = 0x0AC2;
+  t['uvowelsignbengali'] = 0x09C1;
+  t['uvowelsigndeva'] = 0x0941;
+  t['uvowelsigngujarati'] = 0x0AC1;
+  t['v'] = 0x0076;
+  t['vadeva'] = 0x0935;
+  t['vagujarati'] = 0x0AB5;
+  t['vagurmukhi'] = 0x0A35;
+  t['vakatakana'] = 0x30F7;
+  t['vav'] = 0x05D5;
+  t['vavdagesh'] = 0xFB35;
+  t['vavdagesh65'] = 0xFB35;
+  t['vavdageshhebrew'] = 0xFB35;
+  t['vavhebrew'] = 0x05D5;
+  t['vavholam'] = 0xFB4B;
+  t['vavholamhebrew'] = 0xFB4B;
+  t['vavvavhebrew'] = 0x05F0;
+  t['vavyodhebrew'] = 0x05F1;
+  t['vcircle'] = 0x24E5;
+  t['vdotbelow'] = 0x1E7F;
+  t['vecyrillic'] = 0x0432;
+  t['veharabic'] = 0x06A4;
+  t['vehfinalarabic'] = 0xFB6B;
+  t['vehinitialarabic'] = 0xFB6C;
+  t['vehmedialarabic'] = 0xFB6D;
+  t['vekatakana'] = 0x30F9;
+  t['venus'] = 0x2640;
+  t['verticalbar'] = 0x007C;
+  t['verticallineabovecmb'] = 0x030D;
+  t['verticallinebelowcmb'] = 0x0329;
+  t['verticallinelowmod'] = 0x02CC;
+  t['verticallinemod'] = 0x02C8;
+  t['vewarmenian'] = 0x057E;
+  t['vhook'] = 0x028B;
+  t['vikatakana'] = 0x30F8;
+  t['viramabengali'] = 0x09CD;
+  t['viramadeva'] = 0x094D;
+  t['viramagujarati'] = 0x0ACD;
+  t['visargabengali'] = 0x0983;
+  t['visargadeva'] = 0x0903;
+  t['visargagujarati'] = 0x0A83;
+  t['vmonospace'] = 0xFF56;
+  t['voarmenian'] = 0x0578;
+  t['voicediterationhiragana'] = 0x309E;
+  t['voicediterationkatakana'] = 0x30FE;
+  t['voicedmarkkana'] = 0x309B;
+  t['voicedmarkkanahalfwidth'] = 0xFF9E;
+  t['vokatakana'] = 0x30FA;
+  t['vparen'] = 0x24B1;
+  t['vtilde'] = 0x1E7D;
+  t['vturned'] = 0x028C;
+  t['vuhiragana'] = 0x3094;
+  t['vukatakana'] = 0x30F4;
+  t['w'] = 0x0077;
+  t['wacute'] = 0x1E83;
+  t['waekorean'] = 0x3159;
+  t['wahiragana'] = 0x308F;
+  t['wakatakana'] = 0x30EF;
+  t['wakatakanahalfwidth'] = 0xFF9C;
+  t['wakorean'] = 0x3158;
+  t['wasmallhiragana'] = 0x308E;
+  t['wasmallkatakana'] = 0x30EE;
+  t['wattosquare'] = 0x3357;
+  t['wavedash'] = 0x301C;
+  t['wavyunderscorevertical'] = 0xFE34;
+  t['wawarabic'] = 0x0648;
+  t['wawfinalarabic'] = 0xFEEE;
+  t['wawhamzaabovearabic'] = 0x0624;
+  t['wawhamzaabovefinalarabic'] = 0xFE86;
+  t['wbsquare'] = 0x33DD;
+  t['wcircle'] = 0x24E6;
+  t['wcircumflex'] = 0x0175;
+  t['wdieresis'] = 0x1E85;
+  t['wdotaccent'] = 0x1E87;
+  t['wdotbelow'] = 0x1E89;
+  t['wehiragana'] = 0x3091;
+  t['weierstrass'] = 0x2118;
+  t['wekatakana'] = 0x30F1;
+  t['wekorean'] = 0x315E;
+  t['weokorean'] = 0x315D;
+  t['wgrave'] = 0x1E81;
+  t['whitebullet'] = 0x25E6;
+  t['whitecircle'] = 0x25CB;
+  t['whitecircleinverse'] = 0x25D9;
+  t['whitecornerbracketleft'] = 0x300E;
+  t['whitecornerbracketleftvertical'] = 0xFE43;
+  t['whitecornerbracketright'] = 0x300F;
+  t['whitecornerbracketrightvertical'] = 0xFE44;
+  t['whitediamond'] = 0x25C7;
+  t['whitediamondcontainingblacksmalldiamond'] = 0x25C8;
+  t['whitedownpointingsmalltriangle'] = 0x25BF;
+  t['whitedownpointingtriangle'] = 0x25BD;
+  t['whiteleftpointingsmalltriangle'] = 0x25C3;
+  t['whiteleftpointingtriangle'] = 0x25C1;
+  t['whitelenticularbracketleft'] = 0x3016;
+  t['whitelenticularbracketright'] = 0x3017;
+  t['whiterightpointingsmalltriangle'] = 0x25B9;
+  t['whiterightpointingtriangle'] = 0x25B7;
+  t['whitesmallsquare'] = 0x25AB;
+  t['whitesmilingface'] = 0x263A;
+  t['whitesquare'] = 0x25A1;
+  t['whitestar'] = 0x2606;
+  t['whitetelephone'] = 0x260F;
+  t['whitetortoiseshellbracketleft'] = 0x3018;
+  t['whitetortoiseshellbracketright'] = 0x3019;
+  t['whiteuppointingsmalltriangle'] = 0x25B5;
+  t['whiteuppointingtriangle'] = 0x25B3;
+  t['wihiragana'] = 0x3090;
+  t['wikatakana'] = 0x30F0;
+  t['wikorean'] = 0x315F;
+  t['wmonospace'] = 0xFF57;
+  t['wohiragana'] = 0x3092;
+  t['wokatakana'] = 0x30F2;
+  t['wokatakanahalfwidth'] = 0xFF66;
+  t['won'] = 0x20A9;
+  t['wonmonospace'] = 0xFFE6;
+  t['wowaenthai'] = 0x0E27;
+  t['wparen'] = 0x24B2;
+  t['wring'] = 0x1E98;
+  t['wsuperior'] = 0x02B7;
+  t['wturned'] = 0x028D;
+  t['wynn'] = 0x01BF;
+  t['x'] = 0x0078;
+  t['xabovecmb'] = 0x033D;
+  t['xbopomofo'] = 0x3112;
+  t['xcircle'] = 0x24E7;
+  t['xdieresis'] = 0x1E8D;
+  t['xdotaccent'] = 0x1E8B;
+  t['xeharmenian'] = 0x056D;
+  t['xi'] = 0x03BE;
+  t['xmonospace'] = 0xFF58;
+  t['xparen'] = 0x24B3;
+  t['xsuperior'] = 0x02E3;
+  t['y'] = 0x0079;
+  t['yaadosquare'] = 0x334E;
+  t['yabengali'] = 0x09AF;
+  t['yacute'] = 0x00FD;
+  t['yadeva'] = 0x092F;
+  t['yaekorean'] = 0x3152;
+  t['yagujarati'] = 0x0AAF;
+  t['yagurmukhi'] = 0x0A2F;
+  t['yahiragana'] = 0x3084;
+  t['yakatakana'] = 0x30E4;
+  t['yakatakanahalfwidth'] = 0xFF94;
+  t['yakorean'] = 0x3151;
+  t['yamakkanthai'] = 0x0E4E;
+  t['yasmallhiragana'] = 0x3083;
+  t['yasmallkatakana'] = 0x30E3;
+  t['yasmallkatakanahalfwidth'] = 0xFF6C;
+  t['yatcyrillic'] = 0x0463;
+  t['ycircle'] = 0x24E8;
+  t['ycircumflex'] = 0x0177;
+  t['ydieresis'] = 0x00FF;
+  t['ydotaccent'] = 0x1E8F;
+  t['ydotbelow'] = 0x1EF5;
+  t['yeharabic'] = 0x064A;
+  t['yehbarreearabic'] = 0x06D2;
+  t['yehbarreefinalarabic'] = 0xFBAF;
+  t['yehfinalarabic'] = 0xFEF2;
+  t['yehhamzaabovearabic'] = 0x0626;
+  t['yehhamzaabovefinalarabic'] = 0xFE8A;
+  t['yehhamzaaboveinitialarabic'] = 0xFE8B;
+  t['yehhamzaabovemedialarabic'] = 0xFE8C;
+  t['yehinitialarabic'] = 0xFEF3;
+  t['yehmedialarabic'] = 0xFEF4;
+  t['yehmeeminitialarabic'] = 0xFCDD;
+  t['yehmeemisolatedarabic'] = 0xFC58;
+  t['yehnoonfinalarabic'] = 0xFC94;
+  t['yehthreedotsbelowarabic'] = 0x06D1;
+  t['yekorean'] = 0x3156;
+  t['yen'] = 0x00A5;
+  t['yenmonospace'] = 0xFFE5;
+  t['yeokorean'] = 0x3155;
+  t['yeorinhieuhkorean'] = 0x3186;
+  t['yerahbenyomohebrew'] = 0x05AA;
+  t['yerahbenyomolefthebrew'] = 0x05AA;
+  t['yericyrillic'] = 0x044B;
+  t['yerudieresiscyrillic'] = 0x04F9;
+  t['yesieungkorean'] = 0x3181;
+  t['yesieungpansioskorean'] = 0x3183;
+  t['yesieungsioskorean'] = 0x3182;
+  t['yetivhebrew'] = 0x059A;
+  t['ygrave'] = 0x1EF3;
+  t['yhook'] = 0x01B4;
+  t['yhookabove'] = 0x1EF7;
+  t['yiarmenian'] = 0x0575;
+  t['yicyrillic'] = 0x0457;
+  t['yikorean'] = 0x3162;
+  t['yinyang'] = 0x262F;
+  t['yiwnarmenian'] = 0x0582;
+  t['ymonospace'] = 0xFF59;
+  t['yod'] = 0x05D9;
+  t['yoddagesh'] = 0xFB39;
+  t['yoddageshhebrew'] = 0xFB39;
+  t['yodhebrew'] = 0x05D9;
+  t['yodyodhebrew'] = 0x05F2;
+  t['yodyodpatahhebrew'] = 0xFB1F;
+  t['yohiragana'] = 0x3088;
+  t['yoikorean'] = 0x3189;
+  t['yokatakana'] = 0x30E8;
+  t['yokatakanahalfwidth'] = 0xFF96;
+  t['yokorean'] = 0x315B;
+  t['yosmallhiragana'] = 0x3087;
+  t['yosmallkatakana'] = 0x30E7;
+  t['yosmallkatakanahalfwidth'] = 0xFF6E;
+  t['yotgreek'] = 0x03F3;
+  t['yoyaekorean'] = 0x3188;
+  t['yoyakorean'] = 0x3187;
+  t['yoyakthai'] = 0x0E22;
+  t['yoyingthai'] = 0x0E0D;
+  t['yparen'] = 0x24B4;
+  t['ypogegrammeni'] = 0x037A;
+  t['ypogegrammenigreekcmb'] = 0x0345;
+  t['yr'] = 0x01A6;
+  t['yring'] = 0x1E99;
+  t['ysuperior'] = 0x02B8;
+  t['ytilde'] = 0x1EF9;
+  t['yturned'] = 0x028E;
+  t['yuhiragana'] = 0x3086;
+  t['yuikorean'] = 0x318C;
+  t['yukatakana'] = 0x30E6;
+  t['yukatakanahalfwidth'] = 0xFF95;
+  t['yukorean'] = 0x3160;
+  t['yusbigcyrillic'] = 0x046B;
+  t['yusbigiotifiedcyrillic'] = 0x046D;
+  t['yuslittlecyrillic'] = 0x0467;
+  t['yuslittleiotifiedcyrillic'] = 0x0469;
+  t['yusmallhiragana'] = 0x3085;
+  t['yusmallkatakana'] = 0x30E5;
+  t['yusmallkatakanahalfwidth'] = 0xFF6D;
+  t['yuyekorean'] = 0x318B;
+  t['yuyeokorean'] = 0x318A;
+  t['yyabengali'] = 0x09DF;
+  t['yyadeva'] = 0x095F;
+  t['z'] = 0x007A;
+  t['zaarmenian'] = 0x0566;
+  t['zacute'] = 0x017A;
+  t['zadeva'] = 0x095B;
+  t['zagurmukhi'] = 0x0A5B;
+  t['zaharabic'] = 0x0638;
+  t['zahfinalarabic'] = 0xFEC6;
+  t['zahinitialarabic'] = 0xFEC7;
+  t['zahiragana'] = 0x3056;
+  t['zahmedialarabic'] = 0xFEC8;
+  t['zainarabic'] = 0x0632;
+  t['zainfinalarabic'] = 0xFEB0;
+  t['zakatakana'] = 0x30B6;
+  t['zaqefgadolhebrew'] = 0x0595;
+  t['zaqefqatanhebrew'] = 0x0594;
+  t['zarqahebrew'] = 0x0598;
+  t['zayin'] = 0x05D6;
+  t['zayindagesh'] = 0xFB36;
+  t['zayindageshhebrew'] = 0xFB36;
+  t['zayinhebrew'] = 0x05D6;
+  t['zbopomofo'] = 0x3117;
+  t['zcaron'] = 0x017E;
+  t['zcircle'] = 0x24E9;
+  t['zcircumflex'] = 0x1E91;
+  t['zcurl'] = 0x0291;
+  t['zdot'] = 0x017C;
+  t['zdotaccent'] = 0x017C;
+  t['zdotbelow'] = 0x1E93;
+  t['zecyrillic'] = 0x0437;
+  t['zedescendercyrillic'] = 0x0499;
+  t['zedieresiscyrillic'] = 0x04DF;
+  t['zehiragana'] = 0x305C;
+  t['zekatakana'] = 0x30BC;
+  t['zero'] = 0x0030;
+  t['zeroarabic'] = 0x0660;
+  t['zerobengali'] = 0x09E6;
+  t['zerodeva'] = 0x0966;
+  t['zerogujarati'] = 0x0AE6;
+  t['zerogurmukhi'] = 0x0A66;
+  t['zerohackarabic'] = 0x0660;
+  t['zeroinferior'] = 0x2080;
+  t['zeromonospace'] = 0xFF10;
+  t['zerooldstyle'] = 0xF730;
+  t['zeropersian'] = 0x06F0;
+  t['zerosuperior'] = 0x2070;
+  t['zerothai'] = 0x0E50;
+  t['zerowidthjoiner'] = 0xFEFF;
+  t['zerowidthnonjoiner'] = 0x200C;
+  t['zerowidthspace'] = 0x200B;
+  t['zeta'] = 0x03B6;
+  t['zhbopomofo'] = 0x3113;
+  t['zhearmenian'] = 0x056A;
+  t['zhebrevecyrillic'] = 0x04C2;
+  t['zhecyrillic'] = 0x0436;
+  t['zhedescendercyrillic'] = 0x0497;
+  t['zhedieresiscyrillic'] = 0x04DD;
+  t['zihiragana'] = 0x3058;
+  t['zikatakana'] = 0x30B8;
+  t['zinorhebrew'] = 0x05AE;
+  t['zlinebelow'] = 0x1E95;
+  t['zmonospace'] = 0xFF5A;
+  t['zohiragana'] = 0x305E;
+  t['zokatakana'] = 0x30BE;
+  t['zparen'] = 0x24B5;
+  t['zretroflexhook'] = 0x0290;
+  t['zstroke'] = 0x01B6;
+  t['zuhiragana'] = 0x305A;
+  t['zukatakana'] = 0x30BA;
+  t['.notdef'] = 0x0000;
+  t['angbracketleftbig'] = 0x2329;
+  t['angbracketleftBig'] = 0x2329;
+  t['angbracketleftbigg'] = 0x2329;
+  t['angbracketleftBigg'] = 0x2329;
+  t['angbracketrightBig'] = 0x232A;
+  t['angbracketrightbig'] = 0x232A;
+  t['angbracketrightBigg'] = 0x232A;
+  t['angbracketrightbigg'] = 0x232A;
+  t['arrowhookleft'] = 0x21AA;
+  t['arrowhookright'] = 0x21A9;
+  t['arrowlefttophalf'] = 0x21BC;
+  t['arrowleftbothalf'] = 0x21BD;
+  t['arrownortheast'] = 0x2197;
+  t['arrownorthwest'] = 0x2196;
+  t['arrowrighttophalf'] = 0x21C0;
+  t['arrowrightbothalf'] = 0x21C1;
+  t['arrowsoutheast'] = 0x2198;
+  t['arrowsouthwest'] = 0x2199;
+  t['backslashbig'] = 0x2216;
+  t['backslashBig'] = 0x2216;
+  t['backslashBigg'] = 0x2216;
+  t['backslashbigg'] = 0x2216;
+  t['bardbl'] = 0x2016;
+  t['bracehtipdownleft'] = 0xFE37;
+  t['bracehtipdownright'] = 0xFE37;
+  t['bracehtipupleft'] = 0xFE38;
+  t['bracehtipupright'] = 0xFE38;
+  t['braceleftBig'] = 0x007B;
+  t['braceleftbig'] = 0x007B;
+  t['braceleftbigg'] = 0x007B;
+  t['braceleftBigg'] = 0x007B;
+  t['bracerightBig'] = 0x007D;
+  t['bracerightbig'] = 0x007D;
+  t['bracerightbigg'] = 0x007D;
+  t['bracerightBigg'] = 0x007D;
+  t['bracketleftbig'] = 0x005B;
+  t['bracketleftBig'] = 0x005B;
+  t['bracketleftbigg'] = 0x005B;
+  t['bracketleftBigg'] = 0x005B;
+  t['bracketrightBig'] = 0x005D;
+  t['bracketrightbig'] = 0x005D;
+  t['bracketrightbigg'] = 0x005D;
+  t['bracketrightBigg'] = 0x005D;
+  t['ceilingleftbig'] = 0x2308;
+  t['ceilingleftBig'] = 0x2308;
+  t['ceilingleftBigg'] = 0x2308;
+  t['ceilingleftbigg'] = 0x2308;
+  t['ceilingrightbig'] = 0x2309;
+  t['ceilingrightBig'] = 0x2309;
+  t['ceilingrightbigg'] = 0x2309;
+  t['ceilingrightBigg'] = 0x2309;
+  t['circledotdisplay'] = 0x2299;
+  t['circledottext'] = 0x2299;
+  t['circlemultiplydisplay'] = 0x2297;
+  t['circlemultiplytext'] = 0x2297;
+  t['circleplusdisplay'] = 0x2295;
+  t['circleplustext'] = 0x2295;
+  t['contintegraldisplay'] = 0x222E;
+  t['contintegraltext'] = 0x222E;
+  t['coproductdisplay'] = 0x2210;
+  t['coproducttext'] = 0x2210;
+  t['floorleftBig'] = 0x230A;
+  t['floorleftbig'] = 0x230A;
+  t['floorleftbigg'] = 0x230A;
+  t['floorleftBigg'] = 0x230A;
+  t['floorrightbig'] = 0x230B;
+  t['floorrightBig'] = 0x230B;
+  t['floorrightBigg'] = 0x230B;
+  t['floorrightbigg'] = 0x230B;
+  t['hatwide'] = 0x0302;
+  t['hatwider'] = 0x0302;
+  t['hatwidest'] = 0x0302;
+  t['intercal'] = 0x1D40;
+  t['integraldisplay'] = 0x222B;
+  t['integraltext'] = 0x222B;
+  t['intersectiondisplay'] = 0x22C2;
+  t['intersectiontext'] = 0x22C2;
+  t['logicalanddisplay'] = 0x2227;
+  t['logicalandtext'] = 0x2227;
+  t['logicalordisplay'] = 0x2228;
+  t['logicalortext'] = 0x2228;
+  t['parenleftBig'] = 0x0028;
+  t['parenleftbig'] = 0x0028;
+  t['parenleftBigg'] = 0x0028;
+  t['parenleftbigg'] = 0x0028;
+  t['parenrightBig'] = 0x0029;
+  t['parenrightbig'] = 0x0029;
+  t['parenrightBigg'] = 0x0029;
+  t['parenrightbigg'] = 0x0029;
+  t['prime'] = 0x2032;
+  t['productdisplay'] = 0x220F;
+  t['producttext'] = 0x220F;
+  t['radicalbig'] = 0x221A;
+  t['radicalBig'] = 0x221A;
+  t['radicalBigg'] = 0x221A;
+  t['radicalbigg'] = 0x221A;
+  t['radicalbt'] = 0x221A;
+  t['radicaltp'] = 0x221A;
+  t['radicalvertex'] = 0x221A;
+  t['slashbig'] = 0x002F;
+  t['slashBig'] = 0x002F;
+  t['slashBigg'] = 0x002F;
+  t['slashbigg'] = 0x002F;
+  t['summationdisplay'] = 0x2211;
+  t['summationtext'] = 0x2211;
+  t['tildewide'] = 0x02DC;
+  t['tildewider'] = 0x02DC;
+  t['tildewidest'] = 0x02DC;
+  t['uniondisplay'] = 0x22C3;
+  t['unionmultidisplay'] = 0x228E;
+  t['unionmultitext'] = 0x228E;
+  t['unionsqdisplay'] = 0x2294;
+  t['unionsqtext'] = 0x2294;
+  t['uniontext'] = 0x22C3;
+  t['vextenddouble'] = 0x2225;
+  t['vextendsingle'] = 0x2223;
 });
 var getDingbatsGlyphsUnicode = getLookupTableFactory(function (t) {
- t['space'] = 0x0020;
- t['a1'] = 0x2701;
- t['a2'] = 0x2702;
- t['a202'] = 0x2703;
- t['a3'] = 0x2704;
- t['a4'] = 0x260E;
- t['a5'] = 0x2706;
- t['a119'] = 0x2707;
- t['a118'] = 0x2708;
- t['a117'] = 0x2709;
- t['a11'] = 0x261B;
- t['a12'] = 0x261E;
- t['a13'] = 0x270C;
- t['a14'] = 0x270D;
- t['a15'] = 0x270E;
- t['a16'] = 0x270F;
- t['a105'] = 0x2710;
- t['a17'] = 0x2711;
- t['a18'] = 0x2712;
- t['a19'] = 0x2713;
- t['a20'] = 0x2714;
- t['a21'] = 0x2715;
- t['a22'] = 0x2716;
- t['a23'] = 0x2717;
- t['a24'] = 0x2718;
- t['a25'] = 0x2719;
- t['a26'] = 0x271A;
- t['a27'] = 0x271B;
- t['a28'] = 0x271C;
- t['a6'] = 0x271D;
- t['a7'] = 0x271E;
- t['a8'] = 0x271F;
- t['a9'] = 0x2720;
- t['a10'] = 0x2721;
- t['a29'] = 0x2722;
- t['a30'] = 0x2723;
- t['a31'] = 0x2724;
- t['a32'] = 0x2725;
- t['a33'] = 0x2726;
- t['a34'] = 0x2727;
- t['a35'] = 0x2605;
- t['a36'] = 0x2729;
- t['a37'] = 0x272A;
- t['a38'] = 0x272B;
- t['a39'] = 0x272C;
- t['a40'] = 0x272D;
- t['a41'] = 0x272E;
- t['a42'] = 0x272F;
- t['a43'] = 0x2730;
- t['a44'] = 0x2731;
- t['a45'] = 0x2732;
- t['a46'] = 0x2733;
- t['a47'] = 0x2734;
- t['a48'] = 0x2735;
- t['a49'] = 0x2736;
- t['a50'] = 0x2737;
- t['a51'] = 0x2738;
- t['a52'] = 0x2739;
- t['a53'] = 0x273A;
- t['a54'] = 0x273B;
- t['a55'] = 0x273C;
- t['a56'] = 0x273D;
- t['a57'] = 0x273E;
- t['a58'] = 0x273F;
- t['a59'] = 0x2740;
- t['a60'] = 0x2741;
- t['a61'] = 0x2742;
- t['a62'] = 0x2743;
- t['a63'] = 0x2744;
- t['a64'] = 0x2745;
- t['a65'] = 0x2746;
- t['a66'] = 0x2747;
- t['a67'] = 0x2748;
- t['a68'] = 0x2749;
- t['a69'] = 0x274A;
- t['a70'] = 0x274B;
- t['a71'] = 0x25CF;
- t['a72'] = 0x274D;
- t['a73'] = 0x25A0;
- t['a74'] = 0x274F;
- t['a203'] = 0x2750;
- t['a75'] = 0x2751;
- t['a204'] = 0x2752;
- t['a76'] = 0x25B2;
- t['a77'] = 0x25BC;
- t['a78'] = 0x25C6;
- t['a79'] = 0x2756;
- t['a81'] = 0x25D7;
- t['a82'] = 0x2758;
- t['a83'] = 0x2759;
- t['a84'] = 0x275A;
- t['a97'] = 0x275B;
- t['a98'] = 0x275C;
- t['a99'] = 0x275D;
- t['a100'] = 0x275E;
- t['a101'] = 0x2761;
- t['a102'] = 0x2762;
- t['a103'] = 0x2763;
- t['a104'] = 0x2764;
- t['a106'] = 0x2765;
- t['a107'] = 0x2766;
- t['a108'] = 0x2767;
- t['a112'] = 0x2663;
- t['a111'] = 0x2666;
- t['a110'] = 0x2665;
- t['a109'] = 0x2660;
- t['a120'] = 0x2460;
- t['a121'] = 0x2461;
- t['a122'] = 0x2462;
- t['a123'] = 0x2463;
- t['a124'] = 0x2464;
- t['a125'] = 0x2465;
- t['a126'] = 0x2466;
- t['a127'] = 0x2467;
- t['a128'] = 0x2468;
- t['a129'] = 0x2469;
- t['a130'] = 0x2776;
- t['a131'] = 0x2777;
- t['a132'] = 0x2778;
- t['a133'] = 0x2779;
- t['a134'] = 0x277A;
- t['a135'] = 0x277B;
- t['a136'] = 0x277C;
- t['a137'] = 0x277D;
- t['a138'] = 0x277E;
- t['a139'] = 0x277F;
- t['a140'] = 0x2780;
- t['a141'] = 0x2781;
- t['a142'] = 0x2782;
- t['a143'] = 0x2783;
- t['a144'] = 0x2784;
- t['a145'] = 0x2785;
- t['a146'] = 0x2786;
- t['a147'] = 0x2787;
- t['a148'] = 0x2788;
- t['a149'] = 0x2789;
- t['a150'] = 0x278A;
- t['a151'] = 0x278B;
- t['a152'] = 0x278C;
- t['a153'] = 0x278D;
- t['a154'] = 0x278E;
- t['a155'] = 0x278F;
- t['a156'] = 0x2790;
- t['a157'] = 0x2791;
- t['a158'] = 0x2792;
- t['a159'] = 0x2793;
- t['a160'] = 0x2794;
- t['a161'] = 0x2192;
- t['a163'] = 0x2194;
- t['a164'] = 0x2195;
- t['a196'] = 0x2798;
- t['a165'] = 0x2799;
- t['a192'] = 0x279A;
- t['a166'] = 0x279B;
- t['a167'] = 0x279C;
- t['a168'] = 0x279D;
- t['a169'] = 0x279E;
- t['a170'] = 0x279F;
- t['a171'] = 0x27A0;
- t['a172'] = 0x27A1;
- t['a173'] = 0x27A2;
- t['a162'] = 0x27A3;
- t['a174'] = 0x27A4;
- t['a175'] = 0x27A5;
- t['a176'] = 0x27A6;
- t['a177'] = 0x27A7;
- t['a178'] = 0x27A8;
- t['a179'] = 0x27A9;
- t['a193'] = 0x27AA;
- t['a180'] = 0x27AB;
- t['a199'] = 0x27AC;
- t['a181'] = 0x27AD;
- t['a200'] = 0x27AE;
- t['a182'] = 0x27AF;
- t['a201'] = 0x27B1;
- t['a183'] = 0x27B2;
- t['a184'] = 0x27B3;
- t['a197'] = 0x27B4;
- t['a185'] = 0x27B5;
- t['a194'] = 0x27B6;
- t['a198'] = 0x27B7;
- t['a186'] = 0x27B8;
- t['a195'] = 0x27B9;
- t['a187'] = 0x27BA;
- t['a188'] = 0x27BB;
- t['a189'] = 0x27BC;
- t['a190'] = 0x27BD;
- t['a191'] = 0x27BE;
- t['a89'] = 0x2768;
- t['a90'] = 0x2769;
- t['a93'] = 0x276A;
- t['a94'] = 0x276B;
- t['a91'] = 0x276C;
- t['a92'] = 0x276D;
- t['a205'] = 0x276E;
- t['a85'] = 0x276F;
- t['a206'] = 0x2770;
- t['a86'] = 0x2771;
- t['a87'] = 0x2772;
- t['a88'] = 0x2773;
- t['a95'] = 0x2774;
- t['a96'] = 0x2775;
- t['.notdef'] = 0x0000;
+  t['space'] = 0x0020;
+  t['a1'] = 0x2701;
+  t['a2'] = 0x2702;
+  t['a202'] = 0x2703;
+  t['a3'] = 0x2704;
+  t['a4'] = 0x260E;
+  t['a5'] = 0x2706;
+  t['a119'] = 0x2707;
+  t['a118'] = 0x2708;
+  t['a117'] = 0x2709;
+  t['a11'] = 0x261B;
+  t['a12'] = 0x261E;
+  t['a13'] = 0x270C;
+  t['a14'] = 0x270D;
+  t['a15'] = 0x270E;
+  t['a16'] = 0x270F;
+  t['a105'] = 0x2710;
+  t['a17'] = 0x2711;
+  t['a18'] = 0x2712;
+  t['a19'] = 0x2713;
+  t['a20'] = 0x2714;
+  t['a21'] = 0x2715;
+  t['a22'] = 0x2716;
+  t['a23'] = 0x2717;
+  t['a24'] = 0x2718;
+  t['a25'] = 0x2719;
+  t['a26'] = 0x271A;
+  t['a27'] = 0x271B;
+  t['a28'] = 0x271C;
+  t['a6'] = 0x271D;
+  t['a7'] = 0x271E;
+  t['a8'] = 0x271F;
+  t['a9'] = 0x2720;
+  t['a10'] = 0x2721;
+  t['a29'] = 0x2722;
+  t['a30'] = 0x2723;
+  t['a31'] = 0x2724;
+  t['a32'] = 0x2725;
+  t['a33'] = 0x2726;
+  t['a34'] = 0x2727;
+  t['a35'] = 0x2605;
+  t['a36'] = 0x2729;
+  t['a37'] = 0x272A;
+  t['a38'] = 0x272B;
+  t['a39'] = 0x272C;
+  t['a40'] = 0x272D;
+  t['a41'] = 0x272E;
+  t['a42'] = 0x272F;
+  t['a43'] = 0x2730;
+  t['a44'] = 0x2731;
+  t['a45'] = 0x2732;
+  t['a46'] = 0x2733;
+  t['a47'] = 0x2734;
+  t['a48'] = 0x2735;
+  t['a49'] = 0x2736;
+  t['a50'] = 0x2737;
+  t['a51'] = 0x2738;
+  t['a52'] = 0x2739;
+  t['a53'] = 0x273A;
+  t['a54'] = 0x273B;
+  t['a55'] = 0x273C;
+  t['a56'] = 0x273D;
+  t['a57'] = 0x273E;
+  t['a58'] = 0x273F;
+  t['a59'] = 0x2740;
+  t['a60'] = 0x2741;
+  t['a61'] = 0x2742;
+  t['a62'] = 0x2743;
+  t['a63'] = 0x2744;
+  t['a64'] = 0x2745;
+  t['a65'] = 0x2746;
+  t['a66'] = 0x2747;
+  t['a67'] = 0x2748;
+  t['a68'] = 0x2749;
+  t['a69'] = 0x274A;
+  t['a70'] = 0x274B;
+  t['a71'] = 0x25CF;
+  t['a72'] = 0x274D;
+  t['a73'] = 0x25A0;
+  t['a74'] = 0x274F;
+  t['a203'] = 0x2750;
+  t['a75'] = 0x2751;
+  t['a204'] = 0x2752;
+  t['a76'] = 0x25B2;
+  t['a77'] = 0x25BC;
+  t['a78'] = 0x25C6;
+  t['a79'] = 0x2756;
+  t['a81'] = 0x25D7;
+  t['a82'] = 0x2758;
+  t['a83'] = 0x2759;
+  t['a84'] = 0x275A;
+  t['a97'] = 0x275B;
+  t['a98'] = 0x275C;
+  t['a99'] = 0x275D;
+  t['a100'] = 0x275E;
+  t['a101'] = 0x2761;
+  t['a102'] = 0x2762;
+  t['a103'] = 0x2763;
+  t['a104'] = 0x2764;
+  t['a106'] = 0x2765;
+  t['a107'] = 0x2766;
+  t['a108'] = 0x2767;
+  t['a112'] = 0x2663;
+  t['a111'] = 0x2666;
+  t['a110'] = 0x2665;
+  t['a109'] = 0x2660;
+  t['a120'] = 0x2460;
+  t['a121'] = 0x2461;
+  t['a122'] = 0x2462;
+  t['a123'] = 0x2463;
+  t['a124'] = 0x2464;
+  t['a125'] = 0x2465;
+  t['a126'] = 0x2466;
+  t['a127'] = 0x2467;
+  t['a128'] = 0x2468;
+  t['a129'] = 0x2469;
+  t['a130'] = 0x2776;
+  t['a131'] = 0x2777;
+  t['a132'] = 0x2778;
+  t['a133'] = 0x2779;
+  t['a134'] = 0x277A;
+  t['a135'] = 0x277B;
+  t['a136'] = 0x277C;
+  t['a137'] = 0x277D;
+  t['a138'] = 0x277E;
+  t['a139'] = 0x277F;
+  t['a140'] = 0x2780;
+  t['a141'] = 0x2781;
+  t['a142'] = 0x2782;
+  t['a143'] = 0x2783;
+  t['a144'] = 0x2784;
+  t['a145'] = 0x2785;
+  t['a146'] = 0x2786;
+  t['a147'] = 0x2787;
+  t['a148'] = 0x2788;
+  t['a149'] = 0x2789;
+  t['a150'] = 0x278A;
+  t['a151'] = 0x278B;
+  t['a152'] = 0x278C;
+  t['a153'] = 0x278D;
+  t['a154'] = 0x278E;
+  t['a155'] = 0x278F;
+  t['a156'] = 0x2790;
+  t['a157'] = 0x2791;
+  t['a158'] = 0x2792;
+  t['a159'] = 0x2793;
+  t['a160'] = 0x2794;
+  t['a161'] = 0x2192;
+  t['a163'] = 0x2194;
+  t['a164'] = 0x2195;
+  t['a196'] = 0x2798;
+  t['a165'] = 0x2799;
+  t['a192'] = 0x279A;
+  t['a166'] = 0x279B;
+  t['a167'] = 0x279C;
+  t['a168'] = 0x279D;
+  t['a169'] = 0x279E;
+  t['a170'] = 0x279F;
+  t['a171'] = 0x27A0;
+  t['a172'] = 0x27A1;
+  t['a173'] = 0x27A2;
+  t['a162'] = 0x27A3;
+  t['a174'] = 0x27A4;
+  t['a175'] = 0x27A5;
+  t['a176'] = 0x27A6;
+  t['a177'] = 0x27A7;
+  t['a178'] = 0x27A8;
+  t['a179'] = 0x27A9;
+  t['a193'] = 0x27AA;
+  t['a180'] = 0x27AB;
+  t['a199'] = 0x27AC;
+  t['a181'] = 0x27AD;
+  t['a200'] = 0x27AE;
+  t['a182'] = 0x27AF;
+  t['a201'] = 0x27B1;
+  t['a183'] = 0x27B2;
+  t['a184'] = 0x27B3;
+  t['a197'] = 0x27B4;
+  t['a185'] = 0x27B5;
+  t['a194'] = 0x27B6;
+  t['a198'] = 0x27B7;
+  t['a186'] = 0x27B8;
+  t['a195'] = 0x27B9;
+  t['a187'] = 0x27BA;
+  t['a188'] = 0x27BB;
+  t['a189'] = 0x27BC;
+  t['a190'] = 0x27BD;
+  t['a191'] = 0x27BE;
+  t['a89'] = 0x2768;
+  t['a90'] = 0x2769;
+  t['a93'] = 0x276A;
+  t['a94'] = 0x276B;
+  t['a91'] = 0x276C;
+  t['a92'] = 0x276D;
+  t['a205'] = 0x276E;
+  t['a85'] = 0x276F;
+  t['a206'] = 0x2770;
+  t['a86'] = 0x2771;
+  t['a87'] = 0x2772;
+  t['a88'] = 0x2773;
+  t['a95'] = 0x2774;
+  t['a96'] = 0x2775;
+  t['.notdef'] = 0x0000;
 });
 exports.getGlyphsUnicode = getGlyphsUnicode;
 exports.getDingbatsGlyphsUnicode = getDingbatsGlyphsUnicode;

+ 449 - 445
lib/core/image.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var coreColorSpace = require('./colorspace.js');
@@ -31,469 +32,472 @@ var DecodeStream = coreStream.DecodeStream;
 var JpegStream = coreStream.JpegStream;
 var JpxImage = coreJpx.JpxImage;
 var PDFImage = function PDFImageClosure() {
- function handleImageData(image, nativeDecoder) {
-  if (nativeDecoder && nativeDecoder.canDecode(image)) {
-   return nativeDecoder.decode(image);
-  }
-  return Promise.resolve(image);
- }
- function decodeAndClamp(value, addend, coefficient, max) {
-  value = addend + value * coefficient;
-  return value < 0 ? 0 : value > max ? max : value;
- }
- function resizeImageMask(src, bpc, w1, h1, w2, h2) {
-  var length = w2 * h2;
-  var dest = bpc <= 8 ? new Uint8Array(length) : bpc <= 16 ? new Uint16Array(length) : new Uint32Array(length);
-  var xRatio = w1 / w2;
-  var yRatio = h1 / h2;
-  var i, j, py, newIndex = 0, oldIndex;
-  var xScaled = new Uint16Array(w2);
-  var w1Scanline = w1;
-  for (i = 0; i < w2; i++) {
-   xScaled[i] = Math.floor(i * xRatio);
-  }
-  for (i = 0; i < h2; i++) {
-   py = Math.floor(i * yRatio) * w1Scanline;
-   for (j = 0; j < w2; j++) {
-    oldIndex = py + xScaled[j];
-    dest[newIndex++] = src[oldIndex];
-   }
-  }
-  return dest;
- }
- function PDFImage(xref, res, image, inline, smask, mask, isMask) {
-  this.image = image;
-  var dict = image.dict;
-  if (dict.has('Filter')) {
-   var filter = dict.get('Filter').name;
-   if (filter === 'JPXDecode') {
-    var jpxImage = new JpxImage();
-    jpxImage.parseImageProperties(image.stream);
-    image.stream.reset();
-    image.bitsPerComponent = jpxImage.bitsPerComponent;
-    image.numComps = jpxImage.componentsCount;
-   } else if (filter === 'JBIG2Decode') {
-    image.bitsPerComponent = 1;
-    image.numComps = 1;
-   }
-  }
-  this.width = dict.get('Width', 'W');
-  this.height = dict.get('Height', 'H');
-  if (this.width < 1 || this.height < 1) {
-   error('Invalid image width: ' + this.width + ' or height: ' + this.height);
-  }
-  this.interpolate = dict.get('Interpolate', 'I') || false;
-  this.imageMask = dict.get('ImageMask', 'IM') || false;
-  this.matte = dict.get('Matte') || false;
-  var bitsPerComponent = image.bitsPerComponent;
-  if (!bitsPerComponent) {
-   bitsPerComponent = dict.get('BitsPerComponent', 'BPC');
-   if (!bitsPerComponent) {
-    if (this.imageMask) {
-     bitsPerComponent = 1;
-    } else {
-     error('Bits per component missing in image: ' + this.imageMask);
+  function handleImageData(image, nativeDecoder) {
+    if (nativeDecoder && nativeDecoder.canDecode(image)) {
+      return nativeDecoder.decode(image);
     }
-   }
+    return Promise.resolve(image);
   }
-  this.bpc = bitsPerComponent;
-  if (!this.imageMask) {
-   var colorSpace = dict.get('ColorSpace', 'CS');
-   if (!colorSpace) {
-    info('JPX images (which do not require color spaces)');
-    switch (image.numComps) {
-    case 1:
-     colorSpace = Name.get('DeviceGray');
-     break;
-    case 3:
-     colorSpace = Name.get('DeviceRGB');
-     break;
-    case 4:
-     colorSpace = Name.get('DeviceCMYK');
-     break;
-    default:
-     error('JPX images with ' + this.numComps + ' color components not supported.');
-    }
-   }
-   this.colorSpace = ColorSpace.parse(colorSpace, xref, res);
-   this.numComps = this.colorSpace.numComps;
-  }
-  this.decode = dict.getArray('Decode', 'D');
-  this.needsDecode = false;
-  if (this.decode && (this.colorSpace && !this.colorSpace.isDefaultDecode(this.decode) || isMask && !ColorSpace.isDefaultDecode(this.decode, 1))) {
-   this.needsDecode = true;
-   var max = (1 << bitsPerComponent) - 1;
-   this.decodeCoefficients = [];
-   this.decodeAddends = [];
-   for (var i = 0, j = 0; i < this.decode.length; i += 2, ++j) {
-    var dmin = this.decode[i];
-    var dmax = this.decode[i + 1];
-    this.decodeCoefficients[j] = dmax - dmin;
-    this.decodeAddends[j] = max * dmin;
-   }
+  function decodeAndClamp(value, addend, coefficient, max) {
+    value = addend + value * coefficient;
+    return value < 0 ? 0 : value > max ? max : value;
   }
-  if (smask) {
-   this.smask = new PDFImage(xref, res, smask, false);
-  } else if (mask) {
-   if (isStream(mask)) {
-    var maskDict = mask.dict, imageMask = maskDict.get('ImageMask', 'IM');
-    if (!imageMask) {
-     warn('Ignoring /Mask in image without /ImageMask.');
-    } else {
-     this.mask = new PDFImage(xref, res, mask, false, null, null, true);
+  function resizeImageMask(src, bpc, w1, h1, w2, h2) {
+    var length = w2 * h2;
+    var dest = bpc <= 8 ? new Uint8Array(length) : bpc <= 16 ? new Uint16Array(length) : new Uint32Array(length);
+    var xRatio = w1 / w2;
+    var yRatio = h1 / h2;
+    var i,
+        j,
+        py,
+        newIndex = 0,
+        oldIndex;
+    var xScaled = new Uint16Array(w2);
+    var w1Scanline = w1;
+    for (i = 0; i < w2; i++) {
+      xScaled[i] = Math.floor(i * xRatio);
     }
-   } else {
-    this.mask = mask;
-   }
-  }
- }
- PDFImage.buildImage = function PDFImage_buildImage(handler, xref, res, image, inline, nativeDecoder) {
-  var imagePromise = handleImageData(image, nativeDecoder);
-  var smaskPromise;
-  var maskPromise;
-  var smask = image.dict.get('SMask');
-  var mask = image.dict.get('Mask');
-  if (smask) {
-   smaskPromise = handleImageData(smask, nativeDecoder);
-   maskPromise = Promise.resolve(null);
-  } else {
-   smaskPromise = Promise.resolve(null);
-   if (mask) {
-    if (isStream(mask)) {
-     maskPromise = handleImageData(mask, nativeDecoder);
-    } else if (isArray(mask)) {
-     maskPromise = Promise.resolve(mask);
-    } else {
-     warn('Unsupported mask format.');
-     maskPromise = Promise.resolve(null);
+    for (i = 0; i < h2; i++) {
+      py = Math.floor(i * yRatio) * w1Scanline;
+      for (j = 0; j < w2; j++) {
+        oldIndex = py + xScaled[j];
+        dest[newIndex++] = src[oldIndex];
+      }
     }
-   } else {
-    maskPromise = Promise.resolve(null);
-   }
+    return dest;
   }
-  return Promise.all([
-   imagePromise,
-   smaskPromise,
-   maskPromise
-  ]).then(function (results) {
-   var imageData = results[0];
-   var smaskData = results[1];
-   var maskData = results[2];
-   return new PDFImage(xref, res, imageData, inline, smaskData, maskData);
-  });
- };
- PDFImage.createMask = function PDFImage_createMask(imgArray, width, height, imageIsFromDecodeStream, inverseDecode) {
-  var computedLength = (width + 7 >> 3) * height;
-  var actualLength = imgArray.byteLength;
-  var haveFullData = computedLength === actualLength;
-  var data, i;
-  if (imageIsFromDecodeStream && (!inverseDecode || haveFullData)) {
-   data = imgArray;
-  } else if (!inverseDecode) {
-   data = new Uint8Array(actualLength);
-   data.set(imgArray);
-  } else {
-   data = new Uint8Array(computedLength);
-   data.set(imgArray);
-   for (i = actualLength; i < computedLength; i++) {
-    data[i] = 0xff;
-   }
-  }
-  if (inverseDecode) {
-   for (i = 0; i < actualLength; i++) {
-    data[i] = ~data[i];
-   }
-  }
-  return {
-   data: data,
-   width: width,
-   height: height
-  };
- };
- PDFImage.prototype = {
-  get drawWidth() {
-   return Math.max(this.width, this.smask && this.smask.width || 0, this.mask && this.mask.width || 0);
-  },
-  get drawHeight() {
-   return Math.max(this.height, this.smask && this.smask.height || 0, this.mask && this.mask.height || 0);
-  },
-  decodeBuffer: function PDFImage_decodeBuffer(buffer) {
-   var bpc = this.bpc;
-   var numComps = this.numComps;
-   var decodeAddends = this.decodeAddends;
-   var decodeCoefficients = this.decodeCoefficients;
-   var max = (1 << bpc) - 1;
-   var i, ii;
-   if (bpc === 1) {
-    for (i = 0, ii = buffer.length; i < ii; i++) {
-     buffer[i] = +!buffer[i];
-    }
-    return;
-   }
-   var index = 0;
-   for (i = 0, ii = this.width * this.height; i < ii; i++) {
-    for (var j = 0; j < numComps; j++) {
-     buffer[index] = decodeAndClamp(buffer[index], decodeAddends[j], decodeCoefficients[j], max);
-     index++;
-    }
-   }
-  },
-  getComponents: function PDFImage_getComponents(buffer) {
-   var bpc = this.bpc;
-   if (bpc === 8) {
-    return buffer;
-   }
-   var width = this.width;
-   var height = this.height;
-   var numComps = this.numComps;
-   var length = width * height * numComps;
-   var bufferPos = 0;
-   var output = bpc <= 8 ? new Uint8Array(length) : bpc <= 16 ? new Uint16Array(length) : new Uint32Array(length);
-   var rowComps = width * numComps;
-   var max = (1 << bpc) - 1;
-   var i = 0, ii, buf;
-   if (bpc === 1) {
-    var mask, loop1End, loop2End;
-    for (var j = 0; j < height; j++) {
-     loop1End = i + (rowComps & ~7);
-     loop2End = i + rowComps;
-     while (i < loop1End) {
-      buf = buffer[bufferPos++];
-      output[i] = buf >> 7 & 1;
-      output[i + 1] = buf >> 6 & 1;
-      output[i + 2] = buf >> 5 & 1;
-      output[i + 3] = buf >> 4 & 1;
-      output[i + 4] = buf >> 3 & 1;
-      output[i + 5] = buf >> 2 & 1;
-      output[i + 6] = buf >> 1 & 1;
-      output[i + 7] = buf & 1;
-      i += 8;
-     }
-     if (i < loop2End) {
-      buf = buffer[bufferPos++];
-      mask = 128;
-      while (i < loop2End) {
-       output[i++] = +!!(buf & mask);
-       mask >>= 1;
+  function PDFImage(xref, res, image, inline, smask, mask, isMask) {
+    this.image = image;
+    var dict = image.dict;
+    if (dict.has('Filter')) {
+      var filter = dict.get('Filter').name;
+      if (filter === 'JPXDecode') {
+        var jpxImage = new JpxImage();
+        jpxImage.parseImageProperties(image.stream);
+        image.stream.reset();
+        image.bitsPerComponent = jpxImage.bitsPerComponent;
+        image.numComps = jpxImage.componentsCount;
+      } else if (filter === 'JBIG2Decode') {
+        image.bitsPerComponent = 1;
+        image.numComps = 1;
       }
-     }
-    }
-   } else {
-    var bits = 0;
-    buf = 0;
-    for (i = 0, ii = length; i < ii; ++i) {
-     if (i % rowComps === 0) {
-      buf = 0;
-      bits = 0;
-     }
-     while (bits < bpc) {
-      buf = buf << 8 | buffer[bufferPos++];
-      bits += 8;
-     }
-     var remainingBits = bits - bpc;
-     var value = buf >> remainingBits;
-     output[i] = value < 0 ? 0 : value > max ? max : value;
-     buf = buf & (1 << remainingBits) - 1;
-     bits = remainingBits;
     }
-   }
-   return output;
-  },
-  fillOpacity: function PDFImage_fillOpacity(rgbaBuf, width, height, actualHeight, image) {
-   var smask = this.smask;
-   var mask = this.mask;
-   var alphaBuf, sw, sh, i, ii, j;
-   if (smask) {
-    sw = smask.width;
-    sh = smask.height;
-    alphaBuf = new Uint8Array(sw * sh);
-    smask.fillGrayBuffer(alphaBuf);
-    if (sw !== width || sh !== height) {
-     alphaBuf = resizeImageMask(alphaBuf, smask.bpc, sw, sh, width, height);
+    this.width = dict.get('Width', 'W');
+    this.height = dict.get('Height', 'H');
+    if (this.width < 1 || this.height < 1) {
+      error('Invalid image width: ' + this.width + ' or height: ' + this.height);
     }
-   } else if (mask) {
-    if (mask instanceof PDFImage) {
-     sw = mask.width;
-     sh = mask.height;
-     alphaBuf = new Uint8Array(sw * sh);
-     mask.numComps = 1;
-     mask.fillGrayBuffer(alphaBuf);
-     for (i = 0, ii = sw * sh; i < ii; ++i) {
-      alphaBuf[i] = 255 - alphaBuf[i];
-     }
-     if (sw !== width || sh !== height) {
-      alphaBuf = resizeImageMask(alphaBuf, mask.bpc, sw, sh, width, height);
-     }
-    } else if (isArray(mask)) {
-     alphaBuf = new Uint8Array(width * height);
-     var numComps = this.numComps;
-     for (i = 0, ii = width * height; i < ii; ++i) {
-      var opacity = 0;
-      var imageOffset = i * numComps;
-      for (j = 0; j < numComps; ++j) {
-       var color = image[imageOffset + j];
-       var maskOffset = j * 2;
-       if (color < mask[maskOffset] || color > mask[maskOffset + 1]) {
-        opacity = 255;
-        break;
-       }
+    this.interpolate = dict.get('Interpolate', 'I') || false;
+    this.imageMask = dict.get('ImageMask', 'IM') || false;
+    this.matte = dict.get('Matte') || false;
+    var bitsPerComponent = image.bitsPerComponent;
+    if (!bitsPerComponent) {
+      bitsPerComponent = dict.get('BitsPerComponent', 'BPC');
+      if (!bitsPerComponent) {
+        if (this.imageMask) {
+          bitsPerComponent = 1;
+        } else {
+          error('Bits per component missing in image: ' + this.imageMask);
+        }
       }
-      alphaBuf[i] = opacity;
-     }
-    } else {
-     error('Unknown mask format.');
     }
-   }
-   if (alphaBuf) {
-    for (i = 0, j = 3, ii = width * actualHeight; i < ii; ++i, j += 4) {
-     rgbaBuf[j] = alphaBuf[i];
+    this.bpc = bitsPerComponent;
+    if (!this.imageMask) {
+      var colorSpace = dict.get('ColorSpace', 'CS');
+      if (!colorSpace) {
+        info('JPX images (which do not require color spaces)');
+        switch (image.numComps) {
+          case 1:
+            colorSpace = Name.get('DeviceGray');
+            break;
+          case 3:
+            colorSpace = Name.get('DeviceRGB');
+            break;
+          case 4:
+            colorSpace = Name.get('DeviceCMYK');
+            break;
+          default:
+            error('JPX images with ' + this.numComps + ' color components not supported.');
+        }
+      }
+      this.colorSpace = ColorSpace.parse(colorSpace, xref, res);
+      this.numComps = this.colorSpace.numComps;
     }
-   } else {
-    for (i = 0, j = 3, ii = width * actualHeight; i < ii; ++i, j += 4) {
-     rgbaBuf[j] = 255;
+    this.decode = dict.getArray('Decode', 'D');
+    this.needsDecode = false;
+    if (this.decode && (this.colorSpace && !this.colorSpace.isDefaultDecode(this.decode) || isMask && !ColorSpace.isDefaultDecode(this.decode, 1))) {
+      this.needsDecode = true;
+      var max = (1 << bitsPerComponent) - 1;
+      this.decodeCoefficients = [];
+      this.decodeAddends = [];
+      for (var i = 0, j = 0; i < this.decode.length; i += 2, ++j) {
+        var dmin = this.decode[i];
+        var dmax = this.decode[i + 1];
+        this.decodeCoefficients[j] = dmax - dmin;
+        this.decodeAddends[j] = max * dmin;
+      }
     }
-   }
-  },
-  undoPreblend: function PDFImage_undoPreblend(buffer, width, height) {
-   var matte = this.smask && this.smask.matte;
-   if (!matte) {
-    return;
-   }
-   var matteRgb = this.colorSpace.getRgb(matte, 0);
-   var matteR = matteRgb[0];
-   var matteG = matteRgb[1];
-   var matteB = matteRgb[2];
-   var length = width * height * 4;
-   var r, g, b;
-   for (var i = 0; i < length; i += 4) {
-    var alpha = buffer[i + 3];
-    if (alpha === 0) {
-     buffer[i] = 255;
-     buffer[i + 1] = 255;
-     buffer[i + 2] = 255;
-     continue;
+    if (smask) {
+      this.smask = new PDFImage(xref, res, smask, false);
+    } else if (mask) {
+      if (isStream(mask)) {
+        var maskDict = mask.dict,
+            imageMask = maskDict.get('ImageMask', 'IM');
+        if (!imageMask) {
+          warn('Ignoring /Mask in image without /ImageMask.');
+        } else {
+          this.mask = new PDFImage(xref, res, mask, false, null, null, true);
+        }
+      } else {
+        this.mask = mask;
+      }
     }
-    var k = 255 / alpha;
-    r = (buffer[i] - matteR) * k + matteR;
-    g = (buffer[i + 1] - matteG) * k + matteG;
-    b = (buffer[i + 2] - matteB) * k + matteB;
-    buffer[i] = r <= 0 ? 0 : r >= 255 ? 255 : r | 0;
-    buffer[i + 1] = g <= 0 ? 0 : g >= 255 ? 255 : g | 0;
-    buffer[i + 2] = b <= 0 ? 0 : b >= 255 ? 255 : b | 0;
-   }
-  },
-  createImageData: function PDFImage_createImageData(forceRGBA) {
-   var drawWidth = this.drawWidth;
-   var drawHeight = this.drawHeight;
-   var imgData = {
-    width: drawWidth,
-    height: drawHeight
-   };
-   var numComps = this.numComps;
-   var originalWidth = this.width;
-   var originalHeight = this.height;
-   var bpc = this.bpc;
-   var rowBytes = originalWidth * numComps * bpc + 7 >> 3;
-   var imgArray;
-   if (!forceRGBA) {
-    var kind;
-    if (this.colorSpace.name === 'DeviceGray' && bpc === 1) {
-     kind = ImageKind.GRAYSCALE_1BPP;
-    } else if (this.colorSpace.name === 'DeviceRGB' && bpc === 8 && !this.needsDecode) {
-     kind = ImageKind.RGB_24BPP;
+  }
+  PDFImage.buildImage = function PDFImage_buildImage(handler, xref, res, image, inline, nativeDecoder) {
+    var imagePromise = handleImageData(image, nativeDecoder);
+    var smaskPromise;
+    var maskPromise;
+    var smask = image.dict.get('SMask');
+    var mask = image.dict.get('Mask');
+    if (smask) {
+      smaskPromise = handleImageData(smask, nativeDecoder);
+      maskPromise = Promise.resolve(null);
+    } else {
+      smaskPromise = Promise.resolve(null);
+      if (mask) {
+        if (isStream(mask)) {
+          maskPromise = handleImageData(mask, nativeDecoder);
+        } else if (isArray(mask)) {
+          maskPromise = Promise.resolve(mask);
+        } else {
+          warn('Unsupported mask format.');
+          maskPromise = Promise.resolve(null);
+        }
+      } else {
+        maskPromise = Promise.resolve(null);
+      }
     }
-    if (kind && !this.smask && !this.mask && drawWidth === originalWidth && drawHeight === originalHeight) {
-     imgData.kind = kind;
-     imgArray = this.getImageBytes(originalHeight * rowBytes);
-     if (this.image instanceof DecodeStream) {
-      imgData.data = imgArray;
-     } else {
-      var newArray = new Uint8Array(imgArray.length);
-      newArray.set(imgArray);
-      imgData.data = newArray;
-     }
-     if (this.needsDecode) {
-      assert(kind === ImageKind.GRAYSCALE_1BPP);
-      var buffer = imgData.data;
-      for (var i = 0, ii = buffer.length; i < ii; i++) {
-       buffer[i] ^= 0xff;
+    return Promise.all([imagePromise, smaskPromise, maskPromise]).then(function (results) {
+      var imageData = results[0];
+      var smaskData = results[1];
+      var maskData = results[2];
+      return new PDFImage(xref, res, imageData, inline, smaskData, maskData);
+    });
+  };
+  PDFImage.createMask = function PDFImage_createMask(imgArray, width, height, imageIsFromDecodeStream, inverseDecode) {
+    var computedLength = (width + 7 >> 3) * height;
+    var actualLength = imgArray.byteLength;
+    var haveFullData = computedLength === actualLength;
+    var data, i;
+    if (imageIsFromDecodeStream && (!inverseDecode || haveFullData)) {
+      data = imgArray;
+    } else if (!inverseDecode) {
+      data = new Uint8Array(actualLength);
+      data.set(imgArray);
+    } else {
+      data = new Uint8Array(computedLength);
+      data.set(imgArray);
+      for (i = actualLength; i < computedLength; i++) {
+        data[i] = 0xff;
       }
-     }
-     return imgData;
     }
-    if (this.image instanceof JpegStream && !this.smask && !this.mask && (this.colorSpace.name === 'DeviceGray' || this.colorSpace.name === 'DeviceRGB' || this.colorSpace.name === 'DeviceCMYK')) {
-     imgData.kind = ImageKind.RGB_24BPP;
-     imgData.data = this.getImageBytes(originalHeight * rowBytes, drawWidth, drawHeight, true);
-     return imgData;
+    if (inverseDecode) {
+      for (i = 0; i < actualLength; i++) {
+        data[i] = ~data[i];
+      }
     }
-   }
-   imgArray = this.getImageBytes(originalHeight * rowBytes);
-   var actualHeight = 0 | imgArray.length / rowBytes * drawHeight / originalHeight;
-   var comps = this.getComponents(imgArray);
-   var alpha01, maybeUndoPreblend;
-   if (!forceRGBA && !this.smask && !this.mask) {
-    imgData.kind = ImageKind.RGB_24BPP;
-    imgData.data = new Uint8Array(drawWidth * drawHeight * 3);
-    alpha01 = 0;
-    maybeUndoPreblend = false;
-   } else {
-    imgData.kind = ImageKind.RGBA_32BPP;
-    imgData.data = new Uint8Array(drawWidth * drawHeight * 4);
-    alpha01 = 1;
-    maybeUndoPreblend = true;
-    this.fillOpacity(imgData.data, drawWidth, drawHeight, actualHeight, comps);
-   }
-   if (this.needsDecode) {
-    this.decodeBuffer(comps);
-   }
-   this.colorSpace.fillRgb(imgData.data, originalWidth, originalHeight, drawWidth, drawHeight, actualHeight, bpc, comps, alpha01);
-   if (maybeUndoPreblend) {
-    this.undoPreblend(imgData.data, drawWidth, actualHeight);
-   }
-   return imgData;
-  },
-  fillGrayBuffer: function PDFImage_fillGrayBuffer(buffer) {
-   var numComps = this.numComps;
-   if (numComps !== 1) {
-    error('Reading gray scale from a color image: ' + numComps);
-   }
-   var width = this.width;
-   var height = this.height;
-   var bpc = this.bpc;
-   var rowBytes = width * numComps * bpc + 7 >> 3;
-   var imgArray = this.getImageBytes(height * rowBytes);
-   var comps = this.getComponents(imgArray);
-   var i, length;
-   if (bpc === 1) {
-    length = width * height;
-    if (this.needsDecode) {
-     for (i = 0; i < length; ++i) {
-      buffer[i] = comps[i] - 1 & 255;
-     }
-    } else {
-     for (i = 0; i < length; ++i) {
-      buffer[i] = -comps[i] & 255;
-     }
+    return {
+      data: data,
+      width: width,
+      height: height
+    };
+  };
+  PDFImage.prototype = {
+    get drawWidth() {
+      return Math.max(this.width, this.smask && this.smask.width || 0, this.mask && this.mask.width || 0);
+    },
+    get drawHeight() {
+      return Math.max(this.height, this.smask && this.smask.height || 0, this.mask && this.mask.height || 0);
+    },
+    decodeBuffer: function PDFImage_decodeBuffer(buffer) {
+      var bpc = this.bpc;
+      var numComps = this.numComps;
+      var decodeAddends = this.decodeAddends;
+      var decodeCoefficients = this.decodeCoefficients;
+      var max = (1 << bpc) - 1;
+      var i, ii;
+      if (bpc === 1) {
+        for (i = 0, ii = buffer.length; i < ii; i++) {
+          buffer[i] = +!buffer[i];
+        }
+        return;
+      }
+      var index = 0;
+      for (i = 0, ii = this.width * this.height; i < ii; i++) {
+        for (var j = 0; j < numComps; j++) {
+          buffer[index] = decodeAndClamp(buffer[index], decodeAddends[j], decodeCoefficients[j], max);
+          index++;
+        }
+      }
+    },
+    getComponents: function PDFImage_getComponents(buffer) {
+      var bpc = this.bpc;
+      if (bpc === 8) {
+        return buffer;
+      }
+      var width = this.width;
+      var height = this.height;
+      var numComps = this.numComps;
+      var length = width * height * numComps;
+      var bufferPos = 0;
+      var output = bpc <= 8 ? new Uint8Array(length) : bpc <= 16 ? new Uint16Array(length) : new Uint32Array(length);
+      var rowComps = width * numComps;
+      var max = (1 << bpc) - 1;
+      var i = 0,
+          ii,
+          buf;
+      if (bpc === 1) {
+        var mask, loop1End, loop2End;
+        for (var j = 0; j < height; j++) {
+          loop1End = i + (rowComps & ~7);
+          loop2End = i + rowComps;
+          while (i < loop1End) {
+            buf = buffer[bufferPos++];
+            output[i] = buf >> 7 & 1;
+            output[i + 1] = buf >> 6 & 1;
+            output[i + 2] = buf >> 5 & 1;
+            output[i + 3] = buf >> 4 & 1;
+            output[i + 4] = buf >> 3 & 1;
+            output[i + 5] = buf >> 2 & 1;
+            output[i + 6] = buf >> 1 & 1;
+            output[i + 7] = buf & 1;
+            i += 8;
+          }
+          if (i < loop2End) {
+            buf = buffer[bufferPos++];
+            mask = 128;
+            while (i < loop2End) {
+              output[i++] = +!!(buf & mask);
+              mask >>= 1;
+            }
+          }
+        }
+      } else {
+        var bits = 0;
+        buf = 0;
+        for (i = 0, ii = length; i < ii; ++i) {
+          if (i % rowComps === 0) {
+            buf = 0;
+            bits = 0;
+          }
+          while (bits < bpc) {
+            buf = buf << 8 | buffer[bufferPos++];
+            bits += 8;
+          }
+          var remainingBits = bits - bpc;
+          var value = buf >> remainingBits;
+          output[i] = value < 0 ? 0 : value > max ? max : value;
+          buf = buf & (1 << remainingBits) - 1;
+          bits = remainingBits;
+        }
+      }
+      return output;
+    },
+    fillOpacity: function PDFImage_fillOpacity(rgbaBuf, width, height, actualHeight, image) {
+      var smask = this.smask;
+      var mask = this.mask;
+      var alphaBuf, sw, sh, i, ii, j;
+      if (smask) {
+        sw = smask.width;
+        sh = smask.height;
+        alphaBuf = new Uint8Array(sw * sh);
+        smask.fillGrayBuffer(alphaBuf);
+        if (sw !== width || sh !== height) {
+          alphaBuf = resizeImageMask(alphaBuf, smask.bpc, sw, sh, width, height);
+        }
+      } else if (mask) {
+        if (mask instanceof PDFImage) {
+          sw = mask.width;
+          sh = mask.height;
+          alphaBuf = new Uint8Array(sw * sh);
+          mask.numComps = 1;
+          mask.fillGrayBuffer(alphaBuf);
+          for (i = 0, ii = sw * sh; i < ii; ++i) {
+            alphaBuf[i] = 255 - alphaBuf[i];
+          }
+          if (sw !== width || sh !== height) {
+            alphaBuf = resizeImageMask(alphaBuf, mask.bpc, sw, sh, width, height);
+          }
+        } else if (isArray(mask)) {
+          alphaBuf = new Uint8Array(width * height);
+          var numComps = this.numComps;
+          for (i = 0, ii = width * height; i < ii; ++i) {
+            var opacity = 0;
+            var imageOffset = i * numComps;
+            for (j = 0; j < numComps; ++j) {
+              var color = image[imageOffset + j];
+              var maskOffset = j * 2;
+              if (color < mask[maskOffset] || color > mask[maskOffset + 1]) {
+                opacity = 255;
+                break;
+              }
+            }
+            alphaBuf[i] = opacity;
+          }
+        } else {
+          error('Unknown mask format.');
+        }
+      }
+      if (alphaBuf) {
+        for (i = 0, j = 3, ii = width * actualHeight; i < ii; ++i, j += 4) {
+          rgbaBuf[j] = alphaBuf[i];
+        }
+      } else {
+        for (i = 0, j = 3, ii = width * actualHeight; i < ii; ++i, j += 4) {
+          rgbaBuf[j] = 255;
+        }
+      }
+    },
+    undoPreblend: function PDFImage_undoPreblend(buffer, width, height) {
+      var matte = this.smask && this.smask.matte;
+      if (!matte) {
+        return;
+      }
+      var matteRgb = this.colorSpace.getRgb(matte, 0);
+      var matteR = matteRgb[0];
+      var matteG = matteRgb[1];
+      var matteB = matteRgb[2];
+      var length = width * height * 4;
+      var r, g, b;
+      for (var i = 0; i < length; i += 4) {
+        var alpha = buffer[i + 3];
+        if (alpha === 0) {
+          buffer[i] = 255;
+          buffer[i + 1] = 255;
+          buffer[i + 2] = 255;
+          continue;
+        }
+        var k = 255 / alpha;
+        r = (buffer[i] - matteR) * k + matteR;
+        g = (buffer[i + 1] - matteG) * k + matteG;
+        b = (buffer[i + 2] - matteB) * k + matteB;
+        buffer[i] = r <= 0 ? 0 : r >= 255 ? 255 : r | 0;
+        buffer[i + 1] = g <= 0 ? 0 : g >= 255 ? 255 : g | 0;
+        buffer[i + 2] = b <= 0 ? 0 : b >= 255 ? 255 : b | 0;
+      }
+    },
+    createImageData: function PDFImage_createImageData(forceRGBA) {
+      var drawWidth = this.drawWidth;
+      var drawHeight = this.drawHeight;
+      var imgData = {
+        width: drawWidth,
+        height: drawHeight
+      };
+      var numComps = this.numComps;
+      var originalWidth = this.width;
+      var originalHeight = this.height;
+      var bpc = this.bpc;
+      var rowBytes = originalWidth * numComps * bpc + 7 >> 3;
+      var imgArray;
+      if (!forceRGBA) {
+        var kind;
+        if (this.colorSpace.name === 'DeviceGray' && bpc === 1) {
+          kind = ImageKind.GRAYSCALE_1BPP;
+        } else if (this.colorSpace.name === 'DeviceRGB' && bpc === 8 && !this.needsDecode) {
+          kind = ImageKind.RGB_24BPP;
+        }
+        if (kind && !this.smask && !this.mask && drawWidth === originalWidth && drawHeight === originalHeight) {
+          imgData.kind = kind;
+          imgArray = this.getImageBytes(originalHeight * rowBytes);
+          if (this.image instanceof DecodeStream) {
+            imgData.data = imgArray;
+          } else {
+            var newArray = new Uint8Array(imgArray.length);
+            newArray.set(imgArray);
+            imgData.data = newArray;
+          }
+          if (this.needsDecode) {
+            assert(kind === ImageKind.GRAYSCALE_1BPP);
+            var buffer = imgData.data;
+            for (var i = 0, ii = buffer.length; i < ii; i++) {
+              buffer[i] ^= 0xff;
+            }
+          }
+          return imgData;
+        }
+        if (this.image instanceof JpegStream && !this.smask && !this.mask && (this.colorSpace.name === 'DeviceGray' || this.colorSpace.name === 'DeviceRGB' || this.colorSpace.name === 'DeviceCMYK')) {
+          imgData.kind = ImageKind.RGB_24BPP;
+          imgData.data = this.getImageBytes(originalHeight * rowBytes, drawWidth, drawHeight, true);
+          return imgData;
+        }
+      }
+      imgArray = this.getImageBytes(originalHeight * rowBytes);
+      var actualHeight = 0 | imgArray.length / rowBytes * drawHeight / originalHeight;
+      var comps = this.getComponents(imgArray);
+      var alpha01, maybeUndoPreblend;
+      if (!forceRGBA && !this.smask && !this.mask) {
+        imgData.kind = ImageKind.RGB_24BPP;
+        imgData.data = new Uint8Array(drawWidth * drawHeight * 3);
+        alpha01 = 0;
+        maybeUndoPreblend = false;
+      } else {
+        imgData.kind = ImageKind.RGBA_32BPP;
+        imgData.data = new Uint8Array(drawWidth * drawHeight * 4);
+        alpha01 = 1;
+        maybeUndoPreblend = true;
+        this.fillOpacity(imgData.data, drawWidth, drawHeight, actualHeight, comps);
+      }
+      if (this.needsDecode) {
+        this.decodeBuffer(comps);
+      }
+      this.colorSpace.fillRgb(imgData.data, originalWidth, originalHeight, drawWidth, drawHeight, actualHeight, bpc, comps, alpha01);
+      if (maybeUndoPreblend) {
+        this.undoPreblend(imgData.data, drawWidth, actualHeight);
+      }
+      return imgData;
+    },
+    fillGrayBuffer: function PDFImage_fillGrayBuffer(buffer) {
+      var numComps = this.numComps;
+      if (numComps !== 1) {
+        error('Reading gray scale from a color image: ' + numComps);
+      }
+      var width = this.width;
+      var height = this.height;
+      var bpc = this.bpc;
+      var rowBytes = width * numComps * bpc + 7 >> 3;
+      var imgArray = this.getImageBytes(height * rowBytes);
+      var comps = this.getComponents(imgArray);
+      var i, length;
+      if (bpc === 1) {
+        length = width * height;
+        if (this.needsDecode) {
+          for (i = 0; i < length; ++i) {
+            buffer[i] = comps[i] - 1 & 255;
+          }
+        } else {
+          for (i = 0; i < length; ++i) {
+            buffer[i] = -comps[i] & 255;
+          }
+        }
+        return;
+      }
+      if (this.needsDecode) {
+        this.decodeBuffer(comps);
+      }
+      length = width * height;
+      var scale = 255 / ((1 << bpc) - 1);
+      for (i = 0; i < length; ++i) {
+        buffer[i] = scale * comps[i] | 0;
+      }
+    },
+    getImageBytes: function PDFImage_getImageBytes(length, drawWidth, drawHeight, forceRGB) {
+      this.image.reset();
+      this.image.drawWidth = drawWidth || this.width;
+      this.image.drawHeight = drawHeight || this.height;
+      this.image.forceRGB = !!forceRGB;
+      return this.image.getBytes(length);
     }
-    return;
-   }
-   if (this.needsDecode) {
-    this.decodeBuffer(comps);
-   }
-   length = width * height;
-   var scale = 255 / ((1 << bpc) - 1);
-   for (i = 0; i < length; ++i) {
-    buffer[i] = scale * comps[i] | 0;
-   }
-  },
-  getImageBytes: function PDFImage_getImageBytes(length, drawWidth, drawHeight, forceRGB) {
-   this.image.reset();
-   this.image.drawWidth = drawWidth || this.width;
-   this.image.drawHeight = drawHeight || this.height;
-   this.image.forceRGB = !!forceRGB;
-   return this.image.getBytes(length);
-  }
- };
- return PDFImage;
+  };
+  return PDFImage;
 }();
 exports.PDFImage = PDFImage;

+ 900 - 1051
lib/core/jbig2.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var coreArithmeticDecoder = require('./arithmetic_decoder.js');
 var error = sharedUtil.error;
@@ -23,1187 +24,1035 @@ var readUint32 = sharedUtil.readUint32;
 var shadow = sharedUtil.shadow;
 var ArithmeticDecoder = coreArithmeticDecoder.ArithmeticDecoder;
 var Jbig2Image = function Jbig2ImageClosure() {
- function ContextCache() {
- }
- ContextCache.prototype = {
-  getContexts: function (id) {
-   if (id in this) {
-    return this[id];
-   }
-   return this[id] = new Int8Array(1 << 16);
-  }
- };
- function DecodingContext(data, start, end) {
-  this.data = data;
-  this.start = start;
-  this.end = end;
- }
- DecodingContext.prototype = {
-  get decoder() {
-   var decoder = new ArithmeticDecoder(this.data, this.start, this.end);
-   return shadow(this, 'decoder', decoder);
-  },
-  get contextCache() {
-   var cache = new ContextCache();
-   return shadow(this, 'contextCache', cache);
-  }
- };
- function decodeInteger(contextCache, procedure, decoder) {
-  var contexts = contextCache.getContexts(procedure);
-  var prev = 1;
-  function readBits(length) {
-   var v = 0;
-   for (var i = 0; i < length; i++) {
-    var bit = decoder.readBit(contexts, prev);
-    prev = prev < 256 ? prev << 1 | bit : (prev << 1 | bit) & 511 | 256;
-    v = v << 1 | bit;
-   }
-   return v >>> 0;
+  function ContextCache() {}
+  ContextCache.prototype = {
+    getContexts: function (id) {
+      if (id in this) {
+        return this[id];
+      }
+      return this[id] = new Int8Array(1 << 16);
+    }
+  };
+  function DecodingContext(data, start, end) {
+    this.data = data;
+    this.start = start;
+    this.end = end;
   }
-  var sign = readBits(1);
-  var value = readBits(1) ? readBits(1) ? readBits(1) ? readBits(1) ? readBits(1) ? readBits(32) + 4436 : readBits(12) + 340 : readBits(8) + 84 : readBits(6) + 20 : readBits(4) + 4 : readBits(2);
-  return sign === 0 ? value : value > 0 ? -value : null;
- }
- function decodeIAID(contextCache, decoder, codeLength) {
-  var contexts = contextCache.getContexts('IAID');
-  var prev = 1;
-  for (var i = 0; i < codeLength; i++) {
-   var bit = decoder.readBit(contexts, prev);
-   prev = prev << 1 | bit;
+  DecodingContext.prototype = {
+    get decoder() {
+      var decoder = new ArithmeticDecoder(this.data, this.start, this.end);
+      return shadow(this, 'decoder', decoder);
+    },
+    get contextCache() {
+      var cache = new ContextCache();
+      return shadow(this, 'contextCache', cache);
+    }
+  };
+  function decodeInteger(contextCache, procedure, decoder) {
+    var contexts = contextCache.getContexts(procedure);
+    var prev = 1;
+    function readBits(length) {
+      var v = 0;
+      for (var i = 0; i < length; i++) {
+        var bit = decoder.readBit(contexts, prev);
+        prev = prev < 256 ? prev << 1 | bit : (prev << 1 | bit) & 511 | 256;
+        v = v << 1 | bit;
+      }
+      return v >>> 0;
+    }
+    var sign = readBits(1);
+    var value = readBits(1) ? readBits(1) ? readBits(1) ? readBits(1) ? readBits(1) ? readBits(32) + 4436 : readBits(12) + 340 : readBits(8) + 84 : readBits(6) + 20 : readBits(4) + 4 : readBits(2);
+    return sign === 0 ? value : value > 0 ? -value : null;
   }
-  if (codeLength < 31) {
-   return prev & (1 << codeLength) - 1;
+  function decodeIAID(contextCache, decoder, codeLength) {
+    var contexts = contextCache.getContexts('IAID');
+    var prev = 1;
+    for (var i = 0; i < codeLength; i++) {
+      var bit = decoder.readBit(contexts, prev);
+      prev = prev << 1 | bit;
+    }
+    if (codeLength < 31) {
+      return prev & (1 << codeLength) - 1;
+    }
+    return prev & 0x7FFFFFFF;
   }
-  return prev & 0x7FFFFFFF;
- }
- var SegmentTypes = [
-  'SymbolDictionary',
-  null,
-  null,
-  null,
-  'IntermediateTextRegion',
-  null,
-  'ImmediateTextRegion',
-  'ImmediateLosslessTextRegion',
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  'patternDictionary',
-  null,
-  null,
-  null,
-  'IntermediateHalftoneRegion',
-  null,
-  'ImmediateHalftoneRegion',
-  'ImmediateLosslessHalftoneRegion',
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  'IntermediateGenericRegion',
-  null,
-  'ImmediateGenericRegion',
-  'ImmediateLosslessGenericRegion',
-  'IntermediateGenericRefinementRegion',
-  null,
-  'ImmediateGenericRefinementRegion',
-  'ImmediateLosslessGenericRefinementRegion',
-  null,
-  null,
-  null,
-  null,
-  'PageInformation',
-  'EndOfPage',
-  'EndOfStripe',
-  'EndOfFile',
-  'Profiles',
-  'Tables',
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  null,
-  'Extension'
- ];
- var CodingTemplates = [
-  [
-   {
+  var SegmentTypes = ['SymbolDictionary', null, null, null, 'IntermediateTextRegion', null, 'ImmediateTextRegion', 'ImmediateLosslessTextRegion', null, null, null, null, null, null, null, null, 'patternDictionary', null, null, null, 'IntermediateHalftoneRegion', null, 'ImmediateHalftoneRegion', 'ImmediateLosslessHalftoneRegion', null, null, null, null, null, null, null, null, null, null, null, null, 'IntermediateGenericRegion', null, 'ImmediateGenericRegion', 'ImmediateLosslessGenericRegion', 'IntermediateGenericRefinementRegion', null, 'ImmediateGenericRefinementRegion', 'ImmediateLosslessGenericRefinementRegion', null, null, null, null, 'PageInformation', 'EndOfPage', 'EndOfStripe', 'EndOfFile', 'Profiles', 'Tables', null, null, null, null, null, null, null, null, 'Extension'];
+  var CodingTemplates = [[{
     x: -1,
     y: -2
-   },
-   {
+  }, {
     x: 0,
     y: -2
-   },
-   {
+  }, {
     x: 1,
     y: -2
-   },
-   {
+  }, {
     x: -2,
     y: -1
-   },
-   {
+  }, {
     x: -1,
     y: -1
-   },
-   {
+  }, {
     x: 0,
     y: -1
-   },
-   {
+  }, {
     x: 1,
     y: -1
-   },
-   {
+  }, {
     x: 2,
     y: -1
-   },
-   {
+  }, {
     x: -4,
     y: 0
-   },
-   {
+  }, {
     x: -3,
     y: 0
-   },
-   {
+  }, {
     x: -2,
     y: 0
-   },
-   {
+  }, {
     x: -1,
     y: 0
-   }
-  ],
-  [
-   {
+  }], [{
     x: -1,
     y: -2
-   },
-   {
+  }, {
     x: 0,
     y: -2
-   },
-   {
+  }, {
     x: 1,
     y: -2
-   },
-   {
+  }, {
     x: 2,
     y: -2
-   },
-   {
+  }, {
     x: -2,
     y: -1
-   },
-   {
+  }, {
     x: -1,
     y: -1
-   },
-   {
+  }, {
     x: 0,
     y: -1
-   },
-   {
+  }, {
     x: 1,
     y: -1
-   },
-   {
+  }, {
     x: 2,
     y: -1
-   },
-   {
+  }, {
     x: -3,
     y: 0
-   },
-   {
+  }, {
     x: -2,
     y: 0
-   },
-   {
+  }, {
     x: -1,
     y: 0
-   }
-  ],
-  [
-   {
+  }], [{
     x: -1,
     y: -2
-   },
-   {
+  }, {
     x: 0,
     y: -2
-   },
-   {
+  }, {
     x: 1,
     y: -2
-   },
-   {
+  }, {
     x: -2,
     y: -1
-   },
-   {
+  }, {
     x: -1,
     y: -1
-   },
-   {
+  }, {
     x: 0,
     y: -1
-   },
-   {
+  }, {
     x: 1,
     y: -1
-   },
-   {
+  }, {
     x: -2,
     y: 0
-   },
-   {
+  }, {
     x: -1,
     y: 0
-   }
-  ],
-  [
-   {
+  }], [{
     x: -3,
     y: -1
-   },
-   {
+  }, {
     x: -2,
     y: -1
-   },
-   {
+  }, {
     x: -1,
     y: -1
-   },
-   {
+  }, {
     x: 0,
     y: -1
-   },
-   {
+  }, {
     x: 1,
     y: -1
-   },
-   {
+  }, {
     x: -4,
     y: 0
-   },
-   {
+  }, {
     x: -3,
     y: 0
-   },
-   {
+  }, {
     x: -2,
     y: 0
-   },
-   {
+  }, {
     x: -1,
     y: 0
-   }
-  ]
- ];
- var RefinementTemplates = [
-  {
-   coding: [
-    {
-     x: 0,
-     y: -1
-    },
-    {
-     x: 1,
-     y: -1
-    },
-    {
-     x: -1,
-     y: 0
-    }
-   ],
-   reference: [
-    {
-     x: 0,
-     y: -1
-    },
-    {
-     x: 1,
-     y: -1
-    },
-    {
-     x: -1,
-     y: 0
-    },
-    {
-     x: 0,
-     y: 0
-    },
-    {
-     x: 1,
-     y: 0
-    },
-    {
-     x: -1,
-     y: 1
-    },
-    {
-     x: 0,
-     y: 1
-    },
-    {
-     x: 1,
-     y: 1
+  }]];
+  var RefinementTemplates = [{
+    coding: [{
+      x: 0,
+      y: -1
+    }, {
+      x: 1,
+      y: -1
+    }, {
+      x: -1,
+      y: 0
+    }],
+    reference: [{
+      x: 0,
+      y: -1
+    }, {
+      x: 1,
+      y: -1
+    }, {
+      x: -1,
+      y: 0
+    }, {
+      x: 0,
+      y: 0
+    }, {
+      x: 1,
+      y: 0
+    }, {
+      x: -1,
+      y: 1
+    }, {
+      x: 0,
+      y: 1
+    }, {
+      x: 1,
+      y: 1
+    }]
+  }, {
+    coding: [{
+      x: -1,
+      y: -1
+    }, {
+      x: 0,
+      y: -1
+    }, {
+      x: 1,
+      y: -1
+    }, {
+      x: -1,
+      y: 0
+    }],
+    reference: [{
+      x: 0,
+      y: -1
+    }, {
+      x: -1,
+      y: 0
+    }, {
+      x: 0,
+      y: 0
+    }, {
+      x: 1,
+      y: 0
+    }, {
+      x: 0,
+      y: 1
+    }, {
+      x: 1,
+      y: 1
+    }]
+  }];
+  var ReusedContexts = [0x9B25, 0x0795, 0x00E5, 0x0195];
+  var RefinementReusedContexts = [0x0020, 0x0008];
+  function decodeBitmapTemplate0(width, height, decodingContext) {
+    var decoder = decodingContext.decoder;
+    var contexts = decodingContext.contextCache.getContexts('GB');
+    var contextLabel,
+        i,
+        j,
+        pixel,
+        row,
+        row1,
+        row2,
+        bitmap = [];
+    var OLD_PIXEL_MASK = 0x7BF7;
+    for (i = 0; i < height; i++) {
+      row = bitmap[i] = new Uint8Array(width);
+      row1 = i < 1 ? row : bitmap[i - 1];
+      row2 = i < 2 ? row : bitmap[i - 2];
+      contextLabel = row2[0] << 13 | row2[1] << 12 | row2[2] << 11 | row1[0] << 7 | row1[1] << 6 | row1[2] << 5 | row1[3] << 4;
+      for (j = 0; j < width; j++) {
+        row[j] = pixel = decoder.readBit(contexts, contextLabel);
+        contextLabel = (contextLabel & OLD_PIXEL_MASK) << 1 | (j + 3 < width ? row2[j + 3] << 11 : 0) | (j + 4 < width ? row1[j + 4] << 4 : 0) | pixel;
+      }
     }
-   ]
-  },
-  {
-   coding: [
-    {
-     x: -1,
-     y: -1
-    },
-    {
-     x: 0,
-     y: -1
-    },
-    {
-     x: 1,
-     y: -1
-    },
-    {
-     x: -1,
-     y: 0
+    return bitmap;
+  }
+  function decodeBitmap(mmr, width, height, templateIndex, prediction, skip, at, decodingContext) {
+    if (mmr) {
+      error('JBIG2 error: MMR encoding is not supported');
     }
-   ],
-   reference: [
-    {
-     x: 0,
-     y: -1
-    },
-    {
-     x: -1,
-     y: 0
-    },
-    {
-     x: 0,
-     y: 0
-    },
-    {
-     x: 1,
-     y: 0
-    },
-    {
-     x: 0,
-     y: 1
-    },
-    {
-     x: 1,
-     y: 1
+    if (templateIndex === 0 && !skip && !prediction && at.length === 4 && at[0].x === 3 && at[0].y === -1 && at[1].x === -3 && at[1].y === -1 && at[2].x === 2 && at[2].y === -2 && at[3].x === -2 && at[3].y === -2) {
+      return decodeBitmapTemplate0(width, height, decodingContext);
     }
-   ]
-  }
- ];
- var ReusedContexts = [
-  0x9B25,
-  0x0795,
-  0x00E5,
-  0x0195
- ];
- var RefinementReusedContexts = [
-  0x0020,
-  0x0008
- ];
- function decodeBitmapTemplate0(width, height, decodingContext) {
-  var decoder = decodingContext.decoder;
-  var contexts = decodingContext.contextCache.getContexts('GB');
-  var contextLabel, i, j, pixel, row, row1, row2, bitmap = [];
-  var OLD_PIXEL_MASK = 0x7BF7;
-  for (i = 0; i < height; i++) {
-   row = bitmap[i] = new Uint8Array(width);
-   row1 = i < 1 ? row : bitmap[i - 1];
-   row2 = i < 2 ? row : bitmap[i - 2];
-   contextLabel = row2[0] << 13 | row2[1] << 12 | row2[2] << 11 | row1[0] << 7 | row1[1] << 6 | row1[2] << 5 | row1[3] << 4;
-   for (j = 0; j < width; j++) {
-    row[j] = pixel = decoder.readBit(contexts, contextLabel);
-    contextLabel = (contextLabel & OLD_PIXEL_MASK) << 1 | (j + 3 < width ? row2[j + 3] << 11 : 0) | (j + 4 < width ? row1[j + 4] << 4 : 0) | pixel;
-   }
-  }
-  return bitmap;
- }
- function decodeBitmap(mmr, width, height, templateIndex, prediction, skip, at, decodingContext) {
-  if (mmr) {
-   error('JBIG2 error: MMR encoding is not supported');
-  }
-  if (templateIndex === 0 && !skip && !prediction && at.length === 4 && at[0].x === 3 && at[0].y === -1 && at[1].x === -3 && at[1].y === -1 && at[2].x === 2 && at[2].y === -2 && at[3].x === -2 && at[3].y === -2) {
-   return decodeBitmapTemplate0(width, height, decodingContext);
-  }
-  var useskip = !!skip;
-  var template = CodingTemplates[templateIndex].concat(at);
-  template.sort(function (a, b) {
-   return a.y - b.y || a.x - b.x;
-  });
-  var templateLength = template.length;
-  var templateX = new Int8Array(templateLength);
-  var templateY = new Int8Array(templateLength);
-  var changingTemplateEntries = [];
-  var reuseMask = 0, minX = 0, maxX = 0, minY = 0;
-  var c, k;
-  for (k = 0; k < templateLength; k++) {
-   templateX[k] = template[k].x;
-   templateY[k] = template[k].y;
-   minX = Math.min(minX, template[k].x);
-   maxX = Math.max(maxX, template[k].x);
-   minY = Math.min(minY, template[k].y);
-   if (k < templateLength - 1 && template[k].y === template[k + 1].y && template[k].x === template[k + 1].x - 1) {
-    reuseMask |= 1 << templateLength - 1 - k;
-   } else {
-    changingTemplateEntries.push(k);
-   }
-  }
-  var changingEntriesLength = changingTemplateEntries.length;
-  var changingTemplateX = new Int8Array(changingEntriesLength);
-  var changingTemplateY = new Int8Array(changingEntriesLength);
-  var changingTemplateBit = new Uint16Array(changingEntriesLength);
-  for (c = 0; c < changingEntriesLength; c++) {
-   k = changingTemplateEntries[c];
-   changingTemplateX[c] = template[k].x;
-   changingTemplateY[c] = template[k].y;
-   changingTemplateBit[c] = 1 << templateLength - 1 - k;
-  }
-  var sbb_left = -minX;
-  var sbb_top = -minY;
-  var sbb_right = width - maxX;
-  var pseudoPixelContext = ReusedContexts[templateIndex];
-  var row = new Uint8Array(width);
-  var bitmap = [];
-  var decoder = decodingContext.decoder;
-  var contexts = decodingContext.contextCache.getContexts('GB');
-  var ltp = 0, j, i0, j0, contextLabel = 0, bit, shift;
-  for (var i = 0; i < height; i++) {
-   if (prediction) {
-    var sltp = decoder.readBit(contexts, pseudoPixelContext);
-    ltp ^= sltp;
-    if (ltp) {
-     bitmap.push(row);
-     continue;
+    var useskip = !!skip;
+    var template = CodingTemplates[templateIndex].concat(at);
+    template.sort(function (a, b) {
+      return a.y - b.y || a.x - b.x;
+    });
+    var templateLength = template.length;
+    var templateX = new Int8Array(templateLength);
+    var templateY = new Int8Array(templateLength);
+    var changingTemplateEntries = [];
+    var reuseMask = 0,
+        minX = 0,
+        maxX = 0,
+        minY = 0;
+    var c, k;
+    for (k = 0; k < templateLength; k++) {
+      templateX[k] = template[k].x;
+      templateY[k] = template[k].y;
+      minX = Math.min(minX, template[k].x);
+      maxX = Math.max(maxX, template[k].x);
+      minY = Math.min(minY, template[k].y);
+      if (k < templateLength - 1 && template[k].y === template[k + 1].y && template[k].x === template[k + 1].x - 1) {
+        reuseMask |= 1 << templateLength - 1 - k;
+      } else {
+        changingTemplateEntries.push(k);
+      }
     }
-   }
-   row = new Uint8Array(row);
-   bitmap.push(row);
-   for (j = 0; j < width; j++) {
-    if (useskip && skip[i][j]) {
-     row[j] = 0;
-     continue;
+    var changingEntriesLength = changingTemplateEntries.length;
+    var changingTemplateX = new Int8Array(changingEntriesLength);
+    var changingTemplateY = new Int8Array(changingEntriesLength);
+    var changingTemplateBit = new Uint16Array(changingEntriesLength);
+    for (c = 0; c < changingEntriesLength; c++) {
+      k = changingTemplateEntries[c];
+      changingTemplateX[c] = template[k].x;
+      changingTemplateY[c] = template[k].y;
+      changingTemplateBit[c] = 1 << templateLength - 1 - k;
     }
-    if (j >= sbb_left && j < sbb_right && i >= sbb_top) {
-     contextLabel = contextLabel << 1 & reuseMask;
-     for (k = 0; k < changingEntriesLength; k++) {
-      i0 = i + changingTemplateY[k];
-      j0 = j + changingTemplateX[k];
-      bit = bitmap[i0][j0];
-      if (bit) {
-       bit = changingTemplateBit[k];
-       contextLabel |= bit;
+    var sbb_left = -minX;
+    var sbb_top = -minY;
+    var sbb_right = width - maxX;
+    var pseudoPixelContext = ReusedContexts[templateIndex];
+    var row = new Uint8Array(width);
+    var bitmap = [];
+    var decoder = decodingContext.decoder;
+    var contexts = decodingContext.contextCache.getContexts('GB');
+    var ltp = 0,
+        j,
+        i0,
+        j0,
+        contextLabel = 0,
+        bit,
+        shift;
+    for (var i = 0; i < height; i++) {
+      if (prediction) {
+        var sltp = decoder.readBit(contexts, pseudoPixelContext);
+        ltp ^= sltp;
+        if (ltp) {
+          bitmap.push(row);
+          continue;
+        }
       }
-     }
-    } else {
-     contextLabel = 0;
-     shift = templateLength - 1;
-     for (k = 0; k < templateLength; k++, shift--) {
-      j0 = j + templateX[k];
-      if (j0 >= 0 && j0 < width) {
-       i0 = i + templateY[k];
-       if (i0 >= 0) {
-        bit = bitmap[i0][j0];
-        if (bit) {
-         contextLabel |= bit << shift;
+      row = new Uint8Array(row);
+      bitmap.push(row);
+      for (j = 0; j < width; j++) {
+        if (useskip && skip[i][j]) {
+          row[j] = 0;
+          continue;
+        }
+        if (j >= sbb_left && j < sbb_right && i >= sbb_top) {
+          contextLabel = contextLabel << 1 & reuseMask;
+          for (k = 0; k < changingEntriesLength; k++) {
+            i0 = i + changingTemplateY[k];
+            j0 = j + changingTemplateX[k];
+            bit = bitmap[i0][j0];
+            if (bit) {
+              bit = changingTemplateBit[k];
+              contextLabel |= bit;
+            }
+          }
+        } else {
+          contextLabel = 0;
+          shift = templateLength - 1;
+          for (k = 0; k < templateLength; k++, shift--) {
+            j0 = j + templateX[k];
+            if (j0 >= 0 && j0 < width) {
+              i0 = i + templateY[k];
+              if (i0 >= 0) {
+                bit = bitmap[i0][j0];
+                if (bit) {
+                  contextLabel |= bit << shift;
+                }
+              }
+            }
+          }
         }
-       }
+        var pixel = decoder.readBit(contexts, contextLabel);
+        row[j] = pixel;
       }
-     }
     }
-    var pixel = decoder.readBit(contexts, contextLabel);
-    row[j] = pixel;
-   }
+    return bitmap;
   }
-  return bitmap;
- }
- function decodeRefinement(width, height, templateIndex, referenceBitmap, offsetX, offsetY, prediction, at, decodingContext) {
-  var codingTemplate = RefinementTemplates[templateIndex].coding;
-  if (templateIndex === 0) {
-   codingTemplate = codingTemplate.concat([at[0]]);
-  }
-  var codingTemplateLength = codingTemplate.length;
-  var codingTemplateX = new Int32Array(codingTemplateLength);
-  var codingTemplateY = new Int32Array(codingTemplateLength);
-  var k;
-  for (k = 0; k < codingTemplateLength; k++) {
-   codingTemplateX[k] = codingTemplate[k].x;
-   codingTemplateY[k] = codingTemplate[k].y;
-  }
-  var referenceTemplate = RefinementTemplates[templateIndex].reference;
-  if (templateIndex === 0) {
-   referenceTemplate = referenceTemplate.concat([at[1]]);
-  }
-  var referenceTemplateLength = referenceTemplate.length;
-  var referenceTemplateX = new Int32Array(referenceTemplateLength);
-  var referenceTemplateY = new Int32Array(referenceTemplateLength);
-  for (k = 0; k < referenceTemplateLength; k++) {
-   referenceTemplateX[k] = referenceTemplate[k].x;
-   referenceTemplateY[k] = referenceTemplate[k].y;
-  }
-  var referenceWidth = referenceBitmap[0].length;
-  var referenceHeight = referenceBitmap.length;
-  var pseudoPixelContext = RefinementReusedContexts[templateIndex];
-  var bitmap = [];
-  var decoder = decodingContext.decoder;
-  var contexts = decodingContext.contextCache.getContexts('GR');
-  var ltp = 0;
-  for (var i = 0; i < height; i++) {
-   if (prediction) {
-    var sltp = decoder.readBit(contexts, pseudoPixelContext);
-    ltp ^= sltp;
-    if (ltp) {
-     error('JBIG2 error: prediction is not supported');
+  function decodeRefinement(width, height, templateIndex, referenceBitmap, offsetX, offsetY, prediction, at, decodingContext) {
+    var codingTemplate = RefinementTemplates[templateIndex].coding;
+    if (templateIndex === 0) {
+      codingTemplate = codingTemplate.concat([at[0]]);
     }
-   }
-   var row = new Uint8Array(width);
-   bitmap.push(row);
-   for (var j = 0; j < width; j++) {
-    var i0, j0;
-    var contextLabel = 0;
+    var codingTemplateLength = codingTemplate.length;
+    var codingTemplateX = new Int32Array(codingTemplateLength);
+    var codingTemplateY = new Int32Array(codingTemplateLength);
+    var k;
     for (k = 0; k < codingTemplateLength; k++) {
-     i0 = i + codingTemplateY[k];
-     j0 = j + codingTemplateX[k];
-     if (i0 < 0 || j0 < 0 || j0 >= width) {
-      contextLabel <<= 1;
-     } else {
-      contextLabel = contextLabel << 1 | bitmap[i0][j0];
-     }
+      codingTemplateX[k] = codingTemplate[k].x;
+      codingTemplateY[k] = codingTemplate[k].y;
     }
-    for (k = 0; k < referenceTemplateLength; k++) {
-     i0 = i + referenceTemplateY[k] + offsetY;
-     j0 = j + referenceTemplateX[k] + offsetX;
-     if (i0 < 0 || i0 >= referenceHeight || j0 < 0 || j0 >= referenceWidth) {
-      contextLabel <<= 1;
-     } else {
-      contextLabel = contextLabel << 1 | referenceBitmap[i0][j0];
-     }
-    }
-    var pixel = decoder.readBit(contexts, contextLabel);
-    row[j] = pixel;
-   }
-  }
-  return bitmap;
- }
- function decodeSymbolDictionary(huffman, refinement, symbols, numberOfNewSymbols, numberOfExportedSymbols, huffmanTables, templateIndex, at, refinementTemplateIndex, refinementAt, decodingContext) {
-  if (huffman) {
-   error('JBIG2 error: huffman is not supported');
-  }
-  var newSymbols = [];
-  var currentHeight = 0;
-  var symbolCodeLength = log2(symbols.length + numberOfNewSymbols);
-  var decoder = decodingContext.decoder;
-  var contextCache = decodingContext.contextCache;
-  while (newSymbols.length < numberOfNewSymbols) {
-   var deltaHeight = decodeInteger(contextCache, 'IADH', decoder);
-   currentHeight += deltaHeight;
-   var currentWidth = 0;
-   while (true) {
-    var deltaWidth = decodeInteger(contextCache, 'IADW', decoder);
-    if (deltaWidth === null) {
-     break;
+    var referenceTemplate = RefinementTemplates[templateIndex].reference;
+    if (templateIndex === 0) {
+      referenceTemplate = referenceTemplate.concat([at[1]]);
     }
-    currentWidth += deltaWidth;
-    var bitmap;
-    if (refinement) {
-     var numberOfInstances = decodeInteger(contextCache, 'IAAI', decoder);
-     if (numberOfInstances > 1) {
-      bitmap = decodeTextRegion(huffman, refinement, currentWidth, currentHeight, 0, numberOfInstances, 1, symbols.concat(newSymbols), symbolCodeLength, 0, 0, 1, 0, huffmanTables, refinementTemplateIndex, refinementAt, decodingContext);
-     } else {
-      var symbolId = decodeIAID(contextCache, decoder, symbolCodeLength);
-      var rdx = decodeInteger(contextCache, 'IARDX', decoder);
-      var rdy = decodeInteger(contextCache, 'IARDY', decoder);
-      var symbol = symbolId < symbols.length ? symbols[symbolId] : newSymbols[symbolId - symbols.length];
-      bitmap = decodeRefinement(currentWidth, currentHeight, refinementTemplateIndex, symbol, rdx, rdy, false, refinementAt, decodingContext);
-     }
-    } else {
-     bitmap = decodeBitmap(false, currentWidth, currentHeight, templateIndex, false, null, at, decodingContext);
+    var referenceTemplateLength = referenceTemplate.length;
+    var referenceTemplateX = new Int32Array(referenceTemplateLength);
+    var referenceTemplateY = new Int32Array(referenceTemplateLength);
+    for (k = 0; k < referenceTemplateLength; k++) {
+      referenceTemplateX[k] = referenceTemplate[k].x;
+      referenceTemplateY[k] = referenceTemplate[k].y;
     }
-    newSymbols.push(bitmap);
-   }
-  }
-  var exportedSymbols = [];
-  var flags = [], currentFlag = false;
-  var totalSymbolsLength = symbols.length + numberOfNewSymbols;
-  while (flags.length < totalSymbolsLength) {
-   var runLength = decodeInteger(contextCache, 'IAEX', decoder);
-   while (runLength--) {
-    flags.push(currentFlag);
-   }
-   currentFlag = !currentFlag;
-  }
-  for (var i = 0, ii = symbols.length; i < ii; i++) {
-   if (flags[i]) {
-    exportedSymbols.push(symbols[i]);
-   }
-  }
-  for (var j = 0; j < numberOfNewSymbols; i++, j++) {
-   if (flags[i]) {
-    exportedSymbols.push(newSymbols[j]);
-   }
-  }
-  return exportedSymbols;
- }
- function decodeTextRegion(huffman, refinement, width, height, defaultPixelValue, numberOfSymbolInstances, stripSize, inputSymbols, symbolCodeLength, transposed, dsOffset, referenceCorner, combinationOperator, huffmanTables, refinementTemplateIndex, refinementAt, decodingContext) {
-  if (huffman) {
-   error('JBIG2 error: huffman is not supported');
-  }
-  var bitmap = [];
-  var i, row;
-  for (i = 0; i < height; i++) {
-   row = new Uint8Array(width);
-   if (defaultPixelValue) {
-    for (var j = 0; j < width; j++) {
-     row[j] = defaultPixelValue;
+    var referenceWidth = referenceBitmap[0].length;
+    var referenceHeight = referenceBitmap.length;
+    var pseudoPixelContext = RefinementReusedContexts[templateIndex];
+    var bitmap = [];
+    var decoder = decodingContext.decoder;
+    var contexts = decodingContext.contextCache.getContexts('GR');
+    var ltp = 0;
+    for (var i = 0; i < height; i++) {
+      if (prediction) {
+        var sltp = decoder.readBit(contexts, pseudoPixelContext);
+        ltp ^= sltp;
+        if (ltp) {
+          error('JBIG2 error: prediction is not supported');
+        }
+      }
+      var row = new Uint8Array(width);
+      bitmap.push(row);
+      for (var j = 0; j < width; j++) {
+        var i0, j0;
+        var contextLabel = 0;
+        for (k = 0; k < codingTemplateLength; k++) {
+          i0 = i + codingTemplateY[k];
+          j0 = j + codingTemplateX[k];
+          if (i0 < 0 || j0 < 0 || j0 >= width) {
+            contextLabel <<= 1;
+          } else {
+            contextLabel = contextLabel << 1 | bitmap[i0][j0];
+          }
+        }
+        for (k = 0; k < referenceTemplateLength; k++) {
+          i0 = i + referenceTemplateY[k] + offsetY;
+          j0 = j + referenceTemplateX[k] + offsetX;
+          if (i0 < 0 || i0 >= referenceHeight || j0 < 0 || j0 >= referenceWidth) {
+            contextLabel <<= 1;
+          } else {
+            contextLabel = contextLabel << 1 | referenceBitmap[i0][j0];
+          }
+        }
+        var pixel = decoder.readBit(contexts, contextLabel);
+        row[j] = pixel;
+      }
     }
-   }
-   bitmap.push(row);
+    return bitmap;
   }
-  var decoder = decodingContext.decoder;
-  var contextCache = decodingContext.contextCache;
-  var stripT = -decodeInteger(contextCache, 'IADT', decoder);
-  var firstS = 0;
-  i = 0;
-  while (i < numberOfSymbolInstances) {
-   var deltaT = decodeInteger(contextCache, 'IADT', decoder);
-   stripT += deltaT;
-   var deltaFirstS = decodeInteger(contextCache, 'IAFS', decoder);
-   firstS += deltaFirstS;
-   var currentS = firstS;
-   do {
-    var currentT = stripSize === 1 ? 0 : decodeInteger(contextCache, 'IAIT', decoder);
-    var t = stripSize * stripT + currentT;
-    var symbolId = decodeIAID(contextCache, decoder, symbolCodeLength);
-    var applyRefinement = refinement && decodeInteger(contextCache, 'IARI', decoder);
-    var symbolBitmap = inputSymbols[symbolId];
-    var symbolWidth = symbolBitmap[0].length;
-    var symbolHeight = symbolBitmap.length;
-    if (applyRefinement) {
-     var rdw = decodeInteger(contextCache, 'IARDW', decoder);
-     var rdh = decodeInteger(contextCache, 'IARDH', decoder);
-     var rdx = decodeInteger(contextCache, 'IARDX', decoder);
-     var rdy = decodeInteger(contextCache, 'IARDY', decoder);
-     symbolWidth += rdw;
-     symbolHeight += rdh;
-     symbolBitmap = decodeRefinement(symbolWidth, symbolHeight, refinementTemplateIndex, symbolBitmap, (rdw >> 1) + rdx, (rdh >> 1) + rdy, false, refinementAt, decodingContext);
+  function decodeSymbolDictionary(huffman, refinement, symbols, numberOfNewSymbols, numberOfExportedSymbols, huffmanTables, templateIndex, at, refinementTemplateIndex, refinementAt, decodingContext) {
+    if (huffman) {
+      error('JBIG2 error: huffman is not supported');
     }
-    var offsetT = t - (referenceCorner & 1 ? 0 : symbolHeight);
-    var offsetS = currentS - (referenceCorner & 2 ? symbolWidth : 0);
-    var s2, t2, symbolRow;
-    if (transposed) {
-     for (s2 = 0; s2 < symbolHeight; s2++) {
-      row = bitmap[offsetS + s2];
-      if (!row) {
-       continue;
-      }
-      symbolRow = symbolBitmap[s2];
-      var maxWidth = Math.min(width - offsetT, symbolWidth);
-      switch (combinationOperator) {
-      case 0:
-       for (t2 = 0; t2 < maxWidth; t2++) {
-        row[offsetT + t2] |= symbolRow[t2];
-       }
-       break;
-      case 2:
-       for (t2 = 0; t2 < maxWidth; t2++) {
-        row[offsetT + t2] ^= symbolRow[t2];
-       }
-       break;
-      default:
-       error('JBIG2 error: operator ' + combinationOperator + ' is not supported');
+    var newSymbols = [];
+    var currentHeight = 0;
+    var symbolCodeLength = log2(symbols.length + numberOfNewSymbols);
+    var decoder = decodingContext.decoder;
+    var contextCache = decodingContext.contextCache;
+    while (newSymbols.length < numberOfNewSymbols) {
+      var deltaHeight = decodeInteger(contextCache, 'IADH', decoder);
+      currentHeight += deltaHeight;
+      var currentWidth = 0;
+      while (true) {
+        var deltaWidth = decodeInteger(contextCache, 'IADW', decoder);
+        if (deltaWidth === null) {
+          break;
+        }
+        currentWidth += deltaWidth;
+        var bitmap;
+        if (refinement) {
+          var numberOfInstances = decodeInteger(contextCache, 'IAAI', decoder);
+          if (numberOfInstances > 1) {
+            bitmap = decodeTextRegion(huffman, refinement, currentWidth, currentHeight, 0, numberOfInstances, 1, symbols.concat(newSymbols), symbolCodeLength, 0, 0, 1, 0, huffmanTables, refinementTemplateIndex, refinementAt, decodingContext);
+          } else {
+            var symbolId = decodeIAID(contextCache, decoder, symbolCodeLength);
+            var rdx = decodeInteger(contextCache, 'IARDX', decoder);
+            var rdy = decodeInteger(contextCache, 'IARDY', decoder);
+            var symbol = symbolId < symbols.length ? symbols[symbolId] : newSymbols[symbolId - symbols.length];
+            bitmap = decodeRefinement(currentWidth, currentHeight, refinementTemplateIndex, symbol, rdx, rdy, false, refinementAt, decodingContext);
+          }
+        } else {
+          bitmap = decodeBitmap(false, currentWidth, currentHeight, templateIndex, false, null, at, decodingContext);
+        }
+        newSymbols.push(bitmap);
       }
-     }
-     currentS += symbolHeight - 1;
-    } else {
-     for (t2 = 0; t2 < symbolHeight; t2++) {
-      row = bitmap[offsetT + t2];
-      if (!row) {
-       continue;
+    }
+    var exportedSymbols = [];
+    var flags = [],
+        currentFlag = false;
+    var totalSymbolsLength = symbols.length + numberOfNewSymbols;
+    while (flags.length < totalSymbolsLength) {
+      var runLength = decodeInteger(contextCache, 'IAEX', decoder);
+      while (runLength--) {
+        flags.push(currentFlag);
       }
-      symbolRow = symbolBitmap[t2];
-      switch (combinationOperator) {
-      case 0:
-       for (s2 = 0; s2 < symbolWidth; s2++) {
-        row[offsetS + s2] |= symbolRow[s2];
-       }
-       break;
-      case 2:
-       for (s2 = 0; s2 < symbolWidth; s2++) {
-        row[offsetS + s2] ^= symbolRow[s2];
-       }
-       break;
-      default:
-       error('JBIG2 error: operator ' + combinationOperator + ' is not supported');
+      currentFlag = !currentFlag;
+    }
+    for (var i = 0, ii = symbols.length; i < ii; i++) {
+      if (flags[i]) {
+        exportedSymbols.push(symbols[i]);
       }
-     }
-     currentS += symbolWidth - 1;
     }
-    i++;
-    var deltaS = decodeInteger(contextCache, 'IADS', decoder);
-    if (deltaS === null) {
-     break;
+    for (var j = 0; j < numberOfNewSymbols; i++, j++) {
+      if (flags[i]) {
+        exportedSymbols.push(newSymbols[j]);
+      }
     }
-    currentS += deltaS + dsOffset;
-   } while (true);
-  }
-  return bitmap;
- }
- function readSegmentHeader(data, start) {
-  var segmentHeader = {};
-  segmentHeader.number = readUint32(data, start);
-  var flags = data[start + 4];
-  var segmentType = flags & 0x3F;
-  if (!SegmentTypes[segmentType]) {
-   error('JBIG2 error: invalid segment type: ' + segmentType);
+    return exportedSymbols;
   }
-  segmentHeader.type = segmentType;
-  segmentHeader.typeName = SegmentTypes[segmentType];
-  segmentHeader.deferredNonRetain = !!(flags & 0x80);
-  var pageAssociationFieldSize = !!(flags & 0x40);
-  var referredFlags = data[start + 5];
-  var referredToCount = referredFlags >> 5 & 7;
-  var retainBits = [referredFlags & 31];
-  var position = start + 6;
-  if (referredFlags === 7) {
-   referredToCount = readUint32(data, position - 1) & 0x1FFFFFFF;
-   position += 3;
-   var bytes = referredToCount + 7 >> 3;
-   retainBits[0] = data[position++];
-   while (--bytes > 0) {
-    retainBits.push(data[position++]);
-   }
-  } else if (referredFlags === 5 || referredFlags === 6) {
-   error('JBIG2 error: invalid referred-to flags');
-  }
-  segmentHeader.retainBits = retainBits;
-  var referredToSegmentNumberSize = segmentHeader.number <= 256 ? 1 : segmentHeader.number <= 65536 ? 2 : 4;
-  var referredTo = [];
-  var i, ii;
-  for (i = 0; i < referredToCount; i++) {
-   var number = referredToSegmentNumberSize === 1 ? data[position] : referredToSegmentNumberSize === 2 ? readUint16(data, position) : readUint32(data, position);
-   referredTo.push(number);
-   position += referredToSegmentNumberSize;
-  }
-  segmentHeader.referredTo = referredTo;
-  if (!pageAssociationFieldSize) {
-   segmentHeader.pageAssociation = data[position++];
-  } else {
-   segmentHeader.pageAssociation = readUint32(data, position);
-   position += 4;
-  }
-  segmentHeader.length = readUint32(data, position);
-  position += 4;
-  if (segmentHeader.length === 0xFFFFFFFF) {
-   if (segmentType === 38) {
-    var genericRegionInfo = readRegionSegmentInformation(data, position);
-    var genericRegionSegmentFlags = data[position + RegionSegmentInformationFieldLength];
-    var genericRegionMmr = !!(genericRegionSegmentFlags & 1);
-    var searchPatternLength = 6;
-    var searchPattern = new Uint8Array(searchPatternLength);
-    if (!genericRegionMmr) {
-     searchPattern[0] = 0xFF;
-     searchPattern[1] = 0xAC;
+  function decodeTextRegion(huffman, refinement, width, height, defaultPixelValue, numberOfSymbolInstances, stripSize, inputSymbols, symbolCodeLength, transposed, dsOffset, referenceCorner, combinationOperator, huffmanTables, refinementTemplateIndex, refinementAt, decodingContext) {
+    if (huffman) {
+      error('JBIG2 error: huffman is not supported');
     }
-    searchPattern[2] = genericRegionInfo.height >>> 24 & 0xFF;
-    searchPattern[3] = genericRegionInfo.height >> 16 & 0xFF;
-    searchPattern[4] = genericRegionInfo.height >> 8 & 0xFF;
-    searchPattern[5] = genericRegionInfo.height & 0xFF;
-    for (i = position, ii = data.length; i < ii; i++) {
-     var j = 0;
-     while (j < searchPatternLength && searchPattern[j] === data[i + j]) {
-      j++;
-     }
-     if (j === searchPatternLength) {
-      segmentHeader.length = i + searchPatternLength;
-      break;
-     }
+    var bitmap = [];
+    var i, row;
+    for (i = 0; i < height; i++) {
+      row = new Uint8Array(width);
+      if (defaultPixelValue) {
+        for (var j = 0; j < width; j++) {
+          row[j] = defaultPixelValue;
+        }
+      }
+      bitmap.push(row);
     }
-    if (segmentHeader.length === 0xFFFFFFFF) {
-     error('JBIG2 error: segment end was not found');
+    var decoder = decodingContext.decoder;
+    var contextCache = decodingContext.contextCache;
+    var stripT = -decodeInteger(contextCache, 'IADT', decoder);
+    var firstS = 0;
+    i = 0;
+    while (i < numberOfSymbolInstances) {
+      var deltaT = decodeInteger(contextCache, 'IADT', decoder);
+      stripT += deltaT;
+      var deltaFirstS = decodeInteger(contextCache, 'IAFS', decoder);
+      firstS += deltaFirstS;
+      var currentS = firstS;
+      do {
+        var currentT = stripSize === 1 ? 0 : decodeInteger(contextCache, 'IAIT', decoder);
+        var t = stripSize * stripT + currentT;
+        var symbolId = decodeIAID(contextCache, decoder, symbolCodeLength);
+        var applyRefinement = refinement && decodeInteger(contextCache, 'IARI', decoder);
+        var symbolBitmap = inputSymbols[symbolId];
+        var symbolWidth = symbolBitmap[0].length;
+        var symbolHeight = symbolBitmap.length;
+        if (applyRefinement) {
+          var rdw = decodeInteger(contextCache, 'IARDW', decoder);
+          var rdh = decodeInteger(contextCache, 'IARDH', decoder);
+          var rdx = decodeInteger(contextCache, 'IARDX', decoder);
+          var rdy = decodeInteger(contextCache, 'IARDY', decoder);
+          symbolWidth += rdw;
+          symbolHeight += rdh;
+          symbolBitmap = decodeRefinement(symbolWidth, symbolHeight, refinementTemplateIndex, symbolBitmap, (rdw >> 1) + rdx, (rdh >> 1) + rdy, false, refinementAt, decodingContext);
+        }
+        var offsetT = t - (referenceCorner & 1 ? 0 : symbolHeight);
+        var offsetS = currentS - (referenceCorner & 2 ? symbolWidth : 0);
+        var s2, t2, symbolRow;
+        if (transposed) {
+          for (s2 = 0; s2 < symbolHeight; s2++) {
+            row = bitmap[offsetS + s2];
+            if (!row) {
+              continue;
+            }
+            symbolRow = symbolBitmap[s2];
+            var maxWidth = Math.min(width - offsetT, symbolWidth);
+            switch (combinationOperator) {
+              case 0:
+                for (t2 = 0; t2 < maxWidth; t2++) {
+                  row[offsetT + t2] |= symbolRow[t2];
+                }
+                break;
+              case 2:
+                for (t2 = 0; t2 < maxWidth; t2++) {
+                  row[offsetT + t2] ^= symbolRow[t2];
+                }
+                break;
+              default:
+                error('JBIG2 error: operator ' + combinationOperator + ' is not supported');
+            }
+          }
+          currentS += symbolHeight - 1;
+        } else {
+          for (t2 = 0; t2 < symbolHeight; t2++) {
+            row = bitmap[offsetT + t2];
+            if (!row) {
+              continue;
+            }
+            symbolRow = symbolBitmap[t2];
+            switch (combinationOperator) {
+              case 0:
+                for (s2 = 0; s2 < symbolWidth; s2++) {
+                  row[offsetS + s2] |= symbolRow[s2];
+                }
+                break;
+              case 2:
+                for (s2 = 0; s2 < symbolWidth; s2++) {
+                  row[offsetS + s2] ^= symbolRow[s2];
+                }
+                break;
+              default:
+                error('JBIG2 error: operator ' + combinationOperator + ' is not supported');
+            }
+          }
+          currentS += symbolWidth - 1;
+        }
+        i++;
+        var deltaS = decodeInteger(contextCache, 'IADS', decoder);
+        if (deltaS === null) {
+          break;
+        }
+        currentS += deltaS + dsOffset;
+      } while (true);
     }
-   } else {
-    error('JBIG2 error: invalid unknown segment length');
-   }
-  }
-  segmentHeader.headerEnd = position;
-  return segmentHeader;
- }
- function readSegments(header, data, start, end) {
-  var segments = [];
-  var position = start;
-  while (position < end) {
-   var segmentHeader = readSegmentHeader(data, position);
-   position = segmentHeader.headerEnd;
-   var segment = {
-    header: segmentHeader,
-    data: data
-   };
-   if (!header.randomAccess) {
-    segment.start = position;
-    position += segmentHeader.length;
-    segment.end = position;
-   }
-   segments.push(segment);
-   if (segmentHeader.type === 51) {
-    break;
-   }
+    return bitmap;
   }
-  if (header.randomAccess) {
-   for (var i = 0, ii = segments.length; i < ii; i++) {
-    segments[i].start = position;
-    position += segments[i].header.length;
-    segments[i].end = position;
-   }
-  }
-  return segments;
- }
- function readRegionSegmentInformation(data, start) {
-  return {
-   width: readUint32(data, start),
-   height: readUint32(data, start + 4),
-   x: readUint32(data, start + 8),
-   y: readUint32(data, start + 12),
-   combinationOperator: data[start + 16] & 7
-  };
- }
- var RegionSegmentInformationFieldLength = 17;
- function processSegment(segment, visitor) {
-  var header = segment.header;
-  var data = segment.data, position = segment.start, end = segment.end;
-  var args, at, i, atLength;
-  switch (header.type) {
-  case 0:
-   var dictionary = {};
-   var dictionaryFlags = readUint16(data, position);
-   dictionary.huffman = !!(dictionaryFlags & 1);
-   dictionary.refinement = !!(dictionaryFlags & 2);
-   dictionary.huffmanDHSelector = dictionaryFlags >> 2 & 3;
-   dictionary.huffmanDWSelector = dictionaryFlags >> 4 & 3;
-   dictionary.bitmapSizeSelector = dictionaryFlags >> 6 & 1;
-   dictionary.aggregationInstancesSelector = dictionaryFlags >> 7 & 1;
-   dictionary.bitmapCodingContextUsed = !!(dictionaryFlags & 256);
-   dictionary.bitmapCodingContextRetained = !!(dictionaryFlags & 512);
-   dictionary.template = dictionaryFlags >> 10 & 3;
-   dictionary.refinementTemplate = dictionaryFlags >> 12 & 1;
-   position += 2;
-   if (!dictionary.huffman) {
-    atLength = dictionary.template === 0 ? 4 : 1;
-    at = [];
-    for (i = 0; i < atLength; i++) {
-     at.push({
-      x: readInt8(data, position),
-      y: readInt8(data, position + 1)
-     });
-     position += 2;
+  function readSegmentHeader(data, start) {
+    var segmentHeader = {};
+    segmentHeader.number = readUint32(data, start);
+    var flags = data[start + 4];
+    var segmentType = flags & 0x3F;
+    if (!SegmentTypes[segmentType]) {
+      error('JBIG2 error: invalid segment type: ' + segmentType);
     }
-    dictionary.at = at;
-   }
-   if (dictionary.refinement && !dictionary.refinementTemplate) {
-    at = [];
-    for (i = 0; i < 2; i++) {
-     at.push({
-      x: readInt8(data, position),
-      y: readInt8(data, position + 1)
-     });
-     position += 2;
+    segmentHeader.type = segmentType;
+    segmentHeader.typeName = SegmentTypes[segmentType];
+    segmentHeader.deferredNonRetain = !!(flags & 0x80);
+    var pageAssociationFieldSize = !!(flags & 0x40);
+    var referredFlags = data[start + 5];
+    var referredToCount = referredFlags >> 5 & 7;
+    var retainBits = [referredFlags & 31];
+    var position = start + 6;
+    if (referredFlags === 7) {
+      referredToCount = readUint32(data, position - 1) & 0x1FFFFFFF;
+      position += 3;
+      var bytes = referredToCount + 7 >> 3;
+      retainBits[0] = data[position++];
+      while (--bytes > 0) {
+        retainBits.push(data[position++]);
+      }
+    } else if (referredFlags === 5 || referredFlags === 6) {
+      error('JBIG2 error: invalid referred-to flags');
     }
-    dictionary.refinementAt = at;
-   }
-   dictionary.numberOfExportedSymbols = readUint32(data, position);
-   position += 4;
-   dictionary.numberOfNewSymbols = readUint32(data, position);
-   position += 4;
-   args = [
-    dictionary,
-    header.number,
-    header.referredTo,
-    data,
-    position,
-    end
-   ];
-   break;
-  case 6:
-  case 7:
-   var textRegion = {};
-   textRegion.info = readRegionSegmentInformation(data, position);
-   position += RegionSegmentInformationFieldLength;
-   var textRegionSegmentFlags = readUint16(data, position);
-   position += 2;
-   textRegion.huffman = !!(textRegionSegmentFlags & 1);
-   textRegion.refinement = !!(textRegionSegmentFlags & 2);
-   textRegion.stripSize = 1 << (textRegionSegmentFlags >> 2 & 3);
-   textRegion.referenceCorner = textRegionSegmentFlags >> 4 & 3;
-   textRegion.transposed = !!(textRegionSegmentFlags & 64);
-   textRegion.combinationOperator = textRegionSegmentFlags >> 7 & 3;
-   textRegion.defaultPixelValue = textRegionSegmentFlags >> 9 & 1;
-   textRegion.dsOffset = textRegionSegmentFlags << 17 >> 27;
-   textRegion.refinementTemplate = textRegionSegmentFlags >> 15 & 1;
-   if (textRegion.huffman) {
-    var textRegionHuffmanFlags = readUint16(data, position);
-    position += 2;
-    textRegion.huffmanFS = textRegionHuffmanFlags & 3;
-    textRegion.huffmanDS = textRegionHuffmanFlags >> 2 & 3;
-    textRegion.huffmanDT = textRegionHuffmanFlags >> 4 & 3;
-    textRegion.huffmanRefinementDW = textRegionHuffmanFlags >> 6 & 3;
-    textRegion.huffmanRefinementDH = textRegionHuffmanFlags >> 8 & 3;
-    textRegion.huffmanRefinementDX = textRegionHuffmanFlags >> 10 & 3;
-    textRegion.huffmanRefinementDY = textRegionHuffmanFlags >> 12 & 3;
-    textRegion.huffmanRefinementSizeSelector = !!(textRegionHuffmanFlags & 14);
-   }
-   if (textRegion.refinement && !textRegion.refinementTemplate) {
-    at = [];
-    for (i = 0; i < 2; i++) {
-     at.push({
-      x: readInt8(data, position),
-      y: readInt8(data, position + 1)
-     });
-     position += 2;
+    segmentHeader.retainBits = retainBits;
+    var referredToSegmentNumberSize = segmentHeader.number <= 256 ? 1 : segmentHeader.number <= 65536 ? 2 : 4;
+    var referredTo = [];
+    var i, ii;
+    for (i = 0; i < referredToCount; i++) {
+      var number = referredToSegmentNumberSize === 1 ? data[position] : referredToSegmentNumberSize === 2 ? readUint16(data, position) : readUint32(data, position);
+      referredTo.push(number);
+      position += referredToSegmentNumberSize;
     }
-    textRegion.refinementAt = at;
-   }
-   textRegion.numberOfSymbolInstances = readUint32(data, position);
-   position += 4;
-   if (textRegion.huffman) {
-    error('JBIG2 error: huffman is not supported');
-   }
-   args = [
-    textRegion,
-    header.referredTo,
-    data,
-    position,
-    end
-   ];
-   break;
-  case 38:
-  case 39:
-   var genericRegion = {};
-   genericRegion.info = readRegionSegmentInformation(data, position);
-   position += RegionSegmentInformationFieldLength;
-   var genericRegionSegmentFlags = data[position++];
-   genericRegion.mmr = !!(genericRegionSegmentFlags & 1);
-   genericRegion.template = genericRegionSegmentFlags >> 1 & 3;
-   genericRegion.prediction = !!(genericRegionSegmentFlags & 8);
-   if (!genericRegion.mmr) {
-    atLength = genericRegion.template === 0 ? 4 : 1;
-    at = [];
-    for (i = 0; i < atLength; i++) {
-     at.push({
-      x: readInt8(data, position),
-      y: readInt8(data, position + 1)
-     });
-     position += 2;
+    segmentHeader.referredTo = referredTo;
+    if (!pageAssociationFieldSize) {
+      segmentHeader.pageAssociation = data[position++];
+    } else {
+      segmentHeader.pageAssociation = readUint32(data, position);
+      position += 4;
     }
-    genericRegion.at = at;
-   }
-   args = [
-    genericRegion,
-    data,
-    position,
-    end
-   ];
-   break;
-  case 48:
-   var pageInfo = {
-    width: readUint32(data, position),
-    height: readUint32(data, position + 4),
-    resolutionX: readUint32(data, position + 8),
-    resolutionY: readUint32(data, position + 12)
-   };
-   if (pageInfo.height === 0xFFFFFFFF) {
-    delete pageInfo.height;
-   }
-   var pageSegmentFlags = data[position + 16];
-   readUint16(data, position + 17);
-   pageInfo.lossless = !!(pageSegmentFlags & 1);
-   pageInfo.refinement = !!(pageSegmentFlags & 2);
-   pageInfo.defaultPixelValue = pageSegmentFlags >> 2 & 1;
-   pageInfo.combinationOperator = pageSegmentFlags >> 3 & 3;
-   pageInfo.requiresBuffer = !!(pageSegmentFlags & 32);
-   pageInfo.combinationOperatorOverride = !!(pageSegmentFlags & 64);
-   args = [pageInfo];
-   break;
-  case 49:
-   break;
-  case 50:
-   break;
-  case 51:
-   break;
-  case 62:
-   break;
-  default:
-   error('JBIG2 error: segment type ' + header.typeName + '(' + header.type + ') is not implemented');
+    segmentHeader.length = readUint32(data, position);
+    position += 4;
+    if (segmentHeader.length === 0xFFFFFFFF) {
+      if (segmentType === 38) {
+        var genericRegionInfo = readRegionSegmentInformation(data, position);
+        var genericRegionSegmentFlags = data[position + RegionSegmentInformationFieldLength];
+        var genericRegionMmr = !!(genericRegionSegmentFlags & 1);
+        var searchPatternLength = 6;
+        var searchPattern = new Uint8Array(searchPatternLength);
+        if (!genericRegionMmr) {
+          searchPattern[0] = 0xFF;
+          searchPattern[1] = 0xAC;
+        }
+        searchPattern[2] = genericRegionInfo.height >>> 24 & 0xFF;
+        searchPattern[3] = genericRegionInfo.height >> 16 & 0xFF;
+        searchPattern[4] = genericRegionInfo.height >> 8 & 0xFF;
+        searchPattern[5] = genericRegionInfo.height & 0xFF;
+        for (i = position, ii = data.length; i < ii; i++) {
+          var j = 0;
+          while (j < searchPatternLength && searchPattern[j] === data[i + j]) {
+            j++;
+          }
+          if (j === searchPatternLength) {
+            segmentHeader.length = i + searchPatternLength;
+            break;
+          }
+        }
+        if (segmentHeader.length === 0xFFFFFFFF) {
+          error('JBIG2 error: segment end was not found');
+        }
+      } else {
+        error('JBIG2 error: invalid unknown segment length');
+      }
+    }
+    segmentHeader.headerEnd = position;
+    return segmentHeader;
   }
-  var callbackName = 'on' + header.typeName;
-  if (callbackName in visitor) {
-   visitor[callbackName].apply(visitor, args);
+  function readSegments(header, data, start, end) {
+    var segments = [];
+    var position = start;
+    while (position < end) {
+      var segmentHeader = readSegmentHeader(data, position);
+      position = segmentHeader.headerEnd;
+      var segment = {
+        header: segmentHeader,
+        data: data
+      };
+      if (!header.randomAccess) {
+        segment.start = position;
+        position += segmentHeader.length;
+        segment.end = position;
+      }
+      segments.push(segment);
+      if (segmentHeader.type === 51) {
+        break;
+      }
+    }
+    if (header.randomAccess) {
+      for (var i = 0, ii = segments.length; i < ii; i++) {
+        segments[i].start = position;
+        position += segments[i].header.length;
+        segments[i].end = position;
+      }
+    }
+    return segments;
   }
- }
- function processSegments(segments, visitor) {
-  for (var i = 0, ii = segments.length; i < ii; i++) {
-   processSegment(segments[i], visitor);
+  function readRegionSegmentInformation(data, start) {
+    return {
+      width: readUint32(data, start),
+      height: readUint32(data, start + 4),
+      x: readUint32(data, start + 8),
+      y: readUint32(data, start + 12),
+      combinationOperator: data[start + 16] & 7
+    };
   }
- }
- function parseJbig2(data, start, end) {
-  var position = start;
-  if (data[position] !== 0x97 || data[position + 1] !== 0x4A || data[position + 2] !== 0x42 || data[position + 3] !== 0x32 || data[position + 4] !== 0x0D || data[position + 5] !== 0x0A || data[position + 6] !== 0x1A || data[position + 7] !== 0x0A) {
-   error('JBIG2 error: invalid header');
+  var RegionSegmentInformationFieldLength = 17;
+  function processSegment(segment, visitor) {
+    var header = segment.header;
+    var data = segment.data,
+        position = segment.start,
+        end = segment.end;
+    var args, at, i, atLength;
+    switch (header.type) {
+      case 0:
+        var dictionary = {};
+        var dictionaryFlags = readUint16(data, position);
+        dictionary.huffman = !!(dictionaryFlags & 1);
+        dictionary.refinement = !!(dictionaryFlags & 2);
+        dictionary.huffmanDHSelector = dictionaryFlags >> 2 & 3;
+        dictionary.huffmanDWSelector = dictionaryFlags >> 4 & 3;
+        dictionary.bitmapSizeSelector = dictionaryFlags >> 6 & 1;
+        dictionary.aggregationInstancesSelector = dictionaryFlags >> 7 & 1;
+        dictionary.bitmapCodingContextUsed = !!(dictionaryFlags & 256);
+        dictionary.bitmapCodingContextRetained = !!(dictionaryFlags & 512);
+        dictionary.template = dictionaryFlags >> 10 & 3;
+        dictionary.refinementTemplate = dictionaryFlags >> 12 & 1;
+        position += 2;
+        if (!dictionary.huffman) {
+          atLength = dictionary.template === 0 ? 4 : 1;
+          at = [];
+          for (i = 0; i < atLength; i++) {
+            at.push({
+              x: readInt8(data, position),
+              y: readInt8(data, position + 1)
+            });
+            position += 2;
+          }
+          dictionary.at = at;
+        }
+        if (dictionary.refinement && !dictionary.refinementTemplate) {
+          at = [];
+          for (i = 0; i < 2; i++) {
+            at.push({
+              x: readInt8(data, position),
+              y: readInt8(data, position + 1)
+            });
+            position += 2;
+          }
+          dictionary.refinementAt = at;
+        }
+        dictionary.numberOfExportedSymbols = readUint32(data, position);
+        position += 4;
+        dictionary.numberOfNewSymbols = readUint32(data, position);
+        position += 4;
+        args = [dictionary, header.number, header.referredTo, data, position, end];
+        break;
+      case 6:
+      case 7:
+        var textRegion = {};
+        textRegion.info = readRegionSegmentInformation(data, position);
+        position += RegionSegmentInformationFieldLength;
+        var textRegionSegmentFlags = readUint16(data, position);
+        position += 2;
+        textRegion.huffman = !!(textRegionSegmentFlags & 1);
+        textRegion.refinement = !!(textRegionSegmentFlags & 2);
+        textRegion.stripSize = 1 << (textRegionSegmentFlags >> 2 & 3);
+        textRegion.referenceCorner = textRegionSegmentFlags >> 4 & 3;
+        textRegion.transposed = !!(textRegionSegmentFlags & 64);
+        textRegion.combinationOperator = textRegionSegmentFlags >> 7 & 3;
+        textRegion.defaultPixelValue = textRegionSegmentFlags >> 9 & 1;
+        textRegion.dsOffset = textRegionSegmentFlags << 17 >> 27;
+        textRegion.refinementTemplate = textRegionSegmentFlags >> 15 & 1;
+        if (textRegion.huffman) {
+          var textRegionHuffmanFlags = readUint16(data, position);
+          position += 2;
+          textRegion.huffmanFS = textRegionHuffmanFlags & 3;
+          textRegion.huffmanDS = textRegionHuffmanFlags >> 2 & 3;
+          textRegion.huffmanDT = textRegionHuffmanFlags >> 4 & 3;
+          textRegion.huffmanRefinementDW = textRegionHuffmanFlags >> 6 & 3;
+          textRegion.huffmanRefinementDH = textRegionHuffmanFlags >> 8 & 3;
+          textRegion.huffmanRefinementDX = textRegionHuffmanFlags >> 10 & 3;
+          textRegion.huffmanRefinementDY = textRegionHuffmanFlags >> 12 & 3;
+          textRegion.huffmanRefinementSizeSelector = !!(textRegionHuffmanFlags & 14);
+        }
+        if (textRegion.refinement && !textRegion.refinementTemplate) {
+          at = [];
+          for (i = 0; i < 2; i++) {
+            at.push({
+              x: readInt8(data, position),
+              y: readInt8(data, position + 1)
+            });
+            position += 2;
+          }
+          textRegion.refinementAt = at;
+        }
+        textRegion.numberOfSymbolInstances = readUint32(data, position);
+        position += 4;
+        if (textRegion.huffman) {
+          error('JBIG2 error: huffman is not supported');
+        }
+        args = [textRegion, header.referredTo, data, position, end];
+        break;
+      case 38:
+      case 39:
+        var genericRegion = {};
+        genericRegion.info = readRegionSegmentInformation(data, position);
+        position += RegionSegmentInformationFieldLength;
+        var genericRegionSegmentFlags = data[position++];
+        genericRegion.mmr = !!(genericRegionSegmentFlags & 1);
+        genericRegion.template = genericRegionSegmentFlags >> 1 & 3;
+        genericRegion.prediction = !!(genericRegionSegmentFlags & 8);
+        if (!genericRegion.mmr) {
+          atLength = genericRegion.template === 0 ? 4 : 1;
+          at = [];
+          for (i = 0; i < atLength; i++) {
+            at.push({
+              x: readInt8(data, position),
+              y: readInt8(data, position + 1)
+            });
+            position += 2;
+          }
+          genericRegion.at = at;
+        }
+        args = [genericRegion, data, position, end];
+        break;
+      case 48:
+        var pageInfo = {
+          width: readUint32(data, position),
+          height: readUint32(data, position + 4),
+          resolutionX: readUint32(data, position + 8),
+          resolutionY: readUint32(data, position + 12)
+        };
+        if (pageInfo.height === 0xFFFFFFFF) {
+          delete pageInfo.height;
+        }
+        var pageSegmentFlags = data[position + 16];
+        readUint16(data, position + 17);
+        pageInfo.lossless = !!(pageSegmentFlags & 1);
+        pageInfo.refinement = !!(pageSegmentFlags & 2);
+        pageInfo.defaultPixelValue = pageSegmentFlags >> 2 & 1;
+        pageInfo.combinationOperator = pageSegmentFlags >> 3 & 3;
+        pageInfo.requiresBuffer = !!(pageSegmentFlags & 32);
+        pageInfo.combinationOperatorOverride = !!(pageSegmentFlags & 64);
+        args = [pageInfo];
+        break;
+      case 49:
+        break;
+      case 50:
+        break;
+      case 51:
+        break;
+      case 62:
+        break;
+      default:
+        error('JBIG2 error: segment type ' + header.typeName + '(' + header.type + ') is not implemented');
+    }
+    var callbackName = 'on' + header.typeName;
+    if (callbackName in visitor) {
+      visitor[callbackName].apply(visitor, args);
+    }
   }
-  var header = {};
-  position += 8;
-  var flags = data[position++];
-  header.randomAccess = !(flags & 1);
-  if (!(flags & 2)) {
-   header.numberOfPages = readUint32(data, position);
-   position += 4;
+  function processSegments(segments, visitor) {
+    for (var i = 0, ii = segments.length; i < ii; i++) {
+      processSegment(segments[i], visitor);
+    }
   }
-  readSegments(header, data, position, end);
-  error('Not implemented');
- }
- function parseJbig2Chunks(chunks) {
-  var visitor = new SimpleSegmentVisitor();
-  for (var i = 0, ii = chunks.length; i < ii; i++) {
-   var chunk = chunks[i];
-   var segments = readSegments({}, chunk.data, chunk.start, chunk.end);
-   processSegments(segments, visitor);
+  function parseJbig2(data, start, end) {
+    var position = start;
+    if (data[position] !== 0x97 || data[position + 1] !== 0x4A || data[position + 2] !== 0x42 || data[position + 3] !== 0x32 || data[position + 4] !== 0x0D || data[position + 5] !== 0x0A || data[position + 6] !== 0x1A || data[position + 7] !== 0x0A) {
+      error('JBIG2 error: invalid header');
+    }
+    var header = {};
+    position += 8;
+    var flags = data[position++];
+    header.randomAccess = !(flags & 1);
+    if (!(flags & 2)) {
+      header.numberOfPages = readUint32(data, position);
+      position += 4;
+    }
+    readSegments(header, data, position, end);
+    error('Not implemented');
   }
-  return visitor.buffer;
- }
- function SimpleSegmentVisitor() {
- }
- SimpleSegmentVisitor.prototype = {
-  onPageInformation: function SimpleSegmentVisitor_onPageInformation(info) {
-   this.currentPageInfo = info;
-   var rowSize = info.width + 7 >> 3;
-   var buffer = new Uint8Array(rowSize * info.height);
-   if (info.defaultPixelValue) {
-    for (var i = 0, ii = buffer.length; i < ii; i++) {
-     buffer[i] = 0xFF;
+  function parseJbig2Chunks(chunks) {
+    var visitor = new SimpleSegmentVisitor();
+    for (var i = 0, ii = chunks.length; i < ii; i++) {
+      var chunk = chunks[i];
+      var segments = readSegments({}, chunk.data, chunk.start, chunk.end);
+      processSegments(segments, visitor);
     }
-   }
-   this.buffer = buffer;
-  },
-  drawBitmap: function SimpleSegmentVisitor_drawBitmap(regionInfo, bitmap) {
-   var pageInfo = this.currentPageInfo;
-   var width = regionInfo.width, height = regionInfo.height;
-   var rowSize = pageInfo.width + 7 >> 3;
-   var combinationOperator = pageInfo.combinationOperatorOverride ? regionInfo.combinationOperator : pageInfo.combinationOperator;
-   var buffer = this.buffer;
-   var mask0 = 128 >> (regionInfo.x & 7);
-   var offset0 = regionInfo.y * rowSize + (regionInfo.x >> 3);
-   var i, j, mask, offset;
-   switch (combinationOperator) {
-   case 0:
-    for (i = 0; i < height; i++) {
-     mask = mask0;
-     offset = offset0;
-     for (j = 0; j < width; j++) {
-      if (bitmap[i][j]) {
-       buffer[offset] |= mask;
+    return visitor.buffer;
+  }
+  function SimpleSegmentVisitor() {}
+  SimpleSegmentVisitor.prototype = {
+    onPageInformation: function SimpleSegmentVisitor_onPageInformation(info) {
+      this.currentPageInfo = info;
+      var rowSize = info.width + 7 >> 3;
+      var buffer = new Uint8Array(rowSize * info.height);
+      if (info.defaultPixelValue) {
+        for (var i = 0, ii = buffer.length; i < ii; i++) {
+          buffer[i] = 0xFF;
+        }
       }
-      mask >>= 1;
-      if (!mask) {
-       mask = 128;
-       offset++;
+      this.buffer = buffer;
+    },
+    drawBitmap: function SimpleSegmentVisitor_drawBitmap(regionInfo, bitmap) {
+      var pageInfo = this.currentPageInfo;
+      var width = regionInfo.width,
+          height = regionInfo.height;
+      var rowSize = pageInfo.width + 7 >> 3;
+      var combinationOperator = pageInfo.combinationOperatorOverride ? regionInfo.combinationOperator : pageInfo.combinationOperator;
+      var buffer = this.buffer;
+      var mask0 = 128 >> (regionInfo.x & 7);
+      var offset0 = regionInfo.y * rowSize + (regionInfo.x >> 3);
+      var i, j, mask, offset;
+      switch (combinationOperator) {
+        case 0:
+          for (i = 0; i < height; i++) {
+            mask = mask0;
+            offset = offset0;
+            for (j = 0; j < width; j++) {
+              if (bitmap[i][j]) {
+                buffer[offset] |= mask;
+              }
+              mask >>= 1;
+              if (!mask) {
+                mask = 128;
+                offset++;
+              }
+            }
+            offset0 += rowSize;
+          }
+          break;
+        case 2:
+          for (i = 0; i < height; i++) {
+            mask = mask0;
+            offset = offset0;
+            for (j = 0; j < width; j++) {
+              if (bitmap[i][j]) {
+                buffer[offset] ^= mask;
+              }
+              mask >>= 1;
+              if (!mask) {
+                mask = 128;
+                offset++;
+              }
+            }
+            offset0 += rowSize;
+          }
+          break;
+        default:
+          error('JBIG2 error: operator ' + combinationOperator + ' is not supported');
       }
-     }
-     offset0 += rowSize;
-    }
-    break;
-   case 2:
-    for (i = 0; i < height; i++) {
-     mask = mask0;
-     offset = offset0;
-     for (j = 0; j < width; j++) {
-      if (bitmap[i][j]) {
-       buffer[offset] ^= mask;
+    },
+    onImmediateGenericRegion: function SimpleSegmentVisitor_onImmediateGenericRegion(region, data, start, end) {
+      var regionInfo = region.info;
+      var decodingContext = new DecodingContext(data, start, end);
+      var bitmap = decodeBitmap(region.mmr, regionInfo.width, regionInfo.height, region.template, region.prediction, null, region.at, decodingContext);
+      this.drawBitmap(regionInfo, bitmap);
+    },
+    onImmediateLosslessGenericRegion: function SimpleSegmentVisitor_onImmediateLosslessGenericRegion() {
+      this.onImmediateGenericRegion.apply(this, arguments);
+    },
+    onSymbolDictionary: function SimpleSegmentVisitor_onSymbolDictionary(dictionary, currentSegment, referredSegments, data, start, end) {
+      var huffmanTables;
+      if (dictionary.huffman) {
+        error('JBIG2 error: huffman is not supported');
+      }
+      var symbols = this.symbols;
+      if (!symbols) {
+        this.symbols = symbols = {};
       }
-      mask >>= 1;
-      if (!mask) {
-       mask = 128;
-       offset++;
+      var inputSymbols = [];
+      for (var i = 0, ii = referredSegments.length; i < ii; i++) {
+        inputSymbols = inputSymbols.concat(symbols[referredSegments[i]]);
       }
-     }
-     offset0 += rowSize;
+      var decodingContext = new DecodingContext(data, start, end);
+      symbols[currentSegment] = decodeSymbolDictionary(dictionary.huffman, dictionary.refinement, inputSymbols, dictionary.numberOfNewSymbols, dictionary.numberOfExportedSymbols, huffmanTables, dictionary.template, dictionary.at, dictionary.refinementTemplate, dictionary.refinementAt, decodingContext);
+    },
+    onImmediateTextRegion: function SimpleSegmentVisitor_onImmediateTextRegion(region, referredSegments, data, start, end) {
+      var regionInfo = region.info;
+      var huffmanTables;
+      var symbols = this.symbols;
+      var inputSymbols = [];
+      for (var i = 0, ii = referredSegments.length; i < ii; i++) {
+        inputSymbols = inputSymbols.concat(symbols[referredSegments[i]]);
+      }
+      var symbolCodeLength = log2(inputSymbols.length);
+      var decodingContext = new DecodingContext(data, start, end);
+      var bitmap = decodeTextRegion(region.huffman, region.refinement, regionInfo.width, regionInfo.height, region.defaultPixelValue, region.numberOfSymbolInstances, region.stripSize, inputSymbols, symbolCodeLength, region.transposed, region.dsOffset, region.referenceCorner, region.combinationOperator, huffmanTables, region.refinementTemplate, region.refinementAt, decodingContext);
+      this.drawBitmap(regionInfo, bitmap);
+    },
+    onImmediateLosslessTextRegion: function SimpleSegmentVisitor_onImmediateLosslessTextRegion() {
+      this.onImmediateTextRegion.apply(this, arguments);
     }
-    break;
-   default:
-    error('JBIG2 error: operator ' + combinationOperator + ' is not supported');
-   }
-  },
-  onImmediateGenericRegion: function SimpleSegmentVisitor_onImmediateGenericRegion(region, data, start, end) {
-   var regionInfo = region.info;
-   var decodingContext = new DecodingContext(data, start, end);
-   var bitmap = decodeBitmap(region.mmr, regionInfo.width, regionInfo.height, region.template, region.prediction, null, region.at, decodingContext);
-   this.drawBitmap(regionInfo, bitmap);
-  },
-  onImmediateLosslessGenericRegion: function SimpleSegmentVisitor_onImmediateLosslessGenericRegion() {
-   this.onImmediateGenericRegion.apply(this, arguments);
-  },
-  onSymbolDictionary: function SimpleSegmentVisitor_onSymbolDictionary(dictionary, currentSegment, referredSegments, data, start, end) {
-   var huffmanTables;
-   if (dictionary.huffman) {
-    error('JBIG2 error: huffman is not supported');
-   }
-   var symbols = this.symbols;
-   if (!symbols) {
-    this.symbols = symbols = {};
-   }
-   var inputSymbols = [];
-   for (var i = 0, ii = referredSegments.length; i < ii; i++) {
-    inputSymbols = inputSymbols.concat(symbols[referredSegments[i]]);
-   }
-   var decodingContext = new DecodingContext(data, start, end);
-   symbols[currentSegment] = decodeSymbolDictionary(dictionary.huffman, dictionary.refinement, inputSymbols, dictionary.numberOfNewSymbols, dictionary.numberOfExportedSymbols, huffmanTables, dictionary.template, dictionary.at, dictionary.refinementTemplate, dictionary.refinementAt, decodingContext);
-  },
-  onImmediateTextRegion: function SimpleSegmentVisitor_onImmediateTextRegion(region, referredSegments, data, start, end) {
-   var regionInfo = region.info;
-   var huffmanTables;
-   var symbols = this.symbols;
-   var inputSymbols = [];
-   for (var i = 0, ii = referredSegments.length; i < ii; i++) {
-    inputSymbols = inputSymbols.concat(symbols[referredSegments[i]]);
-   }
-   var symbolCodeLength = log2(inputSymbols.length);
-   var decodingContext = new DecodingContext(data, start, end);
-   var bitmap = decodeTextRegion(region.huffman, region.refinement, regionInfo.width, regionInfo.height, region.defaultPixelValue, region.numberOfSymbolInstances, region.stripSize, inputSymbols, symbolCodeLength, region.transposed, region.dsOffset, region.referenceCorner, region.combinationOperator, huffmanTables, region.refinementTemplate, region.refinementAt, decodingContext);
-   this.drawBitmap(regionInfo, bitmap);
-  },
-  onImmediateLosslessTextRegion: function SimpleSegmentVisitor_onImmediateLosslessTextRegion() {
-   this.onImmediateTextRegion.apply(this, arguments);
-  }
- };
- function Jbig2Image() {
- }
- Jbig2Image.prototype = {
-  parseChunks: function Jbig2Image_parseChunks(chunks) {
-   return parseJbig2Chunks(chunks);
-  }
- };
- return Jbig2Image;
+  };
+  function Jbig2Image() {}
+  Jbig2Image.prototype = {
+    parseChunks: function Jbig2Image_parseChunks(chunks) {
+      return parseJbig2Chunks(chunks);
+    }
+  };
+  return Jbig2Image;
 }();
 exports.Jbig2Image = Jbig2Image;

+ 845 - 891
lib/core/jpg.js

@@ -13,927 +13,881 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var warn = sharedUtil.warn;
 var error = sharedUtil.error;
 var JpegImage = function JpegImageClosure() {
- var dctZigZag = new Uint8Array([
-  0,
-  1,
-  8,
-  16,
-  9,
-  2,
-  3,
-  10,
-  17,
-  24,
-  32,
-  25,
-  18,
-  11,
-  4,
-  5,
-  12,
-  19,
-  26,
-  33,
-  40,
-  48,
-  41,
-  34,
-  27,
-  20,
-  13,
-  6,
-  7,
-  14,
-  21,
-  28,
-  35,
-  42,
-  49,
-  56,
-  57,
-  50,
-  43,
-  36,
-  29,
-  22,
-  15,
-  23,
-  30,
-  37,
-  44,
-  51,
-  58,
-  59,
-  52,
-  45,
-  38,
-  31,
-  39,
-  46,
-  53,
-  60,
-  61,
-  54,
-  47,
-  55,
-  62,
-  63
- ]);
- var dctCos1 = 4017;
- var dctSin1 = 799;
- var dctCos3 = 3406;
- var dctSin3 = 2276;
- var dctCos6 = 1567;
- var dctSin6 = 3784;
- var dctSqrt2 = 5793;
- var dctSqrt1d2 = 2896;
- function JpegImage() {
-  this.decodeTransform = null;
-  this.colorTransform = -1;
- }
- function buildHuffmanTable(codeLengths, values) {
-  var k = 0, code = [], i, j, length = 16;
-  while (length > 0 && !codeLengths[length - 1]) {
-   length--;
+  var dctZigZag = new Uint8Array([0, 1, 8, 16, 9, 2, 3, 10, 17, 24, 32, 25, 18, 11, 4, 5, 12, 19, 26, 33, 40, 48, 41, 34, 27, 20, 13, 6, 7, 14, 21, 28, 35, 42, 49, 56, 57, 50, 43, 36, 29, 22, 15, 23, 30, 37, 44, 51, 58, 59, 52, 45, 38, 31, 39, 46, 53, 60, 61, 54, 47, 55, 62, 63]);
+  var dctCos1 = 4017;
+  var dctSin1 = 799;
+  var dctCos3 = 3406;
+  var dctSin3 = 2276;
+  var dctCos6 = 1567;
+  var dctSin6 = 3784;
+  var dctSqrt2 = 5793;
+  var dctSqrt1d2 = 2896;
+  function JpegImage() {
+    this.decodeTransform = null;
+    this.colorTransform = -1;
   }
-  code.push({
-   children: [],
-   index: 0
-  });
-  var p = code[0], q;
-  for (i = 0; i < length; i++) {
-   for (j = 0; j < codeLengths[i]; j++) {
-    p = code.pop();
-    p.children[p.index] = values[k];
-    while (p.index > 0) {
-     p = code.pop();
+  function buildHuffmanTable(codeLengths, values) {
+    var k = 0,
+        code = [],
+        i,
+        j,
+        length = 16;
+    while (length > 0 && !codeLengths[length - 1]) {
+      length--;
     }
-    p.index++;
-    code.push(p);
-    while (code.length <= i) {
-     code.push(q = {
+    code.push({
       children: [],
       index: 0
-     });
-     p.children[p.index] = q.children;
-     p = q;
-    }
-    k++;
-   }
-   if (i + 1 < length) {
-    code.push(q = {
-     children: [],
-     index: 0
     });
-    p.children[p.index] = q.children;
-    p = q;
-   }
-  }
-  return code[0].children;
- }
- function getBlockBufferOffset(component, row, col) {
-  return 64 * ((component.blocksPerLine + 1) * row + col);
- }
- function decodeScan(data, offset, frame, components, resetInterval, spectralStart, spectralEnd, successivePrev, successive) {
-  var mcusPerLine = frame.mcusPerLine;
-  var progressive = frame.progressive;
-  var startOffset = offset, bitsData = 0, bitsCount = 0;
-  function readBit() {
-   if (bitsCount > 0) {
-    bitsCount--;
-    return bitsData >> bitsCount & 1;
-   }
-   bitsData = data[offset++];
-   if (bitsData === 0xFF) {
-    var nextByte = data[offset++];
-    if (nextByte) {
-     error('JPEG error: unexpected marker ' + (bitsData << 8 | nextByte).toString(16));
+    var p = code[0],
+        q;
+    for (i = 0; i < length; i++) {
+      for (j = 0; j < codeLengths[i]; j++) {
+        p = code.pop();
+        p.children[p.index] = values[k];
+        while (p.index > 0) {
+          p = code.pop();
+        }
+        p.index++;
+        code.push(p);
+        while (code.length <= i) {
+          code.push(q = {
+            children: [],
+            index: 0
+          });
+          p.children[p.index] = q.children;
+          p = q;
+        }
+        k++;
+      }
+      if (i + 1 < length) {
+        code.push(q = {
+          children: [],
+          index: 0
+        });
+        p.children[p.index] = q.children;
+        p = q;
+      }
     }
-   }
-   bitsCount = 7;
-   return bitsData >>> 7;
+    return code[0].children;
+  }
+  function getBlockBufferOffset(component, row, col) {
+    return 64 * ((component.blocksPerLine + 1) * row + col);
   }
-  function decodeHuffman(tree) {
-   var node = tree;
-   while (true) {
-    node = node[readBit()];
-    if (typeof node === 'number') {
-     return node;
+  function decodeScan(data, offset, frame, components, resetInterval, spectralStart, spectralEnd, successivePrev, successive) {
+    var mcusPerLine = frame.mcusPerLine;
+    var progressive = frame.progressive;
+    var startOffset = offset,
+        bitsData = 0,
+        bitsCount = 0;
+    function readBit() {
+      if (bitsCount > 0) {
+        bitsCount--;
+        return bitsData >> bitsCount & 1;
+      }
+      bitsData = data[offset++];
+      if (bitsData === 0xFF) {
+        var nextByte = data[offset++];
+        if (nextByte) {
+          error('JPEG error: unexpected marker ' + (bitsData << 8 | nextByte).toString(16));
+        }
+      }
+      bitsCount = 7;
+      return bitsData >>> 7;
     }
-    if (typeof node !== 'object') {
-     error('JPEG error: invalid huffman sequence');
+    function decodeHuffman(tree) {
+      var node = tree;
+      while (true) {
+        node = node[readBit()];
+        if (typeof node === 'number') {
+          return node;
+        }
+        if (typeof node !== 'object') {
+          error('JPEG error: invalid huffman sequence');
+        }
+      }
     }
-   }
-  }
-  function receive(length) {
-   var n = 0;
-   while (length > 0) {
-    n = n << 1 | readBit();
-    length--;
-   }
-   return n;
-  }
-  function receiveAndExtend(length) {
-   if (length === 1) {
-    return readBit() === 1 ? 1 : -1;
-   }
-   var n = receive(length);
-   if (n >= 1 << length - 1) {
-    return n;
-   }
-   return n + (-1 << length) + 1;
-  }
-  function decodeBaseline(component, offset) {
-   var t = decodeHuffman(component.huffmanTableDC);
-   var diff = t === 0 ? 0 : receiveAndExtend(t);
-   component.blockData[offset] = component.pred += diff;
-   var k = 1;
-   while (k < 64) {
-    var rs = decodeHuffman(component.huffmanTableAC);
-    var s = rs & 15, r = rs >> 4;
-    if (s === 0) {
-     if (r < 15) {
-      break;
-     }
-     k += 16;
-     continue;
+    function receive(length) {
+      var n = 0;
+      while (length > 0) {
+        n = n << 1 | readBit();
+        length--;
+      }
+      return n;
     }
-    k += r;
-    var z = dctZigZag[k];
-    component.blockData[offset + z] = receiveAndExtend(s);
-    k++;
-   }
-  }
-  function decodeDCFirst(component, offset) {
-   var t = decodeHuffman(component.huffmanTableDC);
-   var diff = t === 0 ? 0 : receiveAndExtend(t) << successive;
-   component.blockData[offset] = component.pred += diff;
-  }
-  function decodeDCSuccessive(component, offset) {
-   component.blockData[offset] |= readBit() << successive;
-  }
-  var eobrun = 0;
-  function decodeACFirst(component, offset) {
-   if (eobrun > 0) {
-    eobrun--;
-    return;
-   }
-   var k = spectralStart, e = spectralEnd;
-   while (k <= e) {
-    var rs = decodeHuffman(component.huffmanTableAC);
-    var s = rs & 15, r = rs >> 4;
-    if (s === 0) {
-     if (r < 15) {
-      eobrun = receive(r) + (1 << r) - 1;
-      break;
-     }
-     k += 16;
-     continue;
+    function receiveAndExtend(length) {
+      if (length === 1) {
+        return readBit() === 1 ? 1 : -1;
+      }
+      var n = receive(length);
+      if (n >= 1 << length - 1) {
+        return n;
+      }
+      return n + (-1 << length) + 1;
     }
-    k += r;
-    var z = dctZigZag[k];
-    component.blockData[offset + z] = receiveAndExtend(s) * (1 << successive);
-    k++;
-   }
-  }
-  var successiveACState = 0, successiveACNextValue;
-  function decodeACSuccessive(component, offset) {
-   var k = spectralStart;
-   var e = spectralEnd;
-   var r = 0;
-   var s;
-   var rs;
-   while (k <= e) {
-    var z = dctZigZag[k];
-    switch (successiveACState) {
-    case 0:
-     rs = decodeHuffman(component.huffmanTableAC);
-     s = rs & 15;
-     r = rs >> 4;
-     if (s === 0) {
-      if (r < 15) {
-       eobrun = receive(r) + (1 << r);
-       successiveACState = 4;
-      } else {
-       r = 16;
-       successiveACState = 1;
-      }
-     } else {
-      if (s !== 1) {
-       error('JPEG error: invalid ACn encoding');
-      }
-      successiveACNextValue = receiveAndExtend(s);
-      successiveACState = r ? 2 : 3;
-     }
-     continue;
-    case 1:
-    case 2:
-     if (component.blockData[offset + z]) {
-      component.blockData[offset + z] += readBit() << successive;
-     } else {
-      r--;
-      if (r === 0) {
-       successiveACState = successiveACState === 2 ? 3 : 0;
-      }
-     }
-     break;
-    case 3:
-     if (component.blockData[offset + z]) {
-      component.blockData[offset + z] += readBit() << successive;
-     } else {
-      component.blockData[offset + z] = successiveACNextValue << successive;
-      successiveACState = 0;
-     }
-     break;
-    case 4:
-     if (component.blockData[offset + z]) {
-      component.blockData[offset + z] += readBit() << successive;
-     }
-     break;
+    function decodeBaseline(component, offset) {
+      var t = decodeHuffman(component.huffmanTableDC);
+      var diff = t === 0 ? 0 : receiveAndExtend(t);
+      component.blockData[offset] = component.pred += diff;
+      var k = 1;
+      while (k < 64) {
+        var rs = decodeHuffman(component.huffmanTableAC);
+        var s = rs & 15,
+            r = rs >> 4;
+        if (s === 0) {
+          if (r < 15) {
+            break;
+          }
+          k += 16;
+          continue;
+        }
+        k += r;
+        var z = dctZigZag[k];
+        component.blockData[offset + z] = receiveAndExtend(s);
+        k++;
+      }
     }
-    k++;
-   }
-   if (successiveACState === 4) {
-    eobrun--;
-    if (eobrun === 0) {
-     successiveACState = 0;
+    function decodeDCFirst(component, offset) {
+      var t = decodeHuffman(component.huffmanTableDC);
+      var diff = t === 0 ? 0 : receiveAndExtend(t) << successive;
+      component.blockData[offset] = component.pred += diff;
     }
-   }
-  }
-  function decodeMcu(component, decode, mcu, row, col) {
-   var mcuRow = mcu / mcusPerLine | 0;
-   var mcuCol = mcu % mcusPerLine;
-   var blockRow = mcuRow * component.v + row;
-   var blockCol = mcuCol * component.h + col;
-   var offset = getBlockBufferOffset(component, blockRow, blockCol);
-   decode(component, offset);
-  }
-  function decodeBlock(component, decode, mcu) {
-   var blockRow = mcu / component.blocksPerLine | 0;
-   var blockCol = mcu % component.blocksPerLine;
-   var offset = getBlockBufferOffset(component, blockRow, blockCol);
-   decode(component, offset);
-  }
-  var componentsLength = components.length;
-  var component, i, j, k, n;
-  var decodeFn;
-  if (progressive) {
-   if (spectralStart === 0) {
-    decodeFn = successivePrev === 0 ? decodeDCFirst : decodeDCSuccessive;
-   } else {
-    decodeFn = successivePrev === 0 ? decodeACFirst : decodeACSuccessive;
-   }
-  } else {
-   decodeFn = decodeBaseline;
-  }
-  var mcu = 0, marker;
-  var mcuExpected;
-  if (componentsLength === 1) {
-   mcuExpected = components[0].blocksPerLine * components[0].blocksPerColumn;
-  } else {
-   mcuExpected = mcusPerLine * frame.mcusPerColumn;
-  }
-  var h, v;
-  while (mcu < mcuExpected) {
-   var mcuToRead = resetInterval ? Math.min(mcuExpected - mcu, resetInterval) : mcuExpected;
-   for (i = 0; i < componentsLength; i++) {
-    components[i].pred = 0;
-   }
-   eobrun = 0;
-   if (componentsLength === 1) {
-    component = components[0];
-    for (n = 0; n < mcuToRead; n++) {
-     decodeBlock(component, decodeFn, mcu);
-     mcu++;
+    function decodeDCSuccessive(component, offset) {
+      component.blockData[offset] |= readBit() << successive;
     }
-   } else {
-    for (n = 0; n < mcuToRead; n++) {
-     for (i = 0; i < componentsLength; i++) {
-      component = components[i];
-      h = component.h;
-      v = component.v;
-      for (j = 0; j < v; j++) {
-       for (k = 0; k < h; k++) {
-        decodeMcu(component, decodeFn, mcu, j, k);
-       }
-      }
-     }
-     mcu++;
+    var eobrun = 0;
+    function decodeACFirst(component, offset) {
+      if (eobrun > 0) {
+        eobrun--;
+        return;
+      }
+      var k = spectralStart,
+          e = spectralEnd;
+      while (k <= e) {
+        var rs = decodeHuffman(component.huffmanTableAC);
+        var s = rs & 15,
+            r = rs >> 4;
+        if (s === 0) {
+          if (r < 15) {
+            eobrun = receive(r) + (1 << r) - 1;
+            break;
+          }
+          k += 16;
+          continue;
+        }
+        k += r;
+        var z = dctZigZag[k];
+        component.blockData[offset + z] = receiveAndExtend(s) * (1 << successive);
+        k++;
+      }
     }
-   }
-   bitsCount = 0;
-   marker = data[offset] << 8 | data[offset + 1];
-   while (data[offset] === 0x00 && offset < data.length - 1) {
-    offset++;
-    marker = data[offset] << 8 | data[offset + 1];
-   }
-   if (marker <= 0xFF00) {
-    error('JPEG error: marker was not found');
-   }
-   if (marker >= 0xFFD0 && marker <= 0xFFD7) {
-    offset += 2;
-   } else {
-    break;
-   }
-  }
-  return offset - startOffset;
- }
- function quantizeAndInverse(component, blockBufferOffset, p) {
-  var qt = component.quantizationTable, blockData = component.blockData;
-  var v0, v1, v2, v3, v4, v5, v6, v7;
-  var p0, p1, p2, p3, p4, p5, p6, p7;
-  var t;
-  if (!qt) {
-   error('JPEG error: missing required Quantization Table.');
-  }
-  for (var row = 0; row < 64; row += 8) {
-   p0 = blockData[blockBufferOffset + row];
-   p1 = blockData[blockBufferOffset + row + 1];
-   p2 = blockData[blockBufferOffset + row + 2];
-   p3 = blockData[blockBufferOffset + row + 3];
-   p4 = blockData[blockBufferOffset + row + 4];
-   p5 = blockData[blockBufferOffset + row + 5];
-   p6 = blockData[blockBufferOffset + row + 6];
-   p7 = blockData[blockBufferOffset + row + 7];
-   p0 *= qt[row];
-   if ((p1 | p2 | p3 | p4 | p5 | p6 | p7) === 0) {
-    t = dctSqrt2 * p0 + 512 >> 10;
-    p[row] = t;
-    p[row + 1] = t;
-    p[row + 2] = t;
-    p[row + 3] = t;
-    p[row + 4] = t;
-    p[row + 5] = t;
-    p[row + 6] = t;
-    p[row + 7] = t;
-    continue;
-   }
-   p1 *= qt[row + 1];
-   p2 *= qt[row + 2];
-   p3 *= qt[row + 3];
-   p4 *= qt[row + 4];
-   p5 *= qt[row + 5];
-   p6 *= qt[row + 6];
-   p7 *= qt[row + 7];
-   v0 = dctSqrt2 * p0 + 128 >> 8;
-   v1 = dctSqrt2 * p4 + 128 >> 8;
-   v2 = p2;
-   v3 = p6;
-   v4 = dctSqrt1d2 * (p1 - p7) + 128 >> 8;
-   v7 = dctSqrt1d2 * (p1 + p7) + 128 >> 8;
-   v5 = p3 << 4;
-   v6 = p5 << 4;
-   v0 = v0 + v1 + 1 >> 1;
-   v1 = v0 - v1;
-   t = v2 * dctSin6 + v3 * dctCos6 + 128 >> 8;
-   v2 = v2 * dctCos6 - v3 * dctSin6 + 128 >> 8;
-   v3 = t;
-   v4 = v4 + v6 + 1 >> 1;
-   v6 = v4 - v6;
-   v7 = v7 + v5 + 1 >> 1;
-   v5 = v7 - v5;
-   v0 = v0 + v3 + 1 >> 1;
-   v3 = v0 - v3;
-   v1 = v1 + v2 + 1 >> 1;
-   v2 = v1 - v2;
-   t = v4 * dctSin3 + v7 * dctCos3 + 2048 >> 12;
-   v4 = v4 * dctCos3 - v7 * dctSin3 + 2048 >> 12;
-   v7 = t;
-   t = v5 * dctSin1 + v6 * dctCos1 + 2048 >> 12;
-   v5 = v5 * dctCos1 - v6 * dctSin1 + 2048 >> 12;
-   v6 = t;
-   p[row] = v0 + v7;
-   p[row + 7] = v0 - v7;
-   p[row + 1] = v1 + v6;
-   p[row + 6] = v1 - v6;
-   p[row + 2] = v2 + v5;
-   p[row + 5] = v2 - v5;
-   p[row + 3] = v3 + v4;
-   p[row + 4] = v3 - v4;
-  }
-  for (var col = 0; col < 8; ++col) {
-   p0 = p[col];
-   p1 = p[col + 8];
-   p2 = p[col + 16];
-   p3 = p[col + 24];
-   p4 = p[col + 32];
-   p5 = p[col + 40];
-   p6 = p[col + 48];
-   p7 = p[col + 56];
-   if ((p1 | p2 | p3 | p4 | p5 | p6 | p7) === 0) {
-    t = dctSqrt2 * p0 + 8192 >> 14;
-    t = t < -2040 ? 0 : t >= 2024 ? 255 : t + 2056 >> 4;
-    blockData[blockBufferOffset + col] = t;
-    blockData[blockBufferOffset + col + 8] = t;
-    blockData[blockBufferOffset + col + 16] = t;
-    blockData[blockBufferOffset + col + 24] = t;
-    blockData[blockBufferOffset + col + 32] = t;
-    blockData[blockBufferOffset + col + 40] = t;
-    blockData[blockBufferOffset + col + 48] = t;
-    blockData[blockBufferOffset + col + 56] = t;
-    continue;
-   }
-   v0 = dctSqrt2 * p0 + 2048 >> 12;
-   v1 = dctSqrt2 * p4 + 2048 >> 12;
-   v2 = p2;
-   v3 = p6;
-   v4 = dctSqrt1d2 * (p1 - p7) + 2048 >> 12;
-   v7 = dctSqrt1d2 * (p1 + p7) + 2048 >> 12;
-   v5 = p3;
-   v6 = p5;
-   v0 = (v0 + v1 + 1 >> 1) + 4112;
-   v1 = v0 - v1;
-   t = v2 * dctSin6 + v3 * dctCos6 + 2048 >> 12;
-   v2 = v2 * dctCos6 - v3 * dctSin6 + 2048 >> 12;
-   v3 = t;
-   v4 = v4 + v6 + 1 >> 1;
-   v6 = v4 - v6;
-   v7 = v7 + v5 + 1 >> 1;
-   v5 = v7 - v5;
-   v0 = v0 + v3 + 1 >> 1;
-   v3 = v0 - v3;
-   v1 = v1 + v2 + 1 >> 1;
-   v2 = v1 - v2;
-   t = v4 * dctSin3 + v7 * dctCos3 + 2048 >> 12;
-   v4 = v4 * dctCos3 - v7 * dctSin3 + 2048 >> 12;
-   v7 = t;
-   t = v5 * dctSin1 + v6 * dctCos1 + 2048 >> 12;
-   v5 = v5 * dctCos1 - v6 * dctSin1 + 2048 >> 12;
-   v6 = t;
-   p0 = v0 + v7;
-   p7 = v0 - v7;
-   p1 = v1 + v6;
-   p6 = v1 - v6;
-   p2 = v2 + v5;
-   p5 = v2 - v5;
-   p3 = v3 + v4;
-   p4 = v3 - v4;
-   p0 = p0 < 16 ? 0 : p0 >= 4080 ? 255 : p0 >> 4;
-   p1 = p1 < 16 ? 0 : p1 >= 4080 ? 255 : p1 >> 4;
-   p2 = p2 < 16 ? 0 : p2 >= 4080 ? 255 : p2 >> 4;
-   p3 = p3 < 16 ? 0 : p3 >= 4080 ? 255 : p3 >> 4;
-   p4 = p4 < 16 ? 0 : p4 >= 4080 ? 255 : p4 >> 4;
-   p5 = p5 < 16 ? 0 : p5 >= 4080 ? 255 : p5 >> 4;
-   p6 = p6 < 16 ? 0 : p6 >= 4080 ? 255 : p6 >> 4;
-   p7 = p7 < 16 ? 0 : p7 >= 4080 ? 255 : p7 >> 4;
-   blockData[blockBufferOffset + col] = p0;
-   blockData[blockBufferOffset + col + 8] = p1;
-   blockData[blockBufferOffset + col + 16] = p2;
-   blockData[blockBufferOffset + col + 24] = p3;
-   blockData[blockBufferOffset + col + 32] = p4;
-   blockData[blockBufferOffset + col + 40] = p5;
-   blockData[blockBufferOffset + col + 48] = p6;
-   blockData[blockBufferOffset + col + 56] = p7;
-  }
- }
- function buildComponentData(frame, component) {
-  var blocksPerLine = component.blocksPerLine;
-  var blocksPerColumn = component.blocksPerColumn;
-  var computationBuffer = new Int16Array(64);
-  for (var blockRow = 0; blockRow < blocksPerColumn; blockRow++) {
-   for (var blockCol = 0; blockCol < blocksPerLine; blockCol++) {
-    var offset = getBlockBufferOffset(component, blockRow, blockCol);
-    quantizeAndInverse(component, offset, computationBuffer);
-   }
-  }
-  return component.blockData;
- }
- function clamp0to255(a) {
-  return a <= 0 ? 0 : a >= 255 ? 255 : a;
- }
- JpegImage.prototype = {
-  parse: function parse(data) {
-   function readUint16() {
-    var value = data[offset] << 8 | data[offset + 1];
-    offset += 2;
-    return value;
-   }
-   function readDataBlock() {
-    function isValidMarkerAt(pos) {
-     if (pos < data.length - 1) {
-      return data[pos] === 0xFF && data[pos + 1] >= 0xC0 && data[pos + 1] <= 0xFE;
-     }
-     return true;
+    var successiveACState = 0,
+        successiveACNextValue;
+    function decodeACSuccessive(component, offset) {
+      var k = spectralStart;
+      var e = spectralEnd;
+      var r = 0;
+      var s;
+      var rs;
+      while (k <= e) {
+        var z = dctZigZag[k];
+        switch (successiveACState) {
+          case 0:
+            rs = decodeHuffman(component.huffmanTableAC);
+            s = rs & 15;
+            r = rs >> 4;
+            if (s === 0) {
+              if (r < 15) {
+                eobrun = receive(r) + (1 << r);
+                successiveACState = 4;
+              } else {
+                r = 16;
+                successiveACState = 1;
+              }
+            } else {
+              if (s !== 1) {
+                error('JPEG error: invalid ACn encoding');
+              }
+              successiveACNextValue = receiveAndExtend(s);
+              successiveACState = r ? 2 : 3;
+            }
+            continue;
+          case 1:
+          case 2:
+            if (component.blockData[offset + z]) {
+              component.blockData[offset + z] += readBit() << successive;
+            } else {
+              r--;
+              if (r === 0) {
+                successiveACState = successiveACState === 2 ? 3 : 0;
+              }
+            }
+            break;
+          case 3:
+            if (component.blockData[offset + z]) {
+              component.blockData[offset + z] += readBit() << successive;
+            } else {
+              component.blockData[offset + z] = successiveACNextValue << successive;
+              successiveACState = 0;
+            }
+            break;
+          case 4:
+            if (component.blockData[offset + z]) {
+              component.blockData[offset + z] += readBit() << successive;
+            }
+            break;
+        }
+        k++;
+      }
+      if (successiveACState === 4) {
+        eobrun--;
+        if (eobrun === 0) {
+          successiveACState = 0;
+        }
+      }
     }
-    var length = readUint16();
-    var endOffset = offset + length - 2;
-    if (!isValidMarkerAt(endOffset)) {
-     warn('readDataBlock - incorrect length, next marker is: ' + (data[endOffset] << 8 | data[endOffset + 1]).toString('16'));
-     var pos = offset;
-     while (!isValidMarkerAt(pos)) {
-      pos++;
-     }
-     endOffset = pos;
+    function decodeMcu(component, decode, mcu, row, col) {
+      var mcuRow = mcu / mcusPerLine | 0;
+      var mcuCol = mcu % mcusPerLine;
+      var blockRow = mcuRow * component.v + row;
+      var blockCol = mcuCol * component.h + col;
+      var offset = getBlockBufferOffset(component, blockRow, blockCol);
+      decode(component, offset);
     }
-    var array = data.subarray(offset, endOffset);
-    offset += array.length;
-    return array;
-   }
-   function prepareComponents(frame) {
-    var mcusPerLine = Math.ceil(frame.samplesPerLine / 8 / frame.maxH);
-    var mcusPerColumn = Math.ceil(frame.scanLines / 8 / frame.maxV);
-    for (var i = 0; i < frame.components.length; i++) {
-     component = frame.components[i];
-     var blocksPerLine = Math.ceil(Math.ceil(frame.samplesPerLine / 8) * component.h / frame.maxH);
-     var blocksPerColumn = Math.ceil(Math.ceil(frame.scanLines / 8) * component.v / frame.maxV);
-     var blocksPerLineForMcu = mcusPerLine * component.h;
-     var blocksPerColumnForMcu = mcusPerColumn * component.v;
-     var blocksBufferSize = 64 * blocksPerColumnForMcu * (blocksPerLineForMcu + 1);
-     component.blockData = new Int16Array(blocksBufferSize);
-     component.blocksPerLine = blocksPerLine;
-     component.blocksPerColumn = blocksPerColumn;
+    function decodeBlock(component, decode, mcu) {
+      var blockRow = mcu / component.blocksPerLine | 0;
+      var blockCol = mcu % component.blocksPerLine;
+      var offset = getBlockBufferOffset(component, blockRow, blockCol);
+      decode(component, offset);
     }
-    frame.mcusPerLine = mcusPerLine;
-    frame.mcusPerColumn = mcusPerColumn;
-   }
-   var offset = 0;
-   var jfif = null;
-   var adobe = null;
-   var frame, resetInterval;
-   var quantizationTables = [];
-   var huffmanTablesAC = [], huffmanTablesDC = [];
-   var fileMarker = readUint16();
-   if (fileMarker !== 0xFFD8) {
-    error('JPEG error: SOI not found');
-   }
-   fileMarker = readUint16();
-   while (fileMarker !== 0xFFD9) {
-    var i, j, l;
-    switch (fileMarker) {
-    case 0xFFE0:
-    case 0xFFE1:
-    case 0xFFE2:
-    case 0xFFE3:
-    case 0xFFE4:
-    case 0xFFE5:
-    case 0xFFE6:
-    case 0xFFE7:
-    case 0xFFE8:
-    case 0xFFE9:
-    case 0xFFEA:
-    case 0xFFEB:
-    case 0xFFEC:
-    case 0xFFED:
-    case 0xFFEE:
-    case 0xFFEF:
-    case 0xFFFE:
-     var appData = readDataBlock();
-     if (fileMarker === 0xFFE0) {
-      if (appData[0] === 0x4A && appData[1] === 0x46 && appData[2] === 0x49 && appData[3] === 0x46 && appData[4] === 0) {
-       jfif = {
-        version: {
-         major: appData[5],
-         minor: appData[6]
-        },
-        densityUnits: appData[7],
-        xDensity: appData[8] << 8 | appData[9],
-        yDensity: appData[10] << 8 | appData[11],
-        thumbWidth: appData[12],
-        thumbHeight: appData[13],
-        thumbData: appData.subarray(14, 14 + 3 * appData[12] * appData[13])
-       };
-      }
-     }
-     if (fileMarker === 0xFFEE) {
-      if (appData[0] === 0x41 && appData[1] === 0x64 && appData[2] === 0x6F && appData[3] === 0x62 && appData[4] === 0x65) {
-       adobe = {
-        version: appData[5] << 8 | appData[6],
-        flags0: appData[7] << 8 | appData[8],
-        flags1: appData[9] << 8 | appData[10],
-        transformCode: appData[11]
-       };
-      }
-     }
-     break;
-    case 0xFFDB:
-     var quantizationTablesLength = readUint16();
-     var quantizationTablesEnd = quantizationTablesLength + offset - 2;
-     var z;
-     while (offset < quantizationTablesEnd) {
-      var quantizationTableSpec = data[offset++];
-      var tableData = new Uint16Array(64);
-      if (quantizationTableSpec >> 4 === 0) {
-       for (j = 0; j < 64; j++) {
-        z = dctZigZag[j];
-        tableData[z] = data[offset++];
-       }
-      } else if (quantizationTableSpec >> 4 === 1) {
-       for (j = 0; j < 64; j++) {
-        z = dctZigZag[j];
-        tableData[z] = readUint16();
-       }
+    var componentsLength = components.length;
+    var component, i, j, k, n;
+    var decodeFn;
+    if (progressive) {
+      if (spectralStart === 0) {
+        decodeFn = successivePrev === 0 ? decodeDCFirst : decodeDCSuccessive;
       } else {
-       error('JPEG error: DQT - invalid table spec');
-      }
-      quantizationTables[quantizationTableSpec & 15] = tableData;
-     }
-     break;
-    case 0xFFC0:
-    case 0xFFC1:
-    case 0xFFC2:
-     if (frame) {
-      error('JPEG error: Only single frame JPEGs supported');
-     }
-     readUint16();
-     frame = {};
-     frame.extended = fileMarker === 0xFFC1;
-     frame.progressive = fileMarker === 0xFFC2;
-     frame.precision = data[offset++];
-     frame.scanLines = readUint16();
-     frame.samplesPerLine = readUint16();
-     frame.components = [];
-     frame.componentIds = {};
-     var componentsCount = data[offset++], componentId;
-     var maxH = 0, maxV = 0;
-     for (i = 0; i < componentsCount; i++) {
-      componentId = data[offset];
-      var h = data[offset + 1] >> 4;
-      var v = data[offset + 1] & 15;
-      if (maxH < h) {
-       maxH = h;
-      }
-      if (maxV < v) {
-       maxV = v;
-      }
-      var qId = data[offset + 2];
-      l = frame.components.push({
-       h: h,
-       v: v,
-       quantizationId: qId,
-       quantizationTable: null
-      });
-      frame.componentIds[componentId] = l - 1;
-      offset += 3;
-     }
-     frame.maxH = maxH;
-     frame.maxV = maxV;
-     prepareComponents(frame);
-     break;
-    case 0xFFC4:
-     var huffmanLength = readUint16();
-     for (i = 2; i < huffmanLength;) {
-      var huffmanTableSpec = data[offset++];
-      var codeLengths = new Uint8Array(16);
-      var codeLengthSum = 0;
-      for (j = 0; j < 16; j++, offset++) {
-       codeLengthSum += codeLengths[j] = data[offset];
-      }
-      var huffmanValues = new Uint8Array(codeLengthSum);
-      for (j = 0; j < codeLengthSum; j++, offset++) {
-       huffmanValues[j] = data[offset];
-      }
-      i += 17 + codeLengthSum;
-      (huffmanTableSpec >> 4 === 0 ? huffmanTablesDC : huffmanTablesAC)[huffmanTableSpec & 15] = buildHuffmanTable(codeLengths, huffmanValues);
-     }
-     break;
-    case 0xFFDD:
-     readUint16();
-     resetInterval = readUint16();
-     break;
-    case 0xFFDA:
-     readUint16();
-     var selectorsCount = data[offset++];
-     var components = [], component;
-     for (i = 0; i < selectorsCount; i++) {
-      var componentIndex = frame.componentIds[data[offset++]];
-      component = frame.components[componentIndex];
-      var tableSpec = data[offset++];
-      component.huffmanTableDC = huffmanTablesDC[tableSpec >> 4];
-      component.huffmanTableAC = huffmanTablesAC[tableSpec & 15];
-      components.push(component);
-     }
-     var spectralStart = data[offset++];
-     var spectralEnd = data[offset++];
-     var successiveApproximation = data[offset++];
-     var processed = decodeScan(data, offset, frame, components, resetInterval, spectralStart, spectralEnd, successiveApproximation >> 4, successiveApproximation & 15);
-     offset += processed;
-     break;
-    case 0xFFFF:
-     if (data[offset] !== 0xFF) {
-      offset--;
-     }
-     break;
-    default:
-     if (data[offset - 3] === 0xFF && data[offset - 2] >= 0xC0 && data[offset - 2] <= 0xFE) {
-      offset -= 3;
-      break;
-     }
-     error('JPEG error: unknown marker ' + fileMarker.toString(16));
-    }
-    fileMarker = readUint16();
-   }
-   this.width = frame.samplesPerLine;
-   this.height = frame.scanLines;
-   this.jfif = jfif;
-   this.adobe = adobe;
-   this.components = [];
-   for (i = 0; i < frame.components.length; i++) {
-    component = frame.components[i];
-    var quantizationTable = quantizationTables[component.quantizationId];
-    if (quantizationTable) {
-     component.quantizationTable = quantizationTable;
+        decodeFn = successivePrev === 0 ? decodeACFirst : decodeACSuccessive;
+      }
+    } else {
+      decodeFn = decodeBaseline;
     }
-    this.components.push({
-     output: buildComponentData(frame, component),
-     scaleX: component.h / frame.maxH,
-     scaleY: component.v / frame.maxV,
-     blocksPerLine: component.blocksPerLine,
-     blocksPerColumn: component.blocksPerColumn
-    });
-   }
-   this.numComponents = this.components.length;
-  },
-  _getLinearizedBlockData: function getLinearizedBlockData(width, height) {
-   var scaleX = this.width / width, scaleY = this.height / height;
-   var component, componentScaleX, componentScaleY, blocksPerScanline;
-   var x, y, i, j, k;
-   var index;
-   var offset = 0;
-   var output;
-   var numComponents = this.components.length;
-   var dataLength = width * height * numComponents;
-   var data = new Uint8Array(dataLength);
-   var xScaleBlockOffset = new Uint32Array(width);
-   var mask3LSB = 0xfffffff8;
-   for (i = 0; i < numComponents; i++) {
-    component = this.components[i];
-    componentScaleX = component.scaleX * scaleX;
-    componentScaleY = component.scaleY * scaleY;
-    offset = i;
-    output = component.output;
-    blocksPerScanline = component.blocksPerLine + 1 << 3;
-    for (x = 0; x < width; x++) {
-     j = 0 | x * componentScaleX;
-     xScaleBlockOffset[x] = (j & mask3LSB) << 3 | j & 7;
+    var mcu = 0,
+        marker;
+    var mcuExpected;
+    if (componentsLength === 1) {
+      mcuExpected = components[0].blocksPerLine * components[0].blocksPerColumn;
+    } else {
+      mcuExpected = mcusPerLine * frame.mcusPerColumn;
     }
-    for (y = 0; y < height; y++) {
-     j = 0 | y * componentScaleY;
-     index = blocksPerScanline * (j & mask3LSB) | (j & 7) << 3;
-     for (x = 0; x < width; x++) {
-      data[offset] = output[index + xScaleBlockOffset[x]];
-      offset += numComponents;
-     }
+    var h, v;
+    while (mcu < mcuExpected) {
+      var mcuToRead = resetInterval ? Math.min(mcuExpected - mcu, resetInterval) : mcuExpected;
+      for (i = 0; i < componentsLength; i++) {
+        components[i].pred = 0;
+      }
+      eobrun = 0;
+      if (componentsLength === 1) {
+        component = components[0];
+        for (n = 0; n < mcuToRead; n++) {
+          decodeBlock(component, decodeFn, mcu);
+          mcu++;
+        }
+      } else {
+        for (n = 0; n < mcuToRead; n++) {
+          for (i = 0; i < componentsLength; i++) {
+            component = components[i];
+            h = component.h;
+            v = component.v;
+            for (j = 0; j < v; j++) {
+              for (k = 0; k < h; k++) {
+                decodeMcu(component, decodeFn, mcu, j, k);
+              }
+            }
+          }
+          mcu++;
+        }
+      }
+      bitsCount = 0;
+      marker = data[offset] << 8 | data[offset + 1];
+      while (data[offset] === 0x00 && offset < data.length - 1) {
+        offset++;
+        marker = data[offset] << 8 | data[offset + 1];
+      }
+      if (marker <= 0xFF00) {
+        error('JPEG error: marker was not found');
+      }
+      if (marker >= 0xFFD0 && marker <= 0xFFD7) {
+        offset += 2;
+      } else {
+        break;
+      }
     }
-   }
-   var transform = this.decodeTransform;
-   if (transform) {
-    for (i = 0; i < dataLength;) {
-     for (j = 0, k = 0; j < numComponents; j++, i++, k += 2) {
-      data[i] = (data[i] * transform[k] >> 8) + transform[k + 1];
-     }
+    return offset - startOffset;
+  }
+  function quantizeAndInverse(component, blockBufferOffset, p) {
+    var qt = component.quantizationTable,
+        blockData = component.blockData;
+    var v0, v1, v2, v3, v4, v5, v6, v7;
+    var p0, p1, p2, p3, p4, p5, p6, p7;
+    var t;
+    if (!qt) {
+      error('JPEG error: missing required Quantization Table.');
     }
-   }
-   return data;
-  },
-  _isColorConversionNeeded: function isColorConversionNeeded() {
-   if (this.adobe && this.adobe.transformCode) {
-    return true;
-   } else if (this.numComponents === 3) {
-    if (!this.adobe && this.colorTransform === 0) {
-     return false;
+    for (var row = 0; row < 64; row += 8) {
+      p0 = blockData[blockBufferOffset + row];
+      p1 = blockData[blockBufferOffset + row + 1];
+      p2 = blockData[blockBufferOffset + row + 2];
+      p3 = blockData[blockBufferOffset + row + 3];
+      p4 = blockData[blockBufferOffset + row + 4];
+      p5 = blockData[blockBufferOffset + row + 5];
+      p6 = blockData[blockBufferOffset + row + 6];
+      p7 = blockData[blockBufferOffset + row + 7];
+      p0 *= qt[row];
+      if ((p1 | p2 | p3 | p4 | p5 | p6 | p7) === 0) {
+        t = dctSqrt2 * p0 + 512 >> 10;
+        p[row] = t;
+        p[row + 1] = t;
+        p[row + 2] = t;
+        p[row + 3] = t;
+        p[row + 4] = t;
+        p[row + 5] = t;
+        p[row + 6] = t;
+        p[row + 7] = t;
+        continue;
+      }
+      p1 *= qt[row + 1];
+      p2 *= qt[row + 2];
+      p3 *= qt[row + 3];
+      p4 *= qt[row + 4];
+      p5 *= qt[row + 5];
+      p6 *= qt[row + 6];
+      p7 *= qt[row + 7];
+      v0 = dctSqrt2 * p0 + 128 >> 8;
+      v1 = dctSqrt2 * p4 + 128 >> 8;
+      v2 = p2;
+      v3 = p6;
+      v4 = dctSqrt1d2 * (p1 - p7) + 128 >> 8;
+      v7 = dctSqrt1d2 * (p1 + p7) + 128 >> 8;
+      v5 = p3 << 4;
+      v6 = p5 << 4;
+      v0 = v0 + v1 + 1 >> 1;
+      v1 = v0 - v1;
+      t = v2 * dctSin6 + v3 * dctCos6 + 128 >> 8;
+      v2 = v2 * dctCos6 - v3 * dctSin6 + 128 >> 8;
+      v3 = t;
+      v4 = v4 + v6 + 1 >> 1;
+      v6 = v4 - v6;
+      v7 = v7 + v5 + 1 >> 1;
+      v5 = v7 - v5;
+      v0 = v0 + v3 + 1 >> 1;
+      v3 = v0 - v3;
+      v1 = v1 + v2 + 1 >> 1;
+      v2 = v1 - v2;
+      t = v4 * dctSin3 + v7 * dctCos3 + 2048 >> 12;
+      v4 = v4 * dctCos3 - v7 * dctSin3 + 2048 >> 12;
+      v7 = t;
+      t = v5 * dctSin1 + v6 * dctCos1 + 2048 >> 12;
+      v5 = v5 * dctCos1 - v6 * dctSin1 + 2048 >> 12;
+      v6 = t;
+      p[row] = v0 + v7;
+      p[row + 7] = v0 - v7;
+      p[row + 1] = v1 + v6;
+      p[row + 6] = v1 - v6;
+      p[row + 2] = v2 + v5;
+      p[row + 5] = v2 - v5;
+      p[row + 3] = v3 + v4;
+      p[row + 4] = v3 - v4;
     }
-    return true;
-   }
-   if (!this.adobe && this.colorTransform === 1) {
-    return true;
-   }
-   return false;
-  },
-  _convertYccToRgb: function convertYccToRgb(data) {
-   var Y, Cb, Cr;
-   for (var i = 0, length = data.length; i < length; i += 3) {
-    Y = data[i];
-    Cb = data[i + 1];
-    Cr = data[i + 2];
-    data[i] = clamp0to255(Y - 179.456 + 1.402 * Cr);
-    data[i + 1] = clamp0to255(Y + 135.459 - 0.344 * Cb - 0.714 * Cr);
-    data[i + 2] = clamp0to255(Y - 226.816 + 1.772 * Cb);
-   }
-   return data;
-  },
-  _convertYcckToRgb: function convertYcckToRgb(data) {
-   var Y, Cb, Cr, k;
-   var offset = 0;
-   for (var i = 0, length = data.length; i < length; i += 4) {
-    Y = data[i];
-    Cb = data[i + 1];
-    Cr = data[i + 2];
-    k = data[i + 3];
-    var r = -122.67195406894 + Cb * (-6.60635669420364e-5 * Cb + 0.000437130475926232 * Cr - 5.4080610064599e-5 * Y + 0.00048449797120281 * k - 0.154362151871126) + Cr * (-0.000957964378445773 * Cr + 0.000817076911346625 * Y - 0.00477271405408747 * k + 1.53380253221734) + Y * (0.000961250184130688 * Y - 0.00266257332283933 * k + 0.48357088451265) + k * (-0.000336197177618394 * k + 0.484791561490776);
-    var g = 107.268039397724 + Cb * (2.19927104525741e-5 * Cb - 0.000640992018297945 * Cr + 0.000659397001245577 * Y + 0.000426105652938837 * k - 0.176491792462875) + Cr * (-0.000778269941513683 * Cr + 0.00130872261408275 * Y + 0.000770482631801132 * k - 0.151051492775562) + Y * (0.00126935368114843 * Y - 0.00265090189010898 * k + 0.25802910206845) + k * (-0.000318913117588328 * k - 0.213742400323665);
-    var b = -20.810012546947 + Cb * (-0.000570115196973677 * Cb - 2.63409051004589e-5 * Cr + 0.0020741088115012 * Y - 0.00288260236853442 * k + 0.814272968359295) + Cr * (-1.53496057440975e-5 * Cr - 0.000132689043961446 * Y + 0.000560833691242812 * k - 0.195152027534049) + Y * (0.00174418132927582 * Y - 0.00255243321439347 * k + 0.116935020465145) + k * (-0.000343531996510555 * k + 0.24165260232407);
-    data[offset++] = clamp0to255(r);
-    data[offset++] = clamp0to255(g);
-    data[offset++] = clamp0to255(b);
-   }
-   return data;
-  },
-  _convertYcckToCmyk: function convertYcckToCmyk(data) {
-   var Y, Cb, Cr;
-   for (var i = 0, length = data.length; i < length; i += 4) {
-    Y = data[i];
-    Cb = data[i + 1];
-    Cr = data[i + 2];
-    data[i] = clamp0to255(434.456 - Y - 1.402 * Cr);
-    data[i + 1] = clamp0to255(119.541 - Y + 0.344 * Cb + 0.714 * Cr);
-    data[i + 2] = clamp0to255(481.816 - Y - 1.772 * Cb);
-   }
-   return data;
-  },
-  _convertCmykToRgb: function convertCmykToRgb(data) {
-   var c, m, y, k;
-   var offset = 0;
-   var min = -255 * 255 * 255;
-   var scale = 1 / 255 / 255;
-   for (var i = 0, length = data.length; i < length; i += 4) {
-    c = data[i];
-    m = data[i + 1];
-    y = data[i + 2];
-    k = data[i + 3];
-    var r = c * (-4.387332384609988 * c + 54.48615194189176 * m + 18.82290502165302 * y + 212.25662451639585 * k - 72734.4411664936) + m * (1.7149763477362134 * m - 5.6096736904047315 * y - 17.873870861415444 * k - 1401.7366389350734) + y * (-2.5217340131683033 * y - 21.248923337353073 * k + 4465.541406466231) - k * (21.86122147463605 * k + 48317.86113160301);
-    var g = c * (8.841041422036149 * c + 60.118027045597366 * m + 6.871425592049007 * y + 31.159100130055922 * k - 20220.756542821975) + m * (-15.310361306967817 * m + 17.575251261109482 * y + 131.35250912493976 * k - 48691.05921601825) + y * (4.444339102852739 * y + 9.8632861493405 * k - 6341.191035517494) - k * (20.737325471181034 * k + 47890.15695978492);
-    var b = c * (0.8842522430003296 * c + 8.078677503112928 * m + 30.89978309703729 * y - 0.23883238689178934 * k - 3616.812083916688) + m * (10.49593273432072 * m + 63.02378494754052 * y + 50.606957656360734 * k - 28620.90484698408) + y * (0.03296041114873217 * y + 115.60384449646641 * k - 49363.43385999684) - k * (22.33816807309886 * k + 45932.16563550634);
-    data[offset++] = r >= 0 ? 255 : r <= min ? 0 : 255 + r * scale | 0;
-    data[offset++] = g >= 0 ? 255 : g <= min ? 0 : 255 + g * scale | 0;
-    data[offset++] = b >= 0 ? 255 : b <= min ? 0 : 255 + b * scale | 0;
-   }
-   return data;
-  },
-  getData: function getData(width, height, forceRGBoutput) {
-   if (this.numComponents > 4) {
-    error('JPEG error: Unsupported color mode');
-   }
-   var data = this._getLinearizedBlockData(width, height);
-   if (this.numComponents === 1 && forceRGBoutput) {
-    var dataLength = data.length;
-    var rgbData = new Uint8Array(dataLength * 3);
-    var offset = 0;
-    for (var i = 0; i < dataLength; i++) {
-     var grayColor = data[i];
-     rgbData[offset++] = grayColor;
-     rgbData[offset++] = grayColor;
-     rgbData[offset++] = grayColor;
+    for (var col = 0; col < 8; ++col) {
+      p0 = p[col];
+      p1 = p[col + 8];
+      p2 = p[col + 16];
+      p3 = p[col + 24];
+      p4 = p[col + 32];
+      p5 = p[col + 40];
+      p6 = p[col + 48];
+      p7 = p[col + 56];
+      if ((p1 | p2 | p3 | p4 | p5 | p6 | p7) === 0) {
+        t = dctSqrt2 * p0 + 8192 >> 14;
+        t = t < -2040 ? 0 : t >= 2024 ? 255 : t + 2056 >> 4;
+        blockData[blockBufferOffset + col] = t;
+        blockData[blockBufferOffset + col + 8] = t;
+        blockData[blockBufferOffset + col + 16] = t;
+        blockData[blockBufferOffset + col + 24] = t;
+        blockData[blockBufferOffset + col + 32] = t;
+        blockData[blockBufferOffset + col + 40] = t;
+        blockData[blockBufferOffset + col + 48] = t;
+        blockData[blockBufferOffset + col + 56] = t;
+        continue;
+      }
+      v0 = dctSqrt2 * p0 + 2048 >> 12;
+      v1 = dctSqrt2 * p4 + 2048 >> 12;
+      v2 = p2;
+      v3 = p6;
+      v4 = dctSqrt1d2 * (p1 - p7) + 2048 >> 12;
+      v7 = dctSqrt1d2 * (p1 + p7) + 2048 >> 12;
+      v5 = p3;
+      v6 = p5;
+      v0 = (v0 + v1 + 1 >> 1) + 4112;
+      v1 = v0 - v1;
+      t = v2 * dctSin6 + v3 * dctCos6 + 2048 >> 12;
+      v2 = v2 * dctCos6 - v3 * dctSin6 + 2048 >> 12;
+      v3 = t;
+      v4 = v4 + v6 + 1 >> 1;
+      v6 = v4 - v6;
+      v7 = v7 + v5 + 1 >> 1;
+      v5 = v7 - v5;
+      v0 = v0 + v3 + 1 >> 1;
+      v3 = v0 - v3;
+      v1 = v1 + v2 + 1 >> 1;
+      v2 = v1 - v2;
+      t = v4 * dctSin3 + v7 * dctCos3 + 2048 >> 12;
+      v4 = v4 * dctCos3 - v7 * dctSin3 + 2048 >> 12;
+      v7 = t;
+      t = v5 * dctSin1 + v6 * dctCos1 + 2048 >> 12;
+      v5 = v5 * dctCos1 - v6 * dctSin1 + 2048 >> 12;
+      v6 = t;
+      p0 = v0 + v7;
+      p7 = v0 - v7;
+      p1 = v1 + v6;
+      p6 = v1 - v6;
+      p2 = v2 + v5;
+      p5 = v2 - v5;
+      p3 = v3 + v4;
+      p4 = v3 - v4;
+      p0 = p0 < 16 ? 0 : p0 >= 4080 ? 255 : p0 >> 4;
+      p1 = p1 < 16 ? 0 : p1 >= 4080 ? 255 : p1 >> 4;
+      p2 = p2 < 16 ? 0 : p2 >= 4080 ? 255 : p2 >> 4;
+      p3 = p3 < 16 ? 0 : p3 >= 4080 ? 255 : p3 >> 4;
+      p4 = p4 < 16 ? 0 : p4 >= 4080 ? 255 : p4 >> 4;
+      p5 = p5 < 16 ? 0 : p5 >= 4080 ? 255 : p5 >> 4;
+      p6 = p6 < 16 ? 0 : p6 >= 4080 ? 255 : p6 >> 4;
+      p7 = p7 < 16 ? 0 : p7 >= 4080 ? 255 : p7 >> 4;
+      blockData[blockBufferOffset + col] = p0;
+      blockData[blockBufferOffset + col + 8] = p1;
+      blockData[blockBufferOffset + col + 16] = p2;
+      blockData[blockBufferOffset + col + 24] = p3;
+      blockData[blockBufferOffset + col + 32] = p4;
+      blockData[blockBufferOffset + col + 40] = p5;
+      blockData[blockBufferOffset + col + 48] = p6;
+      blockData[blockBufferOffset + col + 56] = p7;
     }
-    return rgbData;
-   } else if (this.numComponents === 3 && this._isColorConversionNeeded()) {
-    return this._convertYccToRgb(data);
-   } else if (this.numComponents === 4) {
-    if (this._isColorConversionNeeded()) {
-     if (forceRGBoutput) {
-      return this._convertYcckToRgb(data);
-     }
-     return this._convertYcckToCmyk(data);
-    } else if (forceRGBoutput) {
-     return this._convertCmykToRgb(data);
+  }
+  function buildComponentData(frame, component) {
+    var blocksPerLine = component.blocksPerLine;
+    var blocksPerColumn = component.blocksPerColumn;
+    var computationBuffer = new Int16Array(64);
+    for (var blockRow = 0; blockRow < blocksPerColumn; blockRow++) {
+      for (var blockCol = 0; blockCol < blocksPerLine; blockCol++) {
+        var offset = getBlockBufferOffset(component, blockRow, blockCol);
+        quantizeAndInverse(component, offset, computationBuffer);
+      }
     }
-   }
-   return data;
+    return component.blockData;
   }
- };
- return JpegImage;
+  function clamp0to255(a) {
+    return a <= 0 ? 0 : a >= 255 ? 255 : a;
+  }
+  JpegImage.prototype = {
+    parse: function parse(data) {
+      function readUint16() {
+        var value = data[offset] << 8 | data[offset + 1];
+        offset += 2;
+        return value;
+      }
+      function readDataBlock() {
+        function isValidMarkerAt(pos) {
+          if (pos < data.length - 1) {
+            return data[pos] === 0xFF && data[pos + 1] >= 0xC0 && data[pos + 1] <= 0xFE;
+          }
+          return true;
+        }
+        var length = readUint16();
+        var endOffset = offset + length - 2;
+        if (!isValidMarkerAt(endOffset)) {
+          warn('readDataBlock - incorrect length, next marker is: ' + (data[endOffset] << 8 | data[endOffset + 1]).toString('16'));
+          var pos = offset;
+          while (!isValidMarkerAt(pos)) {
+            pos++;
+          }
+          endOffset = pos;
+        }
+        var array = data.subarray(offset, endOffset);
+        offset += array.length;
+        return array;
+      }
+      function prepareComponents(frame) {
+        var mcusPerLine = Math.ceil(frame.samplesPerLine / 8 / frame.maxH);
+        var mcusPerColumn = Math.ceil(frame.scanLines / 8 / frame.maxV);
+        for (var i = 0; i < frame.components.length; i++) {
+          component = frame.components[i];
+          var blocksPerLine = Math.ceil(Math.ceil(frame.samplesPerLine / 8) * component.h / frame.maxH);
+          var blocksPerColumn = Math.ceil(Math.ceil(frame.scanLines / 8) * component.v / frame.maxV);
+          var blocksPerLineForMcu = mcusPerLine * component.h;
+          var blocksPerColumnForMcu = mcusPerColumn * component.v;
+          var blocksBufferSize = 64 * blocksPerColumnForMcu * (blocksPerLineForMcu + 1);
+          component.blockData = new Int16Array(blocksBufferSize);
+          component.blocksPerLine = blocksPerLine;
+          component.blocksPerColumn = blocksPerColumn;
+        }
+        frame.mcusPerLine = mcusPerLine;
+        frame.mcusPerColumn = mcusPerColumn;
+      }
+      var offset = 0;
+      var jfif = null;
+      var adobe = null;
+      var frame, resetInterval;
+      var quantizationTables = [];
+      var huffmanTablesAC = [],
+          huffmanTablesDC = [];
+      var fileMarker = readUint16();
+      if (fileMarker !== 0xFFD8) {
+        error('JPEG error: SOI not found');
+      }
+      fileMarker = readUint16();
+      while (fileMarker !== 0xFFD9) {
+        var i, j, l;
+        switch (fileMarker) {
+          case 0xFFE0:
+          case 0xFFE1:
+          case 0xFFE2:
+          case 0xFFE3:
+          case 0xFFE4:
+          case 0xFFE5:
+          case 0xFFE6:
+          case 0xFFE7:
+          case 0xFFE8:
+          case 0xFFE9:
+          case 0xFFEA:
+          case 0xFFEB:
+          case 0xFFEC:
+          case 0xFFED:
+          case 0xFFEE:
+          case 0xFFEF:
+          case 0xFFFE:
+            var appData = readDataBlock();
+            if (fileMarker === 0xFFE0) {
+              if (appData[0] === 0x4A && appData[1] === 0x46 && appData[2] === 0x49 && appData[3] === 0x46 && appData[4] === 0) {
+                jfif = {
+                  version: {
+                    major: appData[5],
+                    minor: appData[6]
+                  },
+                  densityUnits: appData[7],
+                  xDensity: appData[8] << 8 | appData[9],
+                  yDensity: appData[10] << 8 | appData[11],
+                  thumbWidth: appData[12],
+                  thumbHeight: appData[13],
+                  thumbData: appData.subarray(14, 14 + 3 * appData[12] * appData[13])
+                };
+              }
+            }
+            if (fileMarker === 0xFFEE) {
+              if (appData[0] === 0x41 && appData[1] === 0x64 && appData[2] === 0x6F && appData[3] === 0x62 && appData[4] === 0x65) {
+                adobe = {
+                  version: appData[5] << 8 | appData[6],
+                  flags0: appData[7] << 8 | appData[8],
+                  flags1: appData[9] << 8 | appData[10],
+                  transformCode: appData[11]
+                };
+              }
+            }
+            break;
+          case 0xFFDB:
+            var quantizationTablesLength = readUint16();
+            var quantizationTablesEnd = quantizationTablesLength + offset - 2;
+            var z;
+            while (offset < quantizationTablesEnd) {
+              var quantizationTableSpec = data[offset++];
+              var tableData = new Uint16Array(64);
+              if (quantizationTableSpec >> 4 === 0) {
+                for (j = 0; j < 64; j++) {
+                  z = dctZigZag[j];
+                  tableData[z] = data[offset++];
+                }
+              } else if (quantizationTableSpec >> 4 === 1) {
+                for (j = 0; j < 64; j++) {
+                  z = dctZigZag[j];
+                  tableData[z] = readUint16();
+                }
+              } else {
+                error('JPEG error: DQT - invalid table spec');
+              }
+              quantizationTables[quantizationTableSpec & 15] = tableData;
+            }
+            break;
+          case 0xFFC0:
+          case 0xFFC1:
+          case 0xFFC2:
+            if (frame) {
+              error('JPEG error: Only single frame JPEGs supported');
+            }
+            readUint16();
+            frame = {};
+            frame.extended = fileMarker === 0xFFC1;
+            frame.progressive = fileMarker === 0xFFC2;
+            frame.precision = data[offset++];
+            frame.scanLines = readUint16();
+            frame.samplesPerLine = readUint16();
+            frame.components = [];
+            frame.componentIds = {};
+            var componentsCount = data[offset++],
+                componentId;
+            var maxH = 0,
+                maxV = 0;
+            for (i = 0; i < componentsCount; i++) {
+              componentId = data[offset];
+              var h = data[offset + 1] >> 4;
+              var v = data[offset + 1] & 15;
+              if (maxH < h) {
+                maxH = h;
+              }
+              if (maxV < v) {
+                maxV = v;
+              }
+              var qId = data[offset + 2];
+              l = frame.components.push({
+                h: h,
+                v: v,
+                quantizationId: qId,
+                quantizationTable: null
+              });
+              frame.componentIds[componentId] = l - 1;
+              offset += 3;
+            }
+            frame.maxH = maxH;
+            frame.maxV = maxV;
+            prepareComponents(frame);
+            break;
+          case 0xFFC4:
+            var huffmanLength = readUint16();
+            for (i = 2; i < huffmanLength;) {
+              var huffmanTableSpec = data[offset++];
+              var codeLengths = new Uint8Array(16);
+              var codeLengthSum = 0;
+              for (j = 0; j < 16; j++, offset++) {
+                codeLengthSum += codeLengths[j] = data[offset];
+              }
+              var huffmanValues = new Uint8Array(codeLengthSum);
+              for (j = 0; j < codeLengthSum; j++, offset++) {
+                huffmanValues[j] = data[offset];
+              }
+              i += 17 + codeLengthSum;
+              (huffmanTableSpec >> 4 === 0 ? huffmanTablesDC : huffmanTablesAC)[huffmanTableSpec & 15] = buildHuffmanTable(codeLengths, huffmanValues);
+            }
+            break;
+          case 0xFFDD:
+            readUint16();
+            resetInterval = readUint16();
+            break;
+          case 0xFFDA:
+            readUint16();
+            var selectorsCount = data[offset++];
+            var components = [],
+                component;
+            for (i = 0; i < selectorsCount; i++) {
+              var componentIndex = frame.componentIds[data[offset++]];
+              component = frame.components[componentIndex];
+              var tableSpec = data[offset++];
+              component.huffmanTableDC = huffmanTablesDC[tableSpec >> 4];
+              component.huffmanTableAC = huffmanTablesAC[tableSpec & 15];
+              components.push(component);
+            }
+            var spectralStart = data[offset++];
+            var spectralEnd = data[offset++];
+            var successiveApproximation = data[offset++];
+            var processed = decodeScan(data, offset, frame, components, resetInterval, spectralStart, spectralEnd, successiveApproximation >> 4, successiveApproximation & 15);
+            offset += processed;
+            break;
+          case 0xFFFF:
+            if (data[offset] !== 0xFF) {
+              offset--;
+            }
+            break;
+          default:
+            if (data[offset - 3] === 0xFF && data[offset - 2] >= 0xC0 && data[offset - 2] <= 0xFE) {
+              offset -= 3;
+              break;
+            }
+            error('JPEG error: unknown marker ' + fileMarker.toString(16));
+        }
+        fileMarker = readUint16();
+      }
+      this.width = frame.samplesPerLine;
+      this.height = frame.scanLines;
+      this.jfif = jfif;
+      this.adobe = adobe;
+      this.components = [];
+      for (i = 0; i < frame.components.length; i++) {
+        component = frame.components[i];
+        var quantizationTable = quantizationTables[component.quantizationId];
+        if (quantizationTable) {
+          component.quantizationTable = quantizationTable;
+        }
+        this.components.push({
+          output: buildComponentData(frame, component),
+          scaleX: component.h / frame.maxH,
+          scaleY: component.v / frame.maxV,
+          blocksPerLine: component.blocksPerLine,
+          blocksPerColumn: component.blocksPerColumn
+        });
+      }
+      this.numComponents = this.components.length;
+    },
+    _getLinearizedBlockData: function getLinearizedBlockData(width, height) {
+      var scaleX = this.width / width,
+          scaleY = this.height / height;
+      var component, componentScaleX, componentScaleY, blocksPerScanline;
+      var x, y, i, j, k;
+      var index;
+      var offset = 0;
+      var output;
+      var numComponents = this.components.length;
+      var dataLength = width * height * numComponents;
+      var data = new Uint8Array(dataLength);
+      var xScaleBlockOffset = new Uint32Array(width);
+      var mask3LSB = 0xfffffff8;
+      for (i = 0; i < numComponents; i++) {
+        component = this.components[i];
+        componentScaleX = component.scaleX * scaleX;
+        componentScaleY = component.scaleY * scaleY;
+        offset = i;
+        output = component.output;
+        blocksPerScanline = component.blocksPerLine + 1 << 3;
+        for (x = 0; x < width; x++) {
+          j = 0 | x * componentScaleX;
+          xScaleBlockOffset[x] = (j & mask3LSB) << 3 | j & 7;
+        }
+        for (y = 0; y < height; y++) {
+          j = 0 | y * componentScaleY;
+          index = blocksPerScanline * (j & mask3LSB) | (j & 7) << 3;
+          for (x = 0; x < width; x++) {
+            data[offset] = output[index + xScaleBlockOffset[x]];
+            offset += numComponents;
+          }
+        }
+      }
+      var transform = this.decodeTransform;
+      if (transform) {
+        for (i = 0; i < dataLength;) {
+          for (j = 0, k = 0; j < numComponents; j++, i++, k += 2) {
+            data[i] = (data[i] * transform[k] >> 8) + transform[k + 1];
+          }
+        }
+      }
+      return data;
+    },
+    _isColorConversionNeeded: function isColorConversionNeeded() {
+      if (this.adobe && this.adobe.transformCode) {
+        return true;
+      } else if (this.numComponents === 3) {
+        if (!this.adobe && this.colorTransform === 0) {
+          return false;
+        }
+        return true;
+      }
+      if (!this.adobe && this.colorTransform === 1) {
+        return true;
+      }
+      return false;
+    },
+    _convertYccToRgb: function convertYccToRgb(data) {
+      var Y, Cb, Cr;
+      for (var i = 0, length = data.length; i < length; i += 3) {
+        Y = data[i];
+        Cb = data[i + 1];
+        Cr = data[i + 2];
+        data[i] = clamp0to255(Y - 179.456 + 1.402 * Cr);
+        data[i + 1] = clamp0to255(Y + 135.459 - 0.344 * Cb - 0.714 * Cr);
+        data[i + 2] = clamp0to255(Y - 226.816 + 1.772 * Cb);
+      }
+      return data;
+    },
+    _convertYcckToRgb: function convertYcckToRgb(data) {
+      var Y, Cb, Cr, k;
+      var offset = 0;
+      for (var i = 0, length = data.length; i < length; i += 4) {
+        Y = data[i];
+        Cb = data[i + 1];
+        Cr = data[i + 2];
+        k = data[i + 3];
+        var r = -122.67195406894 + Cb * (-6.60635669420364e-5 * Cb + 0.000437130475926232 * Cr - 5.4080610064599e-5 * Y + 0.00048449797120281 * k - 0.154362151871126) + Cr * (-0.000957964378445773 * Cr + 0.000817076911346625 * Y - 0.00477271405408747 * k + 1.53380253221734) + Y * (0.000961250184130688 * Y - 0.00266257332283933 * k + 0.48357088451265) + k * (-0.000336197177618394 * k + 0.484791561490776);
+        var g = 107.268039397724 + Cb * (2.19927104525741e-5 * Cb - 0.000640992018297945 * Cr + 0.000659397001245577 * Y + 0.000426105652938837 * k - 0.176491792462875) + Cr * (-0.000778269941513683 * Cr + 0.00130872261408275 * Y + 0.000770482631801132 * k - 0.151051492775562) + Y * (0.00126935368114843 * Y - 0.00265090189010898 * k + 0.25802910206845) + k * (-0.000318913117588328 * k - 0.213742400323665);
+        var b = -20.810012546947 + Cb * (-0.000570115196973677 * Cb - 2.63409051004589e-5 * Cr + 0.0020741088115012 * Y - 0.00288260236853442 * k + 0.814272968359295) + Cr * (-1.53496057440975e-5 * Cr - 0.000132689043961446 * Y + 0.000560833691242812 * k - 0.195152027534049) + Y * (0.00174418132927582 * Y - 0.00255243321439347 * k + 0.116935020465145) + k * (-0.000343531996510555 * k + 0.24165260232407);
+        data[offset++] = clamp0to255(r);
+        data[offset++] = clamp0to255(g);
+        data[offset++] = clamp0to255(b);
+      }
+      return data;
+    },
+    _convertYcckToCmyk: function convertYcckToCmyk(data) {
+      var Y, Cb, Cr;
+      for (var i = 0, length = data.length; i < length; i += 4) {
+        Y = data[i];
+        Cb = data[i + 1];
+        Cr = data[i + 2];
+        data[i] = clamp0to255(434.456 - Y - 1.402 * Cr);
+        data[i + 1] = clamp0to255(119.541 - Y + 0.344 * Cb + 0.714 * Cr);
+        data[i + 2] = clamp0to255(481.816 - Y - 1.772 * Cb);
+      }
+      return data;
+    },
+    _convertCmykToRgb: function convertCmykToRgb(data) {
+      var c, m, y, k;
+      var offset = 0;
+      var min = -255 * 255 * 255;
+      var scale = 1 / 255 / 255;
+      for (var i = 0, length = data.length; i < length; i += 4) {
+        c = data[i];
+        m = data[i + 1];
+        y = data[i + 2];
+        k = data[i + 3];
+        var r = c * (-4.387332384609988 * c + 54.48615194189176 * m + 18.82290502165302 * y + 212.25662451639585 * k - 72734.4411664936) + m * (1.7149763477362134 * m - 5.6096736904047315 * y - 17.873870861415444 * k - 1401.7366389350734) + y * (-2.5217340131683033 * y - 21.248923337353073 * k + 4465.541406466231) - k * (21.86122147463605 * k + 48317.86113160301);
+        var g = c * (8.841041422036149 * c + 60.118027045597366 * m + 6.871425592049007 * y + 31.159100130055922 * k - 20220.756542821975) + m * (-15.310361306967817 * m + 17.575251261109482 * y + 131.35250912493976 * k - 48691.05921601825) + y * (4.444339102852739 * y + 9.8632861493405 * k - 6341.191035517494) - k * (20.737325471181034 * k + 47890.15695978492);
+        var b = c * (0.8842522430003296 * c + 8.078677503112928 * m + 30.89978309703729 * y - 0.23883238689178934 * k - 3616.812083916688) + m * (10.49593273432072 * m + 63.02378494754052 * y + 50.606957656360734 * k - 28620.90484698408) + y * (0.03296041114873217 * y + 115.60384449646641 * k - 49363.43385999684) - k * (22.33816807309886 * k + 45932.16563550634);
+        data[offset++] = r >= 0 ? 255 : r <= min ? 0 : 255 + r * scale | 0;
+        data[offset++] = g >= 0 ? 255 : g <= min ? 0 : 255 + g * scale | 0;
+        data[offset++] = b >= 0 ? 255 : b <= min ? 0 : 255 + b * scale | 0;
+      }
+      return data;
+    },
+    getData: function getData(width, height, forceRGBoutput) {
+      if (this.numComponents > 4) {
+        error('JPEG error: Unsupported color mode');
+      }
+      var data = this._getLinearizedBlockData(width, height);
+      if (this.numComponents === 1 && forceRGBoutput) {
+        var dataLength = data.length;
+        var rgbData = new Uint8Array(dataLength * 3);
+        var offset = 0;
+        for (var i = 0; i < dataLength; i++) {
+          var grayColor = data[i];
+          rgbData[offset++] = grayColor;
+          rgbData[offset++] = grayColor;
+          rgbData[offset++] = grayColor;
+        }
+        return rgbData;
+      } else if (this.numComponents === 3 && this._isColorConversionNeeded()) {
+        return this._convertYccToRgb(data);
+      } else if (this.numComponents === 4) {
+        if (this._isColorConversionNeeded()) {
+          if (forceRGBoutput) {
+            return this._convertYcckToRgb(data);
+          }
+          return this._convertYcckToCmyk(data);
+        } else if (forceRGBoutput) {
+          return this._convertCmykToRgb(data);
+        }
+      }
+      return data;
+    }
+  };
+  return JpegImage;
 }();
 exports.JpegImage = JpegImage;

+ 1848 - 2025
lib/core/jpx.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var coreArithmeticDecoder = require('./arithmetic_decoder.js');
 var info = sharedUtil.info;
@@ -23,2098 +24,1920 @@ var readUint16 = sharedUtil.readUint16;
 var readUint32 = sharedUtil.readUint32;
 var ArithmeticDecoder = coreArithmeticDecoder.ArithmeticDecoder;
 var JpxImage = function JpxImageClosure() {
- var SubbandsGainLog2 = {
-  'LL': 0,
-  'LH': 1,
-  'HL': 1,
-  'HH': 2
- };
- function JpxImage() {
-  this.failOnCorruptedImage = false;
- }
- JpxImage.prototype = {
-  parse: function JpxImage_parse(data) {
-   var head = readUint16(data, 0);
-   if (head === 0xFF4F) {
-    this.parseCodestream(data, 0, data.length);
-    return;
-   }
-   var position = 0, length = data.length;
-   while (position < length) {
-    var headerSize = 8;
-    var lbox = readUint32(data, position);
-    var tbox = readUint32(data, position + 4);
-    position += headerSize;
-    if (lbox === 1) {
-     lbox = readUint32(data, position) * 4294967296 + readUint32(data, position + 4);
-     position += 8;
-     headerSize += 8;
-    }
-    if (lbox === 0) {
-     lbox = length - position + headerSize;
-    }
-    if (lbox < headerSize) {
-     error('JPX Error: Invalid box field size');
-    }
-    var dataLength = lbox - headerSize;
-    var jumpDataLength = true;
-    switch (tbox) {
-    case 0x6A703268:
-     jumpDataLength = false;
-     break;
-    case 0x636F6C72:
-     var method = data[position];
-     if (method === 1) {
-      var colorspace = readUint32(data, position + 3);
-      switch (colorspace) {
-      case 16:
-      case 17:
-      case 18:
-       break;
-      default:
-       warn('Unknown colorspace ' + colorspace);
-       break;
-      }
-     } else if (method === 2) {
-      info('ICC profile not supported');
-     }
-     break;
-    case 0x6A703263:
-     this.parseCodestream(data, position, position + dataLength);
-     break;
-    case 0x6A502020:
-     if (readUint32(data, position) !== 0x0d0a870a) {
-      warn('Invalid JP2 signature');
-     }
-     break;
-    case 0x6A501A1A:
-    case 0x66747970:
-    case 0x72726571:
-    case 0x72657320:
-    case 0x69686472:
-     break;
-    default:
-     var headerType = String.fromCharCode(tbox >> 24 & 0xFF, tbox >> 16 & 0xFF, tbox >> 8 & 0xFF, tbox & 0xFF);
-     warn('Unsupported header type ' + tbox + ' (' + headerType + ')');
-     break;
-    }
-    if (jumpDataLength) {
-     position += dataLength;
-    }
-   }
-  },
-  parseImageProperties: function JpxImage_parseImageProperties(stream) {
-   var newByte = stream.getByte();
-   while (newByte >= 0) {
-    var oldByte = newByte;
-    newByte = stream.getByte();
-    var code = oldByte << 8 | newByte;
-    if (code === 0xFF51) {
-     stream.skip(4);
-     var Xsiz = stream.getInt32() >>> 0;
-     var Ysiz = stream.getInt32() >>> 0;
-     var XOsiz = stream.getInt32() >>> 0;
-     var YOsiz = stream.getInt32() >>> 0;
-     stream.skip(16);
-     var Csiz = stream.getUint16();
-     this.width = Xsiz - XOsiz;
-     this.height = Ysiz - YOsiz;
-     this.componentsCount = Csiz;
-     this.bitsPerComponent = 8;
-     return;
-    }
-   }
-   error('JPX Error: No size marker found in JPX stream');
-  },
-  parseCodestream: function JpxImage_parseCodestream(data, start, end) {
-   var context = {};
-   var doNotRecover = false;
-   try {
-    var position = start;
-    while (position + 1 < end) {
-     var code = readUint16(data, position);
-     position += 2;
-     var length = 0, j, sqcd, spqcds, spqcdSize, scalarExpounded, tile;
-     switch (code) {
-     case 0xFF4F:
-      context.mainHeader = true;
-      break;
-     case 0xFFD9:
-      break;
-     case 0xFF51:
-      length = readUint16(data, position);
-      var siz = {};
-      siz.Xsiz = readUint32(data, position + 4);
-      siz.Ysiz = readUint32(data, position + 8);
-      siz.XOsiz = readUint32(data, position + 12);
-      siz.YOsiz = readUint32(data, position + 16);
-      siz.XTsiz = readUint32(data, position + 20);
-      siz.YTsiz = readUint32(data, position + 24);
-      siz.XTOsiz = readUint32(data, position + 28);
-      siz.YTOsiz = readUint32(data, position + 32);
-      var componentsCount = readUint16(data, position + 36);
-      siz.Csiz = componentsCount;
-      var components = [];
-      j = position + 38;
-      for (var i = 0; i < componentsCount; i++) {
-       var component = {
-        precision: (data[j] & 0x7F) + 1,
-        isSigned: !!(data[j] & 0x80),
-        XRsiz: data[j + 1],
-        YRsiz: data[j + 1]
-       };
-       calculateComponentDimensions(component, siz);
-       components.push(component);
-      }
-      context.SIZ = siz;
-      context.components = components;
-      calculateTileGrids(context, components);
-      context.QCC = [];
-      context.COC = [];
-      break;
-     case 0xFF5C:
-      length = readUint16(data, position);
-      var qcd = {};
-      j = position + 2;
-      sqcd = data[j++];
-      switch (sqcd & 0x1F) {
-      case 0:
-       spqcdSize = 8;
-       scalarExpounded = true;
-       break;
-      case 1:
-       spqcdSize = 16;
-       scalarExpounded = false;
-       break;
-      case 2:
-       spqcdSize = 16;
-       scalarExpounded = true;
-       break;
-      default:
-       throw new Error('Invalid SQcd value ' + sqcd);
-      }
-      qcd.noQuantization = spqcdSize === 8;
-      qcd.scalarExpounded = scalarExpounded;
-      qcd.guardBits = sqcd >> 5;
-      spqcds = [];
-      while (j < length + position) {
-       var spqcd = {};
-       if (spqcdSize === 8) {
-        spqcd.epsilon = data[j++] >> 3;
-        spqcd.mu = 0;
-       } else {
-        spqcd.epsilon = data[j] >> 3;
-        spqcd.mu = (data[j] & 0x7) << 8 | data[j + 1];
-        j += 2;
-       }
-       spqcds.push(spqcd);
-      }
-      qcd.SPqcds = spqcds;
-      if (context.mainHeader) {
-       context.QCD = qcd;
-      } else {
-       context.currentTile.QCD = qcd;
-       context.currentTile.QCC = [];
-      }
-      break;
-     case 0xFF5D:
-      length = readUint16(data, position);
-      var qcc = {};
-      j = position + 2;
-      var cqcc;
-      if (context.SIZ.Csiz < 257) {
-       cqcc = data[j++];
-      } else {
-       cqcc = readUint16(data, j);
-       j += 2;
-      }
-      sqcd = data[j++];
-      switch (sqcd & 0x1F) {
-      case 0:
-       spqcdSize = 8;
-       scalarExpounded = true;
-       break;
-      case 1:
-       spqcdSize = 16;
-       scalarExpounded = false;
-       break;
-      case 2:
-       spqcdSize = 16;
-       scalarExpounded = true;
-       break;
-      default:
-       throw new Error('Invalid SQcd value ' + sqcd);
-      }
-      qcc.noQuantization = spqcdSize === 8;
-      qcc.scalarExpounded = scalarExpounded;
-      qcc.guardBits = sqcd >> 5;
-      spqcds = [];
-      while (j < length + position) {
-       spqcd = {};
-       if (spqcdSize === 8) {
-        spqcd.epsilon = data[j++] >> 3;
-        spqcd.mu = 0;
-       } else {
-        spqcd.epsilon = data[j] >> 3;
-        spqcd.mu = (data[j] & 0x7) << 8 | data[j + 1];
-        j += 2;
-       }
-       spqcds.push(spqcd);
-      }
-      qcc.SPqcds = spqcds;
-      if (context.mainHeader) {
-       context.QCC[cqcc] = qcc;
-      } else {
-       context.currentTile.QCC[cqcc] = qcc;
-      }
-      break;
-     case 0xFF52:
-      length = readUint16(data, position);
-      var cod = {};
-      j = position + 2;
-      var scod = data[j++];
-      cod.entropyCoderWithCustomPrecincts = !!(scod & 1);
-      cod.sopMarkerUsed = !!(scod & 2);
-      cod.ephMarkerUsed = !!(scod & 4);
-      cod.progressionOrder = data[j++];
-      cod.layersCount = readUint16(data, j);
-      j += 2;
-      cod.multipleComponentTransform = data[j++];
-      cod.decompositionLevelsCount = data[j++];
-      cod.xcb = (data[j++] & 0xF) + 2;
-      cod.ycb = (data[j++] & 0xF) + 2;
-      var blockStyle = data[j++];
-      cod.selectiveArithmeticCodingBypass = !!(blockStyle & 1);
-      cod.resetContextProbabilities = !!(blockStyle & 2);
-      cod.terminationOnEachCodingPass = !!(blockStyle & 4);
-      cod.verticalyStripe = !!(blockStyle & 8);
-      cod.predictableTermination = !!(blockStyle & 16);
-      cod.segmentationSymbolUsed = !!(blockStyle & 32);
-      cod.reversibleTransformation = data[j++];
-      if (cod.entropyCoderWithCustomPrecincts) {
-       var precinctsSizes = [];
-       while (j < length + position) {
-        var precinctsSize = data[j++];
-        precinctsSizes.push({
-         PPx: precinctsSize & 0xF,
-         PPy: precinctsSize >> 4
-        });
-       }
-       cod.precinctsSizes = precinctsSizes;
-      }
-      var unsupported = [];
-      if (cod.selectiveArithmeticCodingBypass) {
-       unsupported.push('selectiveArithmeticCodingBypass');
-      }
-      if (cod.resetContextProbabilities) {
-       unsupported.push('resetContextProbabilities');
-      }
-      if (cod.terminationOnEachCodingPass) {
-       unsupported.push('terminationOnEachCodingPass');
-      }
-      if (cod.verticalyStripe) {
-       unsupported.push('verticalyStripe');
+  var SubbandsGainLog2 = {
+    'LL': 0,
+    'LH': 1,
+    'HL': 1,
+    'HH': 2
+  };
+  function JpxImage() {
+    this.failOnCorruptedImage = false;
+  }
+  JpxImage.prototype = {
+    parse: function JpxImage_parse(data) {
+      var head = readUint16(data, 0);
+      if (head === 0xFF4F) {
+        this.parseCodestream(data, 0, data.length);
+        return;
       }
-      if (cod.predictableTermination) {
-       unsupported.push('predictableTermination');
+      var position = 0,
+          length = data.length;
+      while (position < length) {
+        var headerSize = 8;
+        var lbox = readUint32(data, position);
+        var tbox = readUint32(data, position + 4);
+        position += headerSize;
+        if (lbox === 1) {
+          lbox = readUint32(data, position) * 4294967296 + readUint32(data, position + 4);
+          position += 8;
+          headerSize += 8;
+        }
+        if (lbox === 0) {
+          lbox = length - position + headerSize;
+        }
+        if (lbox < headerSize) {
+          error('JPX Error: Invalid box field size');
+        }
+        var dataLength = lbox - headerSize;
+        var jumpDataLength = true;
+        switch (tbox) {
+          case 0x6A703268:
+            jumpDataLength = false;
+            break;
+          case 0x636F6C72:
+            var method = data[position];
+            if (method === 1) {
+              var colorspace = readUint32(data, position + 3);
+              switch (colorspace) {
+                case 16:
+                case 17:
+                case 18:
+                  break;
+                default:
+                  warn('Unknown colorspace ' + colorspace);
+                  break;
+              }
+            } else if (method === 2) {
+              info('ICC profile not supported');
+            }
+            break;
+          case 0x6A703263:
+            this.parseCodestream(data, position, position + dataLength);
+            break;
+          case 0x6A502020:
+            if (readUint32(data, position) !== 0x0d0a870a) {
+              warn('Invalid JP2 signature');
+            }
+            break;
+          case 0x6A501A1A:
+          case 0x66747970:
+          case 0x72726571:
+          case 0x72657320:
+          case 0x69686472:
+            break;
+          default:
+            var headerType = String.fromCharCode(tbox >> 24 & 0xFF, tbox >> 16 & 0xFF, tbox >> 8 & 0xFF, tbox & 0xFF);
+            warn('Unsupported header type ' + tbox + ' (' + headerType + ')');
+            break;
+        }
+        if (jumpDataLength) {
+          position += dataLength;
+        }
       }
-      if (unsupported.length > 0) {
-       doNotRecover = true;
-       throw new Error('Unsupported COD options (' + unsupported.join(', ') + ')');
+    },
+    parseImageProperties: function JpxImage_parseImageProperties(stream) {
+      var newByte = stream.getByte();
+      while (newByte >= 0) {
+        var oldByte = newByte;
+        newByte = stream.getByte();
+        var code = oldByte << 8 | newByte;
+        if (code === 0xFF51) {
+          stream.skip(4);
+          var Xsiz = stream.getInt32() >>> 0;
+          var Ysiz = stream.getInt32() >>> 0;
+          var XOsiz = stream.getInt32() >>> 0;
+          var YOsiz = stream.getInt32() >>> 0;
+          stream.skip(16);
+          var Csiz = stream.getUint16();
+          this.width = Xsiz - XOsiz;
+          this.height = Ysiz - YOsiz;
+          this.componentsCount = Csiz;
+          this.bitsPerComponent = 8;
+          return;
+        }
       }
-      if (context.mainHeader) {
-       context.COD = cod;
-      } else {
-       context.currentTile.COD = cod;
-       context.currentTile.COC = [];
+      error('JPX Error: No size marker found in JPX stream');
+    },
+    parseCodestream: function JpxImage_parseCodestream(data, start, end) {
+      var context = {};
+      var doNotRecover = false;
+      try {
+        var position = start;
+        while (position + 1 < end) {
+          var code = readUint16(data, position);
+          position += 2;
+          var length = 0,
+              j,
+              sqcd,
+              spqcds,
+              spqcdSize,
+              scalarExpounded,
+              tile;
+          switch (code) {
+            case 0xFF4F:
+              context.mainHeader = true;
+              break;
+            case 0xFFD9:
+              break;
+            case 0xFF51:
+              length = readUint16(data, position);
+              var siz = {};
+              siz.Xsiz = readUint32(data, position + 4);
+              siz.Ysiz = readUint32(data, position + 8);
+              siz.XOsiz = readUint32(data, position + 12);
+              siz.YOsiz = readUint32(data, position + 16);
+              siz.XTsiz = readUint32(data, position + 20);
+              siz.YTsiz = readUint32(data, position + 24);
+              siz.XTOsiz = readUint32(data, position + 28);
+              siz.YTOsiz = readUint32(data, position + 32);
+              var componentsCount = readUint16(data, position + 36);
+              siz.Csiz = componentsCount;
+              var components = [];
+              j = position + 38;
+              for (var i = 0; i < componentsCount; i++) {
+                var component = {
+                  precision: (data[j] & 0x7F) + 1,
+                  isSigned: !!(data[j] & 0x80),
+                  XRsiz: data[j + 1],
+                  YRsiz: data[j + 1]
+                };
+                calculateComponentDimensions(component, siz);
+                components.push(component);
+              }
+              context.SIZ = siz;
+              context.components = components;
+              calculateTileGrids(context, components);
+              context.QCC = [];
+              context.COC = [];
+              break;
+            case 0xFF5C:
+              length = readUint16(data, position);
+              var qcd = {};
+              j = position + 2;
+              sqcd = data[j++];
+              switch (sqcd & 0x1F) {
+                case 0:
+                  spqcdSize = 8;
+                  scalarExpounded = true;
+                  break;
+                case 1:
+                  spqcdSize = 16;
+                  scalarExpounded = false;
+                  break;
+                case 2:
+                  spqcdSize = 16;
+                  scalarExpounded = true;
+                  break;
+                default:
+                  throw new Error('Invalid SQcd value ' + sqcd);
+              }
+              qcd.noQuantization = spqcdSize === 8;
+              qcd.scalarExpounded = scalarExpounded;
+              qcd.guardBits = sqcd >> 5;
+              spqcds = [];
+              while (j < length + position) {
+                var spqcd = {};
+                if (spqcdSize === 8) {
+                  spqcd.epsilon = data[j++] >> 3;
+                  spqcd.mu = 0;
+                } else {
+                  spqcd.epsilon = data[j] >> 3;
+                  spqcd.mu = (data[j] & 0x7) << 8 | data[j + 1];
+                  j += 2;
+                }
+                spqcds.push(spqcd);
+              }
+              qcd.SPqcds = spqcds;
+              if (context.mainHeader) {
+                context.QCD = qcd;
+              } else {
+                context.currentTile.QCD = qcd;
+                context.currentTile.QCC = [];
+              }
+              break;
+            case 0xFF5D:
+              length = readUint16(data, position);
+              var qcc = {};
+              j = position + 2;
+              var cqcc;
+              if (context.SIZ.Csiz < 257) {
+                cqcc = data[j++];
+              } else {
+                cqcc = readUint16(data, j);
+                j += 2;
+              }
+              sqcd = data[j++];
+              switch (sqcd & 0x1F) {
+                case 0:
+                  spqcdSize = 8;
+                  scalarExpounded = true;
+                  break;
+                case 1:
+                  spqcdSize = 16;
+                  scalarExpounded = false;
+                  break;
+                case 2:
+                  spqcdSize = 16;
+                  scalarExpounded = true;
+                  break;
+                default:
+                  throw new Error('Invalid SQcd value ' + sqcd);
+              }
+              qcc.noQuantization = spqcdSize === 8;
+              qcc.scalarExpounded = scalarExpounded;
+              qcc.guardBits = sqcd >> 5;
+              spqcds = [];
+              while (j < length + position) {
+                spqcd = {};
+                if (spqcdSize === 8) {
+                  spqcd.epsilon = data[j++] >> 3;
+                  spqcd.mu = 0;
+                } else {
+                  spqcd.epsilon = data[j] >> 3;
+                  spqcd.mu = (data[j] & 0x7) << 8 | data[j + 1];
+                  j += 2;
+                }
+                spqcds.push(spqcd);
+              }
+              qcc.SPqcds = spqcds;
+              if (context.mainHeader) {
+                context.QCC[cqcc] = qcc;
+              } else {
+                context.currentTile.QCC[cqcc] = qcc;
+              }
+              break;
+            case 0xFF52:
+              length = readUint16(data, position);
+              var cod = {};
+              j = position + 2;
+              var scod = data[j++];
+              cod.entropyCoderWithCustomPrecincts = !!(scod & 1);
+              cod.sopMarkerUsed = !!(scod & 2);
+              cod.ephMarkerUsed = !!(scod & 4);
+              cod.progressionOrder = data[j++];
+              cod.layersCount = readUint16(data, j);
+              j += 2;
+              cod.multipleComponentTransform = data[j++];
+              cod.decompositionLevelsCount = data[j++];
+              cod.xcb = (data[j++] & 0xF) + 2;
+              cod.ycb = (data[j++] & 0xF) + 2;
+              var blockStyle = data[j++];
+              cod.selectiveArithmeticCodingBypass = !!(blockStyle & 1);
+              cod.resetContextProbabilities = !!(blockStyle & 2);
+              cod.terminationOnEachCodingPass = !!(blockStyle & 4);
+              cod.verticalyStripe = !!(blockStyle & 8);
+              cod.predictableTermination = !!(blockStyle & 16);
+              cod.segmentationSymbolUsed = !!(blockStyle & 32);
+              cod.reversibleTransformation = data[j++];
+              if (cod.entropyCoderWithCustomPrecincts) {
+                var precinctsSizes = [];
+                while (j < length + position) {
+                  var precinctsSize = data[j++];
+                  precinctsSizes.push({
+                    PPx: precinctsSize & 0xF,
+                    PPy: precinctsSize >> 4
+                  });
+                }
+                cod.precinctsSizes = precinctsSizes;
+              }
+              var unsupported = [];
+              if (cod.selectiveArithmeticCodingBypass) {
+                unsupported.push('selectiveArithmeticCodingBypass');
+              }
+              if (cod.resetContextProbabilities) {
+                unsupported.push('resetContextProbabilities');
+              }
+              if (cod.terminationOnEachCodingPass) {
+                unsupported.push('terminationOnEachCodingPass');
+              }
+              if (cod.verticalyStripe) {
+                unsupported.push('verticalyStripe');
+              }
+              if (cod.predictableTermination) {
+                unsupported.push('predictableTermination');
+              }
+              if (unsupported.length > 0) {
+                doNotRecover = true;
+                throw new Error('Unsupported COD options (' + unsupported.join(', ') + ')');
+              }
+              if (context.mainHeader) {
+                context.COD = cod;
+              } else {
+                context.currentTile.COD = cod;
+                context.currentTile.COC = [];
+              }
+              break;
+            case 0xFF90:
+              length = readUint16(data, position);
+              tile = {};
+              tile.index = readUint16(data, position + 2);
+              tile.length = readUint32(data, position + 4);
+              tile.dataEnd = tile.length + position - 2;
+              tile.partIndex = data[position + 8];
+              tile.partsCount = data[position + 9];
+              context.mainHeader = false;
+              if (tile.partIndex === 0) {
+                tile.COD = context.COD;
+                tile.COC = context.COC.slice(0);
+                tile.QCD = context.QCD;
+                tile.QCC = context.QCC.slice(0);
+              }
+              context.currentTile = tile;
+              break;
+            case 0xFF93:
+              tile = context.currentTile;
+              if (tile.partIndex === 0) {
+                initializeTile(context, tile.index);
+                buildPackets(context);
+              }
+              length = tile.dataEnd - position;
+              parseTilePackets(context, data, position, length);
+              break;
+            case 0xFF55:
+            case 0xFF57:
+            case 0xFF58:
+            case 0xFF64:
+              length = readUint16(data, position);
+              break;
+            case 0xFF53:
+              throw new Error('Codestream code 0xFF53 (COC) is ' + 'not implemented');
+            default:
+              throw new Error('Unknown codestream code: ' + code.toString(16));
+          }
+          position += length;
+        }
+      } catch (e) {
+        if (doNotRecover || this.failOnCorruptedImage) {
+          error('JPX Error: ' + e.message);
+        } else {
+          warn('JPX: Trying to recover from: ' + e.message);
+        }
       }
-      break;
-     case 0xFF90:
-      length = readUint16(data, position);
-      tile = {};
-      tile.index = readUint16(data, position + 2);
-      tile.length = readUint32(data, position + 4);
-      tile.dataEnd = tile.length + position - 2;
-      tile.partIndex = data[position + 8];
-      tile.partsCount = data[position + 9];
-      context.mainHeader = false;
-      if (tile.partIndex === 0) {
-       tile.COD = context.COD;
-       tile.COC = context.COC.slice(0);
-       tile.QCD = context.QCD;
-       tile.QCC = context.QCC.slice(0);
+      this.tiles = transformComponents(context);
+      this.width = context.SIZ.Xsiz - context.SIZ.XOsiz;
+      this.height = context.SIZ.Ysiz - context.SIZ.YOsiz;
+      this.componentsCount = context.SIZ.Csiz;
+    }
+  };
+  function calculateComponentDimensions(component, siz) {
+    component.x0 = Math.ceil(siz.XOsiz / component.XRsiz);
+    component.x1 = Math.ceil(siz.Xsiz / component.XRsiz);
+    component.y0 = Math.ceil(siz.YOsiz / component.YRsiz);
+    component.y1 = Math.ceil(siz.Ysiz / component.YRsiz);
+    component.width = component.x1 - component.x0;
+    component.height = component.y1 - component.y0;
+  }
+  function calculateTileGrids(context, components) {
+    var siz = context.SIZ;
+    var tile,
+        tiles = [];
+    var numXtiles = Math.ceil((siz.Xsiz - siz.XTOsiz) / siz.XTsiz);
+    var numYtiles = Math.ceil((siz.Ysiz - siz.YTOsiz) / siz.YTsiz);
+    for (var q = 0; q < numYtiles; q++) {
+      for (var p = 0; p < numXtiles; p++) {
+        tile = {};
+        tile.tx0 = Math.max(siz.XTOsiz + p * siz.XTsiz, siz.XOsiz);
+        tile.ty0 = Math.max(siz.YTOsiz + q * siz.YTsiz, siz.YOsiz);
+        tile.tx1 = Math.min(siz.XTOsiz + (p + 1) * siz.XTsiz, siz.Xsiz);
+        tile.ty1 = Math.min(siz.YTOsiz + (q + 1) * siz.YTsiz, siz.Ysiz);
+        tile.width = tile.tx1 - tile.tx0;
+        tile.height = tile.ty1 - tile.ty0;
+        tile.components = [];
+        tiles.push(tile);
       }
-      context.currentTile = tile;
-      break;
-     case 0xFF93:
-      tile = context.currentTile;
-      if (tile.partIndex === 0) {
-       initializeTile(context, tile.index);
-       buildPackets(context);
+    }
+    context.tiles = tiles;
+    var componentsCount = siz.Csiz;
+    for (var i = 0, ii = componentsCount; i < ii; i++) {
+      var component = components[i];
+      for (var j = 0, jj = tiles.length; j < jj; j++) {
+        var tileComponent = {};
+        tile = tiles[j];
+        tileComponent.tcx0 = Math.ceil(tile.tx0 / component.XRsiz);
+        tileComponent.tcy0 = Math.ceil(tile.ty0 / component.YRsiz);
+        tileComponent.tcx1 = Math.ceil(tile.tx1 / component.XRsiz);
+        tileComponent.tcy1 = Math.ceil(tile.ty1 / component.YRsiz);
+        tileComponent.width = tileComponent.tcx1 - tileComponent.tcx0;
+        tileComponent.height = tileComponent.tcy1 - tileComponent.tcy0;
+        tile.components[i] = tileComponent;
       }
-      length = tile.dataEnd - position;
-      parseTilePackets(context, data, position, length);
-      break;
-     case 0xFF55:
-     case 0xFF57:
-     case 0xFF58:
-     case 0xFF64:
-      length = readUint16(data, position);
-      break;
-     case 0xFF53:
-      throw new Error('Codestream code 0xFF53 (COC) is ' + 'not implemented');
-     default:
-      throw new Error('Unknown codestream code: ' + code.toString(16));
-     }
-     position += length;
     }
-   } catch (e) {
-    if (doNotRecover || this.failOnCorruptedImage) {
-     error('JPX Error: ' + e.message);
+  }
+  function getBlocksDimensions(context, component, r) {
+    var codOrCoc = component.codingStyleParameters;
+    var result = {};
+    if (!codOrCoc.entropyCoderWithCustomPrecincts) {
+      result.PPx = 15;
+      result.PPy = 15;
     } else {
-     warn('JPX: Trying to recover from: ' + e.message);
+      result.PPx = codOrCoc.precinctsSizes[r].PPx;
+      result.PPy = codOrCoc.precinctsSizes[r].PPy;
     }
-   }
-   this.tiles = transformComponents(context);
-   this.width = context.SIZ.Xsiz - context.SIZ.XOsiz;
-   this.height = context.SIZ.Ysiz - context.SIZ.YOsiz;
-   this.componentsCount = context.SIZ.Csiz;
-  }
- };
- function calculateComponentDimensions(component, siz) {
-  component.x0 = Math.ceil(siz.XOsiz / component.XRsiz);
-  component.x1 = Math.ceil(siz.Xsiz / component.XRsiz);
-  component.y0 = Math.ceil(siz.YOsiz / component.YRsiz);
-  component.y1 = Math.ceil(siz.Ysiz / component.YRsiz);
-  component.width = component.x1 - component.x0;
-  component.height = component.y1 - component.y0;
- }
- function calculateTileGrids(context, components) {
-  var siz = context.SIZ;
-  var tile, tiles = [];
-  var numXtiles = Math.ceil((siz.Xsiz - siz.XTOsiz) / siz.XTsiz);
-  var numYtiles = Math.ceil((siz.Ysiz - siz.YTOsiz) / siz.YTsiz);
-  for (var q = 0; q < numYtiles; q++) {
-   for (var p = 0; p < numXtiles; p++) {
-    tile = {};
-    tile.tx0 = Math.max(siz.XTOsiz + p * siz.XTsiz, siz.XOsiz);
-    tile.ty0 = Math.max(siz.YTOsiz + q * siz.YTsiz, siz.YOsiz);
-    tile.tx1 = Math.min(siz.XTOsiz + (p + 1) * siz.XTsiz, siz.Xsiz);
-    tile.ty1 = Math.min(siz.YTOsiz + (q + 1) * siz.YTsiz, siz.Ysiz);
-    tile.width = tile.tx1 - tile.tx0;
-    tile.height = tile.ty1 - tile.ty0;
-    tile.components = [];
-    tiles.push(tile);
-   }
-  }
-  context.tiles = tiles;
-  var componentsCount = siz.Csiz;
-  for (var i = 0, ii = componentsCount; i < ii; i++) {
-   var component = components[i];
-   for (var j = 0, jj = tiles.length; j < jj; j++) {
-    var tileComponent = {};
-    tile = tiles[j];
-    tileComponent.tcx0 = Math.ceil(tile.tx0 / component.XRsiz);
-    tileComponent.tcy0 = Math.ceil(tile.ty0 / component.YRsiz);
-    tileComponent.tcx1 = Math.ceil(tile.tx1 / component.XRsiz);
-    tileComponent.tcy1 = Math.ceil(tile.ty1 / component.YRsiz);
-    tileComponent.width = tileComponent.tcx1 - tileComponent.tcx0;
-    tileComponent.height = tileComponent.tcy1 - tileComponent.tcy0;
-    tile.components[i] = tileComponent;
-   }
-  }
- }
- function getBlocksDimensions(context, component, r) {
-  var codOrCoc = component.codingStyleParameters;
-  var result = {};
-  if (!codOrCoc.entropyCoderWithCustomPrecincts) {
-   result.PPx = 15;
-   result.PPy = 15;
-  } else {
-   result.PPx = codOrCoc.precinctsSizes[r].PPx;
-   result.PPy = codOrCoc.precinctsSizes[r].PPy;
+    result.xcb_ = r > 0 ? Math.min(codOrCoc.xcb, result.PPx - 1) : Math.min(codOrCoc.xcb, result.PPx);
+    result.ycb_ = r > 0 ? Math.min(codOrCoc.ycb, result.PPy - 1) : Math.min(codOrCoc.ycb, result.PPy);
+    return result;
   }
-  result.xcb_ = r > 0 ? Math.min(codOrCoc.xcb, result.PPx - 1) : Math.min(codOrCoc.xcb, result.PPx);
-  result.ycb_ = r > 0 ? Math.min(codOrCoc.ycb, result.PPy - 1) : Math.min(codOrCoc.ycb, result.PPy);
-  return result;
- }
- function buildPrecincts(context, resolution, dimensions) {
-  var precinctWidth = 1 << dimensions.PPx;
-  var precinctHeight = 1 << dimensions.PPy;
-  var isZeroRes = resolution.resLevel === 0;
-  var precinctWidthInSubband = 1 << dimensions.PPx + (isZeroRes ? 0 : -1);
-  var precinctHeightInSubband = 1 << dimensions.PPy + (isZeroRes ? 0 : -1);
-  var numprecinctswide = resolution.trx1 > resolution.trx0 ? Math.ceil(resolution.trx1 / precinctWidth) - Math.floor(resolution.trx0 / precinctWidth) : 0;
-  var numprecinctshigh = resolution.try1 > resolution.try0 ? Math.ceil(resolution.try1 / precinctHeight) - Math.floor(resolution.try0 / precinctHeight) : 0;
-  var numprecincts = numprecinctswide * numprecinctshigh;
-  resolution.precinctParameters = {
-   precinctWidth: precinctWidth,
-   precinctHeight: precinctHeight,
-   numprecinctswide: numprecinctswide,
-   numprecinctshigh: numprecinctshigh,
-   numprecincts: numprecincts,
-   precinctWidthInSubband: precinctWidthInSubband,
-   precinctHeightInSubband: precinctHeightInSubband
-  };
- }
- function buildCodeblocks(context, subband, dimensions) {
-  var xcb_ = dimensions.xcb_;
-  var ycb_ = dimensions.ycb_;
-  var codeblockWidth = 1 << xcb_;
-  var codeblockHeight = 1 << ycb_;
-  var cbx0 = subband.tbx0 >> xcb_;
-  var cby0 = subband.tby0 >> ycb_;
-  var cbx1 = subband.tbx1 + codeblockWidth - 1 >> xcb_;
-  var cby1 = subband.tby1 + codeblockHeight - 1 >> ycb_;
-  var precinctParameters = subband.resolution.precinctParameters;
-  var codeblocks = [];
-  var precincts = [];
-  var i, j, codeblock, precinctNumber;
-  for (j = cby0; j < cby1; j++) {
-   for (i = cbx0; i < cbx1; i++) {
-    codeblock = {
-     cbx: i,
-     cby: j,
-     tbx0: codeblockWidth * i,
-     tby0: codeblockHeight * j,
-     tbx1: codeblockWidth * (i + 1),
-     tby1: codeblockHeight * (j + 1)
+  function buildPrecincts(context, resolution, dimensions) {
+    var precinctWidth = 1 << dimensions.PPx;
+    var precinctHeight = 1 << dimensions.PPy;
+    var isZeroRes = resolution.resLevel === 0;
+    var precinctWidthInSubband = 1 << dimensions.PPx + (isZeroRes ? 0 : -1);
+    var precinctHeightInSubband = 1 << dimensions.PPy + (isZeroRes ? 0 : -1);
+    var numprecinctswide = resolution.trx1 > resolution.trx0 ? Math.ceil(resolution.trx1 / precinctWidth) - Math.floor(resolution.trx0 / precinctWidth) : 0;
+    var numprecinctshigh = resolution.try1 > resolution.try0 ? Math.ceil(resolution.try1 / precinctHeight) - Math.floor(resolution.try0 / precinctHeight) : 0;
+    var numprecincts = numprecinctswide * numprecinctshigh;
+    resolution.precinctParameters = {
+      precinctWidth: precinctWidth,
+      precinctHeight: precinctHeight,
+      numprecinctswide: numprecinctswide,
+      numprecinctshigh: numprecinctshigh,
+      numprecincts: numprecincts,
+      precinctWidthInSubband: precinctWidthInSubband,
+      precinctHeightInSubband: precinctHeightInSubband
     };
-    codeblock.tbx0_ = Math.max(subband.tbx0, codeblock.tbx0);
-    codeblock.tby0_ = Math.max(subband.tby0, codeblock.tby0);
-    codeblock.tbx1_ = Math.min(subband.tbx1, codeblock.tbx1);
-    codeblock.tby1_ = Math.min(subband.tby1, codeblock.tby1);
-    var pi = Math.floor((codeblock.tbx0_ - subband.tbx0) / precinctParameters.precinctWidthInSubband);
-    var pj = Math.floor((codeblock.tby0_ - subband.tby0) / precinctParameters.precinctHeightInSubband);
-    precinctNumber = pi + pj * precinctParameters.numprecinctswide;
-    codeblock.precinctNumber = precinctNumber;
-    codeblock.subbandType = subband.type;
-    codeblock.Lblock = 3;
-    if (codeblock.tbx1_ <= codeblock.tbx0_ || codeblock.tby1_ <= codeblock.tby0_) {
-     continue;
-    }
-    codeblocks.push(codeblock);
-    var precinct = precincts[precinctNumber];
-    if (precinct !== undefined) {
-     if (i < precinct.cbxMin) {
-      precinct.cbxMin = i;
-     } else if (i > precinct.cbxMax) {
-      precinct.cbxMax = i;
-     }
-     if (j < precinct.cbyMin) {
-      precinct.cbxMin = j;
-     } else if (j > precinct.cbyMax) {
-      precinct.cbyMax = j;
-     }
-    } else {
-     precincts[precinctNumber] = precinct = {
-      cbxMin: i,
-      cbyMin: j,
-      cbxMax: i,
-      cbyMax: j
-     };
-    }
-    codeblock.precinct = precinct;
-   }
   }
-  subband.codeblockParameters = {
-   codeblockWidth: xcb_,
-   codeblockHeight: ycb_,
-   numcodeblockwide: cbx1 - cbx0 + 1,
-   numcodeblockhigh: cby1 - cby0 + 1
-  };
-  subband.codeblocks = codeblocks;
-  subband.precincts = precincts;
- }
- function createPacket(resolution, precinctNumber, layerNumber) {
-  var precinctCodeblocks = [];
-  var subbands = resolution.subbands;
-  for (var i = 0, ii = subbands.length; i < ii; i++) {
-   var subband = subbands[i];
-   var codeblocks = subband.codeblocks;
-   for (var j = 0, jj = codeblocks.length; j < jj; j++) {
-    var codeblock = codeblocks[j];
-    if (codeblock.precinctNumber !== precinctNumber) {
-     continue;
+  function buildCodeblocks(context, subband, dimensions) {
+    var xcb_ = dimensions.xcb_;
+    var ycb_ = dimensions.ycb_;
+    var codeblockWidth = 1 << xcb_;
+    var codeblockHeight = 1 << ycb_;
+    var cbx0 = subband.tbx0 >> xcb_;
+    var cby0 = subband.tby0 >> ycb_;
+    var cbx1 = subband.tbx1 + codeblockWidth - 1 >> xcb_;
+    var cby1 = subband.tby1 + codeblockHeight - 1 >> ycb_;
+    var precinctParameters = subband.resolution.precinctParameters;
+    var codeblocks = [];
+    var precincts = [];
+    var i, j, codeblock, precinctNumber;
+    for (j = cby0; j < cby1; j++) {
+      for (i = cbx0; i < cbx1; i++) {
+        codeblock = {
+          cbx: i,
+          cby: j,
+          tbx0: codeblockWidth * i,
+          tby0: codeblockHeight * j,
+          tbx1: codeblockWidth * (i + 1),
+          tby1: codeblockHeight * (j + 1)
+        };
+        codeblock.tbx0_ = Math.max(subband.tbx0, codeblock.tbx0);
+        codeblock.tby0_ = Math.max(subband.tby0, codeblock.tby0);
+        codeblock.tbx1_ = Math.min(subband.tbx1, codeblock.tbx1);
+        codeblock.tby1_ = Math.min(subband.tby1, codeblock.tby1);
+        var pi = Math.floor((codeblock.tbx0_ - subband.tbx0) / precinctParameters.precinctWidthInSubband);
+        var pj = Math.floor((codeblock.tby0_ - subband.tby0) / precinctParameters.precinctHeightInSubband);
+        precinctNumber = pi + pj * precinctParameters.numprecinctswide;
+        codeblock.precinctNumber = precinctNumber;
+        codeblock.subbandType = subband.type;
+        codeblock.Lblock = 3;
+        if (codeblock.tbx1_ <= codeblock.tbx0_ || codeblock.tby1_ <= codeblock.tby0_) {
+          continue;
+        }
+        codeblocks.push(codeblock);
+        var precinct = precincts[precinctNumber];
+        if (precinct !== undefined) {
+          if (i < precinct.cbxMin) {
+            precinct.cbxMin = i;
+          } else if (i > precinct.cbxMax) {
+            precinct.cbxMax = i;
+          }
+          if (j < precinct.cbyMin) {
+            precinct.cbxMin = j;
+          } else if (j > precinct.cbyMax) {
+            precinct.cbyMax = j;
+          }
+        } else {
+          precincts[precinctNumber] = precinct = {
+            cbxMin: i,
+            cbyMin: j,
+            cbxMax: i,
+            cbyMax: j
+          };
+        }
+        codeblock.precinct = precinct;
+      }
     }
-    precinctCodeblocks.push(codeblock);
-   }
-  }
-  return {
-   layerNumber: layerNumber,
-   codeblocks: precinctCodeblocks
-  };
- }
- function LayerResolutionComponentPositionIterator(context) {
-  var siz = context.SIZ;
-  var tileIndex = context.currentTile.index;
-  var tile = context.tiles[tileIndex];
-  var layersCount = tile.codingStyleDefaultParameters.layersCount;
-  var componentsCount = siz.Csiz;
-  var maxDecompositionLevelsCount = 0;
-  for (var q = 0; q < componentsCount; q++) {
-   maxDecompositionLevelsCount = Math.max(maxDecompositionLevelsCount, tile.components[q].codingStyleParameters.decompositionLevelsCount);
+    subband.codeblockParameters = {
+      codeblockWidth: xcb_,
+      codeblockHeight: ycb_,
+      numcodeblockwide: cbx1 - cbx0 + 1,
+      numcodeblockhigh: cby1 - cby0 + 1
+    };
+    subband.codeblocks = codeblocks;
+    subband.precincts = precincts;
   }
-  var l = 0, r = 0, i = 0, k = 0;
-  this.nextPacket = function JpxImage_nextPacket() {
-   for (; l < layersCount; l++) {
-    for (; r <= maxDecompositionLevelsCount; r++) {
-     for (; i < componentsCount; i++) {
-      var component = tile.components[i];
-      if (r > component.codingStyleParameters.decompositionLevelsCount) {
-       continue;
-      }
-      var resolution = component.resolutions[r];
-      var numprecincts = resolution.precinctParameters.numprecincts;
-      for (; k < numprecincts;) {
-       var packet = createPacket(resolution, k, l);
-       k++;
-       return packet;
+  function createPacket(resolution, precinctNumber, layerNumber) {
+    var precinctCodeblocks = [];
+    var subbands = resolution.subbands;
+    for (var i = 0, ii = subbands.length; i < ii; i++) {
+      var subband = subbands[i];
+      var codeblocks = subband.codeblocks;
+      for (var j = 0, jj = codeblocks.length; j < jj; j++) {
+        var codeblock = codeblocks[j];
+        if (codeblock.precinctNumber !== precinctNumber) {
+          continue;
+        }
+        precinctCodeblocks.push(codeblock);
       }
-      k = 0;
-     }
-     i = 0;
     }
-    r = 0;
-   }
-   error('JPX Error: Out of packets');
-  };
- }
- function ResolutionLayerComponentPositionIterator(context) {
-  var siz = context.SIZ;
-  var tileIndex = context.currentTile.index;
-  var tile = context.tiles[tileIndex];
-  var layersCount = tile.codingStyleDefaultParameters.layersCount;
-  var componentsCount = siz.Csiz;
-  var maxDecompositionLevelsCount = 0;
-  for (var q = 0; q < componentsCount; q++) {
-   maxDecompositionLevelsCount = Math.max(maxDecompositionLevelsCount, tile.components[q].codingStyleParameters.decompositionLevelsCount);
+    return {
+      layerNumber: layerNumber,
+      codeblocks: precinctCodeblocks
+    };
   }
-  var r = 0, l = 0, i = 0, k = 0;
-  this.nextPacket = function JpxImage_nextPacket() {
-   for (; r <= maxDecompositionLevelsCount; r++) {
-    for (; l < layersCount; l++) {
-     for (; i < componentsCount; i++) {
-      var component = tile.components[i];
-      if (r > component.codingStyleParameters.decompositionLevelsCount) {
-       continue;
-      }
-      var resolution = component.resolutions[r];
-      var numprecincts = resolution.precinctParameters.numprecincts;
-      for (; k < numprecincts;) {
-       var packet = createPacket(resolution, k, l);
-       k++;
-       return packet;
-      }
-      k = 0;
-     }
-     i = 0;
+  function LayerResolutionComponentPositionIterator(context) {
+    var siz = context.SIZ;
+    var tileIndex = context.currentTile.index;
+    var tile = context.tiles[tileIndex];
+    var layersCount = tile.codingStyleDefaultParameters.layersCount;
+    var componentsCount = siz.Csiz;
+    var maxDecompositionLevelsCount = 0;
+    for (var q = 0; q < componentsCount; q++) {
+      maxDecompositionLevelsCount = Math.max(maxDecompositionLevelsCount, tile.components[q].codingStyleParameters.decompositionLevelsCount);
     }
-    l = 0;
-   }
-   error('JPX Error: Out of packets');
-  };
- }
- function ResolutionPositionComponentLayerIterator(context) {
-  var siz = context.SIZ;
-  var tileIndex = context.currentTile.index;
-  var tile = context.tiles[tileIndex];
-  var layersCount = tile.codingStyleDefaultParameters.layersCount;
-  var componentsCount = siz.Csiz;
-  var l, r, c, p;
-  var maxDecompositionLevelsCount = 0;
-  for (c = 0; c < componentsCount; c++) {
-   var component = tile.components[c];
-   maxDecompositionLevelsCount = Math.max(maxDecompositionLevelsCount, component.codingStyleParameters.decompositionLevelsCount);
+    var l = 0,
+        r = 0,
+        i = 0,
+        k = 0;
+    this.nextPacket = function JpxImage_nextPacket() {
+      for (; l < layersCount; l++) {
+        for (; r <= maxDecompositionLevelsCount; r++) {
+          for (; i < componentsCount; i++) {
+            var component = tile.components[i];
+            if (r > component.codingStyleParameters.decompositionLevelsCount) {
+              continue;
+            }
+            var resolution = component.resolutions[r];
+            var numprecincts = resolution.precinctParameters.numprecincts;
+            for (; k < numprecincts;) {
+              var packet = createPacket(resolution, k, l);
+              k++;
+              return packet;
+            }
+            k = 0;
+          }
+          i = 0;
+        }
+        r = 0;
+      }
+      error('JPX Error: Out of packets');
+    };
   }
-  var maxNumPrecinctsInLevel = new Int32Array(maxDecompositionLevelsCount + 1);
-  for (r = 0; r <= maxDecompositionLevelsCount; ++r) {
-   var maxNumPrecincts = 0;
-   for (c = 0; c < componentsCount; ++c) {
-    var resolutions = tile.components[c].resolutions;
-    if (r < resolutions.length) {
-     maxNumPrecincts = Math.max(maxNumPrecincts, resolutions[r].precinctParameters.numprecincts);
+  function ResolutionLayerComponentPositionIterator(context) {
+    var siz = context.SIZ;
+    var tileIndex = context.currentTile.index;
+    var tile = context.tiles[tileIndex];
+    var layersCount = tile.codingStyleDefaultParameters.layersCount;
+    var componentsCount = siz.Csiz;
+    var maxDecompositionLevelsCount = 0;
+    for (var q = 0; q < componentsCount; q++) {
+      maxDecompositionLevelsCount = Math.max(maxDecompositionLevelsCount, tile.components[q].codingStyleParameters.decompositionLevelsCount);
     }
-   }
-   maxNumPrecinctsInLevel[r] = maxNumPrecincts;
+    var r = 0,
+        l = 0,
+        i = 0,
+        k = 0;
+    this.nextPacket = function JpxImage_nextPacket() {
+      for (; r <= maxDecompositionLevelsCount; r++) {
+        for (; l < layersCount; l++) {
+          for (; i < componentsCount; i++) {
+            var component = tile.components[i];
+            if (r > component.codingStyleParameters.decompositionLevelsCount) {
+              continue;
+            }
+            var resolution = component.resolutions[r];
+            var numprecincts = resolution.precinctParameters.numprecincts;
+            for (; k < numprecincts;) {
+              var packet = createPacket(resolution, k, l);
+              k++;
+              return packet;
+            }
+            k = 0;
+          }
+          i = 0;
+        }
+        l = 0;
+      }
+      error('JPX Error: Out of packets');
+    };
   }
-  l = 0;
-  r = 0;
-  c = 0;
-  p = 0;
-  this.nextPacket = function JpxImage_nextPacket() {
-   for (; r <= maxDecompositionLevelsCount; r++) {
-    for (; p < maxNumPrecinctsInLevel[r]; p++) {
-     for (; c < componentsCount; c++) {
+  function ResolutionPositionComponentLayerIterator(context) {
+    var siz = context.SIZ;
+    var tileIndex = context.currentTile.index;
+    var tile = context.tiles[tileIndex];
+    var layersCount = tile.codingStyleDefaultParameters.layersCount;
+    var componentsCount = siz.Csiz;
+    var l, r, c, p;
+    var maxDecompositionLevelsCount = 0;
+    for (c = 0; c < componentsCount; c++) {
       var component = tile.components[c];
-      if (r > component.codingStyleParameters.decompositionLevelsCount) {
-       continue;
-      }
-      var resolution = component.resolutions[r];
-      var numprecincts = resolution.precinctParameters.numprecincts;
-      if (p >= numprecincts) {
-       continue;
-      }
-      for (; l < layersCount;) {
-       var packet = createPacket(resolution, p, l);
-       l++;
-       return packet;
+      maxDecompositionLevelsCount = Math.max(maxDecompositionLevelsCount, component.codingStyleParameters.decompositionLevelsCount);
+    }
+    var maxNumPrecinctsInLevel = new Int32Array(maxDecompositionLevelsCount + 1);
+    for (r = 0; r <= maxDecompositionLevelsCount; ++r) {
+      var maxNumPrecincts = 0;
+      for (c = 0; c < componentsCount; ++c) {
+        var resolutions = tile.components[c].resolutions;
+        if (r < resolutions.length) {
+          maxNumPrecincts = Math.max(maxNumPrecincts, resolutions[r].precinctParameters.numprecincts);
+        }
       }
-      l = 0;
-     }
-     c = 0;
+      maxNumPrecinctsInLevel[r] = maxNumPrecincts;
     }
+    l = 0;
+    r = 0;
+    c = 0;
     p = 0;
-   }
-   error('JPX Error: Out of packets');
-  };
- }
- function PositionComponentResolutionLayerIterator(context) {
-  var siz = context.SIZ;
-  var tileIndex = context.currentTile.index;
-  var tile = context.tiles[tileIndex];
-  var layersCount = tile.codingStyleDefaultParameters.layersCount;
-  var componentsCount = siz.Csiz;
-  var precinctsSizes = getPrecinctSizesInImageScale(tile);
-  var precinctsIterationSizes = precinctsSizes;
-  var l = 0, r = 0, c = 0, px = 0, py = 0;
-  this.nextPacket = function JpxImage_nextPacket() {
-   for (; py < precinctsIterationSizes.maxNumHigh; py++) {
-    for (; px < precinctsIterationSizes.maxNumWide; px++) {
-     for (; c < componentsCount; c++) {
-      var component = tile.components[c];
-      var decompositionLevelsCount = component.codingStyleParameters.decompositionLevelsCount;
-      for (; r <= decompositionLevelsCount; r++) {
-       var resolution = component.resolutions[r];
-       var sizeInImageScale = precinctsSizes.components[c].resolutions[r];
-       var k = getPrecinctIndexIfExist(px, py, sizeInImageScale, precinctsIterationSizes, resolution);
-       if (k === null) {
-        continue;
-       }
-       for (; l < layersCount;) {
-        var packet = createPacket(resolution, k, l);
-        l++;
-        return packet;
-       }
-       l = 0;
+    this.nextPacket = function JpxImage_nextPacket() {
+      for (; r <= maxDecompositionLevelsCount; r++) {
+        for (; p < maxNumPrecinctsInLevel[r]; p++) {
+          for (; c < componentsCount; c++) {
+            var component = tile.components[c];
+            if (r > component.codingStyleParameters.decompositionLevelsCount) {
+              continue;
+            }
+            var resolution = component.resolutions[r];
+            var numprecincts = resolution.precinctParameters.numprecincts;
+            if (p >= numprecincts) {
+              continue;
+            }
+            for (; l < layersCount;) {
+              var packet = createPacket(resolution, p, l);
+              l++;
+              return packet;
+            }
+            l = 0;
+          }
+          c = 0;
+        }
+        p = 0;
       }
-      r = 0;
-     }
-     c = 0;
-    }
-    px = 0;
-   }
-   error('JPX Error: Out of packets');
-  };
- }
- function ComponentPositionResolutionLayerIterator(context) {
-  var siz = context.SIZ;
-  var tileIndex = context.currentTile.index;
-  var tile = context.tiles[tileIndex];
-  var layersCount = tile.codingStyleDefaultParameters.layersCount;
-  var componentsCount = siz.Csiz;
-  var precinctsSizes = getPrecinctSizesInImageScale(tile);
-  var l = 0, r = 0, c = 0, px = 0, py = 0;
-  this.nextPacket = function JpxImage_nextPacket() {
-   for (; c < componentsCount; ++c) {
-    var component = tile.components[c];
-    var precinctsIterationSizes = precinctsSizes.components[c];
-    var decompositionLevelsCount = component.codingStyleParameters.decompositionLevelsCount;
-    for (; py < precinctsIterationSizes.maxNumHigh; py++) {
-     for (; px < precinctsIterationSizes.maxNumWide; px++) {
-      for (; r <= decompositionLevelsCount; r++) {
-       var resolution = component.resolutions[r];
-       var sizeInImageScale = precinctsIterationSizes.resolutions[r];
-       var k = getPrecinctIndexIfExist(px, py, sizeInImageScale, precinctsIterationSizes, resolution);
-       if (k === null) {
-        continue;
-       }
-       for (; l < layersCount;) {
-        var packet = createPacket(resolution, k, l);
-        l++;
-        return packet;
-       }
-       l = 0;
+      error('JPX Error: Out of packets');
+    };
+  }
+  function PositionComponentResolutionLayerIterator(context) {
+    var siz = context.SIZ;
+    var tileIndex = context.currentTile.index;
+    var tile = context.tiles[tileIndex];
+    var layersCount = tile.codingStyleDefaultParameters.layersCount;
+    var componentsCount = siz.Csiz;
+    var precinctsSizes = getPrecinctSizesInImageScale(tile);
+    var precinctsIterationSizes = precinctsSizes;
+    var l = 0,
+        r = 0,
+        c = 0,
+        px = 0,
+        py = 0;
+    this.nextPacket = function JpxImage_nextPacket() {
+      for (; py < precinctsIterationSizes.maxNumHigh; py++) {
+        for (; px < precinctsIterationSizes.maxNumWide; px++) {
+          for (; c < componentsCount; c++) {
+            var component = tile.components[c];
+            var decompositionLevelsCount = component.codingStyleParameters.decompositionLevelsCount;
+            for (; r <= decompositionLevelsCount; r++) {
+              var resolution = component.resolutions[r];
+              var sizeInImageScale = precinctsSizes.components[c].resolutions[r];
+              var k = getPrecinctIndexIfExist(px, py, sizeInImageScale, precinctsIterationSizes, resolution);
+              if (k === null) {
+                continue;
+              }
+              for (; l < layersCount;) {
+                var packet = createPacket(resolution, k, l);
+                l++;
+                return packet;
+              }
+              l = 0;
+            }
+            r = 0;
+          }
+          c = 0;
+        }
+        px = 0;
       }
-      r = 0;
-     }
-     px = 0;
-    }
-    py = 0;
-   }
-   error('JPX Error: Out of packets');
-  };
- }
- function getPrecinctIndexIfExist(pxIndex, pyIndex, sizeInImageScale, precinctIterationSizes, resolution) {
-  var posX = pxIndex * precinctIterationSizes.minWidth;
-  var posY = pyIndex * precinctIterationSizes.minHeight;
-  if (posX % sizeInImageScale.width !== 0 || posY % sizeInImageScale.height !== 0) {
-   return null;
+      error('JPX Error: Out of packets');
+    };
   }
-  var startPrecinctRowIndex = posY / sizeInImageScale.width * resolution.precinctParameters.numprecinctswide;
-  return posX / sizeInImageScale.height + startPrecinctRowIndex;
- }
- function getPrecinctSizesInImageScale(tile) {
-  var componentsCount = tile.components.length;
-  var minWidth = Number.MAX_VALUE;
-  var minHeight = Number.MAX_VALUE;
-  var maxNumWide = 0;
-  var maxNumHigh = 0;
-  var sizePerComponent = new Array(componentsCount);
-  for (var c = 0; c < componentsCount; c++) {
-   var component = tile.components[c];
-   var decompositionLevelsCount = component.codingStyleParameters.decompositionLevelsCount;
-   var sizePerResolution = new Array(decompositionLevelsCount + 1);
-   var minWidthCurrentComponent = Number.MAX_VALUE;
-   var minHeightCurrentComponent = Number.MAX_VALUE;
-   var maxNumWideCurrentComponent = 0;
-   var maxNumHighCurrentComponent = 0;
-   var scale = 1;
-   for (var r = decompositionLevelsCount; r >= 0; --r) {
-    var resolution = component.resolutions[r];
-    var widthCurrentResolution = scale * resolution.precinctParameters.precinctWidth;
-    var heightCurrentResolution = scale * resolution.precinctParameters.precinctHeight;
-    minWidthCurrentComponent = Math.min(minWidthCurrentComponent, widthCurrentResolution);
-    minHeightCurrentComponent = Math.min(minHeightCurrentComponent, heightCurrentResolution);
-    maxNumWideCurrentComponent = Math.max(maxNumWideCurrentComponent, resolution.precinctParameters.numprecinctswide);
-    maxNumHighCurrentComponent = Math.max(maxNumHighCurrentComponent, resolution.precinctParameters.numprecinctshigh);
-    sizePerResolution[r] = {
-     width: widthCurrentResolution,
-     height: heightCurrentResolution
+  function ComponentPositionResolutionLayerIterator(context) {
+    var siz = context.SIZ;
+    var tileIndex = context.currentTile.index;
+    var tile = context.tiles[tileIndex];
+    var layersCount = tile.codingStyleDefaultParameters.layersCount;
+    var componentsCount = siz.Csiz;
+    var precinctsSizes = getPrecinctSizesInImageScale(tile);
+    var l = 0,
+        r = 0,
+        c = 0,
+        px = 0,
+        py = 0;
+    this.nextPacket = function JpxImage_nextPacket() {
+      for (; c < componentsCount; ++c) {
+        var component = tile.components[c];
+        var precinctsIterationSizes = precinctsSizes.components[c];
+        var decompositionLevelsCount = component.codingStyleParameters.decompositionLevelsCount;
+        for (; py < precinctsIterationSizes.maxNumHigh; py++) {
+          for (; px < precinctsIterationSizes.maxNumWide; px++) {
+            for (; r <= decompositionLevelsCount; r++) {
+              var resolution = component.resolutions[r];
+              var sizeInImageScale = precinctsIterationSizes.resolutions[r];
+              var k = getPrecinctIndexIfExist(px, py, sizeInImageScale, precinctsIterationSizes, resolution);
+              if (k === null) {
+                continue;
+              }
+              for (; l < layersCount;) {
+                var packet = createPacket(resolution, k, l);
+                l++;
+                return packet;
+              }
+              l = 0;
+            }
+            r = 0;
+          }
+          px = 0;
+        }
+        py = 0;
+      }
+      error('JPX Error: Out of packets');
     };
-    scale <<= 1;
-   }
-   minWidth = Math.min(minWidth, minWidthCurrentComponent);
-   minHeight = Math.min(minHeight, minHeightCurrentComponent);
-   maxNumWide = Math.max(maxNumWide, maxNumWideCurrentComponent);
-   maxNumHigh = Math.max(maxNumHigh, maxNumHighCurrentComponent);
-   sizePerComponent[c] = {
-    resolutions: sizePerResolution,
-    minWidth: minWidthCurrentComponent,
-    minHeight: minHeightCurrentComponent,
-    maxNumWide: maxNumWideCurrentComponent,
-    maxNumHigh: maxNumHighCurrentComponent
-   };
   }
-  return {
-   components: sizePerComponent,
-   minWidth: minWidth,
-   minHeight: minHeight,
-   maxNumWide: maxNumWide,
-   maxNumHigh: maxNumHigh
-  };
- }
- function buildPackets(context) {
-  var siz = context.SIZ;
-  var tileIndex = context.currentTile.index;
-  var tile = context.tiles[tileIndex];
-  var componentsCount = siz.Csiz;
-  for (var c = 0; c < componentsCount; c++) {
-   var component = tile.components[c];
-   var decompositionLevelsCount = component.codingStyleParameters.decompositionLevelsCount;
-   var resolutions = [];
-   var subbands = [];
-   for (var r = 0; r <= decompositionLevelsCount; r++) {
-    var blocksDimensions = getBlocksDimensions(context, component, r);
-    var resolution = {};
-    var scale = 1 << decompositionLevelsCount - r;
-    resolution.trx0 = Math.ceil(component.tcx0 / scale);
-    resolution.try0 = Math.ceil(component.tcy0 / scale);
-    resolution.trx1 = Math.ceil(component.tcx1 / scale);
-    resolution.try1 = Math.ceil(component.tcy1 / scale);
-    resolution.resLevel = r;
-    buildPrecincts(context, resolution, blocksDimensions);
-    resolutions.push(resolution);
-    var subband;
-    if (r === 0) {
-     subband = {};
-     subband.type = 'LL';
-     subband.tbx0 = Math.ceil(component.tcx0 / scale);
-     subband.tby0 = Math.ceil(component.tcy0 / scale);
-     subband.tbx1 = Math.ceil(component.tcx1 / scale);
-     subband.tby1 = Math.ceil(component.tcy1 / scale);
-     subband.resolution = resolution;
-     buildCodeblocks(context, subband, blocksDimensions);
-     subbands.push(subband);
-     resolution.subbands = [subband];
-    } else {
-     var bscale = 1 << decompositionLevelsCount - r + 1;
-     var resolutionSubbands = [];
-     subband = {};
-     subband.type = 'HL';
-     subband.tbx0 = Math.ceil(component.tcx0 / bscale - 0.5);
-     subband.tby0 = Math.ceil(component.tcy0 / bscale);
-     subband.tbx1 = Math.ceil(component.tcx1 / bscale - 0.5);
-     subband.tby1 = Math.ceil(component.tcy1 / bscale);
-     subband.resolution = resolution;
-     buildCodeblocks(context, subband, blocksDimensions);
-     subbands.push(subband);
-     resolutionSubbands.push(subband);
-     subband = {};
-     subband.type = 'LH';
-     subband.tbx0 = Math.ceil(component.tcx0 / bscale);
-     subband.tby0 = Math.ceil(component.tcy0 / bscale - 0.5);
-     subband.tbx1 = Math.ceil(component.tcx1 / bscale);
-     subband.tby1 = Math.ceil(component.tcy1 / bscale - 0.5);
-     subband.resolution = resolution;
-     buildCodeblocks(context, subband, blocksDimensions);
-     subbands.push(subband);
-     resolutionSubbands.push(subband);
-     subband = {};
-     subband.type = 'HH';
-     subband.tbx0 = Math.ceil(component.tcx0 / bscale - 0.5);
-     subband.tby0 = Math.ceil(component.tcy0 / bscale - 0.5);
-     subband.tbx1 = Math.ceil(component.tcx1 / bscale - 0.5);
-     subband.tby1 = Math.ceil(component.tcy1 / bscale - 0.5);
-     subband.resolution = resolution;
-     buildCodeblocks(context, subband, blocksDimensions);
-     subbands.push(subband);
-     resolutionSubbands.push(subband);
-     resolution.subbands = resolutionSubbands;
+  function getPrecinctIndexIfExist(pxIndex, pyIndex, sizeInImageScale, precinctIterationSizes, resolution) {
+    var posX = pxIndex * precinctIterationSizes.minWidth;
+    var posY = pyIndex * precinctIterationSizes.minHeight;
+    if (posX % sizeInImageScale.width !== 0 || posY % sizeInImageScale.height !== 0) {
+      return null;
     }
-   }
-   component.resolutions = resolutions;
-   component.subbands = subbands;
+    var startPrecinctRowIndex = posY / sizeInImageScale.width * resolution.precinctParameters.numprecinctswide;
+    return posX / sizeInImageScale.height + startPrecinctRowIndex;
   }
-  var progressionOrder = tile.codingStyleDefaultParameters.progressionOrder;
-  switch (progressionOrder) {
-  case 0:
-   tile.packetsIterator = new LayerResolutionComponentPositionIterator(context);
-   break;
-  case 1:
-   tile.packetsIterator = new ResolutionLayerComponentPositionIterator(context);
-   break;
-  case 2:
-   tile.packetsIterator = new ResolutionPositionComponentLayerIterator(context);
-   break;
-  case 3:
-   tile.packetsIterator = new PositionComponentResolutionLayerIterator(context);
-   break;
-  case 4:
-   tile.packetsIterator = new ComponentPositionResolutionLayerIterator(context);
-   break;
-  default:
-   error('JPX Error: Unsupported progression order ' + progressionOrder);
+  function getPrecinctSizesInImageScale(tile) {
+    var componentsCount = tile.components.length;
+    var minWidth = Number.MAX_VALUE;
+    var minHeight = Number.MAX_VALUE;
+    var maxNumWide = 0;
+    var maxNumHigh = 0;
+    var sizePerComponent = new Array(componentsCount);
+    for (var c = 0; c < componentsCount; c++) {
+      var component = tile.components[c];
+      var decompositionLevelsCount = component.codingStyleParameters.decompositionLevelsCount;
+      var sizePerResolution = new Array(decompositionLevelsCount + 1);
+      var minWidthCurrentComponent = Number.MAX_VALUE;
+      var minHeightCurrentComponent = Number.MAX_VALUE;
+      var maxNumWideCurrentComponent = 0;
+      var maxNumHighCurrentComponent = 0;
+      var scale = 1;
+      for (var r = decompositionLevelsCount; r >= 0; --r) {
+        var resolution = component.resolutions[r];
+        var widthCurrentResolution = scale * resolution.precinctParameters.precinctWidth;
+        var heightCurrentResolution = scale * resolution.precinctParameters.precinctHeight;
+        minWidthCurrentComponent = Math.min(minWidthCurrentComponent, widthCurrentResolution);
+        minHeightCurrentComponent = Math.min(minHeightCurrentComponent, heightCurrentResolution);
+        maxNumWideCurrentComponent = Math.max(maxNumWideCurrentComponent, resolution.precinctParameters.numprecinctswide);
+        maxNumHighCurrentComponent = Math.max(maxNumHighCurrentComponent, resolution.precinctParameters.numprecinctshigh);
+        sizePerResolution[r] = {
+          width: widthCurrentResolution,
+          height: heightCurrentResolution
+        };
+        scale <<= 1;
+      }
+      minWidth = Math.min(minWidth, minWidthCurrentComponent);
+      minHeight = Math.min(minHeight, minHeightCurrentComponent);
+      maxNumWide = Math.max(maxNumWide, maxNumWideCurrentComponent);
+      maxNumHigh = Math.max(maxNumHigh, maxNumHighCurrentComponent);
+      sizePerComponent[c] = {
+        resolutions: sizePerResolution,
+        minWidth: minWidthCurrentComponent,
+        minHeight: minHeightCurrentComponent,
+        maxNumWide: maxNumWideCurrentComponent,
+        maxNumHigh: maxNumHighCurrentComponent
+      };
+    }
+    return {
+      components: sizePerComponent,
+      minWidth: minWidth,
+      minHeight: minHeight,
+      maxNumWide: maxNumWide,
+      maxNumHigh: maxNumHigh
+    };
   }
- }
- function parseTilePackets(context, data, offset, dataLength) {
-  var position = 0;
-  var buffer, bufferSize = 0, skipNextBit = false;
-  function readBits(count) {
-   while (bufferSize < count) {
-    var b = data[offset + position];
-    position++;
-    if (skipNextBit) {
-     buffer = buffer << 7 | b;
-     bufferSize += 7;
-     skipNextBit = false;
-    } else {
-     buffer = buffer << 8 | b;
-     bufferSize += 8;
+  function buildPackets(context) {
+    var siz = context.SIZ;
+    var tileIndex = context.currentTile.index;
+    var tile = context.tiles[tileIndex];
+    var componentsCount = siz.Csiz;
+    for (var c = 0; c < componentsCount; c++) {
+      var component = tile.components[c];
+      var decompositionLevelsCount = component.codingStyleParameters.decompositionLevelsCount;
+      var resolutions = [];
+      var subbands = [];
+      for (var r = 0; r <= decompositionLevelsCount; r++) {
+        var blocksDimensions = getBlocksDimensions(context, component, r);
+        var resolution = {};
+        var scale = 1 << decompositionLevelsCount - r;
+        resolution.trx0 = Math.ceil(component.tcx0 / scale);
+        resolution.try0 = Math.ceil(component.tcy0 / scale);
+        resolution.trx1 = Math.ceil(component.tcx1 / scale);
+        resolution.try1 = Math.ceil(component.tcy1 / scale);
+        resolution.resLevel = r;
+        buildPrecincts(context, resolution, blocksDimensions);
+        resolutions.push(resolution);
+        var subband;
+        if (r === 0) {
+          subband = {};
+          subband.type = 'LL';
+          subband.tbx0 = Math.ceil(component.tcx0 / scale);
+          subband.tby0 = Math.ceil(component.tcy0 / scale);
+          subband.tbx1 = Math.ceil(component.tcx1 / scale);
+          subband.tby1 = Math.ceil(component.tcy1 / scale);
+          subband.resolution = resolution;
+          buildCodeblocks(context, subband, blocksDimensions);
+          subbands.push(subband);
+          resolution.subbands = [subband];
+        } else {
+          var bscale = 1 << decompositionLevelsCount - r + 1;
+          var resolutionSubbands = [];
+          subband = {};
+          subband.type = 'HL';
+          subband.tbx0 = Math.ceil(component.tcx0 / bscale - 0.5);
+          subband.tby0 = Math.ceil(component.tcy0 / bscale);
+          subband.tbx1 = Math.ceil(component.tcx1 / bscale - 0.5);
+          subband.tby1 = Math.ceil(component.tcy1 / bscale);
+          subband.resolution = resolution;
+          buildCodeblocks(context, subband, blocksDimensions);
+          subbands.push(subband);
+          resolutionSubbands.push(subband);
+          subband = {};
+          subband.type = 'LH';
+          subband.tbx0 = Math.ceil(component.tcx0 / bscale);
+          subband.tby0 = Math.ceil(component.tcy0 / bscale - 0.5);
+          subband.tbx1 = Math.ceil(component.tcx1 / bscale);
+          subband.tby1 = Math.ceil(component.tcy1 / bscale - 0.5);
+          subband.resolution = resolution;
+          buildCodeblocks(context, subband, blocksDimensions);
+          subbands.push(subband);
+          resolutionSubbands.push(subband);
+          subband = {};
+          subband.type = 'HH';
+          subband.tbx0 = Math.ceil(component.tcx0 / bscale - 0.5);
+          subband.tby0 = Math.ceil(component.tcy0 / bscale - 0.5);
+          subband.tbx1 = Math.ceil(component.tcx1 / bscale - 0.5);
+          subband.tby1 = Math.ceil(component.tcy1 / bscale - 0.5);
+          subband.resolution = resolution;
+          buildCodeblocks(context, subband, blocksDimensions);
+          subbands.push(subband);
+          resolutionSubbands.push(subband);
+          resolution.subbands = resolutionSubbands;
+        }
+      }
+      component.resolutions = resolutions;
+      component.subbands = subbands;
     }
-    if (b === 0xFF) {
-     skipNextBit = true;
+    var progressionOrder = tile.codingStyleDefaultParameters.progressionOrder;
+    switch (progressionOrder) {
+      case 0:
+        tile.packetsIterator = new LayerResolutionComponentPositionIterator(context);
+        break;
+      case 1:
+        tile.packetsIterator = new ResolutionLayerComponentPositionIterator(context);
+        break;
+      case 2:
+        tile.packetsIterator = new ResolutionPositionComponentLayerIterator(context);
+        break;
+      case 3:
+        tile.packetsIterator = new PositionComponentResolutionLayerIterator(context);
+        break;
+      case 4:
+        tile.packetsIterator = new ComponentPositionResolutionLayerIterator(context);
+        break;
+      default:
+        error('JPX Error: Unsupported progression order ' + progressionOrder);
     }
-   }
-   bufferSize -= count;
-   return buffer >>> bufferSize & (1 << count) - 1;
-  }
-  function skipMarkerIfEqual(value) {
-   if (data[offset + position - 1] === 0xFF && data[offset + position] === value) {
-    skipBytes(1);
-    return true;
-   } else if (data[offset + position] === 0xFF && data[offset + position + 1] === value) {
-    skipBytes(2);
-    return true;
-   }
-   return false;
-  }
-  function skipBytes(count) {
-   position += count;
   }
-  function alignToByte() {
-   bufferSize = 0;
-   if (skipNextBit) {
-    position++;
-    skipNextBit = false;
-   }
-  }
-  function readCodingpasses() {
-   if (readBits(1) === 0) {
-    return 1;
-   }
-   if (readBits(1) === 0) {
-    return 2;
-   }
-   var value = readBits(2);
-   if (value < 3) {
-    return value + 3;
-   }
-   value = readBits(5);
-   if (value < 31) {
-    return value + 6;
-   }
-   value = readBits(7);
-   return value + 37;
-  }
-  var tileIndex = context.currentTile.index;
-  var tile = context.tiles[tileIndex];
-  var sopMarkerUsed = context.COD.sopMarkerUsed;
-  var ephMarkerUsed = context.COD.ephMarkerUsed;
-  var packetsIterator = tile.packetsIterator;
-  while (position < dataLength) {
-   alignToByte();
-   if (sopMarkerUsed && skipMarkerIfEqual(0x91)) {
-    skipBytes(4);
-   }
-   var packet = packetsIterator.nextPacket();
-   if (!readBits(1)) {
-    continue;
-   }
-   var layerNumber = packet.layerNumber;
-   var queue = [], codeblock;
-   for (var i = 0, ii = packet.codeblocks.length; i < ii; i++) {
-    codeblock = packet.codeblocks[i];
-    var precinct = codeblock.precinct;
-    var codeblockColumn = codeblock.cbx - precinct.cbxMin;
-    var codeblockRow = codeblock.cby - precinct.cbyMin;
-    var codeblockIncluded = false;
-    var firstTimeInclusion = false;
-    var valueReady;
-    if (codeblock['included'] !== undefined) {
-     codeblockIncluded = !!readBits(1);
-    } else {
-     precinct = codeblock.precinct;
-     var inclusionTree, zeroBitPlanesTree;
-     if (precinct['inclusionTree'] !== undefined) {
-      inclusionTree = precinct.inclusionTree;
-     } else {
-      var width = precinct.cbxMax - precinct.cbxMin + 1;
-      var height = precinct.cbyMax - precinct.cbyMin + 1;
-      inclusionTree = new InclusionTree(width, height, layerNumber);
-      zeroBitPlanesTree = new TagTree(width, height);
-      precinct.inclusionTree = inclusionTree;
-      precinct.zeroBitPlanesTree = zeroBitPlanesTree;
-     }
-     if (inclusionTree.reset(codeblockColumn, codeblockRow, layerNumber)) {
-      while (true) {
-       if (readBits(1)) {
-        valueReady = !inclusionTree.nextLevel();
-        if (valueReady) {
-         codeblock.included = true;
-         codeblockIncluded = firstTimeInclusion = true;
-         break;
+  function parseTilePackets(context, data, offset, dataLength) {
+    var position = 0;
+    var buffer,
+        bufferSize = 0,
+        skipNextBit = false;
+    function readBits(count) {
+      while (bufferSize < count) {
+        var b = data[offset + position];
+        position++;
+        if (skipNextBit) {
+          buffer = buffer << 7 | b;
+          bufferSize += 7;
+          skipNextBit = false;
+        } else {
+          buffer = buffer << 8 | b;
+          bufferSize += 8;
+        }
+        if (b === 0xFF) {
+          skipNextBit = true;
         }
-       } else {
-        inclusionTree.incrementValue(layerNumber);
-        break;
-       }
       }
-     }
-    }
-    if (!codeblockIncluded) {
-     continue;
+      bufferSize -= count;
+      return buffer >>> bufferSize & (1 << count) - 1;
     }
-    if (firstTimeInclusion) {
-     zeroBitPlanesTree = precinct.zeroBitPlanesTree;
-     zeroBitPlanesTree.reset(codeblockColumn, codeblockRow);
-     while (true) {
-      if (readBits(1)) {
-       valueReady = !zeroBitPlanesTree.nextLevel();
-       if (valueReady) {
-        break;
-       }
-      } else {
-       zeroBitPlanesTree.incrementValue();
+    function skipMarkerIfEqual(value) {
+      if (data[offset + position - 1] === 0xFF && data[offset + position] === value) {
+        skipBytes(1);
+        return true;
+      } else if (data[offset + position] === 0xFF && data[offset + position + 1] === value) {
+        skipBytes(2);
+        return true;
       }
-     }
-     codeblock.zeroBitPlanes = zeroBitPlanesTree.value;
+      return false;
     }
-    var codingpasses = readCodingpasses();
-    while (readBits(1)) {
-     codeblock.Lblock++;
+    function skipBytes(count) {
+      position += count;
     }
-    var codingpassesLog2 = log2(codingpasses);
-    var bits = (codingpasses < 1 << codingpassesLog2 ? codingpassesLog2 - 1 : codingpassesLog2) + codeblock.Lblock;
-    var codedDataLength = readBits(bits);
-    queue.push({
-     codeblock: codeblock,
-     codingpasses: codingpasses,
-     dataLength: codedDataLength
-    });
-   }
-   alignToByte();
-   if (ephMarkerUsed) {
-    skipMarkerIfEqual(0x92);
-   }
-   while (queue.length > 0) {
-    var packetItem = queue.shift();
-    codeblock = packetItem.codeblock;
-    if (codeblock['data'] === undefined) {
-     codeblock.data = [];
+    function alignToByte() {
+      bufferSize = 0;
+      if (skipNextBit) {
+        position++;
+        skipNextBit = false;
+      }
     }
-    codeblock.data.push({
-     data: data,
-     start: offset + position,
-     end: offset + position + packetItem.dataLength,
-     codingpasses: packetItem.codingpasses
-    });
-    position += packetItem.dataLength;
-   }
-  }
-  return position;
- }
- function copyCoefficients(coefficients, levelWidth, levelHeight, subband, delta, mb, reversible, segmentationSymbolUsed) {
-  var x0 = subband.tbx0;
-  var y0 = subband.tby0;
-  var width = subband.tbx1 - subband.tbx0;
-  var codeblocks = subband.codeblocks;
-  var right = subband.type.charAt(0) === 'H' ? 1 : 0;
-  var bottom = subband.type.charAt(1) === 'H' ? levelWidth : 0;
-  for (var i = 0, ii = codeblocks.length; i < ii; ++i) {
-   var codeblock = codeblocks[i];
-   var blockWidth = codeblock.tbx1_ - codeblock.tbx0_;
-   var blockHeight = codeblock.tby1_ - codeblock.tby0_;
-   if (blockWidth === 0 || blockHeight === 0) {
-    continue;
-   }
-   if (codeblock['data'] === undefined) {
-    continue;
-   }
-   var bitModel, currentCodingpassType;
-   bitModel = new BitModel(blockWidth, blockHeight, codeblock.subbandType, codeblock.zeroBitPlanes, mb);
-   currentCodingpassType = 2;
-   var data = codeblock.data, totalLength = 0, codingpasses = 0;
-   var j, jj, dataItem;
-   for (j = 0, jj = data.length; j < jj; j++) {
-    dataItem = data[j];
-    totalLength += dataItem.end - dataItem.start;
-    codingpasses += dataItem.codingpasses;
-   }
-   var encodedData = new Uint8Array(totalLength);
-   var position = 0;
-   for (j = 0, jj = data.length; j < jj; j++) {
-    dataItem = data[j];
-    var chunk = dataItem.data.subarray(dataItem.start, dataItem.end);
-    encodedData.set(chunk, position);
-    position += chunk.length;
-   }
-   var decoder = new ArithmeticDecoder(encodedData, 0, totalLength);
-   bitModel.setDecoder(decoder);
-   for (j = 0; j < codingpasses; j++) {
-    switch (currentCodingpassType) {
-    case 0:
-     bitModel.runSignificancePropagationPass();
-     break;
-    case 1:
-     bitModel.runMagnitudeRefinementPass();
-     break;
-    case 2:
-     bitModel.runCleanupPass();
-     if (segmentationSymbolUsed) {
-      bitModel.checkSegmentationSymbol();
-     }
-     break;
+    function readCodingpasses() {
+      if (readBits(1) === 0) {
+        return 1;
+      }
+      if (readBits(1) === 0) {
+        return 2;
+      }
+      var value = readBits(2);
+      if (value < 3) {
+        return value + 3;
+      }
+      value = readBits(5);
+      if (value < 31) {
+        return value + 6;
+      }
+      value = readBits(7);
+      return value + 37;
     }
-    currentCodingpassType = (currentCodingpassType + 1) % 3;
-   }
-   var offset = codeblock.tbx0_ - x0 + (codeblock.tby0_ - y0) * width;
-   var sign = bitModel.coefficentsSign;
-   var magnitude = bitModel.coefficentsMagnitude;
-   var bitsDecoded = bitModel.bitsDecoded;
-   var magnitudeCorrection = reversible ? 0 : 0.5;
-   var k, n, nb;
-   position = 0;
-   var interleave = subband.type !== 'LL';
-   for (j = 0; j < blockHeight; j++) {
-    var row = offset / width | 0;
-    var levelOffset = 2 * row * (levelWidth - width) + right + bottom;
-    for (k = 0; k < blockWidth; k++) {
-     n = magnitude[position];
-     if (n !== 0) {
-      n = (n + magnitudeCorrection) * delta;
-      if (sign[position] !== 0) {
-       n = -n;
+    var tileIndex = context.currentTile.index;
+    var tile = context.tiles[tileIndex];
+    var sopMarkerUsed = context.COD.sopMarkerUsed;
+    var ephMarkerUsed = context.COD.ephMarkerUsed;
+    var packetsIterator = tile.packetsIterator;
+    while (position < dataLength) {
+      alignToByte();
+      if (sopMarkerUsed && skipMarkerIfEqual(0x91)) {
+        skipBytes(4);
       }
-      nb = bitsDecoded[position];
-      var pos = interleave ? levelOffset + (offset << 1) : offset;
-      if (reversible && nb >= mb) {
-       coefficients[pos] = n;
-      } else {
-       coefficients[pos] = n * (1 << mb - nb);
+      var packet = packetsIterator.nextPacket();
+      if (!readBits(1)) {
+        continue;
+      }
+      var layerNumber = packet.layerNumber;
+      var queue = [],
+          codeblock;
+      for (var i = 0, ii = packet.codeblocks.length; i < ii; i++) {
+        codeblock = packet.codeblocks[i];
+        var precinct = codeblock.precinct;
+        var codeblockColumn = codeblock.cbx - precinct.cbxMin;
+        var codeblockRow = codeblock.cby - precinct.cbyMin;
+        var codeblockIncluded = false;
+        var firstTimeInclusion = false;
+        var valueReady;
+        if (codeblock['included'] !== undefined) {
+          codeblockIncluded = !!readBits(1);
+        } else {
+          precinct = codeblock.precinct;
+          var inclusionTree, zeroBitPlanesTree;
+          if (precinct['inclusionTree'] !== undefined) {
+            inclusionTree = precinct.inclusionTree;
+          } else {
+            var width = precinct.cbxMax - precinct.cbxMin + 1;
+            var height = precinct.cbyMax - precinct.cbyMin + 1;
+            inclusionTree = new InclusionTree(width, height, layerNumber);
+            zeroBitPlanesTree = new TagTree(width, height);
+            precinct.inclusionTree = inclusionTree;
+            precinct.zeroBitPlanesTree = zeroBitPlanesTree;
+          }
+          if (inclusionTree.reset(codeblockColumn, codeblockRow, layerNumber)) {
+            while (true) {
+              if (readBits(1)) {
+                valueReady = !inclusionTree.nextLevel();
+                if (valueReady) {
+                  codeblock.included = true;
+                  codeblockIncluded = firstTimeInclusion = true;
+                  break;
+                }
+              } else {
+                inclusionTree.incrementValue(layerNumber);
+                break;
+              }
+            }
+          }
+        }
+        if (!codeblockIncluded) {
+          continue;
+        }
+        if (firstTimeInclusion) {
+          zeroBitPlanesTree = precinct.zeroBitPlanesTree;
+          zeroBitPlanesTree.reset(codeblockColumn, codeblockRow);
+          while (true) {
+            if (readBits(1)) {
+              valueReady = !zeroBitPlanesTree.nextLevel();
+              if (valueReady) {
+                break;
+              }
+            } else {
+              zeroBitPlanesTree.incrementValue();
+            }
+          }
+          codeblock.zeroBitPlanes = zeroBitPlanesTree.value;
+        }
+        var codingpasses = readCodingpasses();
+        while (readBits(1)) {
+          codeblock.Lblock++;
+        }
+        var codingpassesLog2 = log2(codingpasses);
+        var bits = (codingpasses < 1 << codingpassesLog2 ? codingpassesLog2 - 1 : codingpassesLog2) + codeblock.Lblock;
+        var codedDataLength = readBits(bits);
+        queue.push({
+          codeblock: codeblock,
+          codingpasses: codingpasses,
+          dataLength: codedDataLength
+        });
+      }
+      alignToByte();
+      if (ephMarkerUsed) {
+        skipMarkerIfEqual(0x92);
+      }
+      while (queue.length > 0) {
+        var packetItem = queue.shift();
+        codeblock = packetItem.codeblock;
+        if (codeblock['data'] === undefined) {
+          codeblock.data = [];
+        }
+        codeblock.data.push({
+          data: data,
+          start: offset + position,
+          end: offset + position + packetItem.dataLength,
+          codingpasses: packetItem.codingpasses
+        });
+        position += packetItem.dataLength;
       }
-     }
-     offset++;
-     position++;
     }
-    offset += width - blockWidth;
-   }
+    return position;
   }
- }
- function transformTile(context, tile, c) {
-  var component = tile.components[c];
-  var codingStyleParameters = component.codingStyleParameters;
-  var quantizationParameters = component.quantizationParameters;
-  var decompositionLevelsCount = codingStyleParameters.decompositionLevelsCount;
-  var spqcds = quantizationParameters.SPqcds;
-  var scalarExpounded = quantizationParameters.scalarExpounded;
-  var guardBits = quantizationParameters.guardBits;
-  var segmentationSymbolUsed = codingStyleParameters.segmentationSymbolUsed;
-  var precision = context.components[c].precision;
-  var reversible = codingStyleParameters.reversibleTransformation;
-  var transform = reversible ? new ReversibleTransform() : new IrreversibleTransform();
-  var subbandCoefficients = [];
-  var b = 0;
-  for (var i = 0; i <= decompositionLevelsCount; i++) {
-   var resolution = component.resolutions[i];
-   var width = resolution.trx1 - resolution.trx0;
-   var height = resolution.try1 - resolution.try0;
-   var coefficients = new Float32Array(width * height);
-   for (var j = 0, jj = resolution.subbands.length; j < jj; j++) {
-    var mu, epsilon;
-    if (!scalarExpounded) {
-     mu = spqcds[0].mu;
-     epsilon = spqcds[0].epsilon + (i > 0 ? 1 - i : 0);
-    } else {
-     mu = spqcds[b].mu;
-     epsilon = spqcds[b].epsilon;
-     b++;
+  function copyCoefficients(coefficients, levelWidth, levelHeight, subband, delta, mb, reversible, segmentationSymbolUsed) {
+    var x0 = subband.tbx0;
+    var y0 = subband.tby0;
+    var width = subband.tbx1 - subband.tbx0;
+    var codeblocks = subband.codeblocks;
+    var right = subband.type.charAt(0) === 'H' ? 1 : 0;
+    var bottom = subband.type.charAt(1) === 'H' ? levelWidth : 0;
+    for (var i = 0, ii = codeblocks.length; i < ii; ++i) {
+      var codeblock = codeblocks[i];
+      var blockWidth = codeblock.tbx1_ - codeblock.tbx0_;
+      var blockHeight = codeblock.tby1_ - codeblock.tby0_;
+      if (blockWidth === 0 || blockHeight === 0) {
+        continue;
+      }
+      if (codeblock['data'] === undefined) {
+        continue;
+      }
+      var bitModel, currentCodingpassType;
+      bitModel = new BitModel(blockWidth, blockHeight, codeblock.subbandType, codeblock.zeroBitPlanes, mb);
+      currentCodingpassType = 2;
+      var data = codeblock.data,
+          totalLength = 0,
+          codingpasses = 0;
+      var j, jj, dataItem;
+      for (j = 0, jj = data.length; j < jj; j++) {
+        dataItem = data[j];
+        totalLength += dataItem.end - dataItem.start;
+        codingpasses += dataItem.codingpasses;
+      }
+      var encodedData = new Uint8Array(totalLength);
+      var position = 0;
+      for (j = 0, jj = data.length; j < jj; j++) {
+        dataItem = data[j];
+        var chunk = dataItem.data.subarray(dataItem.start, dataItem.end);
+        encodedData.set(chunk, position);
+        position += chunk.length;
+      }
+      var decoder = new ArithmeticDecoder(encodedData, 0, totalLength);
+      bitModel.setDecoder(decoder);
+      for (j = 0; j < codingpasses; j++) {
+        switch (currentCodingpassType) {
+          case 0:
+            bitModel.runSignificancePropagationPass();
+            break;
+          case 1:
+            bitModel.runMagnitudeRefinementPass();
+            break;
+          case 2:
+            bitModel.runCleanupPass();
+            if (segmentationSymbolUsed) {
+              bitModel.checkSegmentationSymbol();
+            }
+            break;
+        }
+        currentCodingpassType = (currentCodingpassType + 1) % 3;
+      }
+      var offset = codeblock.tbx0_ - x0 + (codeblock.tby0_ - y0) * width;
+      var sign = bitModel.coefficentsSign;
+      var magnitude = bitModel.coefficentsMagnitude;
+      var bitsDecoded = bitModel.bitsDecoded;
+      var magnitudeCorrection = reversible ? 0 : 0.5;
+      var k, n, nb;
+      position = 0;
+      var interleave = subband.type !== 'LL';
+      for (j = 0; j < blockHeight; j++) {
+        var row = offset / width | 0;
+        var levelOffset = 2 * row * (levelWidth - width) + right + bottom;
+        for (k = 0; k < blockWidth; k++) {
+          n = magnitude[position];
+          if (n !== 0) {
+            n = (n + magnitudeCorrection) * delta;
+            if (sign[position] !== 0) {
+              n = -n;
+            }
+            nb = bitsDecoded[position];
+            var pos = interleave ? levelOffset + (offset << 1) : offset;
+            if (reversible && nb >= mb) {
+              coefficients[pos] = n;
+            } else {
+              coefficients[pos] = n * (1 << mb - nb);
+            }
+          }
+          offset++;
+          position++;
+        }
+        offset += width - blockWidth;
+      }
     }
-    var subband = resolution.subbands[j];
-    var gainLog2 = SubbandsGainLog2[subband.type];
-    var delta = reversible ? 1 : Math.pow(2, precision + gainLog2 - epsilon) * (1 + mu / 2048);
-    var mb = guardBits + epsilon - 1;
-    copyCoefficients(coefficients, width, height, subband, delta, mb, reversible, segmentationSymbolUsed);
-   }
-   subbandCoefficients.push({
-    width: width,
-    height: height,
-    items: coefficients
-   });
   }
-  var result = transform.calculate(subbandCoefficients, component.tcx0, component.tcy0);
-  return {
-   left: component.tcx0,
-   top: component.tcy0,
-   width: result.width,
-   height: result.height,
-   items: result.items
-  };
- }
- function transformComponents(context) {
-  var siz = context.SIZ;
-  var components = context.components;
-  var componentsCount = siz.Csiz;
-  var resultImages = [];
-  for (var i = 0, ii = context.tiles.length; i < ii; i++) {
-   var tile = context.tiles[i];
-   var transformedTiles = [];
-   var c;
-   for (c = 0; c < componentsCount; c++) {
-    transformedTiles[c] = transformTile(context, tile, c);
-   }
-   var tile0 = transformedTiles[0];
-   var out = new Uint8Array(tile0.items.length * componentsCount);
-   var result = {
-    left: tile0.left,
-    top: tile0.top,
-    width: tile0.width,
-    height: tile0.height,
-    items: out
-   };
-   var shift, offset, max, min, maxK;
-   var pos = 0, j, jj, y0, y1, y2, r, g, b, k, val;
-   if (tile.codingStyleDefaultParameters.multipleComponentTransform) {
-    var fourComponents = componentsCount === 4;
-    var y0items = transformedTiles[0].items;
-    var y1items = transformedTiles[1].items;
-    var y2items = transformedTiles[2].items;
-    var y3items = fourComponents ? transformedTiles[3].items : null;
-    shift = components[0].precision - 8;
-    offset = (128 << shift) + 0.5;
-    max = 255 * (1 << shift);
-    maxK = max * 0.5;
-    min = -maxK;
-    var component0 = tile.components[0];
-    var alpha01 = componentsCount - 3;
-    jj = y0items.length;
-    if (!component0.codingStyleParameters.reversibleTransformation) {
-     for (j = 0; j < jj; j++, pos += alpha01) {
-      y0 = y0items[j] + offset;
-      y1 = y1items[j];
-      y2 = y2items[j];
-      r = y0 + 1.402 * y2;
-      g = y0 - 0.34413 * y1 - 0.71414 * y2;
-      b = y0 + 1.772 * y1;
-      out[pos++] = r <= 0 ? 0 : r >= max ? 255 : r >> shift;
-      out[pos++] = g <= 0 ? 0 : g >= max ? 255 : g >> shift;
-      out[pos++] = b <= 0 ? 0 : b >= max ? 255 : b >> shift;
-     }
-    } else {
-     for (j = 0; j < jj; j++, pos += alpha01) {
-      y0 = y0items[j] + offset;
-      y1 = y1items[j];
-      y2 = y2items[j];
-      g = y0 - (y2 + y1 >> 2);
-      r = g + y2;
-      b = g + y1;
-      out[pos++] = r <= 0 ? 0 : r >= max ? 255 : r >> shift;
-      out[pos++] = g <= 0 ? 0 : g >= max ? 255 : g >> shift;
-      out[pos++] = b <= 0 ? 0 : b >= max ? 255 : b >> shift;
-     }
-    }
-    if (fourComponents) {
-     for (j = 0, pos = 3; j < jj; j++, pos += 4) {
-      k = y3items[j];
-      out[pos] = k <= min ? 0 : k >= maxK ? 255 : k + offset >> shift;
-     }
-    }
-   } else {
-    for (c = 0; c < componentsCount; c++) {
-     var items = transformedTiles[c].items;
-     shift = components[c].precision - 8;
-     offset = (128 << shift) + 0.5;
-     max = 127.5 * (1 << shift);
-     min = -max;
-     for (pos = c, j = 0, jj = items.length; j < jj; j++) {
-      val = items[j];
-      out[pos] = val <= min ? 0 : val >= max ? 255 : val + offset >> shift;
-      pos += componentsCount;
-     }
+  function transformTile(context, tile, c) {
+    var component = tile.components[c];
+    var codingStyleParameters = component.codingStyleParameters;
+    var quantizationParameters = component.quantizationParameters;
+    var decompositionLevelsCount = codingStyleParameters.decompositionLevelsCount;
+    var spqcds = quantizationParameters.SPqcds;
+    var scalarExpounded = quantizationParameters.scalarExpounded;
+    var guardBits = quantizationParameters.guardBits;
+    var segmentationSymbolUsed = codingStyleParameters.segmentationSymbolUsed;
+    var precision = context.components[c].precision;
+    var reversible = codingStyleParameters.reversibleTransformation;
+    var transform = reversible ? new ReversibleTransform() : new IrreversibleTransform();
+    var subbandCoefficients = [];
+    var b = 0;
+    for (var i = 0; i <= decompositionLevelsCount; i++) {
+      var resolution = component.resolutions[i];
+      var width = resolution.trx1 - resolution.trx0;
+      var height = resolution.try1 - resolution.try0;
+      var coefficients = new Float32Array(width * height);
+      for (var j = 0, jj = resolution.subbands.length; j < jj; j++) {
+        var mu, epsilon;
+        if (!scalarExpounded) {
+          mu = spqcds[0].mu;
+          epsilon = spqcds[0].epsilon + (i > 0 ? 1 - i : 0);
+        } else {
+          mu = spqcds[b].mu;
+          epsilon = spqcds[b].epsilon;
+          b++;
+        }
+        var subband = resolution.subbands[j];
+        var gainLog2 = SubbandsGainLog2[subband.type];
+        var delta = reversible ? 1 : Math.pow(2, precision + gainLog2 - epsilon) * (1 + mu / 2048);
+        var mb = guardBits + epsilon - 1;
+        copyCoefficients(coefficients, width, height, subband, delta, mb, reversible, segmentationSymbolUsed);
+      }
+      subbandCoefficients.push({
+        width: width,
+        height: height,
+        items: coefficients
+      });
     }
-   }
-   resultImages.push(result);
-  }
-  return resultImages;
- }
- function initializeTile(context, tileIndex) {
-  var siz = context.SIZ;
-  var componentsCount = siz.Csiz;
-  var tile = context.tiles[tileIndex];
-  for (var c = 0; c < componentsCount; c++) {
-   var component = tile.components[c];
-   var qcdOrQcc = context.currentTile.QCC[c] !== undefined ? context.currentTile.QCC[c] : context.currentTile.QCD;
-   component.quantizationParameters = qcdOrQcc;
-   var codOrCoc = context.currentTile.COC[c] !== undefined ? context.currentTile.COC[c] : context.currentTile.COD;
-   component.codingStyleParameters = codOrCoc;
-  }
-  tile.codingStyleDefaultParameters = context.currentTile.COD;
- }
- var TagTree = function TagTreeClosure() {
-  function TagTree(width, height) {
-   var levelsLength = log2(Math.max(width, height)) + 1;
-   this.levels = [];
-   for (var i = 0; i < levelsLength; i++) {
-    var level = {
-     width: width,
-     height: height,
-     items: []
+    var result = transform.calculate(subbandCoefficients, component.tcx0, component.tcy0);
+    return {
+      left: component.tcx0,
+      top: component.tcy0,
+      width: result.width,
+      height: result.height,
+      items: result.items
     };
-    this.levels.push(level);
-    width = Math.ceil(width / 2);
-    height = Math.ceil(height / 2);
-   }
   }
-  TagTree.prototype = {
-   reset: function TagTree_reset(i, j) {
-    var currentLevel = 0, value = 0, level;
-    while (currentLevel < this.levels.length) {
-     level = this.levels[currentLevel];
-     var index = i + j * level.width;
-     if (level.items[index] !== undefined) {
-      value = level.items[index];
-      break;
-     }
-     level.index = index;
-     i >>= 1;
-     j >>= 1;
-     currentLevel++;
-    }
-    currentLevel--;
-    level = this.levels[currentLevel];
-    level.items[level.index] = value;
-    this.currentLevel = currentLevel;
-    delete this.value;
-   },
-   incrementValue: function TagTree_incrementValue() {
-    var level = this.levels[this.currentLevel];
-    level.items[level.index]++;
-   },
-   nextLevel: function TagTree_nextLevel() {
-    var currentLevel = this.currentLevel;
-    var level = this.levels[currentLevel];
-    var value = level.items[level.index];
-    currentLevel--;
-    if (currentLevel < 0) {
-     this.value = value;
-     return false;
-    }
-    this.currentLevel = currentLevel;
-    level = this.levels[currentLevel];
-    level.items[level.index] = value;
-    return true;
-   }
-  };
-  return TagTree;
- }();
- var InclusionTree = function InclusionTreeClosure() {
-  function InclusionTree(width, height, defaultValue) {
-   var levelsLength = log2(Math.max(width, height)) + 1;
-   this.levels = [];
-   for (var i = 0; i < levelsLength; i++) {
-    var items = new Uint8Array(width * height);
-    for (var j = 0, jj = items.length; j < jj; j++) {
-     items[j] = defaultValue;
+  function transformComponents(context) {
+    var siz = context.SIZ;
+    var components = context.components;
+    var componentsCount = siz.Csiz;
+    var resultImages = [];
+    for (var i = 0, ii = context.tiles.length; i < ii; i++) {
+      var tile = context.tiles[i];
+      var transformedTiles = [];
+      var c;
+      for (c = 0; c < componentsCount; c++) {
+        transformedTiles[c] = transformTile(context, tile, c);
+      }
+      var tile0 = transformedTiles[0];
+      var out = new Uint8Array(tile0.items.length * componentsCount);
+      var result = {
+        left: tile0.left,
+        top: tile0.top,
+        width: tile0.width,
+        height: tile0.height,
+        items: out
+      };
+      var shift, offset, max, min, maxK;
+      var pos = 0,
+          j,
+          jj,
+          y0,
+          y1,
+          y2,
+          r,
+          g,
+          b,
+          k,
+          val;
+      if (tile.codingStyleDefaultParameters.multipleComponentTransform) {
+        var fourComponents = componentsCount === 4;
+        var y0items = transformedTiles[0].items;
+        var y1items = transformedTiles[1].items;
+        var y2items = transformedTiles[2].items;
+        var y3items = fourComponents ? transformedTiles[3].items : null;
+        shift = components[0].precision - 8;
+        offset = (128 << shift) + 0.5;
+        max = 255 * (1 << shift);
+        maxK = max * 0.5;
+        min = -maxK;
+        var component0 = tile.components[0];
+        var alpha01 = componentsCount - 3;
+        jj = y0items.length;
+        if (!component0.codingStyleParameters.reversibleTransformation) {
+          for (j = 0; j < jj; j++, pos += alpha01) {
+            y0 = y0items[j] + offset;
+            y1 = y1items[j];
+            y2 = y2items[j];
+            r = y0 + 1.402 * y2;
+            g = y0 - 0.34413 * y1 - 0.71414 * y2;
+            b = y0 + 1.772 * y1;
+            out[pos++] = r <= 0 ? 0 : r >= max ? 255 : r >> shift;
+            out[pos++] = g <= 0 ? 0 : g >= max ? 255 : g >> shift;
+            out[pos++] = b <= 0 ? 0 : b >= max ? 255 : b >> shift;
+          }
+        } else {
+          for (j = 0; j < jj; j++, pos += alpha01) {
+            y0 = y0items[j] + offset;
+            y1 = y1items[j];
+            y2 = y2items[j];
+            g = y0 - (y2 + y1 >> 2);
+            r = g + y2;
+            b = g + y1;
+            out[pos++] = r <= 0 ? 0 : r >= max ? 255 : r >> shift;
+            out[pos++] = g <= 0 ? 0 : g >= max ? 255 : g >> shift;
+            out[pos++] = b <= 0 ? 0 : b >= max ? 255 : b >> shift;
+          }
+        }
+        if (fourComponents) {
+          for (j = 0, pos = 3; j < jj; j++, pos += 4) {
+            k = y3items[j];
+            out[pos] = k <= min ? 0 : k >= maxK ? 255 : k + offset >> shift;
+          }
+        }
+      } else {
+        for (c = 0; c < componentsCount; c++) {
+          var items = transformedTiles[c].items;
+          shift = components[c].precision - 8;
+          offset = (128 << shift) + 0.5;
+          max = 127.5 * (1 << shift);
+          min = -max;
+          for (pos = c, j = 0, jj = items.length; j < jj; j++) {
+            val = items[j];
+            out[pos] = val <= min ? 0 : val >= max ? 255 : val + offset >> shift;
+            pos += componentsCount;
+          }
+        }
+      }
+      resultImages.push(result);
     }
-    var level = {
-     width: width,
-     height: height,
-     items: items
-    };
-    this.levels.push(level);
-    width = Math.ceil(width / 2);
-    height = Math.ceil(height / 2);
-   }
+    return resultImages;
   }
-  InclusionTree.prototype = {
-   reset: function InclusionTree_reset(i, j, stopValue) {
-    var currentLevel = 0;
-    while (currentLevel < this.levels.length) {
-     var level = this.levels[currentLevel];
-     var index = i + j * level.width;
-     level.index = index;
-     var value = level.items[index];
-     if (value === 0xFF) {
-      break;
-     }
-     if (value > stopValue) {
-      this.currentLevel = currentLevel;
-      this.propagateValues();
-      return false;
-     }
-     i >>= 1;
-     j >>= 1;
-     currentLevel++;
-    }
-    this.currentLevel = currentLevel - 1;
-    return true;
-   },
-   incrementValue: function InclusionTree_incrementValue(stopValue) {
-    var level = this.levels[this.currentLevel];
-    level.items[level.index] = stopValue + 1;
-    this.propagateValues();
-   },
-   propagateValues: function InclusionTree_propagateValues() {
-    var levelIndex = this.currentLevel;
-    var level = this.levels[levelIndex];
-    var currentValue = level.items[level.index];
-    while (--levelIndex >= 0) {
-     level = this.levels[levelIndex];
-     level.items[level.index] = currentValue;
-    }
-   },
-   nextLevel: function InclusionTree_nextLevel() {
-    var currentLevel = this.currentLevel;
-    var level = this.levels[currentLevel];
-    var value = level.items[level.index];
-    level.items[level.index] = 0xFF;
-    currentLevel--;
-    if (currentLevel < 0) {
-     return false;
-    }
-    this.currentLevel = currentLevel;
-    level = this.levels[currentLevel];
-    level.items[level.index] = value;
-    return true;
-   }
-  };
-  return InclusionTree;
- }();
- var BitModel = function BitModelClosure() {
-  var UNIFORM_CONTEXT = 17;
-  var RUNLENGTH_CONTEXT = 18;
-  var LLAndLHContextsLabel = new Uint8Array([
-   0,
-   5,
-   8,
-   0,
-   3,
-   7,
-   8,
-   0,
-   4,
-   7,
-   8,
-   0,
-   0,
-   0,
-   0,
-   0,
-   1,
-   6,
-   8,
-   0,
-   3,
-   7,
-   8,
-   0,
-   4,
-   7,
-   8,
-   0,
-   0,
-   0,
-   0,
-   0,
-   2,
-   6,
-   8,
-   0,
-   3,
-   7,
-   8,
-   0,
-   4,
-   7,
-   8,
-   0,
-   0,
-   0,
-   0,
-   0,
-   2,
-   6,
-   8,
-   0,
-   3,
-   7,
-   8,
-   0,
-   4,
-   7,
-   8,
-   0,
-   0,
-   0,
-   0,
-   0,
-   2,
-   6,
-   8,
-   0,
-   3,
-   7,
-   8,
-   0,
-   4,
-   7,
-   8
-  ]);
-  var HLContextLabel = new Uint8Array([
-   0,
-   3,
-   4,
-   0,
-   5,
-   7,
-   7,
-   0,
-   8,
-   8,
-   8,
-   0,
-   0,
-   0,
-   0,
-   0,
-   1,
-   3,
-   4,
-   0,
-   6,
-   7,
-   7,
-   0,
-   8,
-   8,
-   8,
-   0,
-   0,
-   0,
-   0,
-   0,
-   2,
-   3,
-   4,
-   0,
-   6,
-   7,
-   7,
-   0,
-   8,
-   8,
-   8,
-   0,
-   0,
-   0,
-   0,
-   0,
-   2,
-   3,
-   4,
-   0,
-   6,
-   7,
-   7,
-   0,
-   8,
-   8,
-   8,
-   0,
-   0,
-   0,
-   0,
-   0,
-   2,
-   3,
-   4,
-   0,
-   6,
-   7,
-   7,
-   0,
-   8,
-   8,
-   8
-  ]);
-  var HHContextLabel = new Uint8Array([
-   0,
-   1,
-   2,
-   0,
-   1,
-   2,
-   2,
-   0,
-   2,
-   2,
-   2,
-   0,
-   0,
-   0,
-   0,
-   0,
-   3,
-   4,
-   5,
-   0,
-   4,
-   5,
-   5,
-   0,
-   5,
-   5,
-   5,
-   0,
-   0,
-   0,
-   0,
-   0,
-   6,
-   7,
-   7,
-   0,
-   7,
-   7,
-   7,
-   0,
-   7,
-   7,
-   7,
-   0,
-   0,
-   0,
-   0,
-   0,
-   8,
-   8,
-   8,
-   0,
-   8,
-   8,
-   8,
-   0,
-   8,
-   8,
-   8,
-   0,
-   0,
-   0,
-   0,
-   0,
-   8,
-   8,
-   8,
-   0,
-   8,
-   8,
-   8,
-   0,
-   8,
-   8,
-   8
-  ]);
-  function BitModel(width, height, subband, zeroBitPlanes, mb) {
-   this.width = width;
-   this.height = height;
-   this.contextLabelTable = subband === 'HH' ? HHContextLabel : subband === 'HL' ? HLContextLabel : LLAndLHContextsLabel;
-   var coefficientCount = width * height;
-   this.neighborsSignificance = new Uint8Array(coefficientCount);
-   this.coefficentsSign = new Uint8Array(coefficientCount);
-   this.coefficentsMagnitude = mb > 14 ? new Uint32Array(coefficientCount) : mb > 6 ? new Uint16Array(coefficientCount) : new Uint8Array(coefficientCount);
-   this.processingFlags = new Uint8Array(coefficientCount);
-   var bitsDecoded = new Uint8Array(coefficientCount);
-   if (zeroBitPlanes !== 0) {
-    for (var i = 0; i < coefficientCount; i++) {
-     bitsDecoded[i] = zeroBitPlanes;
+  function initializeTile(context, tileIndex) {
+    var siz = context.SIZ;
+    var componentsCount = siz.Csiz;
+    var tile = context.tiles[tileIndex];
+    for (var c = 0; c < componentsCount; c++) {
+      var component = tile.components[c];
+      var qcdOrQcc = context.currentTile.QCC[c] !== undefined ? context.currentTile.QCC[c] : context.currentTile.QCD;
+      component.quantizationParameters = qcdOrQcc;
+      var codOrCoc = context.currentTile.COC[c] !== undefined ? context.currentTile.COC[c] : context.currentTile.COD;
+      component.codingStyleParameters = codOrCoc;
     }
-   }
-   this.bitsDecoded = bitsDecoded;
-   this.reset();
+    tile.codingStyleDefaultParameters = context.currentTile.COD;
   }
-  BitModel.prototype = {
-   setDecoder: function BitModel_setDecoder(decoder) {
-    this.decoder = decoder;
-   },
-   reset: function BitModel_reset() {
-    this.contexts = new Int8Array(19);
-    this.contexts[0] = 4 << 1 | 0;
-    this.contexts[UNIFORM_CONTEXT] = 46 << 1 | 0;
-    this.contexts[RUNLENGTH_CONTEXT] = 3 << 1 | 0;
-   },
-   setNeighborsSignificance: function BitModel_setNeighborsSignificance(row, column, index) {
-    var neighborsSignificance = this.neighborsSignificance;
-    var width = this.width, height = this.height;
-    var left = column > 0;
-    var right = column + 1 < width;
-    var i;
-    if (row > 0) {
-     i = index - width;
-     if (left) {
-      neighborsSignificance[i - 1] += 0x10;
-     }
-     if (right) {
-      neighborsSignificance[i + 1] += 0x10;
-     }
-     neighborsSignificance[i] += 0x04;
-    }
-    if (row + 1 < height) {
-     i = index + width;
-     if (left) {
-      neighborsSignificance[i - 1] += 0x10;
-     }
-     if (right) {
-      neighborsSignificance[i + 1] += 0x10;
-     }
-     neighborsSignificance[i] += 0x04;
-    }
-    if (left) {
-     neighborsSignificance[index - 1] += 0x01;
-    }
-    if (right) {
-     neighborsSignificance[index + 1] += 0x01;
-    }
-    neighborsSignificance[index] |= 0x80;
-   },
-   runSignificancePropagationPass: function BitModel_runSignificancePropagationPass() {
-    var decoder = this.decoder;
-    var width = this.width, height = this.height;
-    var coefficentsMagnitude = this.coefficentsMagnitude;
-    var coefficentsSign = this.coefficentsSign;
-    var neighborsSignificance = this.neighborsSignificance;
-    var processingFlags = this.processingFlags;
-    var contexts = this.contexts;
-    var labels = this.contextLabelTable;
-    var bitsDecoded = this.bitsDecoded;
-    var processedInverseMask = ~1;
-    var processedMask = 1;
-    var firstMagnitudeBitMask = 2;
-    for (var i0 = 0; i0 < height; i0 += 4) {
-     for (var j = 0; j < width; j++) {
-      var index = i0 * width + j;
-      for (var i1 = 0; i1 < 4; i1++, index += width) {
-       var i = i0 + i1;
-       if (i >= height) {
-        break;
-       }
-       processingFlags[index] &= processedInverseMask;
-       if (coefficentsMagnitude[index] || !neighborsSignificance[index]) {
-        continue;
-       }
-       var contextLabel = labels[neighborsSignificance[index]];
-       var decision = decoder.readBit(contexts, contextLabel);
-       if (decision) {
-        var sign = this.decodeSignBit(i, j, index);
-        coefficentsSign[index] = sign;
-        coefficentsMagnitude[index] = 1;
-        this.setNeighborsSignificance(i, j, index);
-        processingFlags[index] |= firstMagnitudeBitMask;
-       }
-       bitsDecoded[index]++;
-       processingFlags[index] |= processedMask;
+  var TagTree = function TagTreeClosure() {
+    function TagTree(width, height) {
+      var levelsLength = log2(Math.max(width, height)) + 1;
+      this.levels = [];
+      for (var i = 0; i < levelsLength; i++) {
+        var level = {
+          width: width,
+          height: height,
+          items: []
+        };
+        this.levels.push(level);
+        width = Math.ceil(width / 2);
+        height = Math.ceil(height / 2);
       }
-     }
-    }
-   },
-   decodeSignBit: function BitModel_decodeSignBit(row, column, index) {
-    var width = this.width, height = this.height;
-    var coefficentsMagnitude = this.coefficentsMagnitude;
-    var coefficentsSign = this.coefficentsSign;
-    var contribution, sign0, sign1, significance1;
-    var contextLabel, decoded;
-    significance1 = column > 0 && coefficentsMagnitude[index - 1] !== 0;
-    if (column + 1 < width && coefficentsMagnitude[index + 1] !== 0) {
-     sign1 = coefficentsSign[index + 1];
-     if (significance1) {
-      sign0 = coefficentsSign[index - 1];
-      contribution = 1 - sign1 - sign0;
-     } else {
-      contribution = 1 - sign1 - sign1;
-     }
-    } else if (significance1) {
-     sign0 = coefficentsSign[index - 1];
-     contribution = 1 - sign0 - sign0;
-    } else {
-     contribution = 0;
     }
-    var horizontalContribution = 3 * contribution;
-    significance1 = row > 0 && coefficentsMagnitude[index - width] !== 0;
-    if (row + 1 < height && coefficentsMagnitude[index + width] !== 0) {
-     sign1 = coefficentsSign[index + width];
-     if (significance1) {
-      sign0 = coefficentsSign[index - width];
-      contribution = 1 - sign1 - sign0 + horizontalContribution;
-     } else {
-      contribution = 1 - sign1 - sign1 + horizontalContribution;
-     }
-    } else if (significance1) {
-     sign0 = coefficentsSign[index - width];
-     contribution = 1 - sign0 - sign0 + horizontalContribution;
-    } else {
-     contribution = horizontalContribution;
-    }
-    if (contribution >= 0) {
-     contextLabel = 9 + contribution;
-     decoded = this.decoder.readBit(this.contexts, contextLabel);
-    } else {
-     contextLabel = 9 - contribution;
-     decoded = this.decoder.readBit(this.contexts, contextLabel) ^ 1;
-    }
-    return decoded;
-   },
-   runMagnitudeRefinementPass: function BitModel_runMagnitudeRefinementPass() {
-    var decoder = this.decoder;
-    var width = this.width, height = this.height;
-    var coefficentsMagnitude = this.coefficentsMagnitude;
-    var neighborsSignificance = this.neighborsSignificance;
-    var contexts = this.contexts;
-    var bitsDecoded = this.bitsDecoded;
-    var processingFlags = this.processingFlags;
-    var processedMask = 1;
-    var firstMagnitudeBitMask = 2;
-    var length = width * height;
-    var width4 = width * 4;
-    for (var index0 = 0, indexNext; index0 < length; index0 = indexNext) {
-     indexNext = Math.min(length, index0 + width4);
-     for (var j = 0; j < width; j++) {
-      for (var index = index0 + j; index < indexNext; index += width) {
-       if (!coefficentsMagnitude[index] || (processingFlags[index] & processedMask) !== 0) {
-        continue;
-       }
-       var contextLabel = 16;
-       if ((processingFlags[index] & firstMagnitudeBitMask) !== 0) {
-        processingFlags[index] ^= firstMagnitudeBitMask;
-        var significance = neighborsSignificance[index] & 127;
-        contextLabel = significance === 0 ? 15 : 14;
-       }
-       var bit = decoder.readBit(contexts, contextLabel);
-       coefficentsMagnitude[index] = coefficentsMagnitude[index] << 1 | bit;
-       bitsDecoded[index]++;
-       processingFlags[index] |= processedMask;
+    TagTree.prototype = {
+      reset: function TagTree_reset(i, j) {
+        var currentLevel = 0,
+            value = 0,
+            level;
+        while (currentLevel < this.levels.length) {
+          level = this.levels[currentLevel];
+          var index = i + j * level.width;
+          if (level.items[index] !== undefined) {
+            value = level.items[index];
+            break;
+          }
+          level.index = index;
+          i >>= 1;
+          j >>= 1;
+          currentLevel++;
+        }
+        currentLevel--;
+        level = this.levels[currentLevel];
+        level.items[level.index] = value;
+        this.currentLevel = currentLevel;
+        delete this.value;
+      },
+      incrementValue: function TagTree_incrementValue() {
+        var level = this.levels[this.currentLevel];
+        level.items[level.index]++;
+      },
+      nextLevel: function TagTree_nextLevel() {
+        var currentLevel = this.currentLevel;
+        var level = this.levels[currentLevel];
+        var value = level.items[level.index];
+        currentLevel--;
+        if (currentLevel < 0) {
+          this.value = value;
+          return false;
+        }
+        this.currentLevel = currentLevel;
+        level = this.levels[currentLevel];
+        level.items[level.index] = value;
+        return true;
+      }
+    };
+    return TagTree;
+  }();
+  var InclusionTree = function InclusionTreeClosure() {
+    function InclusionTree(width, height, defaultValue) {
+      var levelsLength = log2(Math.max(width, height)) + 1;
+      this.levels = [];
+      for (var i = 0; i < levelsLength; i++) {
+        var items = new Uint8Array(width * height);
+        for (var j = 0, jj = items.length; j < jj; j++) {
+          items[j] = defaultValue;
+        }
+        var level = {
+          width: width,
+          height: height,
+          items: items
+        };
+        this.levels.push(level);
+        width = Math.ceil(width / 2);
+        height = Math.ceil(height / 2);
       }
-     }
     }
-   },
-   runCleanupPass: function BitModel_runCleanupPass() {
-    var decoder = this.decoder;
-    var width = this.width, height = this.height;
-    var neighborsSignificance = this.neighborsSignificance;
-    var coefficentsMagnitude = this.coefficentsMagnitude;
-    var coefficentsSign = this.coefficentsSign;
-    var contexts = this.contexts;
-    var labels = this.contextLabelTable;
-    var bitsDecoded = this.bitsDecoded;
-    var processingFlags = this.processingFlags;
-    var processedMask = 1;
-    var firstMagnitudeBitMask = 2;
-    var oneRowDown = width;
-    var twoRowsDown = width * 2;
-    var threeRowsDown = width * 3;
-    var iNext;
-    for (var i0 = 0; i0 < height; i0 = iNext) {
-     iNext = Math.min(i0 + 4, height);
-     var indexBase = i0 * width;
-     var checkAllEmpty = i0 + 3 < height;
-     for (var j = 0; j < width; j++) {
-      var index0 = indexBase + j;
-      var allEmpty = checkAllEmpty && processingFlags[index0] === 0 && processingFlags[index0 + oneRowDown] === 0 && processingFlags[index0 + twoRowsDown] === 0 && processingFlags[index0 + threeRowsDown] === 0 && neighborsSignificance[index0] === 0 && neighborsSignificance[index0 + oneRowDown] === 0 && neighborsSignificance[index0 + twoRowsDown] === 0 && neighborsSignificance[index0 + threeRowsDown] === 0;
-      var i1 = 0, index = index0;
-      var i = i0, sign;
-      if (allEmpty) {
-       var hasSignificantCoefficent = decoder.readBit(contexts, RUNLENGTH_CONTEXT);
-       if (!hasSignificantCoefficent) {
-        bitsDecoded[index0]++;
-        bitsDecoded[index0 + oneRowDown]++;
-        bitsDecoded[index0 + twoRowsDown]++;
-        bitsDecoded[index0 + threeRowsDown]++;
-        continue;
-       }
-       i1 = decoder.readBit(contexts, UNIFORM_CONTEXT) << 1 | decoder.readBit(contexts, UNIFORM_CONTEXT);
-       if (i1 !== 0) {
-        i = i0 + i1;
-        index += i1 * width;
-       }
-       sign = this.decodeSignBit(i, j, index);
-       coefficentsSign[index] = sign;
-       coefficentsMagnitude[index] = 1;
-       this.setNeighborsSignificance(i, j, index);
-       processingFlags[index] |= firstMagnitudeBitMask;
-       index = index0;
-       for (var i2 = i0; i2 <= i; i2++, index += width) {
-        bitsDecoded[index]++;
-       }
-       i1++;
+    InclusionTree.prototype = {
+      reset: function InclusionTree_reset(i, j, stopValue) {
+        var currentLevel = 0;
+        while (currentLevel < this.levels.length) {
+          var level = this.levels[currentLevel];
+          var index = i + j * level.width;
+          level.index = index;
+          var value = level.items[index];
+          if (value === 0xFF) {
+            break;
+          }
+          if (value > stopValue) {
+            this.currentLevel = currentLevel;
+            this.propagateValues();
+            return false;
+          }
+          i >>= 1;
+          j >>= 1;
+          currentLevel++;
+        }
+        this.currentLevel = currentLevel - 1;
+        return true;
+      },
+      incrementValue: function InclusionTree_incrementValue(stopValue) {
+        var level = this.levels[this.currentLevel];
+        level.items[level.index] = stopValue + 1;
+        this.propagateValues();
+      },
+      propagateValues: function InclusionTree_propagateValues() {
+        var levelIndex = this.currentLevel;
+        var level = this.levels[levelIndex];
+        var currentValue = level.items[level.index];
+        while (--levelIndex >= 0) {
+          level = this.levels[levelIndex];
+          level.items[level.index] = currentValue;
+        }
+      },
+      nextLevel: function InclusionTree_nextLevel() {
+        var currentLevel = this.currentLevel;
+        var level = this.levels[currentLevel];
+        var value = level.items[level.index];
+        level.items[level.index] = 0xFF;
+        currentLevel--;
+        if (currentLevel < 0) {
+          return false;
+        }
+        this.currentLevel = currentLevel;
+        level = this.levels[currentLevel];
+        level.items[level.index] = value;
+        return true;
       }
-      for (i = i0 + i1; i < iNext; i++, index += width) {
-       if (coefficentsMagnitude[index] || (processingFlags[index] & processedMask) !== 0) {
-        continue;
-       }
-       var contextLabel = labels[neighborsSignificance[index]];
-       var decision = decoder.readBit(contexts, contextLabel);
-       if (decision === 1) {
-        sign = this.decodeSignBit(i, j, index);
-        coefficentsSign[index] = sign;
-        coefficentsMagnitude[index] = 1;
-        this.setNeighborsSignificance(i, j, index);
-        processingFlags[index] |= firstMagnitudeBitMask;
-       }
-       bitsDecoded[index]++;
+    };
+    return InclusionTree;
+  }();
+  var BitModel = function BitModelClosure() {
+    var UNIFORM_CONTEXT = 17;
+    var RUNLENGTH_CONTEXT = 18;
+    var LLAndLHContextsLabel = new Uint8Array([0, 5, 8, 0, 3, 7, 8, 0, 4, 7, 8, 0, 0, 0, 0, 0, 1, 6, 8, 0, 3, 7, 8, 0, 4, 7, 8, 0, 0, 0, 0, 0, 2, 6, 8, 0, 3, 7, 8, 0, 4, 7, 8, 0, 0, 0, 0, 0, 2, 6, 8, 0, 3, 7, 8, 0, 4, 7, 8, 0, 0, 0, 0, 0, 2, 6, 8, 0, 3, 7, 8, 0, 4, 7, 8]);
+    var HLContextLabel = new Uint8Array([0, 3, 4, 0, 5, 7, 7, 0, 8, 8, 8, 0, 0, 0, 0, 0, 1, 3, 4, 0, 6, 7, 7, 0, 8, 8, 8, 0, 0, 0, 0, 0, 2, 3, 4, 0, 6, 7, 7, 0, 8, 8, 8, 0, 0, 0, 0, 0, 2, 3, 4, 0, 6, 7, 7, 0, 8, 8, 8, 0, 0, 0, 0, 0, 2, 3, 4, 0, 6, 7, 7, 0, 8, 8, 8]);
+    var HHContextLabel = new Uint8Array([0, 1, 2, 0, 1, 2, 2, 0, 2, 2, 2, 0, 0, 0, 0, 0, 3, 4, 5, 0, 4, 5, 5, 0, 5, 5, 5, 0, 0, 0, 0, 0, 6, 7, 7, 0, 7, 7, 7, 0, 7, 7, 7, 0, 0, 0, 0, 0, 8, 8, 8, 0, 8, 8, 8, 0, 8, 8, 8, 0, 0, 0, 0, 0, 8, 8, 8, 0, 8, 8, 8, 0, 8, 8, 8]);
+    function BitModel(width, height, subband, zeroBitPlanes, mb) {
+      this.width = width;
+      this.height = height;
+      this.contextLabelTable = subband === 'HH' ? HHContextLabel : subband === 'HL' ? HLContextLabel : LLAndLHContextsLabel;
+      var coefficientCount = width * height;
+      this.neighborsSignificance = new Uint8Array(coefficientCount);
+      this.coefficentsSign = new Uint8Array(coefficientCount);
+      this.coefficentsMagnitude = mb > 14 ? new Uint32Array(coefficientCount) : mb > 6 ? new Uint16Array(coefficientCount) : new Uint8Array(coefficientCount);
+      this.processingFlags = new Uint8Array(coefficientCount);
+      var bitsDecoded = new Uint8Array(coefficientCount);
+      if (zeroBitPlanes !== 0) {
+        for (var i = 0; i < coefficientCount; i++) {
+          bitsDecoded[i] = zeroBitPlanes;
+        }
       }
-     }
-    }
-   },
-   checkSegmentationSymbol: function BitModel_checkSegmentationSymbol() {
-    var decoder = this.decoder;
-    var contexts = this.contexts;
-    var symbol = decoder.readBit(contexts, UNIFORM_CONTEXT) << 3 | decoder.readBit(contexts, UNIFORM_CONTEXT) << 2 | decoder.readBit(contexts, UNIFORM_CONTEXT) << 1 | decoder.readBit(contexts, UNIFORM_CONTEXT);
-    if (symbol !== 0xA) {
-     error('JPX Error: Invalid segmentation symbol');
-    }
-   }
-  };
-  return BitModel;
- }();
- var Transform = function TransformClosure() {
-  function Transform() {
-  }
-  Transform.prototype.calculate = function transformCalculate(subbands, u0, v0) {
-   var ll = subbands[0];
-   for (var i = 1, ii = subbands.length; i < ii; i++) {
-    ll = this.iterate(ll, subbands[i], u0, v0);
-   }
-   return ll;
-  };
-  Transform.prototype.extend = function extend(buffer, offset, size) {
-   var i1 = offset - 1, j1 = offset + 1;
-   var i2 = offset + size - 2, j2 = offset + size;
-   buffer[i1--] = buffer[j1++];
-   buffer[j2++] = buffer[i2--];
-   buffer[i1--] = buffer[j1++];
-   buffer[j2++] = buffer[i2--];
-   buffer[i1--] = buffer[j1++];
-   buffer[j2++] = buffer[i2--];
-   buffer[i1] = buffer[j1];
-   buffer[j2] = buffer[i2];
-  };
-  Transform.prototype.iterate = function Transform_iterate(ll, hl_lh_hh, u0, v0) {
-   var llWidth = ll.width, llHeight = ll.height, llItems = ll.items;
-   var width = hl_lh_hh.width;
-   var height = hl_lh_hh.height;
-   var items = hl_lh_hh.items;
-   var i, j, k, l, u, v;
-   for (k = 0, i = 0; i < llHeight; i++) {
-    l = i * 2 * width;
-    for (j = 0; j < llWidth; j++, k++, l += 2) {
-     items[l] = llItems[k];
-    }
-   }
-   llItems = ll.items = null;
-   var bufferPadding = 4;
-   var rowBuffer = new Float32Array(width + 2 * bufferPadding);
-   if (width === 1) {
-    if ((u0 & 1) !== 0) {
-     for (v = 0, k = 0; v < height; v++, k += width) {
-      items[k] *= 0.5;
-     }
-    }
-   } else {
-    for (v = 0, k = 0; v < height; v++, k += width) {
-     rowBuffer.set(items.subarray(k, k + width), bufferPadding);
-     this.extend(rowBuffer, bufferPadding, width);
-     this.filter(rowBuffer, bufferPadding, width);
-     items.set(rowBuffer.subarray(bufferPadding, bufferPadding + width), k);
+      this.bitsDecoded = bitsDecoded;
+      this.reset();
     }
-   }
-   var numBuffers = 16;
-   var colBuffers = [];
-   for (i = 0; i < numBuffers; i++) {
-    colBuffers.push(new Float32Array(height + 2 * bufferPadding));
-   }
-   var b, currentBuffer = 0;
-   ll = bufferPadding + height;
-   if (height === 1) {
-    if ((v0 & 1) !== 0) {
-     for (u = 0; u < width; u++) {
-      items[u] *= 0.5;
-     }
-    }
-   } else {
-    for (u = 0; u < width; u++) {
-     if (currentBuffer === 0) {
-      numBuffers = Math.min(width - u, numBuffers);
-      for (k = u, l = bufferPadding; l < ll; k += width, l++) {
-       for (b = 0; b < numBuffers; b++) {
-        colBuffers[b][l] = items[k + b];
-       }
+    BitModel.prototype = {
+      setDecoder: function BitModel_setDecoder(decoder) {
+        this.decoder = decoder;
+      },
+      reset: function BitModel_reset() {
+        this.contexts = new Int8Array(19);
+        this.contexts[0] = 4 << 1 | 0;
+        this.contexts[UNIFORM_CONTEXT] = 46 << 1 | 0;
+        this.contexts[RUNLENGTH_CONTEXT] = 3 << 1 | 0;
+      },
+      setNeighborsSignificance: function BitModel_setNeighborsSignificance(row, column, index) {
+        var neighborsSignificance = this.neighborsSignificance;
+        var width = this.width,
+            height = this.height;
+        var left = column > 0;
+        var right = column + 1 < width;
+        var i;
+        if (row > 0) {
+          i = index - width;
+          if (left) {
+            neighborsSignificance[i - 1] += 0x10;
+          }
+          if (right) {
+            neighborsSignificance[i + 1] += 0x10;
+          }
+          neighborsSignificance[i] += 0x04;
+        }
+        if (row + 1 < height) {
+          i = index + width;
+          if (left) {
+            neighborsSignificance[i - 1] += 0x10;
+          }
+          if (right) {
+            neighborsSignificance[i + 1] += 0x10;
+          }
+          neighborsSignificance[i] += 0x04;
+        }
+        if (left) {
+          neighborsSignificance[index - 1] += 0x01;
+        }
+        if (right) {
+          neighborsSignificance[index + 1] += 0x01;
+        }
+        neighborsSignificance[index] |= 0x80;
+      },
+      runSignificancePropagationPass: function BitModel_runSignificancePropagationPass() {
+        var decoder = this.decoder;
+        var width = this.width,
+            height = this.height;
+        var coefficentsMagnitude = this.coefficentsMagnitude;
+        var coefficentsSign = this.coefficentsSign;
+        var neighborsSignificance = this.neighborsSignificance;
+        var processingFlags = this.processingFlags;
+        var contexts = this.contexts;
+        var labels = this.contextLabelTable;
+        var bitsDecoded = this.bitsDecoded;
+        var processedInverseMask = ~1;
+        var processedMask = 1;
+        var firstMagnitudeBitMask = 2;
+        for (var i0 = 0; i0 < height; i0 += 4) {
+          for (var j = 0; j < width; j++) {
+            var index = i0 * width + j;
+            for (var i1 = 0; i1 < 4; i1++, index += width) {
+              var i = i0 + i1;
+              if (i >= height) {
+                break;
+              }
+              processingFlags[index] &= processedInverseMask;
+              if (coefficentsMagnitude[index] || !neighborsSignificance[index]) {
+                continue;
+              }
+              var contextLabel = labels[neighborsSignificance[index]];
+              var decision = decoder.readBit(contexts, contextLabel);
+              if (decision) {
+                var sign = this.decodeSignBit(i, j, index);
+                coefficentsSign[index] = sign;
+                coefficentsMagnitude[index] = 1;
+                this.setNeighborsSignificance(i, j, index);
+                processingFlags[index] |= firstMagnitudeBitMask;
+              }
+              bitsDecoded[index]++;
+              processingFlags[index] |= processedMask;
+            }
+          }
+        }
+      },
+      decodeSignBit: function BitModel_decodeSignBit(row, column, index) {
+        var width = this.width,
+            height = this.height;
+        var coefficentsMagnitude = this.coefficentsMagnitude;
+        var coefficentsSign = this.coefficentsSign;
+        var contribution, sign0, sign1, significance1;
+        var contextLabel, decoded;
+        significance1 = column > 0 && coefficentsMagnitude[index - 1] !== 0;
+        if (column + 1 < width && coefficentsMagnitude[index + 1] !== 0) {
+          sign1 = coefficentsSign[index + 1];
+          if (significance1) {
+            sign0 = coefficentsSign[index - 1];
+            contribution = 1 - sign1 - sign0;
+          } else {
+            contribution = 1 - sign1 - sign1;
+          }
+        } else if (significance1) {
+          sign0 = coefficentsSign[index - 1];
+          contribution = 1 - sign0 - sign0;
+        } else {
+          contribution = 0;
+        }
+        var horizontalContribution = 3 * contribution;
+        significance1 = row > 0 && coefficentsMagnitude[index - width] !== 0;
+        if (row + 1 < height && coefficentsMagnitude[index + width] !== 0) {
+          sign1 = coefficentsSign[index + width];
+          if (significance1) {
+            sign0 = coefficentsSign[index - width];
+            contribution = 1 - sign1 - sign0 + horizontalContribution;
+          } else {
+            contribution = 1 - sign1 - sign1 + horizontalContribution;
+          }
+        } else if (significance1) {
+          sign0 = coefficentsSign[index - width];
+          contribution = 1 - sign0 - sign0 + horizontalContribution;
+        } else {
+          contribution = horizontalContribution;
+        }
+        if (contribution >= 0) {
+          contextLabel = 9 + contribution;
+          decoded = this.decoder.readBit(this.contexts, contextLabel);
+        } else {
+          contextLabel = 9 - contribution;
+          decoded = this.decoder.readBit(this.contexts, contextLabel) ^ 1;
+        }
+        return decoded;
+      },
+      runMagnitudeRefinementPass: function BitModel_runMagnitudeRefinementPass() {
+        var decoder = this.decoder;
+        var width = this.width,
+            height = this.height;
+        var coefficentsMagnitude = this.coefficentsMagnitude;
+        var neighborsSignificance = this.neighborsSignificance;
+        var contexts = this.contexts;
+        var bitsDecoded = this.bitsDecoded;
+        var processingFlags = this.processingFlags;
+        var processedMask = 1;
+        var firstMagnitudeBitMask = 2;
+        var length = width * height;
+        var width4 = width * 4;
+        for (var index0 = 0, indexNext; index0 < length; index0 = indexNext) {
+          indexNext = Math.min(length, index0 + width4);
+          for (var j = 0; j < width; j++) {
+            for (var index = index0 + j; index < indexNext; index += width) {
+              if (!coefficentsMagnitude[index] || (processingFlags[index] & processedMask) !== 0) {
+                continue;
+              }
+              var contextLabel = 16;
+              if ((processingFlags[index] & firstMagnitudeBitMask) !== 0) {
+                processingFlags[index] ^= firstMagnitudeBitMask;
+                var significance = neighborsSignificance[index] & 127;
+                contextLabel = significance === 0 ? 15 : 14;
+              }
+              var bit = decoder.readBit(contexts, contextLabel);
+              coefficentsMagnitude[index] = coefficentsMagnitude[index] << 1 | bit;
+              bitsDecoded[index]++;
+              processingFlags[index] |= processedMask;
+            }
+          }
+        }
+      },
+      runCleanupPass: function BitModel_runCleanupPass() {
+        var decoder = this.decoder;
+        var width = this.width,
+            height = this.height;
+        var neighborsSignificance = this.neighborsSignificance;
+        var coefficentsMagnitude = this.coefficentsMagnitude;
+        var coefficentsSign = this.coefficentsSign;
+        var contexts = this.contexts;
+        var labels = this.contextLabelTable;
+        var bitsDecoded = this.bitsDecoded;
+        var processingFlags = this.processingFlags;
+        var processedMask = 1;
+        var firstMagnitudeBitMask = 2;
+        var oneRowDown = width;
+        var twoRowsDown = width * 2;
+        var threeRowsDown = width * 3;
+        var iNext;
+        for (var i0 = 0; i0 < height; i0 = iNext) {
+          iNext = Math.min(i0 + 4, height);
+          var indexBase = i0 * width;
+          var checkAllEmpty = i0 + 3 < height;
+          for (var j = 0; j < width; j++) {
+            var index0 = indexBase + j;
+            var allEmpty = checkAllEmpty && processingFlags[index0] === 0 && processingFlags[index0 + oneRowDown] === 0 && processingFlags[index0 + twoRowsDown] === 0 && processingFlags[index0 + threeRowsDown] === 0 && neighborsSignificance[index0] === 0 && neighborsSignificance[index0 + oneRowDown] === 0 && neighborsSignificance[index0 + twoRowsDown] === 0 && neighborsSignificance[index0 + threeRowsDown] === 0;
+            var i1 = 0,
+                index = index0;
+            var i = i0,
+                sign;
+            if (allEmpty) {
+              var hasSignificantCoefficent = decoder.readBit(contexts, RUNLENGTH_CONTEXT);
+              if (!hasSignificantCoefficent) {
+                bitsDecoded[index0]++;
+                bitsDecoded[index0 + oneRowDown]++;
+                bitsDecoded[index0 + twoRowsDown]++;
+                bitsDecoded[index0 + threeRowsDown]++;
+                continue;
+              }
+              i1 = decoder.readBit(contexts, UNIFORM_CONTEXT) << 1 | decoder.readBit(contexts, UNIFORM_CONTEXT);
+              if (i1 !== 0) {
+                i = i0 + i1;
+                index += i1 * width;
+              }
+              sign = this.decodeSignBit(i, j, index);
+              coefficentsSign[index] = sign;
+              coefficentsMagnitude[index] = 1;
+              this.setNeighborsSignificance(i, j, index);
+              processingFlags[index] |= firstMagnitudeBitMask;
+              index = index0;
+              for (var i2 = i0; i2 <= i; i2++, index += width) {
+                bitsDecoded[index]++;
+              }
+              i1++;
+            }
+            for (i = i0 + i1; i < iNext; i++, index += width) {
+              if (coefficentsMagnitude[index] || (processingFlags[index] & processedMask) !== 0) {
+                continue;
+              }
+              var contextLabel = labels[neighborsSignificance[index]];
+              var decision = decoder.readBit(contexts, contextLabel);
+              if (decision === 1) {
+                sign = this.decodeSignBit(i, j, index);
+                coefficentsSign[index] = sign;
+                coefficentsMagnitude[index] = 1;
+                this.setNeighborsSignificance(i, j, index);
+                processingFlags[index] |= firstMagnitudeBitMask;
+              }
+              bitsDecoded[index]++;
+            }
+          }
+        }
+      },
+      checkSegmentationSymbol: function BitModel_checkSegmentationSymbol() {
+        var decoder = this.decoder;
+        var contexts = this.contexts;
+        var symbol = decoder.readBit(contexts, UNIFORM_CONTEXT) << 3 | decoder.readBit(contexts, UNIFORM_CONTEXT) << 2 | decoder.readBit(contexts, UNIFORM_CONTEXT) << 1 | decoder.readBit(contexts, UNIFORM_CONTEXT);
+        if (symbol !== 0xA) {
+          error('JPX Error: Invalid segmentation symbol');
+        }
       }
-      currentBuffer = numBuffers;
-     }
-     currentBuffer--;
-     var buffer = colBuffers[currentBuffer];
-     this.extend(buffer, bufferPadding, height);
-     this.filter(buffer, bufferPadding, height);
-     if (currentBuffer === 0) {
-      k = u - numBuffers + 1;
-      for (l = bufferPadding; l < ll; k += width, l++) {
-       for (b = 0; b < numBuffers; b++) {
-        items[k + b] = colBuffers[b][l];
-       }
+    };
+    return BitModel;
+  }();
+  var Transform = function TransformClosure() {
+    function Transform() {}
+    Transform.prototype.calculate = function transformCalculate(subbands, u0, v0) {
+      var ll = subbands[0];
+      for (var i = 1, ii = subbands.length; i < ii; i++) {
+        ll = this.iterate(ll, subbands[i], u0, v0);
       }
-     }
-    }
-   }
-   return {
-    width: width,
-    height: height,
-    items: items
-   };
-  };
-  return Transform;
- }();
- var IrreversibleTransform = function IrreversibleTransformClosure() {
-  function IrreversibleTransform() {
-   Transform.call(this);
-  }
-  IrreversibleTransform.prototype = Object.create(Transform.prototype);
-  IrreversibleTransform.prototype.filter = function irreversibleTransformFilter(x, offset, length) {
-   var len = length >> 1;
-   offset = offset | 0;
-   var j, n, current, next;
-   var alpha = -1.586134342059924;
-   var beta = -0.052980118572961;
-   var gamma = 0.882911075530934;
-   var delta = 0.443506852043971;
-   var K = 1.230174104914001;
-   var K_ = 1 / K;
-   j = offset - 3;
-   for (n = len + 4; n--; j += 2) {
-    x[j] *= K_;
-   }
-   j = offset - 2;
-   current = delta * x[j - 1];
-   for (n = len + 3; n--; j += 2) {
-    next = delta * x[j + 1];
-    x[j] = K * x[j] - current - next;
-    if (n--) {
-     j += 2;
-     current = delta * x[j + 1];
-     x[j] = K * x[j] - current - next;
-    } else {
-     break;
-    }
-   }
-   j = offset - 1;
-   current = gamma * x[j - 1];
-   for (n = len + 2; n--; j += 2) {
-    next = gamma * x[j + 1];
-    x[j] -= current + next;
-    if (n--) {
-     j += 2;
-     current = gamma * x[j + 1];
-     x[j] -= current + next;
-    } else {
-     break;
-    }
-   }
-   j = offset;
-   current = beta * x[j - 1];
-   for (n = len + 1; n--; j += 2) {
-    next = beta * x[j + 1];
-    x[j] -= current + next;
-    if (n--) {
-     j += 2;
-     current = beta * x[j + 1];
-     x[j] -= current + next;
-    } else {
-     break;
+      return ll;
+    };
+    Transform.prototype.extend = function extend(buffer, offset, size) {
+      var i1 = offset - 1,
+          j1 = offset + 1;
+      var i2 = offset + size - 2,
+          j2 = offset + size;
+      buffer[i1--] = buffer[j1++];
+      buffer[j2++] = buffer[i2--];
+      buffer[i1--] = buffer[j1++];
+      buffer[j2++] = buffer[i2--];
+      buffer[i1--] = buffer[j1++];
+      buffer[j2++] = buffer[i2--];
+      buffer[i1] = buffer[j1];
+      buffer[j2] = buffer[i2];
+    };
+    Transform.prototype.iterate = function Transform_iterate(ll, hl_lh_hh, u0, v0) {
+      var llWidth = ll.width,
+          llHeight = ll.height,
+          llItems = ll.items;
+      var width = hl_lh_hh.width;
+      var height = hl_lh_hh.height;
+      var items = hl_lh_hh.items;
+      var i, j, k, l, u, v;
+      for (k = 0, i = 0; i < llHeight; i++) {
+        l = i * 2 * width;
+        for (j = 0; j < llWidth; j++, k++, l += 2) {
+          items[l] = llItems[k];
+        }
+      }
+      llItems = ll.items = null;
+      var bufferPadding = 4;
+      var rowBuffer = new Float32Array(width + 2 * bufferPadding);
+      if (width === 1) {
+        if ((u0 & 1) !== 0) {
+          for (v = 0, k = 0; v < height; v++, k += width) {
+            items[k] *= 0.5;
+          }
+        }
+      } else {
+        for (v = 0, k = 0; v < height; v++, k += width) {
+          rowBuffer.set(items.subarray(k, k + width), bufferPadding);
+          this.extend(rowBuffer, bufferPadding, width);
+          this.filter(rowBuffer, bufferPadding, width);
+          items.set(rowBuffer.subarray(bufferPadding, bufferPadding + width), k);
+        }
+      }
+      var numBuffers = 16;
+      var colBuffers = [];
+      for (i = 0; i < numBuffers; i++) {
+        colBuffers.push(new Float32Array(height + 2 * bufferPadding));
+      }
+      var b,
+          currentBuffer = 0;
+      ll = bufferPadding + height;
+      if (height === 1) {
+        if ((v0 & 1) !== 0) {
+          for (u = 0; u < width; u++) {
+            items[u] *= 0.5;
+          }
+        }
+      } else {
+        for (u = 0; u < width; u++) {
+          if (currentBuffer === 0) {
+            numBuffers = Math.min(width - u, numBuffers);
+            for (k = u, l = bufferPadding; l < ll; k += width, l++) {
+              for (b = 0; b < numBuffers; b++) {
+                colBuffers[b][l] = items[k + b];
+              }
+            }
+            currentBuffer = numBuffers;
+          }
+          currentBuffer--;
+          var buffer = colBuffers[currentBuffer];
+          this.extend(buffer, bufferPadding, height);
+          this.filter(buffer, bufferPadding, height);
+          if (currentBuffer === 0) {
+            k = u - numBuffers + 1;
+            for (l = bufferPadding; l < ll; k += width, l++) {
+              for (b = 0; b < numBuffers; b++) {
+                items[k + b] = colBuffers[b][l];
+              }
+            }
+          }
+        }
+      }
+      return {
+        width: width,
+        height: height,
+        items: items
+      };
+    };
+    return Transform;
+  }();
+  var IrreversibleTransform = function IrreversibleTransformClosure() {
+    function IrreversibleTransform() {
+      Transform.call(this);
     }
-   }
-   if (len !== 0) {
-    j = offset + 1;
-    current = alpha * x[j - 1];
-    for (n = len; n--; j += 2) {
-     next = alpha * x[j + 1];
-     x[j] -= current + next;
-     if (n--) {
-      j += 2;
-      current = alpha * x[j + 1];
-      x[j] -= current + next;
-     } else {
-      break;
-     }
+    IrreversibleTransform.prototype = Object.create(Transform.prototype);
+    IrreversibleTransform.prototype.filter = function irreversibleTransformFilter(x, offset, length) {
+      var len = length >> 1;
+      offset = offset | 0;
+      var j, n, current, next;
+      var alpha = -1.586134342059924;
+      var beta = -0.052980118572961;
+      var gamma = 0.882911075530934;
+      var delta = 0.443506852043971;
+      var K = 1.230174104914001;
+      var K_ = 1 / K;
+      j = offset - 3;
+      for (n = len + 4; n--; j += 2) {
+        x[j] *= K_;
+      }
+      j = offset - 2;
+      current = delta * x[j - 1];
+      for (n = len + 3; n--; j += 2) {
+        next = delta * x[j + 1];
+        x[j] = K * x[j] - current - next;
+        if (n--) {
+          j += 2;
+          current = delta * x[j + 1];
+          x[j] = K * x[j] - current - next;
+        } else {
+          break;
+        }
+      }
+      j = offset - 1;
+      current = gamma * x[j - 1];
+      for (n = len + 2; n--; j += 2) {
+        next = gamma * x[j + 1];
+        x[j] -= current + next;
+        if (n--) {
+          j += 2;
+          current = gamma * x[j + 1];
+          x[j] -= current + next;
+        } else {
+          break;
+        }
+      }
+      j = offset;
+      current = beta * x[j - 1];
+      for (n = len + 1; n--; j += 2) {
+        next = beta * x[j + 1];
+        x[j] -= current + next;
+        if (n--) {
+          j += 2;
+          current = beta * x[j + 1];
+          x[j] -= current + next;
+        } else {
+          break;
+        }
+      }
+      if (len !== 0) {
+        j = offset + 1;
+        current = alpha * x[j - 1];
+        for (n = len; n--; j += 2) {
+          next = alpha * x[j + 1];
+          x[j] -= current + next;
+          if (n--) {
+            j += 2;
+            current = alpha * x[j + 1];
+            x[j] -= current + next;
+          } else {
+            break;
+          }
+        }
+      }
+    };
+    return IrreversibleTransform;
+  }();
+  var ReversibleTransform = function ReversibleTransformClosure() {
+    function ReversibleTransform() {
+      Transform.call(this);
     }
-   }
-  };
-  return IrreversibleTransform;
- }();
- var ReversibleTransform = function ReversibleTransformClosure() {
-  function ReversibleTransform() {
-   Transform.call(this);
-  }
-  ReversibleTransform.prototype = Object.create(Transform.prototype);
-  ReversibleTransform.prototype.filter = function reversibleTransformFilter(x, offset, length) {
-   var len = length >> 1;
-   offset = offset | 0;
-   var j, n;
-   for (j = offset, n = len + 1; n--; j += 2) {
-    x[j] -= x[j - 1] + x[j + 1] + 2 >> 2;
-   }
-   for (j = offset + 1, n = len; n--; j += 2) {
-    x[j] += x[j - 1] + x[j + 1] >> 1;
-   }
-  };
-  return ReversibleTransform;
- }();
- return JpxImage;
+    ReversibleTransform.prototype = Object.create(Transform.prototype);
+    ReversibleTransform.prototype.filter = function reversibleTransformFilter(x, offset, length) {
+      var len = length >> 1;
+      offset = offset | 0;
+      var j, n;
+      for (j = offset, n = len + 1; n--; j += 2) {
+        x[j] -= x[j - 1] + x[j + 1] + 2 >> 2;
+      }
+      for (j = offset + 1, n = len; n--; j += 2) {
+        x[j] += x[j - 1] + x[j + 1] >> 1;
+      }
+    };
+    return ReversibleTransform;
+  }();
+  return JpxImage;
 }();
 exports.JpxImage = JpxImage;

+ 2937 - 2936
lib/core/metrics.js

@@ -13,2944 +13,2945 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var getLookupTableFactory = sharedUtil.getLookupTableFactory;
 var getMetrics = getLookupTableFactory(function (t) {
- t['Courier'] = 600;
- t['Courier-Bold'] = 600;
- t['Courier-BoldOblique'] = 600;
- t['Courier-Oblique'] = 600;
- t['Helvetica'] = getLookupTableFactory(function (t) {
-  t['space'] = 278;
-  t['exclam'] = 278;
-  t['quotedbl'] = 355;
-  t['numbersign'] = 556;
-  t['dollar'] = 556;
-  t['percent'] = 889;
-  t['ampersand'] = 667;
-  t['quoteright'] = 222;
-  t['parenleft'] = 333;
-  t['parenright'] = 333;
-  t['asterisk'] = 389;
-  t['plus'] = 584;
-  t['comma'] = 278;
-  t['hyphen'] = 333;
-  t['period'] = 278;
-  t['slash'] = 278;
-  t['zero'] = 556;
-  t['one'] = 556;
-  t['two'] = 556;
-  t['three'] = 556;
-  t['four'] = 556;
-  t['five'] = 556;
-  t['six'] = 556;
-  t['seven'] = 556;
-  t['eight'] = 556;
-  t['nine'] = 556;
-  t['colon'] = 278;
-  t['semicolon'] = 278;
-  t['less'] = 584;
-  t['equal'] = 584;
-  t['greater'] = 584;
-  t['question'] = 556;
-  t['at'] = 1015;
-  t['A'] = 667;
-  t['B'] = 667;
-  t['C'] = 722;
-  t['D'] = 722;
-  t['E'] = 667;
-  t['F'] = 611;
-  t['G'] = 778;
-  t['H'] = 722;
-  t['I'] = 278;
-  t['J'] = 500;
-  t['K'] = 667;
-  t['L'] = 556;
-  t['M'] = 833;
-  t['N'] = 722;
-  t['O'] = 778;
-  t['P'] = 667;
-  t['Q'] = 778;
-  t['R'] = 722;
-  t['S'] = 667;
-  t['T'] = 611;
-  t['U'] = 722;
-  t['V'] = 667;
-  t['W'] = 944;
-  t['X'] = 667;
-  t['Y'] = 667;
-  t['Z'] = 611;
-  t['bracketleft'] = 278;
-  t['backslash'] = 278;
-  t['bracketright'] = 278;
-  t['asciicircum'] = 469;
-  t['underscore'] = 556;
-  t['quoteleft'] = 222;
-  t['a'] = 556;
-  t['b'] = 556;
-  t['c'] = 500;
-  t['d'] = 556;
-  t['e'] = 556;
-  t['f'] = 278;
-  t['g'] = 556;
-  t['h'] = 556;
-  t['i'] = 222;
-  t['j'] = 222;
-  t['k'] = 500;
-  t['l'] = 222;
-  t['m'] = 833;
-  t['n'] = 556;
-  t['o'] = 556;
-  t['p'] = 556;
-  t['q'] = 556;
-  t['r'] = 333;
-  t['s'] = 500;
-  t['t'] = 278;
-  t['u'] = 556;
-  t['v'] = 500;
-  t['w'] = 722;
-  t['x'] = 500;
-  t['y'] = 500;
-  t['z'] = 500;
-  t['braceleft'] = 334;
-  t['bar'] = 260;
-  t['braceright'] = 334;
-  t['asciitilde'] = 584;
-  t['exclamdown'] = 333;
-  t['cent'] = 556;
-  t['sterling'] = 556;
-  t['fraction'] = 167;
-  t['yen'] = 556;
-  t['florin'] = 556;
-  t['section'] = 556;
-  t['currency'] = 556;
-  t['quotesingle'] = 191;
-  t['quotedblleft'] = 333;
-  t['guillemotleft'] = 556;
-  t['guilsinglleft'] = 333;
-  t['guilsinglright'] = 333;
-  t['fi'] = 500;
-  t['fl'] = 500;
-  t['endash'] = 556;
-  t['dagger'] = 556;
-  t['daggerdbl'] = 556;
-  t['periodcentered'] = 278;
-  t['paragraph'] = 537;
-  t['bullet'] = 350;
-  t['quotesinglbase'] = 222;
-  t['quotedblbase'] = 333;
-  t['quotedblright'] = 333;
-  t['guillemotright'] = 556;
-  t['ellipsis'] = 1000;
-  t['perthousand'] = 1000;
-  t['questiondown'] = 611;
-  t['grave'] = 333;
-  t['acute'] = 333;
-  t['circumflex'] = 333;
-  t['tilde'] = 333;
-  t['macron'] = 333;
-  t['breve'] = 333;
-  t['dotaccent'] = 333;
-  t['dieresis'] = 333;
-  t['ring'] = 333;
-  t['cedilla'] = 333;
-  t['hungarumlaut'] = 333;
-  t['ogonek'] = 333;
-  t['caron'] = 333;
-  t['emdash'] = 1000;
-  t['AE'] = 1000;
-  t['ordfeminine'] = 370;
-  t['Lslash'] = 556;
-  t['Oslash'] = 778;
-  t['OE'] = 1000;
-  t['ordmasculine'] = 365;
-  t['ae'] = 889;
-  t['dotlessi'] = 278;
-  t['lslash'] = 222;
-  t['oslash'] = 611;
-  t['oe'] = 944;
-  t['germandbls'] = 611;
-  t['Idieresis'] = 278;
-  t['eacute'] = 556;
-  t['abreve'] = 556;
-  t['uhungarumlaut'] = 556;
-  t['ecaron'] = 556;
-  t['Ydieresis'] = 667;
-  t['divide'] = 584;
-  t['Yacute'] = 667;
-  t['Acircumflex'] = 667;
-  t['aacute'] = 556;
-  t['Ucircumflex'] = 722;
-  t['yacute'] = 500;
-  t['scommaaccent'] = 500;
-  t['ecircumflex'] = 556;
-  t['Uring'] = 722;
-  t['Udieresis'] = 722;
-  t['aogonek'] = 556;
-  t['Uacute'] = 722;
-  t['uogonek'] = 556;
-  t['Edieresis'] = 667;
-  t['Dcroat'] = 722;
-  t['commaaccent'] = 250;
-  t['copyright'] = 737;
-  t['Emacron'] = 667;
-  t['ccaron'] = 500;
-  t['aring'] = 556;
-  t['Ncommaaccent'] = 722;
-  t['lacute'] = 222;
-  t['agrave'] = 556;
-  t['Tcommaaccent'] = 611;
-  t['Cacute'] = 722;
-  t['atilde'] = 556;
-  t['Edotaccent'] = 667;
-  t['scaron'] = 500;
-  t['scedilla'] = 500;
-  t['iacute'] = 278;
-  t['lozenge'] = 471;
-  t['Rcaron'] = 722;
-  t['Gcommaaccent'] = 778;
-  t['ucircumflex'] = 556;
-  t['acircumflex'] = 556;
-  t['Amacron'] = 667;
-  t['rcaron'] = 333;
-  t['ccedilla'] = 500;
-  t['Zdotaccent'] = 611;
-  t['Thorn'] = 667;
-  t['Omacron'] = 778;
-  t['Racute'] = 722;
-  t['Sacute'] = 667;
-  t['dcaron'] = 643;
-  t['Umacron'] = 722;
-  t['uring'] = 556;
-  t['threesuperior'] = 333;
-  t['Ograve'] = 778;
-  t['Agrave'] = 667;
-  t['Abreve'] = 667;
-  t['multiply'] = 584;
-  t['uacute'] = 556;
-  t['Tcaron'] = 611;
-  t['partialdiff'] = 476;
-  t['ydieresis'] = 500;
-  t['Nacute'] = 722;
-  t['icircumflex'] = 278;
-  t['Ecircumflex'] = 667;
-  t['adieresis'] = 556;
-  t['edieresis'] = 556;
-  t['cacute'] = 500;
-  t['nacute'] = 556;
-  t['umacron'] = 556;
-  t['Ncaron'] = 722;
-  t['Iacute'] = 278;
-  t['plusminus'] = 584;
-  t['brokenbar'] = 260;
-  t['registered'] = 737;
-  t['Gbreve'] = 778;
-  t['Idotaccent'] = 278;
-  t['summation'] = 600;
-  t['Egrave'] = 667;
-  t['racute'] = 333;
-  t['omacron'] = 556;
-  t['Zacute'] = 611;
-  t['Zcaron'] = 611;
-  t['greaterequal'] = 549;
-  t['Eth'] = 722;
-  t['Ccedilla'] = 722;
-  t['lcommaaccent'] = 222;
-  t['tcaron'] = 317;
-  t['eogonek'] = 556;
-  t['Uogonek'] = 722;
-  t['Aacute'] = 667;
-  t['Adieresis'] = 667;
-  t['egrave'] = 556;
-  t['zacute'] = 500;
-  t['iogonek'] = 222;
-  t['Oacute'] = 778;
-  t['oacute'] = 556;
-  t['amacron'] = 556;
-  t['sacute'] = 500;
-  t['idieresis'] = 278;
-  t['Ocircumflex'] = 778;
-  t['Ugrave'] = 722;
-  t['Delta'] = 612;
-  t['thorn'] = 556;
-  t['twosuperior'] = 333;
-  t['Odieresis'] = 778;
-  t['mu'] = 556;
-  t['igrave'] = 278;
-  t['ohungarumlaut'] = 556;
-  t['Eogonek'] = 667;
-  t['dcroat'] = 556;
-  t['threequarters'] = 834;
-  t['Scedilla'] = 667;
-  t['lcaron'] = 299;
-  t['Kcommaaccent'] = 667;
-  t['Lacute'] = 556;
-  t['trademark'] = 1000;
-  t['edotaccent'] = 556;
-  t['Igrave'] = 278;
-  t['Imacron'] = 278;
-  t['Lcaron'] = 556;
-  t['onehalf'] = 834;
-  t['lessequal'] = 549;
-  t['ocircumflex'] = 556;
-  t['ntilde'] = 556;
-  t['Uhungarumlaut'] = 722;
-  t['Eacute'] = 667;
-  t['emacron'] = 556;
-  t['gbreve'] = 556;
-  t['onequarter'] = 834;
-  t['Scaron'] = 667;
-  t['Scommaaccent'] = 667;
-  t['Ohungarumlaut'] = 778;
-  t['degree'] = 400;
-  t['ograve'] = 556;
-  t['Ccaron'] = 722;
-  t['ugrave'] = 556;
-  t['radical'] = 453;
-  t['Dcaron'] = 722;
-  t['rcommaaccent'] = 333;
-  t['Ntilde'] = 722;
-  t['otilde'] = 556;
-  t['Rcommaaccent'] = 722;
-  t['Lcommaaccent'] = 556;
-  t['Atilde'] = 667;
-  t['Aogonek'] = 667;
-  t['Aring'] = 667;
-  t['Otilde'] = 778;
-  t['zdotaccent'] = 500;
-  t['Ecaron'] = 667;
-  t['Iogonek'] = 278;
-  t['kcommaaccent'] = 500;
-  t['minus'] = 584;
-  t['Icircumflex'] = 278;
-  t['ncaron'] = 556;
-  t['tcommaaccent'] = 278;
-  t['logicalnot'] = 584;
-  t['odieresis'] = 556;
-  t['udieresis'] = 556;
-  t['notequal'] = 549;
-  t['gcommaaccent'] = 556;
-  t['eth'] = 556;
-  t['zcaron'] = 500;
-  t['ncommaaccent'] = 556;
-  t['onesuperior'] = 333;
-  t['imacron'] = 278;
-  t['Euro'] = 556;
- });
- t['Helvetica-Bold'] = getLookupTableFactory(function (t) {
-  t['space'] = 278;
-  t['exclam'] = 333;
-  t['quotedbl'] = 474;
-  t['numbersign'] = 556;
-  t['dollar'] = 556;
-  t['percent'] = 889;
-  t['ampersand'] = 722;
-  t['quoteright'] = 278;
-  t['parenleft'] = 333;
-  t['parenright'] = 333;
-  t['asterisk'] = 389;
-  t['plus'] = 584;
-  t['comma'] = 278;
-  t['hyphen'] = 333;
-  t['period'] = 278;
-  t['slash'] = 278;
-  t['zero'] = 556;
-  t['one'] = 556;
-  t['two'] = 556;
-  t['three'] = 556;
-  t['four'] = 556;
-  t['five'] = 556;
-  t['six'] = 556;
-  t['seven'] = 556;
-  t['eight'] = 556;
-  t['nine'] = 556;
-  t['colon'] = 333;
-  t['semicolon'] = 333;
-  t['less'] = 584;
-  t['equal'] = 584;
-  t['greater'] = 584;
-  t['question'] = 611;
-  t['at'] = 975;
-  t['A'] = 722;
-  t['B'] = 722;
-  t['C'] = 722;
-  t['D'] = 722;
-  t['E'] = 667;
-  t['F'] = 611;
-  t['G'] = 778;
-  t['H'] = 722;
-  t['I'] = 278;
-  t['J'] = 556;
-  t['K'] = 722;
-  t['L'] = 611;
-  t['M'] = 833;
-  t['N'] = 722;
-  t['O'] = 778;
-  t['P'] = 667;
-  t['Q'] = 778;
-  t['R'] = 722;
-  t['S'] = 667;
-  t['T'] = 611;
-  t['U'] = 722;
-  t['V'] = 667;
-  t['W'] = 944;
-  t['X'] = 667;
-  t['Y'] = 667;
-  t['Z'] = 611;
-  t['bracketleft'] = 333;
-  t['backslash'] = 278;
-  t['bracketright'] = 333;
-  t['asciicircum'] = 584;
-  t['underscore'] = 556;
-  t['quoteleft'] = 278;
-  t['a'] = 556;
-  t['b'] = 611;
-  t['c'] = 556;
-  t['d'] = 611;
-  t['e'] = 556;
-  t['f'] = 333;
-  t['g'] = 611;
-  t['h'] = 611;
-  t['i'] = 278;
-  t['j'] = 278;
-  t['k'] = 556;
-  t['l'] = 278;
-  t['m'] = 889;
-  t['n'] = 611;
-  t['o'] = 611;
-  t['p'] = 611;
-  t['q'] = 611;
-  t['r'] = 389;
-  t['s'] = 556;
-  t['t'] = 333;
-  t['u'] = 611;
-  t['v'] = 556;
-  t['w'] = 778;
-  t['x'] = 556;
-  t['y'] = 556;
-  t['z'] = 500;
-  t['braceleft'] = 389;
-  t['bar'] = 280;
-  t['braceright'] = 389;
-  t['asciitilde'] = 584;
-  t['exclamdown'] = 333;
-  t['cent'] = 556;
-  t['sterling'] = 556;
-  t['fraction'] = 167;
-  t['yen'] = 556;
-  t['florin'] = 556;
-  t['section'] = 556;
-  t['currency'] = 556;
-  t['quotesingle'] = 238;
-  t['quotedblleft'] = 500;
-  t['guillemotleft'] = 556;
-  t['guilsinglleft'] = 333;
-  t['guilsinglright'] = 333;
-  t['fi'] = 611;
-  t['fl'] = 611;
-  t['endash'] = 556;
-  t['dagger'] = 556;
-  t['daggerdbl'] = 556;
-  t['periodcentered'] = 278;
-  t['paragraph'] = 556;
-  t['bullet'] = 350;
-  t['quotesinglbase'] = 278;
-  t['quotedblbase'] = 500;
-  t['quotedblright'] = 500;
-  t['guillemotright'] = 556;
-  t['ellipsis'] = 1000;
-  t['perthousand'] = 1000;
-  t['questiondown'] = 611;
-  t['grave'] = 333;
-  t['acute'] = 333;
-  t['circumflex'] = 333;
-  t['tilde'] = 333;
-  t['macron'] = 333;
-  t['breve'] = 333;
-  t['dotaccent'] = 333;
-  t['dieresis'] = 333;
-  t['ring'] = 333;
-  t['cedilla'] = 333;
-  t['hungarumlaut'] = 333;
-  t['ogonek'] = 333;
-  t['caron'] = 333;
-  t['emdash'] = 1000;
-  t['AE'] = 1000;
-  t['ordfeminine'] = 370;
-  t['Lslash'] = 611;
-  t['Oslash'] = 778;
-  t['OE'] = 1000;
-  t['ordmasculine'] = 365;
-  t['ae'] = 889;
-  t['dotlessi'] = 278;
-  t['lslash'] = 278;
-  t['oslash'] = 611;
-  t['oe'] = 944;
-  t['germandbls'] = 611;
-  t['Idieresis'] = 278;
-  t['eacute'] = 556;
-  t['abreve'] = 556;
-  t['uhungarumlaut'] = 611;
-  t['ecaron'] = 556;
-  t['Ydieresis'] = 667;
-  t['divide'] = 584;
-  t['Yacute'] = 667;
-  t['Acircumflex'] = 722;
-  t['aacute'] = 556;
-  t['Ucircumflex'] = 722;
-  t['yacute'] = 556;
-  t['scommaaccent'] = 556;
-  t['ecircumflex'] = 556;
-  t['Uring'] = 722;
-  t['Udieresis'] = 722;
-  t['aogonek'] = 556;
-  t['Uacute'] = 722;
-  t['uogonek'] = 611;
-  t['Edieresis'] = 667;
-  t['Dcroat'] = 722;
-  t['commaaccent'] = 250;
-  t['copyright'] = 737;
-  t['Emacron'] = 667;
-  t['ccaron'] = 556;
-  t['aring'] = 556;
-  t['Ncommaaccent'] = 722;
-  t['lacute'] = 278;
-  t['agrave'] = 556;
-  t['Tcommaaccent'] = 611;
-  t['Cacute'] = 722;
-  t['atilde'] = 556;
-  t['Edotaccent'] = 667;
-  t['scaron'] = 556;
-  t['scedilla'] = 556;
-  t['iacute'] = 278;
-  t['lozenge'] = 494;
-  t['Rcaron'] = 722;
-  t['Gcommaaccent'] = 778;
-  t['ucircumflex'] = 611;
-  t['acircumflex'] = 556;
-  t['Amacron'] = 722;
-  t['rcaron'] = 389;
-  t['ccedilla'] = 556;
-  t['Zdotaccent'] = 611;
-  t['Thorn'] = 667;
-  t['Omacron'] = 778;
-  t['Racute'] = 722;
-  t['Sacute'] = 667;
-  t['dcaron'] = 743;
-  t['Umacron'] = 722;
-  t['uring'] = 611;
-  t['threesuperior'] = 333;
-  t['Ograve'] = 778;
-  t['Agrave'] = 722;
-  t['Abreve'] = 722;
-  t['multiply'] = 584;
-  t['uacute'] = 611;
-  t['Tcaron'] = 611;
-  t['partialdiff'] = 494;
-  t['ydieresis'] = 556;
-  t['Nacute'] = 722;
-  t['icircumflex'] = 278;
-  t['Ecircumflex'] = 667;
-  t['adieresis'] = 556;
-  t['edieresis'] = 556;
-  t['cacute'] = 556;
-  t['nacute'] = 611;
-  t['umacron'] = 611;
-  t['Ncaron'] = 722;
-  t['Iacute'] = 278;
-  t['plusminus'] = 584;
-  t['brokenbar'] = 280;
-  t['registered'] = 737;
-  t['Gbreve'] = 778;
-  t['Idotaccent'] = 278;
-  t['summation'] = 600;
-  t['Egrave'] = 667;
-  t['racute'] = 389;
-  t['omacron'] = 611;
-  t['Zacute'] = 611;
-  t['Zcaron'] = 611;
-  t['greaterequal'] = 549;
-  t['Eth'] = 722;
-  t['Ccedilla'] = 722;
-  t['lcommaaccent'] = 278;
-  t['tcaron'] = 389;
-  t['eogonek'] = 556;
-  t['Uogonek'] = 722;
-  t['Aacute'] = 722;
-  t['Adieresis'] = 722;
-  t['egrave'] = 556;
-  t['zacute'] = 500;
-  t['iogonek'] = 278;
-  t['Oacute'] = 778;
-  t['oacute'] = 611;
-  t['amacron'] = 556;
-  t['sacute'] = 556;
-  t['idieresis'] = 278;
-  t['Ocircumflex'] = 778;
-  t['Ugrave'] = 722;
-  t['Delta'] = 612;
-  t['thorn'] = 611;
-  t['twosuperior'] = 333;
-  t['Odieresis'] = 778;
-  t['mu'] = 611;
-  t['igrave'] = 278;
-  t['ohungarumlaut'] = 611;
-  t['Eogonek'] = 667;
-  t['dcroat'] = 611;
-  t['threequarters'] = 834;
-  t['Scedilla'] = 667;
-  t['lcaron'] = 400;
-  t['Kcommaaccent'] = 722;
-  t['Lacute'] = 611;
-  t['trademark'] = 1000;
-  t['edotaccent'] = 556;
-  t['Igrave'] = 278;
-  t['Imacron'] = 278;
-  t['Lcaron'] = 611;
-  t['onehalf'] = 834;
-  t['lessequal'] = 549;
-  t['ocircumflex'] = 611;
-  t['ntilde'] = 611;
-  t['Uhungarumlaut'] = 722;
-  t['Eacute'] = 667;
-  t['emacron'] = 556;
-  t['gbreve'] = 611;
-  t['onequarter'] = 834;
-  t['Scaron'] = 667;
-  t['Scommaaccent'] = 667;
-  t['Ohungarumlaut'] = 778;
-  t['degree'] = 400;
-  t['ograve'] = 611;
-  t['Ccaron'] = 722;
-  t['ugrave'] = 611;
-  t['radical'] = 549;
-  t['Dcaron'] = 722;
-  t['rcommaaccent'] = 389;
-  t['Ntilde'] = 722;
-  t['otilde'] = 611;
-  t['Rcommaaccent'] = 722;
-  t['Lcommaaccent'] = 611;
-  t['Atilde'] = 722;
-  t['Aogonek'] = 722;
-  t['Aring'] = 722;
-  t['Otilde'] = 778;
-  t['zdotaccent'] = 500;
-  t['Ecaron'] = 667;
-  t['Iogonek'] = 278;
-  t['kcommaaccent'] = 556;
-  t['minus'] = 584;
-  t['Icircumflex'] = 278;
-  t['ncaron'] = 611;
-  t['tcommaaccent'] = 333;
-  t['logicalnot'] = 584;
-  t['odieresis'] = 611;
-  t['udieresis'] = 611;
-  t['notequal'] = 549;
-  t['gcommaaccent'] = 611;
-  t['eth'] = 611;
-  t['zcaron'] = 500;
-  t['ncommaaccent'] = 611;
-  t['onesuperior'] = 333;
-  t['imacron'] = 278;
-  t['Euro'] = 556;
- });
- t['Helvetica-BoldOblique'] = getLookupTableFactory(function (t) {
-  t['space'] = 278;
-  t['exclam'] = 333;
-  t['quotedbl'] = 474;
-  t['numbersign'] = 556;
-  t['dollar'] = 556;
-  t['percent'] = 889;
-  t['ampersand'] = 722;
-  t['quoteright'] = 278;
-  t['parenleft'] = 333;
-  t['parenright'] = 333;
-  t['asterisk'] = 389;
-  t['plus'] = 584;
-  t['comma'] = 278;
-  t['hyphen'] = 333;
-  t['period'] = 278;
-  t['slash'] = 278;
-  t['zero'] = 556;
-  t['one'] = 556;
-  t['two'] = 556;
-  t['three'] = 556;
-  t['four'] = 556;
-  t['five'] = 556;
-  t['six'] = 556;
-  t['seven'] = 556;
-  t['eight'] = 556;
-  t['nine'] = 556;
-  t['colon'] = 333;
-  t['semicolon'] = 333;
-  t['less'] = 584;
-  t['equal'] = 584;
-  t['greater'] = 584;
-  t['question'] = 611;
-  t['at'] = 975;
-  t['A'] = 722;
-  t['B'] = 722;
-  t['C'] = 722;
-  t['D'] = 722;
-  t['E'] = 667;
-  t['F'] = 611;
-  t['G'] = 778;
-  t['H'] = 722;
-  t['I'] = 278;
-  t['J'] = 556;
-  t['K'] = 722;
-  t['L'] = 611;
-  t['M'] = 833;
-  t['N'] = 722;
-  t['O'] = 778;
-  t['P'] = 667;
-  t['Q'] = 778;
-  t['R'] = 722;
-  t['S'] = 667;
-  t['T'] = 611;
-  t['U'] = 722;
-  t['V'] = 667;
-  t['W'] = 944;
-  t['X'] = 667;
-  t['Y'] = 667;
-  t['Z'] = 611;
-  t['bracketleft'] = 333;
-  t['backslash'] = 278;
-  t['bracketright'] = 333;
-  t['asciicircum'] = 584;
-  t['underscore'] = 556;
-  t['quoteleft'] = 278;
-  t['a'] = 556;
-  t['b'] = 611;
-  t['c'] = 556;
-  t['d'] = 611;
-  t['e'] = 556;
-  t['f'] = 333;
-  t['g'] = 611;
-  t['h'] = 611;
-  t['i'] = 278;
-  t['j'] = 278;
-  t['k'] = 556;
-  t['l'] = 278;
-  t['m'] = 889;
-  t['n'] = 611;
-  t['o'] = 611;
-  t['p'] = 611;
-  t['q'] = 611;
-  t['r'] = 389;
-  t['s'] = 556;
-  t['t'] = 333;
-  t['u'] = 611;
-  t['v'] = 556;
-  t['w'] = 778;
-  t['x'] = 556;
-  t['y'] = 556;
-  t['z'] = 500;
-  t['braceleft'] = 389;
-  t['bar'] = 280;
-  t['braceright'] = 389;
-  t['asciitilde'] = 584;
-  t['exclamdown'] = 333;
-  t['cent'] = 556;
-  t['sterling'] = 556;
-  t['fraction'] = 167;
-  t['yen'] = 556;
-  t['florin'] = 556;
-  t['section'] = 556;
-  t['currency'] = 556;
-  t['quotesingle'] = 238;
-  t['quotedblleft'] = 500;
-  t['guillemotleft'] = 556;
-  t['guilsinglleft'] = 333;
-  t['guilsinglright'] = 333;
-  t['fi'] = 611;
-  t['fl'] = 611;
-  t['endash'] = 556;
-  t['dagger'] = 556;
-  t['daggerdbl'] = 556;
-  t['periodcentered'] = 278;
-  t['paragraph'] = 556;
-  t['bullet'] = 350;
-  t['quotesinglbase'] = 278;
-  t['quotedblbase'] = 500;
-  t['quotedblright'] = 500;
-  t['guillemotright'] = 556;
-  t['ellipsis'] = 1000;
-  t['perthousand'] = 1000;
-  t['questiondown'] = 611;
-  t['grave'] = 333;
-  t['acute'] = 333;
-  t['circumflex'] = 333;
-  t['tilde'] = 333;
-  t['macron'] = 333;
-  t['breve'] = 333;
-  t['dotaccent'] = 333;
-  t['dieresis'] = 333;
-  t['ring'] = 333;
-  t['cedilla'] = 333;
-  t['hungarumlaut'] = 333;
-  t['ogonek'] = 333;
-  t['caron'] = 333;
-  t['emdash'] = 1000;
-  t['AE'] = 1000;
-  t['ordfeminine'] = 370;
-  t['Lslash'] = 611;
-  t['Oslash'] = 778;
-  t['OE'] = 1000;
-  t['ordmasculine'] = 365;
-  t['ae'] = 889;
-  t['dotlessi'] = 278;
-  t['lslash'] = 278;
-  t['oslash'] = 611;
-  t['oe'] = 944;
-  t['germandbls'] = 611;
-  t['Idieresis'] = 278;
-  t['eacute'] = 556;
-  t['abreve'] = 556;
-  t['uhungarumlaut'] = 611;
-  t['ecaron'] = 556;
-  t['Ydieresis'] = 667;
-  t['divide'] = 584;
-  t['Yacute'] = 667;
-  t['Acircumflex'] = 722;
-  t['aacute'] = 556;
-  t['Ucircumflex'] = 722;
-  t['yacute'] = 556;
-  t['scommaaccent'] = 556;
-  t['ecircumflex'] = 556;
-  t['Uring'] = 722;
-  t['Udieresis'] = 722;
-  t['aogonek'] = 556;
-  t['Uacute'] = 722;
-  t['uogonek'] = 611;
-  t['Edieresis'] = 667;
-  t['Dcroat'] = 722;
-  t['commaaccent'] = 250;
-  t['copyright'] = 737;
-  t['Emacron'] = 667;
-  t['ccaron'] = 556;
-  t['aring'] = 556;
-  t['Ncommaaccent'] = 722;
-  t['lacute'] = 278;
-  t['agrave'] = 556;
-  t['Tcommaaccent'] = 611;
-  t['Cacute'] = 722;
-  t['atilde'] = 556;
-  t['Edotaccent'] = 667;
-  t['scaron'] = 556;
-  t['scedilla'] = 556;
-  t['iacute'] = 278;
-  t['lozenge'] = 494;
-  t['Rcaron'] = 722;
-  t['Gcommaaccent'] = 778;
-  t['ucircumflex'] = 611;
-  t['acircumflex'] = 556;
-  t['Amacron'] = 722;
-  t['rcaron'] = 389;
-  t['ccedilla'] = 556;
-  t['Zdotaccent'] = 611;
-  t['Thorn'] = 667;
-  t['Omacron'] = 778;
-  t['Racute'] = 722;
-  t['Sacute'] = 667;
-  t['dcaron'] = 743;
-  t['Umacron'] = 722;
-  t['uring'] = 611;
-  t['threesuperior'] = 333;
-  t['Ograve'] = 778;
-  t['Agrave'] = 722;
-  t['Abreve'] = 722;
-  t['multiply'] = 584;
-  t['uacute'] = 611;
-  t['Tcaron'] = 611;
-  t['partialdiff'] = 494;
-  t['ydieresis'] = 556;
-  t['Nacute'] = 722;
-  t['icircumflex'] = 278;
-  t['Ecircumflex'] = 667;
-  t['adieresis'] = 556;
-  t['edieresis'] = 556;
-  t['cacute'] = 556;
-  t['nacute'] = 611;
-  t['umacron'] = 611;
-  t['Ncaron'] = 722;
-  t['Iacute'] = 278;
-  t['plusminus'] = 584;
-  t['brokenbar'] = 280;
-  t['registered'] = 737;
-  t['Gbreve'] = 778;
-  t['Idotaccent'] = 278;
-  t['summation'] = 600;
-  t['Egrave'] = 667;
-  t['racute'] = 389;
-  t['omacron'] = 611;
-  t['Zacute'] = 611;
-  t['Zcaron'] = 611;
-  t['greaterequal'] = 549;
-  t['Eth'] = 722;
-  t['Ccedilla'] = 722;
-  t['lcommaaccent'] = 278;
-  t['tcaron'] = 389;
-  t['eogonek'] = 556;
-  t['Uogonek'] = 722;
-  t['Aacute'] = 722;
-  t['Adieresis'] = 722;
-  t['egrave'] = 556;
-  t['zacute'] = 500;
-  t['iogonek'] = 278;
-  t['Oacute'] = 778;
-  t['oacute'] = 611;
-  t['amacron'] = 556;
-  t['sacute'] = 556;
-  t['idieresis'] = 278;
-  t['Ocircumflex'] = 778;
-  t['Ugrave'] = 722;
-  t['Delta'] = 612;
-  t['thorn'] = 611;
-  t['twosuperior'] = 333;
-  t['Odieresis'] = 778;
-  t['mu'] = 611;
-  t['igrave'] = 278;
-  t['ohungarumlaut'] = 611;
-  t['Eogonek'] = 667;
-  t['dcroat'] = 611;
-  t['threequarters'] = 834;
-  t['Scedilla'] = 667;
-  t['lcaron'] = 400;
-  t['Kcommaaccent'] = 722;
-  t['Lacute'] = 611;
-  t['trademark'] = 1000;
-  t['edotaccent'] = 556;
-  t['Igrave'] = 278;
-  t['Imacron'] = 278;
-  t['Lcaron'] = 611;
-  t['onehalf'] = 834;
-  t['lessequal'] = 549;
-  t['ocircumflex'] = 611;
-  t['ntilde'] = 611;
-  t['Uhungarumlaut'] = 722;
-  t['Eacute'] = 667;
-  t['emacron'] = 556;
-  t['gbreve'] = 611;
-  t['onequarter'] = 834;
-  t['Scaron'] = 667;
-  t['Scommaaccent'] = 667;
-  t['Ohungarumlaut'] = 778;
-  t['degree'] = 400;
-  t['ograve'] = 611;
-  t['Ccaron'] = 722;
-  t['ugrave'] = 611;
-  t['radical'] = 549;
-  t['Dcaron'] = 722;
-  t['rcommaaccent'] = 389;
-  t['Ntilde'] = 722;
-  t['otilde'] = 611;
-  t['Rcommaaccent'] = 722;
-  t['Lcommaaccent'] = 611;
-  t['Atilde'] = 722;
-  t['Aogonek'] = 722;
-  t['Aring'] = 722;
-  t['Otilde'] = 778;
-  t['zdotaccent'] = 500;
-  t['Ecaron'] = 667;
-  t['Iogonek'] = 278;
-  t['kcommaaccent'] = 556;
-  t['minus'] = 584;
-  t['Icircumflex'] = 278;
-  t['ncaron'] = 611;
-  t['tcommaaccent'] = 333;
-  t['logicalnot'] = 584;
-  t['odieresis'] = 611;
-  t['udieresis'] = 611;
-  t['notequal'] = 549;
-  t['gcommaaccent'] = 611;
-  t['eth'] = 611;
-  t['zcaron'] = 500;
-  t['ncommaaccent'] = 611;
-  t['onesuperior'] = 333;
-  t['imacron'] = 278;
-  t['Euro'] = 556;
- });
- t['Helvetica-Oblique'] = getLookupTableFactory(function (t) {
-  t['space'] = 278;
-  t['exclam'] = 278;
-  t['quotedbl'] = 355;
-  t['numbersign'] = 556;
-  t['dollar'] = 556;
-  t['percent'] = 889;
-  t['ampersand'] = 667;
-  t['quoteright'] = 222;
-  t['parenleft'] = 333;
-  t['parenright'] = 333;
-  t['asterisk'] = 389;
-  t['plus'] = 584;
-  t['comma'] = 278;
-  t['hyphen'] = 333;
-  t['period'] = 278;
-  t['slash'] = 278;
-  t['zero'] = 556;
-  t['one'] = 556;
-  t['two'] = 556;
-  t['three'] = 556;
-  t['four'] = 556;
-  t['five'] = 556;
-  t['six'] = 556;
-  t['seven'] = 556;
-  t['eight'] = 556;
-  t['nine'] = 556;
-  t['colon'] = 278;
-  t['semicolon'] = 278;
-  t['less'] = 584;
-  t['equal'] = 584;
-  t['greater'] = 584;
-  t['question'] = 556;
-  t['at'] = 1015;
-  t['A'] = 667;
-  t['B'] = 667;
-  t['C'] = 722;
-  t['D'] = 722;
-  t['E'] = 667;
-  t['F'] = 611;
-  t['G'] = 778;
-  t['H'] = 722;
-  t['I'] = 278;
-  t['J'] = 500;
-  t['K'] = 667;
-  t['L'] = 556;
-  t['M'] = 833;
-  t['N'] = 722;
-  t['O'] = 778;
-  t['P'] = 667;
-  t['Q'] = 778;
-  t['R'] = 722;
-  t['S'] = 667;
-  t['T'] = 611;
-  t['U'] = 722;
-  t['V'] = 667;
-  t['W'] = 944;
-  t['X'] = 667;
-  t['Y'] = 667;
-  t['Z'] = 611;
-  t['bracketleft'] = 278;
-  t['backslash'] = 278;
-  t['bracketright'] = 278;
-  t['asciicircum'] = 469;
-  t['underscore'] = 556;
-  t['quoteleft'] = 222;
-  t['a'] = 556;
-  t['b'] = 556;
-  t['c'] = 500;
-  t['d'] = 556;
-  t['e'] = 556;
-  t['f'] = 278;
-  t['g'] = 556;
-  t['h'] = 556;
-  t['i'] = 222;
-  t['j'] = 222;
-  t['k'] = 500;
-  t['l'] = 222;
-  t['m'] = 833;
-  t['n'] = 556;
-  t['o'] = 556;
-  t['p'] = 556;
-  t['q'] = 556;
-  t['r'] = 333;
-  t['s'] = 500;
-  t['t'] = 278;
-  t['u'] = 556;
-  t['v'] = 500;
-  t['w'] = 722;
-  t['x'] = 500;
-  t['y'] = 500;
-  t['z'] = 500;
-  t['braceleft'] = 334;
-  t['bar'] = 260;
-  t['braceright'] = 334;
-  t['asciitilde'] = 584;
-  t['exclamdown'] = 333;
-  t['cent'] = 556;
-  t['sterling'] = 556;
-  t['fraction'] = 167;
-  t['yen'] = 556;
-  t['florin'] = 556;
-  t['section'] = 556;
-  t['currency'] = 556;
-  t['quotesingle'] = 191;
-  t['quotedblleft'] = 333;
-  t['guillemotleft'] = 556;
-  t['guilsinglleft'] = 333;
-  t['guilsinglright'] = 333;
-  t['fi'] = 500;
-  t['fl'] = 500;
-  t['endash'] = 556;
-  t['dagger'] = 556;
-  t['daggerdbl'] = 556;
-  t['periodcentered'] = 278;
-  t['paragraph'] = 537;
-  t['bullet'] = 350;
-  t['quotesinglbase'] = 222;
-  t['quotedblbase'] = 333;
-  t['quotedblright'] = 333;
-  t['guillemotright'] = 556;
-  t['ellipsis'] = 1000;
-  t['perthousand'] = 1000;
-  t['questiondown'] = 611;
-  t['grave'] = 333;
-  t['acute'] = 333;
-  t['circumflex'] = 333;
-  t['tilde'] = 333;
-  t['macron'] = 333;
-  t['breve'] = 333;
-  t['dotaccent'] = 333;
-  t['dieresis'] = 333;
-  t['ring'] = 333;
-  t['cedilla'] = 333;
-  t['hungarumlaut'] = 333;
-  t['ogonek'] = 333;
-  t['caron'] = 333;
-  t['emdash'] = 1000;
-  t['AE'] = 1000;
-  t['ordfeminine'] = 370;
-  t['Lslash'] = 556;
-  t['Oslash'] = 778;
-  t['OE'] = 1000;
-  t['ordmasculine'] = 365;
-  t['ae'] = 889;
-  t['dotlessi'] = 278;
-  t['lslash'] = 222;
-  t['oslash'] = 611;
-  t['oe'] = 944;
-  t['germandbls'] = 611;
-  t['Idieresis'] = 278;
-  t['eacute'] = 556;
-  t['abreve'] = 556;
-  t['uhungarumlaut'] = 556;
-  t['ecaron'] = 556;
-  t['Ydieresis'] = 667;
-  t['divide'] = 584;
-  t['Yacute'] = 667;
-  t['Acircumflex'] = 667;
-  t['aacute'] = 556;
-  t['Ucircumflex'] = 722;
-  t['yacute'] = 500;
-  t['scommaaccent'] = 500;
-  t['ecircumflex'] = 556;
-  t['Uring'] = 722;
-  t['Udieresis'] = 722;
-  t['aogonek'] = 556;
-  t['Uacute'] = 722;
-  t['uogonek'] = 556;
-  t['Edieresis'] = 667;
-  t['Dcroat'] = 722;
-  t['commaaccent'] = 250;
-  t['copyright'] = 737;
-  t['Emacron'] = 667;
-  t['ccaron'] = 500;
-  t['aring'] = 556;
-  t['Ncommaaccent'] = 722;
-  t['lacute'] = 222;
-  t['agrave'] = 556;
-  t['Tcommaaccent'] = 611;
-  t['Cacute'] = 722;
-  t['atilde'] = 556;
-  t['Edotaccent'] = 667;
-  t['scaron'] = 500;
-  t['scedilla'] = 500;
-  t['iacute'] = 278;
-  t['lozenge'] = 471;
-  t['Rcaron'] = 722;
-  t['Gcommaaccent'] = 778;
-  t['ucircumflex'] = 556;
-  t['acircumflex'] = 556;
-  t['Amacron'] = 667;
-  t['rcaron'] = 333;
-  t['ccedilla'] = 500;
-  t['Zdotaccent'] = 611;
-  t['Thorn'] = 667;
-  t['Omacron'] = 778;
-  t['Racute'] = 722;
-  t['Sacute'] = 667;
-  t['dcaron'] = 643;
-  t['Umacron'] = 722;
-  t['uring'] = 556;
-  t['threesuperior'] = 333;
-  t['Ograve'] = 778;
-  t['Agrave'] = 667;
-  t['Abreve'] = 667;
-  t['multiply'] = 584;
-  t['uacute'] = 556;
-  t['Tcaron'] = 611;
-  t['partialdiff'] = 476;
-  t['ydieresis'] = 500;
-  t['Nacute'] = 722;
-  t['icircumflex'] = 278;
-  t['Ecircumflex'] = 667;
-  t['adieresis'] = 556;
-  t['edieresis'] = 556;
-  t['cacute'] = 500;
-  t['nacute'] = 556;
-  t['umacron'] = 556;
-  t['Ncaron'] = 722;
-  t['Iacute'] = 278;
-  t['plusminus'] = 584;
-  t['brokenbar'] = 260;
-  t['registered'] = 737;
-  t['Gbreve'] = 778;
-  t['Idotaccent'] = 278;
-  t['summation'] = 600;
-  t['Egrave'] = 667;
-  t['racute'] = 333;
-  t['omacron'] = 556;
-  t['Zacute'] = 611;
-  t['Zcaron'] = 611;
-  t['greaterequal'] = 549;
-  t['Eth'] = 722;
-  t['Ccedilla'] = 722;
-  t['lcommaaccent'] = 222;
-  t['tcaron'] = 317;
-  t['eogonek'] = 556;
-  t['Uogonek'] = 722;
-  t['Aacute'] = 667;
-  t['Adieresis'] = 667;
-  t['egrave'] = 556;
-  t['zacute'] = 500;
-  t['iogonek'] = 222;
-  t['Oacute'] = 778;
-  t['oacute'] = 556;
-  t['amacron'] = 556;
-  t['sacute'] = 500;
-  t['idieresis'] = 278;
-  t['Ocircumflex'] = 778;
-  t['Ugrave'] = 722;
-  t['Delta'] = 612;
-  t['thorn'] = 556;
-  t['twosuperior'] = 333;
-  t['Odieresis'] = 778;
-  t['mu'] = 556;
-  t['igrave'] = 278;
-  t['ohungarumlaut'] = 556;
-  t['Eogonek'] = 667;
-  t['dcroat'] = 556;
-  t['threequarters'] = 834;
-  t['Scedilla'] = 667;
-  t['lcaron'] = 299;
-  t['Kcommaaccent'] = 667;
-  t['Lacute'] = 556;
-  t['trademark'] = 1000;
-  t['edotaccent'] = 556;
-  t['Igrave'] = 278;
-  t['Imacron'] = 278;
-  t['Lcaron'] = 556;
-  t['onehalf'] = 834;
-  t['lessequal'] = 549;
-  t['ocircumflex'] = 556;
-  t['ntilde'] = 556;
-  t['Uhungarumlaut'] = 722;
-  t['Eacute'] = 667;
-  t['emacron'] = 556;
-  t['gbreve'] = 556;
-  t['onequarter'] = 834;
-  t['Scaron'] = 667;
-  t['Scommaaccent'] = 667;
-  t['Ohungarumlaut'] = 778;
-  t['degree'] = 400;
-  t['ograve'] = 556;
-  t['Ccaron'] = 722;
-  t['ugrave'] = 556;
-  t['radical'] = 453;
-  t['Dcaron'] = 722;
-  t['rcommaaccent'] = 333;
-  t['Ntilde'] = 722;
-  t['otilde'] = 556;
-  t['Rcommaaccent'] = 722;
-  t['Lcommaaccent'] = 556;
-  t['Atilde'] = 667;
-  t['Aogonek'] = 667;
-  t['Aring'] = 667;
-  t['Otilde'] = 778;
-  t['zdotaccent'] = 500;
-  t['Ecaron'] = 667;
-  t['Iogonek'] = 278;
-  t['kcommaaccent'] = 500;
-  t['minus'] = 584;
-  t['Icircumflex'] = 278;
-  t['ncaron'] = 556;
-  t['tcommaaccent'] = 278;
-  t['logicalnot'] = 584;
-  t['odieresis'] = 556;
-  t['udieresis'] = 556;
-  t['notequal'] = 549;
-  t['gcommaaccent'] = 556;
-  t['eth'] = 556;
-  t['zcaron'] = 500;
-  t['ncommaaccent'] = 556;
-  t['onesuperior'] = 333;
-  t['imacron'] = 278;
-  t['Euro'] = 556;
- });
- t['Symbol'] = getLookupTableFactory(function (t) {
-  t['space'] = 250;
-  t['exclam'] = 333;
-  t['universal'] = 713;
-  t['numbersign'] = 500;
-  t['existential'] = 549;
-  t['percent'] = 833;
-  t['ampersand'] = 778;
-  t['suchthat'] = 439;
-  t['parenleft'] = 333;
-  t['parenright'] = 333;
-  t['asteriskmath'] = 500;
-  t['plus'] = 549;
-  t['comma'] = 250;
-  t['minus'] = 549;
-  t['period'] = 250;
-  t['slash'] = 278;
-  t['zero'] = 500;
-  t['one'] = 500;
-  t['two'] = 500;
-  t['three'] = 500;
-  t['four'] = 500;
-  t['five'] = 500;
-  t['six'] = 500;
-  t['seven'] = 500;
-  t['eight'] = 500;
-  t['nine'] = 500;
-  t['colon'] = 278;
-  t['semicolon'] = 278;
-  t['less'] = 549;
-  t['equal'] = 549;
-  t['greater'] = 549;
-  t['question'] = 444;
-  t['congruent'] = 549;
-  t['Alpha'] = 722;
-  t['Beta'] = 667;
-  t['Chi'] = 722;
-  t['Delta'] = 612;
-  t['Epsilon'] = 611;
-  t['Phi'] = 763;
-  t['Gamma'] = 603;
-  t['Eta'] = 722;
-  t['Iota'] = 333;
-  t['theta1'] = 631;
-  t['Kappa'] = 722;
-  t['Lambda'] = 686;
-  t['Mu'] = 889;
-  t['Nu'] = 722;
-  t['Omicron'] = 722;
-  t['Pi'] = 768;
-  t['Theta'] = 741;
-  t['Rho'] = 556;
-  t['Sigma'] = 592;
-  t['Tau'] = 611;
-  t['Upsilon'] = 690;
-  t['sigma1'] = 439;
-  t['Omega'] = 768;
-  t['Xi'] = 645;
-  t['Psi'] = 795;
-  t['Zeta'] = 611;
-  t['bracketleft'] = 333;
-  t['therefore'] = 863;
-  t['bracketright'] = 333;
-  t['perpendicular'] = 658;
-  t['underscore'] = 500;
-  t['radicalex'] = 500;
-  t['alpha'] = 631;
-  t['beta'] = 549;
-  t['chi'] = 549;
-  t['delta'] = 494;
-  t['epsilon'] = 439;
-  t['phi'] = 521;
-  t['gamma'] = 411;
-  t['eta'] = 603;
-  t['iota'] = 329;
-  t['phi1'] = 603;
-  t['kappa'] = 549;
-  t['lambda'] = 549;
-  t['mu'] = 576;
-  t['nu'] = 521;
-  t['omicron'] = 549;
-  t['pi'] = 549;
-  t['theta'] = 521;
-  t['rho'] = 549;
-  t['sigma'] = 603;
-  t['tau'] = 439;
-  t['upsilon'] = 576;
-  t['omega1'] = 713;
-  t['omega'] = 686;
-  t['xi'] = 493;
-  t['psi'] = 686;
-  t['zeta'] = 494;
-  t['braceleft'] = 480;
-  t['bar'] = 200;
-  t['braceright'] = 480;
-  t['similar'] = 549;
-  t['Euro'] = 750;
-  t['Upsilon1'] = 620;
-  t['minute'] = 247;
-  t['lessequal'] = 549;
-  t['fraction'] = 167;
-  t['infinity'] = 713;
-  t['florin'] = 500;
-  t['club'] = 753;
-  t['diamond'] = 753;
-  t['heart'] = 753;
-  t['spade'] = 753;
-  t['arrowboth'] = 1042;
-  t['arrowleft'] = 987;
-  t['arrowup'] = 603;
-  t['arrowright'] = 987;
-  t['arrowdown'] = 603;
-  t['degree'] = 400;
-  t['plusminus'] = 549;
-  t['second'] = 411;
-  t['greaterequal'] = 549;
-  t['multiply'] = 549;
-  t['proportional'] = 713;
-  t['partialdiff'] = 494;
-  t['bullet'] = 460;
-  t['divide'] = 549;
-  t['notequal'] = 549;
-  t['equivalence'] = 549;
-  t['approxequal'] = 549;
-  t['ellipsis'] = 1000;
-  t['arrowvertex'] = 603;
-  t['arrowhorizex'] = 1000;
-  t['carriagereturn'] = 658;
-  t['aleph'] = 823;
-  t['Ifraktur'] = 686;
-  t['Rfraktur'] = 795;
-  t['weierstrass'] = 987;
-  t['circlemultiply'] = 768;
-  t['circleplus'] = 768;
-  t['emptyset'] = 823;
-  t['intersection'] = 768;
-  t['union'] = 768;
-  t['propersuperset'] = 713;
-  t['reflexsuperset'] = 713;
-  t['notsubset'] = 713;
-  t['propersubset'] = 713;
-  t['reflexsubset'] = 713;
-  t['element'] = 713;
-  t['notelement'] = 713;
-  t['angle'] = 768;
-  t['gradient'] = 713;
-  t['registerserif'] = 790;
-  t['copyrightserif'] = 790;
-  t['trademarkserif'] = 890;
-  t['product'] = 823;
-  t['radical'] = 549;
-  t['dotmath'] = 250;
-  t['logicalnot'] = 713;
-  t['logicaland'] = 603;
-  t['logicalor'] = 603;
-  t['arrowdblboth'] = 1042;
-  t['arrowdblleft'] = 987;
-  t['arrowdblup'] = 603;
-  t['arrowdblright'] = 987;
-  t['arrowdbldown'] = 603;
-  t['lozenge'] = 494;
-  t['angleleft'] = 329;
-  t['registersans'] = 790;
-  t['copyrightsans'] = 790;
-  t['trademarksans'] = 786;
-  t['summation'] = 713;
-  t['parenlefttp'] = 384;
-  t['parenleftex'] = 384;
-  t['parenleftbt'] = 384;
-  t['bracketlefttp'] = 384;
-  t['bracketleftex'] = 384;
-  t['bracketleftbt'] = 384;
-  t['bracelefttp'] = 494;
-  t['braceleftmid'] = 494;
-  t['braceleftbt'] = 494;
-  t['braceex'] = 494;
-  t['angleright'] = 329;
-  t['integral'] = 274;
-  t['integraltp'] = 686;
-  t['integralex'] = 686;
-  t['integralbt'] = 686;
-  t['parenrighttp'] = 384;
-  t['parenrightex'] = 384;
-  t['parenrightbt'] = 384;
-  t['bracketrighttp'] = 384;
-  t['bracketrightex'] = 384;
-  t['bracketrightbt'] = 384;
-  t['bracerighttp'] = 494;
-  t['bracerightmid'] = 494;
-  t['bracerightbt'] = 494;
-  t['apple'] = 790;
- });
- t['Times-Roman'] = getLookupTableFactory(function (t) {
-  t['space'] = 250;
-  t['exclam'] = 333;
-  t['quotedbl'] = 408;
-  t['numbersign'] = 500;
-  t['dollar'] = 500;
-  t['percent'] = 833;
-  t['ampersand'] = 778;
-  t['quoteright'] = 333;
-  t['parenleft'] = 333;
-  t['parenright'] = 333;
-  t['asterisk'] = 500;
-  t['plus'] = 564;
-  t['comma'] = 250;
-  t['hyphen'] = 333;
-  t['period'] = 250;
-  t['slash'] = 278;
-  t['zero'] = 500;
-  t['one'] = 500;
-  t['two'] = 500;
-  t['three'] = 500;
-  t['four'] = 500;
-  t['five'] = 500;
-  t['six'] = 500;
-  t['seven'] = 500;
-  t['eight'] = 500;
-  t['nine'] = 500;
-  t['colon'] = 278;
-  t['semicolon'] = 278;
-  t['less'] = 564;
-  t['equal'] = 564;
-  t['greater'] = 564;
-  t['question'] = 444;
-  t['at'] = 921;
-  t['A'] = 722;
-  t['B'] = 667;
-  t['C'] = 667;
-  t['D'] = 722;
-  t['E'] = 611;
-  t['F'] = 556;
-  t['G'] = 722;
-  t['H'] = 722;
-  t['I'] = 333;
-  t['J'] = 389;
-  t['K'] = 722;
-  t['L'] = 611;
-  t['M'] = 889;
-  t['N'] = 722;
-  t['O'] = 722;
-  t['P'] = 556;
-  t['Q'] = 722;
-  t['R'] = 667;
-  t['S'] = 556;
-  t['T'] = 611;
-  t['U'] = 722;
-  t['V'] = 722;
-  t['W'] = 944;
-  t['X'] = 722;
-  t['Y'] = 722;
-  t['Z'] = 611;
-  t['bracketleft'] = 333;
-  t['backslash'] = 278;
-  t['bracketright'] = 333;
-  t['asciicircum'] = 469;
-  t['underscore'] = 500;
-  t['quoteleft'] = 333;
-  t['a'] = 444;
-  t['b'] = 500;
-  t['c'] = 444;
-  t['d'] = 500;
-  t['e'] = 444;
-  t['f'] = 333;
-  t['g'] = 500;
-  t['h'] = 500;
-  t['i'] = 278;
-  t['j'] = 278;
-  t['k'] = 500;
-  t['l'] = 278;
-  t['m'] = 778;
-  t['n'] = 500;
-  t['o'] = 500;
-  t['p'] = 500;
-  t['q'] = 500;
-  t['r'] = 333;
-  t['s'] = 389;
-  t['t'] = 278;
-  t['u'] = 500;
-  t['v'] = 500;
-  t['w'] = 722;
-  t['x'] = 500;
-  t['y'] = 500;
-  t['z'] = 444;
-  t['braceleft'] = 480;
-  t['bar'] = 200;
-  t['braceright'] = 480;
-  t['asciitilde'] = 541;
-  t['exclamdown'] = 333;
-  t['cent'] = 500;
-  t['sterling'] = 500;
-  t['fraction'] = 167;
-  t['yen'] = 500;
-  t['florin'] = 500;
-  t['section'] = 500;
-  t['currency'] = 500;
-  t['quotesingle'] = 180;
-  t['quotedblleft'] = 444;
-  t['guillemotleft'] = 500;
-  t['guilsinglleft'] = 333;
-  t['guilsinglright'] = 333;
-  t['fi'] = 556;
-  t['fl'] = 556;
-  t['endash'] = 500;
-  t['dagger'] = 500;
-  t['daggerdbl'] = 500;
-  t['periodcentered'] = 250;
-  t['paragraph'] = 453;
-  t['bullet'] = 350;
-  t['quotesinglbase'] = 333;
-  t['quotedblbase'] = 444;
-  t['quotedblright'] = 444;
-  t['guillemotright'] = 500;
-  t['ellipsis'] = 1000;
-  t['perthousand'] = 1000;
-  t['questiondown'] = 444;
-  t['grave'] = 333;
-  t['acute'] = 333;
-  t['circumflex'] = 333;
-  t['tilde'] = 333;
-  t['macron'] = 333;
-  t['breve'] = 333;
-  t['dotaccent'] = 333;
-  t['dieresis'] = 333;
-  t['ring'] = 333;
-  t['cedilla'] = 333;
-  t['hungarumlaut'] = 333;
-  t['ogonek'] = 333;
-  t['caron'] = 333;
-  t['emdash'] = 1000;
-  t['AE'] = 889;
-  t['ordfeminine'] = 276;
-  t['Lslash'] = 611;
-  t['Oslash'] = 722;
-  t['OE'] = 889;
-  t['ordmasculine'] = 310;
-  t['ae'] = 667;
-  t['dotlessi'] = 278;
-  t['lslash'] = 278;
-  t['oslash'] = 500;
-  t['oe'] = 722;
-  t['germandbls'] = 500;
-  t['Idieresis'] = 333;
-  t['eacute'] = 444;
-  t['abreve'] = 444;
-  t['uhungarumlaut'] = 500;
-  t['ecaron'] = 444;
-  t['Ydieresis'] = 722;
-  t['divide'] = 564;
-  t['Yacute'] = 722;
-  t['Acircumflex'] = 722;
-  t['aacute'] = 444;
-  t['Ucircumflex'] = 722;
-  t['yacute'] = 500;
-  t['scommaaccent'] = 389;
-  t['ecircumflex'] = 444;
-  t['Uring'] = 722;
-  t['Udieresis'] = 722;
-  t['aogonek'] = 444;
-  t['Uacute'] = 722;
-  t['uogonek'] = 500;
-  t['Edieresis'] = 611;
-  t['Dcroat'] = 722;
-  t['commaaccent'] = 250;
-  t['copyright'] = 760;
-  t['Emacron'] = 611;
-  t['ccaron'] = 444;
-  t['aring'] = 444;
-  t['Ncommaaccent'] = 722;
-  t['lacute'] = 278;
-  t['agrave'] = 444;
-  t['Tcommaaccent'] = 611;
-  t['Cacute'] = 667;
-  t['atilde'] = 444;
-  t['Edotaccent'] = 611;
-  t['scaron'] = 389;
-  t['scedilla'] = 389;
-  t['iacute'] = 278;
-  t['lozenge'] = 471;
-  t['Rcaron'] = 667;
-  t['Gcommaaccent'] = 722;
-  t['ucircumflex'] = 500;
-  t['acircumflex'] = 444;
-  t['Amacron'] = 722;
-  t['rcaron'] = 333;
-  t['ccedilla'] = 444;
-  t['Zdotaccent'] = 611;
-  t['Thorn'] = 556;
-  t['Omacron'] = 722;
-  t['Racute'] = 667;
-  t['Sacute'] = 556;
-  t['dcaron'] = 588;
-  t['Umacron'] = 722;
-  t['uring'] = 500;
-  t['threesuperior'] = 300;
-  t['Ograve'] = 722;
-  t['Agrave'] = 722;
-  t['Abreve'] = 722;
-  t['multiply'] = 564;
-  t['uacute'] = 500;
-  t['Tcaron'] = 611;
-  t['partialdiff'] = 476;
-  t['ydieresis'] = 500;
-  t['Nacute'] = 722;
-  t['icircumflex'] = 278;
-  t['Ecircumflex'] = 611;
-  t['adieresis'] = 444;
-  t['edieresis'] = 444;
-  t['cacute'] = 444;
-  t['nacute'] = 500;
-  t['umacron'] = 500;
-  t['Ncaron'] = 722;
-  t['Iacute'] = 333;
-  t['plusminus'] = 564;
-  t['brokenbar'] = 200;
-  t['registered'] = 760;
-  t['Gbreve'] = 722;
-  t['Idotaccent'] = 333;
-  t['summation'] = 600;
-  t['Egrave'] = 611;
-  t['racute'] = 333;
-  t['omacron'] = 500;
-  t['Zacute'] = 611;
-  t['Zcaron'] = 611;
-  t['greaterequal'] = 549;
-  t['Eth'] = 722;
-  t['Ccedilla'] = 667;
-  t['lcommaaccent'] = 278;
-  t['tcaron'] = 326;
-  t['eogonek'] = 444;
-  t['Uogonek'] = 722;
-  t['Aacute'] = 722;
-  t['Adieresis'] = 722;
-  t['egrave'] = 444;
-  t['zacute'] = 444;
-  t['iogonek'] = 278;
-  t['Oacute'] = 722;
-  t['oacute'] = 500;
-  t['amacron'] = 444;
-  t['sacute'] = 389;
-  t['idieresis'] = 278;
-  t['Ocircumflex'] = 722;
-  t['Ugrave'] = 722;
-  t['Delta'] = 612;
-  t['thorn'] = 500;
-  t['twosuperior'] = 300;
-  t['Odieresis'] = 722;
-  t['mu'] = 500;
-  t['igrave'] = 278;
-  t['ohungarumlaut'] = 500;
-  t['Eogonek'] = 611;
-  t['dcroat'] = 500;
-  t['threequarters'] = 750;
-  t['Scedilla'] = 556;
-  t['lcaron'] = 344;
-  t['Kcommaaccent'] = 722;
-  t['Lacute'] = 611;
-  t['trademark'] = 980;
-  t['edotaccent'] = 444;
-  t['Igrave'] = 333;
-  t['Imacron'] = 333;
-  t['Lcaron'] = 611;
-  t['onehalf'] = 750;
-  t['lessequal'] = 549;
-  t['ocircumflex'] = 500;
-  t['ntilde'] = 500;
-  t['Uhungarumlaut'] = 722;
-  t['Eacute'] = 611;
-  t['emacron'] = 444;
-  t['gbreve'] = 500;
-  t['onequarter'] = 750;
-  t['Scaron'] = 556;
-  t['Scommaaccent'] = 556;
-  t['Ohungarumlaut'] = 722;
-  t['degree'] = 400;
-  t['ograve'] = 500;
-  t['Ccaron'] = 667;
-  t['ugrave'] = 500;
-  t['radical'] = 453;
-  t['Dcaron'] = 722;
-  t['rcommaaccent'] = 333;
-  t['Ntilde'] = 722;
-  t['otilde'] = 500;
-  t['Rcommaaccent'] = 667;
-  t['Lcommaaccent'] = 611;
-  t['Atilde'] = 722;
-  t['Aogonek'] = 722;
-  t['Aring'] = 722;
-  t['Otilde'] = 722;
-  t['zdotaccent'] = 444;
-  t['Ecaron'] = 611;
-  t['Iogonek'] = 333;
-  t['kcommaaccent'] = 500;
-  t['minus'] = 564;
-  t['Icircumflex'] = 333;
-  t['ncaron'] = 500;
-  t['tcommaaccent'] = 278;
-  t['logicalnot'] = 564;
-  t['odieresis'] = 500;
-  t['udieresis'] = 500;
-  t['notequal'] = 549;
-  t['gcommaaccent'] = 500;
-  t['eth'] = 500;
-  t['zcaron'] = 444;
-  t['ncommaaccent'] = 500;
-  t['onesuperior'] = 300;
-  t['imacron'] = 278;
-  t['Euro'] = 500;
- });
- t['Times-Bold'] = getLookupTableFactory(function (t) {
-  t['space'] = 250;
-  t['exclam'] = 333;
-  t['quotedbl'] = 555;
-  t['numbersign'] = 500;
-  t['dollar'] = 500;
-  t['percent'] = 1000;
-  t['ampersand'] = 833;
-  t['quoteright'] = 333;
-  t['parenleft'] = 333;
-  t['parenright'] = 333;
-  t['asterisk'] = 500;
-  t['plus'] = 570;
-  t['comma'] = 250;
-  t['hyphen'] = 333;
-  t['period'] = 250;
-  t['slash'] = 278;
-  t['zero'] = 500;
-  t['one'] = 500;
-  t['two'] = 500;
-  t['three'] = 500;
-  t['four'] = 500;
-  t['five'] = 500;
-  t['six'] = 500;
-  t['seven'] = 500;
-  t['eight'] = 500;
-  t['nine'] = 500;
-  t['colon'] = 333;
-  t['semicolon'] = 333;
-  t['less'] = 570;
-  t['equal'] = 570;
-  t['greater'] = 570;
-  t['question'] = 500;
-  t['at'] = 930;
-  t['A'] = 722;
-  t['B'] = 667;
-  t['C'] = 722;
-  t['D'] = 722;
-  t['E'] = 667;
-  t['F'] = 611;
-  t['G'] = 778;
-  t['H'] = 778;
-  t['I'] = 389;
-  t['J'] = 500;
-  t['K'] = 778;
-  t['L'] = 667;
-  t['M'] = 944;
-  t['N'] = 722;
-  t['O'] = 778;
-  t['P'] = 611;
-  t['Q'] = 778;
-  t['R'] = 722;
-  t['S'] = 556;
-  t['T'] = 667;
-  t['U'] = 722;
-  t['V'] = 722;
-  t['W'] = 1000;
-  t['X'] = 722;
-  t['Y'] = 722;
-  t['Z'] = 667;
-  t['bracketleft'] = 333;
-  t['backslash'] = 278;
-  t['bracketright'] = 333;
-  t['asciicircum'] = 581;
-  t['underscore'] = 500;
-  t['quoteleft'] = 333;
-  t['a'] = 500;
-  t['b'] = 556;
-  t['c'] = 444;
-  t['d'] = 556;
-  t['e'] = 444;
-  t['f'] = 333;
-  t['g'] = 500;
-  t['h'] = 556;
-  t['i'] = 278;
-  t['j'] = 333;
-  t['k'] = 556;
-  t['l'] = 278;
-  t['m'] = 833;
-  t['n'] = 556;
-  t['o'] = 500;
-  t['p'] = 556;
-  t['q'] = 556;
-  t['r'] = 444;
-  t['s'] = 389;
-  t['t'] = 333;
-  t['u'] = 556;
-  t['v'] = 500;
-  t['w'] = 722;
-  t['x'] = 500;
-  t['y'] = 500;
-  t['z'] = 444;
-  t['braceleft'] = 394;
-  t['bar'] = 220;
-  t['braceright'] = 394;
-  t['asciitilde'] = 520;
-  t['exclamdown'] = 333;
-  t['cent'] = 500;
-  t['sterling'] = 500;
-  t['fraction'] = 167;
-  t['yen'] = 500;
-  t['florin'] = 500;
-  t['section'] = 500;
-  t['currency'] = 500;
-  t['quotesingle'] = 278;
-  t['quotedblleft'] = 500;
-  t['guillemotleft'] = 500;
-  t['guilsinglleft'] = 333;
-  t['guilsinglright'] = 333;
-  t['fi'] = 556;
-  t['fl'] = 556;
-  t['endash'] = 500;
-  t['dagger'] = 500;
-  t['daggerdbl'] = 500;
-  t['periodcentered'] = 250;
-  t['paragraph'] = 540;
-  t['bullet'] = 350;
-  t['quotesinglbase'] = 333;
-  t['quotedblbase'] = 500;
-  t['quotedblright'] = 500;
-  t['guillemotright'] = 500;
-  t['ellipsis'] = 1000;
-  t['perthousand'] = 1000;
-  t['questiondown'] = 500;
-  t['grave'] = 333;
-  t['acute'] = 333;
-  t['circumflex'] = 333;
-  t['tilde'] = 333;
-  t['macron'] = 333;
-  t['breve'] = 333;
-  t['dotaccent'] = 333;
-  t['dieresis'] = 333;
-  t['ring'] = 333;
-  t['cedilla'] = 333;
-  t['hungarumlaut'] = 333;
-  t['ogonek'] = 333;
-  t['caron'] = 333;
-  t['emdash'] = 1000;
-  t['AE'] = 1000;
-  t['ordfeminine'] = 300;
-  t['Lslash'] = 667;
-  t['Oslash'] = 778;
-  t['OE'] = 1000;
-  t['ordmasculine'] = 330;
-  t['ae'] = 722;
-  t['dotlessi'] = 278;
-  t['lslash'] = 278;
-  t['oslash'] = 500;
-  t['oe'] = 722;
-  t['germandbls'] = 556;
-  t['Idieresis'] = 389;
-  t['eacute'] = 444;
-  t['abreve'] = 500;
-  t['uhungarumlaut'] = 556;
-  t['ecaron'] = 444;
-  t['Ydieresis'] = 722;
-  t['divide'] = 570;
-  t['Yacute'] = 722;
-  t['Acircumflex'] = 722;
-  t['aacute'] = 500;
-  t['Ucircumflex'] = 722;
-  t['yacute'] = 500;
-  t['scommaaccent'] = 389;
-  t['ecircumflex'] = 444;
-  t['Uring'] = 722;
-  t['Udieresis'] = 722;
-  t['aogonek'] = 500;
-  t['Uacute'] = 722;
-  t['uogonek'] = 556;
-  t['Edieresis'] = 667;
-  t['Dcroat'] = 722;
-  t['commaaccent'] = 250;
-  t['copyright'] = 747;
-  t['Emacron'] = 667;
-  t['ccaron'] = 444;
-  t['aring'] = 500;
-  t['Ncommaaccent'] = 722;
-  t['lacute'] = 278;
-  t['agrave'] = 500;
-  t['Tcommaaccent'] = 667;
-  t['Cacute'] = 722;
-  t['atilde'] = 500;
-  t['Edotaccent'] = 667;
-  t['scaron'] = 389;
-  t['scedilla'] = 389;
-  t['iacute'] = 278;
-  t['lozenge'] = 494;
-  t['Rcaron'] = 722;
-  t['Gcommaaccent'] = 778;
-  t['ucircumflex'] = 556;
-  t['acircumflex'] = 500;
-  t['Amacron'] = 722;
-  t['rcaron'] = 444;
-  t['ccedilla'] = 444;
-  t['Zdotaccent'] = 667;
-  t['Thorn'] = 611;
-  t['Omacron'] = 778;
-  t['Racute'] = 722;
-  t['Sacute'] = 556;
-  t['dcaron'] = 672;
-  t['Umacron'] = 722;
-  t['uring'] = 556;
-  t['threesuperior'] = 300;
-  t['Ograve'] = 778;
-  t['Agrave'] = 722;
-  t['Abreve'] = 722;
-  t['multiply'] = 570;
-  t['uacute'] = 556;
-  t['Tcaron'] = 667;
-  t['partialdiff'] = 494;
-  t['ydieresis'] = 500;
-  t['Nacute'] = 722;
-  t['icircumflex'] = 278;
-  t['Ecircumflex'] = 667;
-  t['adieresis'] = 500;
-  t['edieresis'] = 444;
-  t['cacute'] = 444;
-  t['nacute'] = 556;
-  t['umacron'] = 556;
-  t['Ncaron'] = 722;
-  t['Iacute'] = 389;
-  t['plusminus'] = 570;
-  t['brokenbar'] = 220;
-  t['registered'] = 747;
-  t['Gbreve'] = 778;
-  t['Idotaccent'] = 389;
-  t['summation'] = 600;
-  t['Egrave'] = 667;
-  t['racute'] = 444;
-  t['omacron'] = 500;
-  t['Zacute'] = 667;
-  t['Zcaron'] = 667;
-  t['greaterequal'] = 549;
-  t['Eth'] = 722;
-  t['Ccedilla'] = 722;
-  t['lcommaaccent'] = 278;
-  t['tcaron'] = 416;
-  t['eogonek'] = 444;
-  t['Uogonek'] = 722;
-  t['Aacute'] = 722;
-  t['Adieresis'] = 722;
-  t['egrave'] = 444;
-  t['zacute'] = 444;
-  t['iogonek'] = 278;
-  t['Oacute'] = 778;
-  t['oacute'] = 500;
-  t['amacron'] = 500;
-  t['sacute'] = 389;
-  t['idieresis'] = 278;
-  t['Ocircumflex'] = 778;
-  t['Ugrave'] = 722;
-  t['Delta'] = 612;
-  t['thorn'] = 556;
-  t['twosuperior'] = 300;
-  t['Odieresis'] = 778;
-  t['mu'] = 556;
-  t['igrave'] = 278;
-  t['ohungarumlaut'] = 500;
-  t['Eogonek'] = 667;
-  t['dcroat'] = 556;
-  t['threequarters'] = 750;
-  t['Scedilla'] = 556;
-  t['lcaron'] = 394;
-  t['Kcommaaccent'] = 778;
-  t['Lacute'] = 667;
-  t['trademark'] = 1000;
-  t['edotaccent'] = 444;
-  t['Igrave'] = 389;
-  t['Imacron'] = 389;
-  t['Lcaron'] = 667;
-  t['onehalf'] = 750;
-  t['lessequal'] = 549;
-  t['ocircumflex'] = 500;
-  t['ntilde'] = 556;
-  t['Uhungarumlaut'] = 722;
-  t['Eacute'] = 667;
-  t['emacron'] = 444;
-  t['gbreve'] = 500;
-  t['onequarter'] = 750;
-  t['Scaron'] = 556;
-  t['Scommaaccent'] = 556;
-  t['Ohungarumlaut'] = 778;
-  t['degree'] = 400;
-  t['ograve'] = 500;
-  t['Ccaron'] = 722;
-  t['ugrave'] = 556;
-  t['radical'] = 549;
-  t['Dcaron'] = 722;
-  t['rcommaaccent'] = 444;
-  t['Ntilde'] = 722;
-  t['otilde'] = 500;
-  t['Rcommaaccent'] = 722;
-  t['Lcommaaccent'] = 667;
-  t['Atilde'] = 722;
-  t['Aogonek'] = 722;
-  t['Aring'] = 722;
-  t['Otilde'] = 778;
-  t['zdotaccent'] = 444;
-  t['Ecaron'] = 667;
-  t['Iogonek'] = 389;
-  t['kcommaaccent'] = 556;
-  t['minus'] = 570;
-  t['Icircumflex'] = 389;
-  t['ncaron'] = 556;
-  t['tcommaaccent'] = 333;
-  t['logicalnot'] = 570;
-  t['odieresis'] = 500;
-  t['udieresis'] = 556;
-  t['notequal'] = 549;
-  t['gcommaaccent'] = 500;
-  t['eth'] = 500;
-  t['zcaron'] = 444;
-  t['ncommaaccent'] = 556;
-  t['onesuperior'] = 300;
-  t['imacron'] = 278;
-  t['Euro'] = 500;
- });
- t['Times-BoldItalic'] = getLookupTableFactory(function (t) {
-  t['space'] = 250;
-  t['exclam'] = 389;
-  t['quotedbl'] = 555;
-  t['numbersign'] = 500;
-  t['dollar'] = 500;
-  t['percent'] = 833;
-  t['ampersand'] = 778;
-  t['quoteright'] = 333;
-  t['parenleft'] = 333;
-  t['parenright'] = 333;
-  t['asterisk'] = 500;
-  t['plus'] = 570;
-  t['comma'] = 250;
-  t['hyphen'] = 333;
-  t['period'] = 250;
-  t['slash'] = 278;
-  t['zero'] = 500;
-  t['one'] = 500;
-  t['two'] = 500;
-  t['three'] = 500;
-  t['four'] = 500;
-  t['five'] = 500;
-  t['six'] = 500;
-  t['seven'] = 500;
-  t['eight'] = 500;
-  t['nine'] = 500;
-  t['colon'] = 333;
-  t['semicolon'] = 333;
-  t['less'] = 570;
-  t['equal'] = 570;
-  t['greater'] = 570;
-  t['question'] = 500;
-  t['at'] = 832;
-  t['A'] = 667;
-  t['B'] = 667;
-  t['C'] = 667;
-  t['D'] = 722;
-  t['E'] = 667;
-  t['F'] = 667;
-  t['G'] = 722;
-  t['H'] = 778;
-  t['I'] = 389;
-  t['J'] = 500;
-  t['K'] = 667;
-  t['L'] = 611;
-  t['M'] = 889;
-  t['N'] = 722;
-  t['O'] = 722;
-  t['P'] = 611;
-  t['Q'] = 722;
-  t['R'] = 667;
-  t['S'] = 556;
-  t['T'] = 611;
-  t['U'] = 722;
-  t['V'] = 667;
-  t['W'] = 889;
-  t['X'] = 667;
-  t['Y'] = 611;
-  t['Z'] = 611;
-  t['bracketleft'] = 333;
-  t['backslash'] = 278;
-  t['bracketright'] = 333;
-  t['asciicircum'] = 570;
-  t['underscore'] = 500;
-  t['quoteleft'] = 333;
-  t['a'] = 500;
-  t['b'] = 500;
-  t['c'] = 444;
-  t['d'] = 500;
-  t['e'] = 444;
-  t['f'] = 333;
-  t['g'] = 500;
-  t['h'] = 556;
-  t['i'] = 278;
-  t['j'] = 278;
-  t['k'] = 500;
-  t['l'] = 278;
-  t['m'] = 778;
-  t['n'] = 556;
-  t['o'] = 500;
-  t['p'] = 500;
-  t['q'] = 500;
-  t['r'] = 389;
-  t['s'] = 389;
-  t['t'] = 278;
-  t['u'] = 556;
-  t['v'] = 444;
-  t['w'] = 667;
-  t['x'] = 500;
-  t['y'] = 444;
-  t['z'] = 389;
-  t['braceleft'] = 348;
-  t['bar'] = 220;
-  t['braceright'] = 348;
-  t['asciitilde'] = 570;
-  t['exclamdown'] = 389;
-  t['cent'] = 500;
-  t['sterling'] = 500;
-  t['fraction'] = 167;
-  t['yen'] = 500;
-  t['florin'] = 500;
-  t['section'] = 500;
-  t['currency'] = 500;
-  t['quotesingle'] = 278;
-  t['quotedblleft'] = 500;
-  t['guillemotleft'] = 500;
-  t['guilsinglleft'] = 333;
-  t['guilsinglright'] = 333;
-  t['fi'] = 556;
-  t['fl'] = 556;
-  t['endash'] = 500;
-  t['dagger'] = 500;
-  t['daggerdbl'] = 500;
-  t['periodcentered'] = 250;
-  t['paragraph'] = 500;
-  t['bullet'] = 350;
-  t['quotesinglbase'] = 333;
-  t['quotedblbase'] = 500;
-  t['quotedblright'] = 500;
-  t['guillemotright'] = 500;
-  t['ellipsis'] = 1000;
-  t['perthousand'] = 1000;
-  t['questiondown'] = 500;
-  t['grave'] = 333;
-  t['acute'] = 333;
-  t['circumflex'] = 333;
-  t['tilde'] = 333;
-  t['macron'] = 333;
-  t['breve'] = 333;
-  t['dotaccent'] = 333;
-  t['dieresis'] = 333;
-  t['ring'] = 333;
-  t['cedilla'] = 333;
-  t['hungarumlaut'] = 333;
-  t['ogonek'] = 333;
-  t['caron'] = 333;
-  t['emdash'] = 1000;
-  t['AE'] = 944;
-  t['ordfeminine'] = 266;
-  t['Lslash'] = 611;
-  t['Oslash'] = 722;
-  t['OE'] = 944;
-  t['ordmasculine'] = 300;
-  t['ae'] = 722;
-  t['dotlessi'] = 278;
-  t['lslash'] = 278;
-  t['oslash'] = 500;
-  t['oe'] = 722;
-  t['germandbls'] = 500;
-  t['Idieresis'] = 389;
-  t['eacute'] = 444;
-  t['abreve'] = 500;
-  t['uhungarumlaut'] = 556;
-  t['ecaron'] = 444;
-  t['Ydieresis'] = 611;
-  t['divide'] = 570;
-  t['Yacute'] = 611;
-  t['Acircumflex'] = 667;
-  t['aacute'] = 500;
-  t['Ucircumflex'] = 722;
-  t['yacute'] = 444;
-  t['scommaaccent'] = 389;
-  t['ecircumflex'] = 444;
-  t['Uring'] = 722;
-  t['Udieresis'] = 722;
-  t['aogonek'] = 500;
-  t['Uacute'] = 722;
-  t['uogonek'] = 556;
-  t['Edieresis'] = 667;
-  t['Dcroat'] = 722;
-  t['commaaccent'] = 250;
-  t['copyright'] = 747;
-  t['Emacron'] = 667;
-  t['ccaron'] = 444;
-  t['aring'] = 500;
-  t['Ncommaaccent'] = 722;
-  t['lacute'] = 278;
-  t['agrave'] = 500;
-  t['Tcommaaccent'] = 611;
-  t['Cacute'] = 667;
-  t['atilde'] = 500;
-  t['Edotaccent'] = 667;
-  t['scaron'] = 389;
-  t['scedilla'] = 389;
-  t['iacute'] = 278;
-  t['lozenge'] = 494;
-  t['Rcaron'] = 667;
-  t['Gcommaaccent'] = 722;
-  t['ucircumflex'] = 556;
-  t['acircumflex'] = 500;
-  t['Amacron'] = 667;
-  t['rcaron'] = 389;
-  t['ccedilla'] = 444;
-  t['Zdotaccent'] = 611;
-  t['Thorn'] = 611;
-  t['Omacron'] = 722;
-  t['Racute'] = 667;
-  t['Sacute'] = 556;
-  t['dcaron'] = 608;
-  t['Umacron'] = 722;
-  t['uring'] = 556;
-  t['threesuperior'] = 300;
-  t['Ograve'] = 722;
-  t['Agrave'] = 667;
-  t['Abreve'] = 667;
-  t['multiply'] = 570;
-  t['uacute'] = 556;
-  t['Tcaron'] = 611;
-  t['partialdiff'] = 494;
-  t['ydieresis'] = 444;
-  t['Nacute'] = 722;
-  t['icircumflex'] = 278;
-  t['Ecircumflex'] = 667;
-  t['adieresis'] = 500;
-  t['edieresis'] = 444;
-  t['cacute'] = 444;
-  t['nacute'] = 556;
-  t['umacron'] = 556;
-  t['Ncaron'] = 722;
-  t['Iacute'] = 389;
-  t['plusminus'] = 570;
-  t['brokenbar'] = 220;
-  t['registered'] = 747;
-  t['Gbreve'] = 722;
-  t['Idotaccent'] = 389;
-  t['summation'] = 600;
-  t['Egrave'] = 667;
-  t['racute'] = 389;
-  t['omacron'] = 500;
-  t['Zacute'] = 611;
-  t['Zcaron'] = 611;
-  t['greaterequal'] = 549;
-  t['Eth'] = 722;
-  t['Ccedilla'] = 667;
-  t['lcommaaccent'] = 278;
-  t['tcaron'] = 366;
-  t['eogonek'] = 444;
-  t['Uogonek'] = 722;
-  t['Aacute'] = 667;
-  t['Adieresis'] = 667;
-  t['egrave'] = 444;
-  t['zacute'] = 389;
-  t['iogonek'] = 278;
-  t['Oacute'] = 722;
-  t['oacute'] = 500;
-  t['amacron'] = 500;
-  t['sacute'] = 389;
-  t['idieresis'] = 278;
-  t['Ocircumflex'] = 722;
-  t['Ugrave'] = 722;
-  t['Delta'] = 612;
-  t['thorn'] = 500;
-  t['twosuperior'] = 300;
-  t['Odieresis'] = 722;
-  t['mu'] = 576;
-  t['igrave'] = 278;
-  t['ohungarumlaut'] = 500;
-  t['Eogonek'] = 667;
-  t['dcroat'] = 500;
-  t['threequarters'] = 750;
-  t['Scedilla'] = 556;
-  t['lcaron'] = 382;
-  t['Kcommaaccent'] = 667;
-  t['Lacute'] = 611;
-  t['trademark'] = 1000;
-  t['edotaccent'] = 444;
-  t['Igrave'] = 389;
-  t['Imacron'] = 389;
-  t['Lcaron'] = 611;
-  t['onehalf'] = 750;
-  t['lessequal'] = 549;
-  t['ocircumflex'] = 500;
-  t['ntilde'] = 556;
-  t['Uhungarumlaut'] = 722;
-  t['Eacute'] = 667;
-  t['emacron'] = 444;
-  t['gbreve'] = 500;
-  t['onequarter'] = 750;
-  t['Scaron'] = 556;
-  t['Scommaaccent'] = 556;
-  t['Ohungarumlaut'] = 722;
-  t['degree'] = 400;
-  t['ograve'] = 500;
-  t['Ccaron'] = 667;
-  t['ugrave'] = 556;
-  t['radical'] = 549;
-  t['Dcaron'] = 722;
-  t['rcommaaccent'] = 389;
-  t['Ntilde'] = 722;
-  t['otilde'] = 500;
-  t['Rcommaaccent'] = 667;
-  t['Lcommaaccent'] = 611;
-  t['Atilde'] = 667;
-  t['Aogonek'] = 667;
-  t['Aring'] = 667;
-  t['Otilde'] = 722;
-  t['zdotaccent'] = 389;
-  t['Ecaron'] = 667;
-  t['Iogonek'] = 389;
-  t['kcommaaccent'] = 500;
-  t['minus'] = 606;
-  t['Icircumflex'] = 389;
-  t['ncaron'] = 556;
-  t['tcommaaccent'] = 278;
-  t['logicalnot'] = 606;
-  t['odieresis'] = 500;
-  t['udieresis'] = 556;
-  t['notequal'] = 549;
-  t['gcommaaccent'] = 500;
-  t['eth'] = 500;
-  t['zcaron'] = 389;
-  t['ncommaaccent'] = 556;
-  t['onesuperior'] = 300;
-  t['imacron'] = 278;
-  t['Euro'] = 500;
- });
- t['Times-Italic'] = getLookupTableFactory(function (t) {
-  t['space'] = 250;
-  t['exclam'] = 333;
-  t['quotedbl'] = 420;
-  t['numbersign'] = 500;
-  t['dollar'] = 500;
-  t['percent'] = 833;
-  t['ampersand'] = 778;
-  t['quoteright'] = 333;
-  t['parenleft'] = 333;
-  t['parenright'] = 333;
-  t['asterisk'] = 500;
-  t['plus'] = 675;
-  t['comma'] = 250;
-  t['hyphen'] = 333;
-  t['period'] = 250;
-  t['slash'] = 278;
-  t['zero'] = 500;
-  t['one'] = 500;
-  t['two'] = 500;
-  t['three'] = 500;
-  t['four'] = 500;
-  t['five'] = 500;
-  t['six'] = 500;
-  t['seven'] = 500;
-  t['eight'] = 500;
-  t['nine'] = 500;
-  t['colon'] = 333;
-  t['semicolon'] = 333;
-  t['less'] = 675;
-  t['equal'] = 675;
-  t['greater'] = 675;
-  t['question'] = 500;
-  t['at'] = 920;
-  t['A'] = 611;
-  t['B'] = 611;
-  t['C'] = 667;
-  t['D'] = 722;
-  t['E'] = 611;
-  t['F'] = 611;
-  t['G'] = 722;
-  t['H'] = 722;
-  t['I'] = 333;
-  t['J'] = 444;
-  t['K'] = 667;
-  t['L'] = 556;
-  t['M'] = 833;
-  t['N'] = 667;
-  t['O'] = 722;
-  t['P'] = 611;
-  t['Q'] = 722;
-  t['R'] = 611;
-  t['S'] = 500;
-  t['T'] = 556;
-  t['U'] = 722;
-  t['V'] = 611;
-  t['W'] = 833;
-  t['X'] = 611;
-  t['Y'] = 556;
-  t['Z'] = 556;
-  t['bracketleft'] = 389;
-  t['backslash'] = 278;
-  t['bracketright'] = 389;
-  t['asciicircum'] = 422;
-  t['underscore'] = 500;
-  t['quoteleft'] = 333;
-  t['a'] = 500;
-  t['b'] = 500;
-  t['c'] = 444;
-  t['d'] = 500;
-  t['e'] = 444;
-  t['f'] = 278;
-  t['g'] = 500;
-  t['h'] = 500;
-  t['i'] = 278;
-  t['j'] = 278;
-  t['k'] = 444;
-  t['l'] = 278;
-  t['m'] = 722;
-  t['n'] = 500;
-  t['o'] = 500;
-  t['p'] = 500;
-  t['q'] = 500;
-  t['r'] = 389;
-  t['s'] = 389;
-  t['t'] = 278;
-  t['u'] = 500;
-  t['v'] = 444;
-  t['w'] = 667;
-  t['x'] = 444;
-  t['y'] = 444;
-  t['z'] = 389;
-  t['braceleft'] = 400;
-  t['bar'] = 275;
-  t['braceright'] = 400;
-  t['asciitilde'] = 541;
-  t['exclamdown'] = 389;
-  t['cent'] = 500;
-  t['sterling'] = 500;
-  t['fraction'] = 167;
-  t['yen'] = 500;
-  t['florin'] = 500;
-  t['section'] = 500;
-  t['currency'] = 500;
-  t['quotesingle'] = 214;
-  t['quotedblleft'] = 556;
-  t['guillemotleft'] = 500;
-  t['guilsinglleft'] = 333;
-  t['guilsinglright'] = 333;
-  t['fi'] = 500;
-  t['fl'] = 500;
-  t['endash'] = 500;
-  t['dagger'] = 500;
-  t['daggerdbl'] = 500;
-  t['periodcentered'] = 250;
-  t['paragraph'] = 523;
-  t['bullet'] = 350;
-  t['quotesinglbase'] = 333;
-  t['quotedblbase'] = 556;
-  t['quotedblright'] = 556;
-  t['guillemotright'] = 500;
-  t['ellipsis'] = 889;
-  t['perthousand'] = 1000;
-  t['questiondown'] = 500;
-  t['grave'] = 333;
-  t['acute'] = 333;
-  t['circumflex'] = 333;
-  t['tilde'] = 333;
-  t['macron'] = 333;
-  t['breve'] = 333;
-  t['dotaccent'] = 333;
-  t['dieresis'] = 333;
-  t['ring'] = 333;
-  t['cedilla'] = 333;
-  t['hungarumlaut'] = 333;
-  t['ogonek'] = 333;
-  t['caron'] = 333;
-  t['emdash'] = 889;
-  t['AE'] = 889;
-  t['ordfeminine'] = 276;
-  t['Lslash'] = 556;
-  t['Oslash'] = 722;
-  t['OE'] = 944;
-  t['ordmasculine'] = 310;
-  t['ae'] = 667;
-  t['dotlessi'] = 278;
-  t['lslash'] = 278;
-  t['oslash'] = 500;
-  t['oe'] = 667;
-  t['germandbls'] = 500;
-  t['Idieresis'] = 333;
-  t['eacute'] = 444;
-  t['abreve'] = 500;
-  t['uhungarumlaut'] = 500;
-  t['ecaron'] = 444;
-  t['Ydieresis'] = 556;
-  t['divide'] = 675;
-  t['Yacute'] = 556;
-  t['Acircumflex'] = 611;
-  t['aacute'] = 500;
-  t['Ucircumflex'] = 722;
-  t['yacute'] = 444;
-  t['scommaaccent'] = 389;
-  t['ecircumflex'] = 444;
-  t['Uring'] = 722;
-  t['Udieresis'] = 722;
-  t['aogonek'] = 500;
-  t['Uacute'] = 722;
-  t['uogonek'] = 500;
-  t['Edieresis'] = 611;
-  t['Dcroat'] = 722;
-  t['commaaccent'] = 250;
-  t['copyright'] = 760;
-  t['Emacron'] = 611;
-  t['ccaron'] = 444;
-  t['aring'] = 500;
-  t['Ncommaaccent'] = 667;
-  t['lacute'] = 278;
-  t['agrave'] = 500;
-  t['Tcommaaccent'] = 556;
-  t['Cacute'] = 667;
-  t['atilde'] = 500;
-  t['Edotaccent'] = 611;
-  t['scaron'] = 389;
-  t['scedilla'] = 389;
-  t['iacute'] = 278;
-  t['lozenge'] = 471;
-  t['Rcaron'] = 611;
-  t['Gcommaaccent'] = 722;
-  t['ucircumflex'] = 500;
-  t['acircumflex'] = 500;
-  t['Amacron'] = 611;
-  t['rcaron'] = 389;
-  t['ccedilla'] = 444;
-  t['Zdotaccent'] = 556;
-  t['Thorn'] = 611;
-  t['Omacron'] = 722;
-  t['Racute'] = 611;
-  t['Sacute'] = 500;
-  t['dcaron'] = 544;
-  t['Umacron'] = 722;
-  t['uring'] = 500;
-  t['threesuperior'] = 300;
-  t['Ograve'] = 722;
-  t['Agrave'] = 611;
-  t['Abreve'] = 611;
-  t['multiply'] = 675;
-  t['uacute'] = 500;
-  t['Tcaron'] = 556;
-  t['partialdiff'] = 476;
-  t['ydieresis'] = 444;
-  t['Nacute'] = 667;
-  t['icircumflex'] = 278;
-  t['Ecircumflex'] = 611;
-  t['adieresis'] = 500;
-  t['edieresis'] = 444;
-  t['cacute'] = 444;
-  t['nacute'] = 500;
-  t['umacron'] = 500;
-  t['Ncaron'] = 667;
-  t['Iacute'] = 333;
-  t['plusminus'] = 675;
-  t['brokenbar'] = 275;
-  t['registered'] = 760;
-  t['Gbreve'] = 722;
-  t['Idotaccent'] = 333;
-  t['summation'] = 600;
-  t['Egrave'] = 611;
-  t['racute'] = 389;
-  t['omacron'] = 500;
-  t['Zacute'] = 556;
-  t['Zcaron'] = 556;
-  t['greaterequal'] = 549;
-  t['Eth'] = 722;
-  t['Ccedilla'] = 667;
-  t['lcommaaccent'] = 278;
-  t['tcaron'] = 300;
-  t['eogonek'] = 444;
-  t['Uogonek'] = 722;
-  t['Aacute'] = 611;
-  t['Adieresis'] = 611;
-  t['egrave'] = 444;
-  t['zacute'] = 389;
-  t['iogonek'] = 278;
-  t['Oacute'] = 722;
-  t['oacute'] = 500;
-  t['amacron'] = 500;
-  t['sacute'] = 389;
-  t['idieresis'] = 278;
-  t['Ocircumflex'] = 722;
-  t['Ugrave'] = 722;
-  t['Delta'] = 612;
-  t['thorn'] = 500;
-  t['twosuperior'] = 300;
-  t['Odieresis'] = 722;
-  t['mu'] = 500;
-  t['igrave'] = 278;
-  t['ohungarumlaut'] = 500;
-  t['Eogonek'] = 611;
-  t['dcroat'] = 500;
-  t['threequarters'] = 750;
-  t['Scedilla'] = 500;
-  t['lcaron'] = 300;
-  t['Kcommaaccent'] = 667;
-  t['Lacute'] = 556;
-  t['trademark'] = 980;
-  t['edotaccent'] = 444;
-  t['Igrave'] = 333;
-  t['Imacron'] = 333;
-  t['Lcaron'] = 611;
-  t['onehalf'] = 750;
-  t['lessequal'] = 549;
-  t['ocircumflex'] = 500;
-  t['ntilde'] = 500;
-  t['Uhungarumlaut'] = 722;
-  t['Eacute'] = 611;
-  t['emacron'] = 444;
-  t['gbreve'] = 500;
-  t['onequarter'] = 750;
-  t['Scaron'] = 500;
-  t['Scommaaccent'] = 500;
-  t['Ohungarumlaut'] = 722;
-  t['degree'] = 400;
-  t['ograve'] = 500;
-  t['Ccaron'] = 667;
-  t['ugrave'] = 500;
-  t['radical'] = 453;
-  t['Dcaron'] = 722;
-  t['rcommaaccent'] = 389;
-  t['Ntilde'] = 667;
-  t['otilde'] = 500;
-  t['Rcommaaccent'] = 611;
-  t['Lcommaaccent'] = 556;
-  t['Atilde'] = 611;
-  t['Aogonek'] = 611;
-  t['Aring'] = 611;
-  t['Otilde'] = 722;
-  t['zdotaccent'] = 389;
-  t['Ecaron'] = 611;
-  t['Iogonek'] = 333;
-  t['kcommaaccent'] = 444;
-  t['minus'] = 675;
-  t['Icircumflex'] = 333;
-  t['ncaron'] = 500;
-  t['tcommaaccent'] = 278;
-  t['logicalnot'] = 675;
-  t['odieresis'] = 500;
-  t['udieresis'] = 500;
-  t['notequal'] = 549;
-  t['gcommaaccent'] = 500;
-  t['eth'] = 500;
-  t['zcaron'] = 389;
-  t['ncommaaccent'] = 500;
-  t['onesuperior'] = 300;
-  t['imacron'] = 278;
-  t['Euro'] = 500;
- });
- t['ZapfDingbats'] = getLookupTableFactory(function (t) {
-  t['space'] = 278;
-  t['a1'] = 974;
-  t['a2'] = 961;
-  t['a202'] = 974;
-  t['a3'] = 980;
-  t['a4'] = 719;
-  t['a5'] = 789;
-  t['a119'] = 790;
-  t['a118'] = 791;
-  t['a117'] = 690;
-  t['a11'] = 960;
-  t['a12'] = 939;
-  t['a13'] = 549;
-  t['a14'] = 855;
-  t['a15'] = 911;
-  t['a16'] = 933;
-  t['a105'] = 911;
-  t['a17'] = 945;
-  t['a18'] = 974;
-  t['a19'] = 755;
-  t['a20'] = 846;
-  t['a21'] = 762;
-  t['a22'] = 761;
-  t['a23'] = 571;
-  t['a24'] = 677;
-  t['a25'] = 763;
-  t['a26'] = 760;
-  t['a27'] = 759;
-  t['a28'] = 754;
-  t['a6'] = 494;
-  t['a7'] = 552;
-  t['a8'] = 537;
-  t['a9'] = 577;
-  t['a10'] = 692;
-  t['a29'] = 786;
-  t['a30'] = 788;
-  t['a31'] = 788;
-  t['a32'] = 790;
-  t['a33'] = 793;
-  t['a34'] = 794;
-  t['a35'] = 816;
-  t['a36'] = 823;
-  t['a37'] = 789;
-  t['a38'] = 841;
-  t['a39'] = 823;
-  t['a40'] = 833;
-  t['a41'] = 816;
-  t['a42'] = 831;
-  t['a43'] = 923;
-  t['a44'] = 744;
-  t['a45'] = 723;
-  t['a46'] = 749;
-  t['a47'] = 790;
-  t['a48'] = 792;
-  t['a49'] = 695;
-  t['a50'] = 776;
-  t['a51'] = 768;
-  t['a52'] = 792;
-  t['a53'] = 759;
-  t['a54'] = 707;
-  t['a55'] = 708;
-  t['a56'] = 682;
-  t['a57'] = 701;
-  t['a58'] = 826;
-  t['a59'] = 815;
-  t['a60'] = 789;
-  t['a61'] = 789;
-  t['a62'] = 707;
-  t['a63'] = 687;
-  t['a64'] = 696;
-  t['a65'] = 689;
-  t['a66'] = 786;
-  t['a67'] = 787;
-  t['a68'] = 713;
-  t['a69'] = 791;
-  t['a70'] = 785;
-  t['a71'] = 791;
-  t['a72'] = 873;
-  t['a73'] = 761;
-  t['a74'] = 762;
-  t['a203'] = 762;
-  t['a75'] = 759;
-  t['a204'] = 759;
-  t['a76'] = 892;
-  t['a77'] = 892;
-  t['a78'] = 788;
-  t['a79'] = 784;
-  t['a81'] = 438;
-  t['a82'] = 138;
-  t['a83'] = 277;
-  t['a84'] = 415;
-  t['a97'] = 392;
-  t['a98'] = 392;
-  t['a99'] = 668;
-  t['a100'] = 668;
-  t['a89'] = 390;
-  t['a90'] = 390;
-  t['a93'] = 317;
-  t['a94'] = 317;
-  t['a91'] = 276;
-  t['a92'] = 276;
-  t['a205'] = 509;
-  t['a85'] = 509;
-  t['a206'] = 410;
-  t['a86'] = 410;
-  t['a87'] = 234;
-  t['a88'] = 234;
-  t['a95'] = 334;
-  t['a96'] = 334;
-  t['a101'] = 732;
-  t['a102'] = 544;
-  t['a103'] = 544;
-  t['a104'] = 910;
-  t['a106'] = 667;
-  t['a107'] = 760;
-  t['a108'] = 760;
-  t['a112'] = 776;
-  t['a111'] = 595;
-  t['a110'] = 694;
-  t['a109'] = 626;
-  t['a120'] = 788;
-  t['a121'] = 788;
-  t['a122'] = 788;
-  t['a123'] = 788;
-  t['a124'] = 788;
-  t['a125'] = 788;
-  t['a126'] = 788;
-  t['a127'] = 788;
-  t['a128'] = 788;
-  t['a129'] = 788;
-  t['a130'] = 788;
-  t['a131'] = 788;
-  t['a132'] = 788;
-  t['a133'] = 788;
-  t['a134'] = 788;
-  t['a135'] = 788;
-  t['a136'] = 788;
-  t['a137'] = 788;
-  t['a138'] = 788;
-  t['a139'] = 788;
-  t['a140'] = 788;
-  t['a141'] = 788;
-  t['a142'] = 788;
-  t['a143'] = 788;
-  t['a144'] = 788;
-  t['a145'] = 788;
-  t['a146'] = 788;
-  t['a147'] = 788;
-  t['a148'] = 788;
-  t['a149'] = 788;
-  t['a150'] = 788;
-  t['a151'] = 788;
-  t['a152'] = 788;
-  t['a153'] = 788;
-  t['a154'] = 788;
-  t['a155'] = 788;
-  t['a156'] = 788;
-  t['a157'] = 788;
-  t['a158'] = 788;
-  t['a159'] = 788;
-  t['a160'] = 894;
-  t['a161'] = 838;
-  t['a163'] = 1016;
-  t['a164'] = 458;
-  t['a196'] = 748;
-  t['a165'] = 924;
-  t['a192'] = 748;
-  t['a166'] = 918;
-  t['a167'] = 927;
-  t['a168'] = 928;
-  t['a169'] = 928;
-  t['a170'] = 834;
-  t['a171'] = 873;
-  t['a172'] = 828;
-  t['a173'] = 924;
-  t['a162'] = 924;
-  t['a174'] = 917;
-  t['a175'] = 930;
-  t['a176'] = 931;
-  t['a177'] = 463;
-  t['a178'] = 883;
-  t['a179'] = 836;
-  t['a193'] = 836;
-  t['a180'] = 867;
-  t['a199'] = 867;
-  t['a181'] = 696;
-  t['a200'] = 696;
-  t['a182'] = 874;
-  t['a201'] = 874;
-  t['a183'] = 760;
-  t['a184'] = 946;
-  t['a197'] = 771;
-  t['a185'] = 865;
-  t['a194'] = 771;
-  t['a198'] = 888;
-  t['a186'] = 967;
-  t['a195'] = 888;
-  t['a187'] = 831;
-  t['a188'] = 873;
-  t['a189'] = 927;
-  t['a190'] = 970;
-  t['a191'] = 918;
- });
+  t['Courier'] = 600;
+  t['Courier-Bold'] = 600;
+  t['Courier-BoldOblique'] = 600;
+  t['Courier-Oblique'] = 600;
+  t['Helvetica'] = getLookupTableFactory(function (t) {
+    t['space'] = 278;
+    t['exclam'] = 278;
+    t['quotedbl'] = 355;
+    t['numbersign'] = 556;
+    t['dollar'] = 556;
+    t['percent'] = 889;
+    t['ampersand'] = 667;
+    t['quoteright'] = 222;
+    t['parenleft'] = 333;
+    t['parenright'] = 333;
+    t['asterisk'] = 389;
+    t['plus'] = 584;
+    t['comma'] = 278;
+    t['hyphen'] = 333;
+    t['period'] = 278;
+    t['slash'] = 278;
+    t['zero'] = 556;
+    t['one'] = 556;
+    t['two'] = 556;
+    t['three'] = 556;
+    t['four'] = 556;
+    t['five'] = 556;
+    t['six'] = 556;
+    t['seven'] = 556;
+    t['eight'] = 556;
+    t['nine'] = 556;
+    t['colon'] = 278;
+    t['semicolon'] = 278;
+    t['less'] = 584;
+    t['equal'] = 584;
+    t['greater'] = 584;
+    t['question'] = 556;
+    t['at'] = 1015;
+    t['A'] = 667;
+    t['B'] = 667;
+    t['C'] = 722;
+    t['D'] = 722;
+    t['E'] = 667;
+    t['F'] = 611;
+    t['G'] = 778;
+    t['H'] = 722;
+    t['I'] = 278;
+    t['J'] = 500;
+    t['K'] = 667;
+    t['L'] = 556;
+    t['M'] = 833;
+    t['N'] = 722;
+    t['O'] = 778;
+    t['P'] = 667;
+    t['Q'] = 778;
+    t['R'] = 722;
+    t['S'] = 667;
+    t['T'] = 611;
+    t['U'] = 722;
+    t['V'] = 667;
+    t['W'] = 944;
+    t['X'] = 667;
+    t['Y'] = 667;
+    t['Z'] = 611;
+    t['bracketleft'] = 278;
+    t['backslash'] = 278;
+    t['bracketright'] = 278;
+    t['asciicircum'] = 469;
+    t['underscore'] = 556;
+    t['quoteleft'] = 222;
+    t['a'] = 556;
+    t['b'] = 556;
+    t['c'] = 500;
+    t['d'] = 556;
+    t['e'] = 556;
+    t['f'] = 278;
+    t['g'] = 556;
+    t['h'] = 556;
+    t['i'] = 222;
+    t['j'] = 222;
+    t['k'] = 500;
+    t['l'] = 222;
+    t['m'] = 833;
+    t['n'] = 556;
+    t['o'] = 556;
+    t['p'] = 556;
+    t['q'] = 556;
+    t['r'] = 333;
+    t['s'] = 500;
+    t['t'] = 278;
+    t['u'] = 556;
+    t['v'] = 500;
+    t['w'] = 722;
+    t['x'] = 500;
+    t['y'] = 500;
+    t['z'] = 500;
+    t['braceleft'] = 334;
+    t['bar'] = 260;
+    t['braceright'] = 334;
+    t['asciitilde'] = 584;
+    t['exclamdown'] = 333;
+    t['cent'] = 556;
+    t['sterling'] = 556;
+    t['fraction'] = 167;
+    t['yen'] = 556;
+    t['florin'] = 556;
+    t['section'] = 556;
+    t['currency'] = 556;
+    t['quotesingle'] = 191;
+    t['quotedblleft'] = 333;
+    t['guillemotleft'] = 556;
+    t['guilsinglleft'] = 333;
+    t['guilsinglright'] = 333;
+    t['fi'] = 500;
+    t['fl'] = 500;
+    t['endash'] = 556;
+    t['dagger'] = 556;
+    t['daggerdbl'] = 556;
+    t['periodcentered'] = 278;
+    t['paragraph'] = 537;
+    t['bullet'] = 350;
+    t['quotesinglbase'] = 222;
+    t['quotedblbase'] = 333;
+    t['quotedblright'] = 333;
+    t['guillemotright'] = 556;
+    t['ellipsis'] = 1000;
+    t['perthousand'] = 1000;
+    t['questiondown'] = 611;
+    t['grave'] = 333;
+    t['acute'] = 333;
+    t['circumflex'] = 333;
+    t['tilde'] = 333;
+    t['macron'] = 333;
+    t['breve'] = 333;
+    t['dotaccent'] = 333;
+    t['dieresis'] = 333;
+    t['ring'] = 333;
+    t['cedilla'] = 333;
+    t['hungarumlaut'] = 333;
+    t['ogonek'] = 333;
+    t['caron'] = 333;
+    t['emdash'] = 1000;
+    t['AE'] = 1000;
+    t['ordfeminine'] = 370;
+    t['Lslash'] = 556;
+    t['Oslash'] = 778;
+    t['OE'] = 1000;
+    t['ordmasculine'] = 365;
+    t['ae'] = 889;
+    t['dotlessi'] = 278;
+    t['lslash'] = 222;
+    t['oslash'] = 611;
+    t['oe'] = 944;
+    t['germandbls'] = 611;
+    t['Idieresis'] = 278;
+    t['eacute'] = 556;
+    t['abreve'] = 556;
+    t['uhungarumlaut'] = 556;
+    t['ecaron'] = 556;
+    t['Ydieresis'] = 667;
+    t['divide'] = 584;
+    t['Yacute'] = 667;
+    t['Acircumflex'] = 667;
+    t['aacute'] = 556;
+    t['Ucircumflex'] = 722;
+    t['yacute'] = 500;
+    t['scommaaccent'] = 500;
+    t['ecircumflex'] = 556;
+    t['Uring'] = 722;
+    t['Udieresis'] = 722;
+    t['aogonek'] = 556;
+    t['Uacute'] = 722;
+    t['uogonek'] = 556;
+    t['Edieresis'] = 667;
+    t['Dcroat'] = 722;
+    t['commaaccent'] = 250;
+    t['copyright'] = 737;
+    t['Emacron'] = 667;
+    t['ccaron'] = 500;
+    t['aring'] = 556;
+    t['Ncommaaccent'] = 722;
+    t['lacute'] = 222;
+    t['agrave'] = 556;
+    t['Tcommaaccent'] = 611;
+    t['Cacute'] = 722;
+    t['atilde'] = 556;
+    t['Edotaccent'] = 667;
+    t['scaron'] = 500;
+    t['scedilla'] = 500;
+    t['iacute'] = 278;
+    t['lozenge'] = 471;
+    t['Rcaron'] = 722;
+    t['Gcommaaccent'] = 778;
+    t['ucircumflex'] = 556;
+    t['acircumflex'] = 556;
+    t['Amacron'] = 667;
+    t['rcaron'] = 333;
+    t['ccedilla'] = 500;
+    t['Zdotaccent'] = 611;
+    t['Thorn'] = 667;
+    t['Omacron'] = 778;
+    t['Racute'] = 722;
+    t['Sacute'] = 667;
+    t['dcaron'] = 643;
+    t['Umacron'] = 722;
+    t['uring'] = 556;
+    t['threesuperior'] = 333;
+    t['Ograve'] = 778;
+    t['Agrave'] = 667;
+    t['Abreve'] = 667;
+    t['multiply'] = 584;
+    t['uacute'] = 556;
+    t['Tcaron'] = 611;
+    t['partialdiff'] = 476;
+    t['ydieresis'] = 500;
+    t['Nacute'] = 722;
+    t['icircumflex'] = 278;
+    t['Ecircumflex'] = 667;
+    t['adieresis'] = 556;
+    t['edieresis'] = 556;
+    t['cacute'] = 500;
+    t['nacute'] = 556;
+    t['umacron'] = 556;
+    t['Ncaron'] = 722;
+    t['Iacute'] = 278;
+    t['plusminus'] = 584;
+    t['brokenbar'] = 260;
+    t['registered'] = 737;
+    t['Gbreve'] = 778;
+    t['Idotaccent'] = 278;
+    t['summation'] = 600;
+    t['Egrave'] = 667;
+    t['racute'] = 333;
+    t['omacron'] = 556;
+    t['Zacute'] = 611;
+    t['Zcaron'] = 611;
+    t['greaterequal'] = 549;
+    t['Eth'] = 722;
+    t['Ccedilla'] = 722;
+    t['lcommaaccent'] = 222;
+    t['tcaron'] = 317;
+    t['eogonek'] = 556;
+    t['Uogonek'] = 722;
+    t['Aacute'] = 667;
+    t['Adieresis'] = 667;
+    t['egrave'] = 556;
+    t['zacute'] = 500;
+    t['iogonek'] = 222;
+    t['Oacute'] = 778;
+    t['oacute'] = 556;
+    t['amacron'] = 556;
+    t['sacute'] = 500;
+    t['idieresis'] = 278;
+    t['Ocircumflex'] = 778;
+    t['Ugrave'] = 722;
+    t['Delta'] = 612;
+    t['thorn'] = 556;
+    t['twosuperior'] = 333;
+    t['Odieresis'] = 778;
+    t['mu'] = 556;
+    t['igrave'] = 278;
+    t['ohungarumlaut'] = 556;
+    t['Eogonek'] = 667;
+    t['dcroat'] = 556;
+    t['threequarters'] = 834;
+    t['Scedilla'] = 667;
+    t['lcaron'] = 299;
+    t['Kcommaaccent'] = 667;
+    t['Lacute'] = 556;
+    t['trademark'] = 1000;
+    t['edotaccent'] = 556;
+    t['Igrave'] = 278;
+    t['Imacron'] = 278;
+    t['Lcaron'] = 556;
+    t['onehalf'] = 834;
+    t['lessequal'] = 549;
+    t['ocircumflex'] = 556;
+    t['ntilde'] = 556;
+    t['Uhungarumlaut'] = 722;
+    t['Eacute'] = 667;
+    t['emacron'] = 556;
+    t['gbreve'] = 556;
+    t['onequarter'] = 834;
+    t['Scaron'] = 667;
+    t['Scommaaccent'] = 667;
+    t['Ohungarumlaut'] = 778;
+    t['degree'] = 400;
+    t['ograve'] = 556;
+    t['Ccaron'] = 722;
+    t['ugrave'] = 556;
+    t['radical'] = 453;
+    t['Dcaron'] = 722;
+    t['rcommaaccent'] = 333;
+    t['Ntilde'] = 722;
+    t['otilde'] = 556;
+    t['Rcommaaccent'] = 722;
+    t['Lcommaaccent'] = 556;
+    t['Atilde'] = 667;
+    t['Aogonek'] = 667;
+    t['Aring'] = 667;
+    t['Otilde'] = 778;
+    t['zdotaccent'] = 500;
+    t['Ecaron'] = 667;
+    t['Iogonek'] = 278;
+    t['kcommaaccent'] = 500;
+    t['minus'] = 584;
+    t['Icircumflex'] = 278;
+    t['ncaron'] = 556;
+    t['tcommaaccent'] = 278;
+    t['logicalnot'] = 584;
+    t['odieresis'] = 556;
+    t['udieresis'] = 556;
+    t['notequal'] = 549;
+    t['gcommaaccent'] = 556;
+    t['eth'] = 556;
+    t['zcaron'] = 500;
+    t['ncommaaccent'] = 556;
+    t['onesuperior'] = 333;
+    t['imacron'] = 278;
+    t['Euro'] = 556;
+  });
+  t['Helvetica-Bold'] = getLookupTableFactory(function (t) {
+    t['space'] = 278;
+    t['exclam'] = 333;
+    t['quotedbl'] = 474;
+    t['numbersign'] = 556;
+    t['dollar'] = 556;
+    t['percent'] = 889;
+    t['ampersand'] = 722;
+    t['quoteright'] = 278;
+    t['parenleft'] = 333;
+    t['parenright'] = 333;
+    t['asterisk'] = 389;
+    t['plus'] = 584;
+    t['comma'] = 278;
+    t['hyphen'] = 333;
+    t['period'] = 278;
+    t['slash'] = 278;
+    t['zero'] = 556;
+    t['one'] = 556;
+    t['two'] = 556;
+    t['three'] = 556;
+    t['four'] = 556;
+    t['five'] = 556;
+    t['six'] = 556;
+    t['seven'] = 556;
+    t['eight'] = 556;
+    t['nine'] = 556;
+    t['colon'] = 333;
+    t['semicolon'] = 333;
+    t['less'] = 584;
+    t['equal'] = 584;
+    t['greater'] = 584;
+    t['question'] = 611;
+    t['at'] = 975;
+    t['A'] = 722;
+    t['B'] = 722;
+    t['C'] = 722;
+    t['D'] = 722;
+    t['E'] = 667;
+    t['F'] = 611;
+    t['G'] = 778;
+    t['H'] = 722;
+    t['I'] = 278;
+    t['J'] = 556;
+    t['K'] = 722;
+    t['L'] = 611;
+    t['M'] = 833;
+    t['N'] = 722;
+    t['O'] = 778;
+    t['P'] = 667;
+    t['Q'] = 778;
+    t['R'] = 722;
+    t['S'] = 667;
+    t['T'] = 611;
+    t['U'] = 722;
+    t['V'] = 667;
+    t['W'] = 944;
+    t['X'] = 667;
+    t['Y'] = 667;
+    t['Z'] = 611;
+    t['bracketleft'] = 333;
+    t['backslash'] = 278;
+    t['bracketright'] = 333;
+    t['asciicircum'] = 584;
+    t['underscore'] = 556;
+    t['quoteleft'] = 278;
+    t['a'] = 556;
+    t['b'] = 611;
+    t['c'] = 556;
+    t['d'] = 611;
+    t['e'] = 556;
+    t['f'] = 333;
+    t['g'] = 611;
+    t['h'] = 611;
+    t['i'] = 278;
+    t['j'] = 278;
+    t['k'] = 556;
+    t['l'] = 278;
+    t['m'] = 889;
+    t['n'] = 611;
+    t['o'] = 611;
+    t['p'] = 611;
+    t['q'] = 611;
+    t['r'] = 389;
+    t['s'] = 556;
+    t['t'] = 333;
+    t['u'] = 611;
+    t['v'] = 556;
+    t['w'] = 778;
+    t['x'] = 556;
+    t['y'] = 556;
+    t['z'] = 500;
+    t['braceleft'] = 389;
+    t['bar'] = 280;
+    t['braceright'] = 389;
+    t['asciitilde'] = 584;
+    t['exclamdown'] = 333;
+    t['cent'] = 556;
+    t['sterling'] = 556;
+    t['fraction'] = 167;
+    t['yen'] = 556;
+    t['florin'] = 556;
+    t['section'] = 556;
+    t['currency'] = 556;
+    t['quotesingle'] = 238;
+    t['quotedblleft'] = 500;
+    t['guillemotleft'] = 556;
+    t['guilsinglleft'] = 333;
+    t['guilsinglright'] = 333;
+    t['fi'] = 611;
+    t['fl'] = 611;
+    t['endash'] = 556;
+    t['dagger'] = 556;
+    t['daggerdbl'] = 556;
+    t['periodcentered'] = 278;
+    t['paragraph'] = 556;
+    t['bullet'] = 350;
+    t['quotesinglbase'] = 278;
+    t['quotedblbase'] = 500;
+    t['quotedblright'] = 500;
+    t['guillemotright'] = 556;
+    t['ellipsis'] = 1000;
+    t['perthousand'] = 1000;
+    t['questiondown'] = 611;
+    t['grave'] = 333;
+    t['acute'] = 333;
+    t['circumflex'] = 333;
+    t['tilde'] = 333;
+    t['macron'] = 333;
+    t['breve'] = 333;
+    t['dotaccent'] = 333;
+    t['dieresis'] = 333;
+    t['ring'] = 333;
+    t['cedilla'] = 333;
+    t['hungarumlaut'] = 333;
+    t['ogonek'] = 333;
+    t['caron'] = 333;
+    t['emdash'] = 1000;
+    t['AE'] = 1000;
+    t['ordfeminine'] = 370;
+    t['Lslash'] = 611;
+    t['Oslash'] = 778;
+    t['OE'] = 1000;
+    t['ordmasculine'] = 365;
+    t['ae'] = 889;
+    t['dotlessi'] = 278;
+    t['lslash'] = 278;
+    t['oslash'] = 611;
+    t['oe'] = 944;
+    t['germandbls'] = 611;
+    t['Idieresis'] = 278;
+    t['eacute'] = 556;
+    t['abreve'] = 556;
+    t['uhungarumlaut'] = 611;
+    t['ecaron'] = 556;
+    t['Ydieresis'] = 667;
+    t['divide'] = 584;
+    t['Yacute'] = 667;
+    t['Acircumflex'] = 722;
+    t['aacute'] = 556;
+    t['Ucircumflex'] = 722;
+    t['yacute'] = 556;
+    t['scommaaccent'] = 556;
+    t['ecircumflex'] = 556;
+    t['Uring'] = 722;
+    t['Udieresis'] = 722;
+    t['aogonek'] = 556;
+    t['Uacute'] = 722;
+    t['uogonek'] = 611;
+    t['Edieresis'] = 667;
+    t['Dcroat'] = 722;
+    t['commaaccent'] = 250;
+    t['copyright'] = 737;
+    t['Emacron'] = 667;
+    t['ccaron'] = 556;
+    t['aring'] = 556;
+    t['Ncommaaccent'] = 722;
+    t['lacute'] = 278;
+    t['agrave'] = 556;
+    t['Tcommaaccent'] = 611;
+    t['Cacute'] = 722;
+    t['atilde'] = 556;
+    t['Edotaccent'] = 667;
+    t['scaron'] = 556;
+    t['scedilla'] = 556;
+    t['iacute'] = 278;
+    t['lozenge'] = 494;
+    t['Rcaron'] = 722;
+    t['Gcommaaccent'] = 778;
+    t['ucircumflex'] = 611;
+    t['acircumflex'] = 556;
+    t['Amacron'] = 722;
+    t['rcaron'] = 389;
+    t['ccedilla'] = 556;
+    t['Zdotaccent'] = 611;
+    t['Thorn'] = 667;
+    t['Omacron'] = 778;
+    t['Racute'] = 722;
+    t['Sacute'] = 667;
+    t['dcaron'] = 743;
+    t['Umacron'] = 722;
+    t['uring'] = 611;
+    t['threesuperior'] = 333;
+    t['Ograve'] = 778;
+    t['Agrave'] = 722;
+    t['Abreve'] = 722;
+    t['multiply'] = 584;
+    t['uacute'] = 611;
+    t['Tcaron'] = 611;
+    t['partialdiff'] = 494;
+    t['ydieresis'] = 556;
+    t['Nacute'] = 722;
+    t['icircumflex'] = 278;
+    t['Ecircumflex'] = 667;
+    t['adieresis'] = 556;
+    t['edieresis'] = 556;
+    t['cacute'] = 556;
+    t['nacute'] = 611;
+    t['umacron'] = 611;
+    t['Ncaron'] = 722;
+    t['Iacute'] = 278;
+    t['plusminus'] = 584;
+    t['brokenbar'] = 280;
+    t['registered'] = 737;
+    t['Gbreve'] = 778;
+    t['Idotaccent'] = 278;
+    t['summation'] = 600;
+    t['Egrave'] = 667;
+    t['racute'] = 389;
+    t['omacron'] = 611;
+    t['Zacute'] = 611;
+    t['Zcaron'] = 611;
+    t['greaterequal'] = 549;
+    t['Eth'] = 722;
+    t['Ccedilla'] = 722;
+    t['lcommaaccent'] = 278;
+    t['tcaron'] = 389;
+    t['eogonek'] = 556;
+    t['Uogonek'] = 722;
+    t['Aacute'] = 722;
+    t['Adieresis'] = 722;
+    t['egrave'] = 556;
+    t['zacute'] = 500;
+    t['iogonek'] = 278;
+    t['Oacute'] = 778;
+    t['oacute'] = 611;
+    t['amacron'] = 556;
+    t['sacute'] = 556;
+    t['idieresis'] = 278;
+    t['Ocircumflex'] = 778;
+    t['Ugrave'] = 722;
+    t['Delta'] = 612;
+    t['thorn'] = 611;
+    t['twosuperior'] = 333;
+    t['Odieresis'] = 778;
+    t['mu'] = 611;
+    t['igrave'] = 278;
+    t['ohungarumlaut'] = 611;
+    t['Eogonek'] = 667;
+    t['dcroat'] = 611;
+    t['threequarters'] = 834;
+    t['Scedilla'] = 667;
+    t['lcaron'] = 400;
+    t['Kcommaaccent'] = 722;
+    t['Lacute'] = 611;
+    t['trademark'] = 1000;
+    t['edotaccent'] = 556;
+    t['Igrave'] = 278;
+    t['Imacron'] = 278;
+    t['Lcaron'] = 611;
+    t['onehalf'] = 834;
+    t['lessequal'] = 549;
+    t['ocircumflex'] = 611;
+    t['ntilde'] = 611;
+    t['Uhungarumlaut'] = 722;
+    t['Eacute'] = 667;
+    t['emacron'] = 556;
+    t['gbreve'] = 611;
+    t['onequarter'] = 834;
+    t['Scaron'] = 667;
+    t['Scommaaccent'] = 667;
+    t['Ohungarumlaut'] = 778;
+    t['degree'] = 400;
+    t['ograve'] = 611;
+    t['Ccaron'] = 722;
+    t['ugrave'] = 611;
+    t['radical'] = 549;
+    t['Dcaron'] = 722;
+    t['rcommaaccent'] = 389;
+    t['Ntilde'] = 722;
+    t['otilde'] = 611;
+    t['Rcommaaccent'] = 722;
+    t['Lcommaaccent'] = 611;
+    t['Atilde'] = 722;
+    t['Aogonek'] = 722;
+    t['Aring'] = 722;
+    t['Otilde'] = 778;
+    t['zdotaccent'] = 500;
+    t['Ecaron'] = 667;
+    t['Iogonek'] = 278;
+    t['kcommaaccent'] = 556;
+    t['minus'] = 584;
+    t['Icircumflex'] = 278;
+    t['ncaron'] = 611;
+    t['tcommaaccent'] = 333;
+    t['logicalnot'] = 584;
+    t['odieresis'] = 611;
+    t['udieresis'] = 611;
+    t['notequal'] = 549;
+    t['gcommaaccent'] = 611;
+    t['eth'] = 611;
+    t['zcaron'] = 500;
+    t['ncommaaccent'] = 611;
+    t['onesuperior'] = 333;
+    t['imacron'] = 278;
+    t['Euro'] = 556;
+  });
+  t['Helvetica-BoldOblique'] = getLookupTableFactory(function (t) {
+    t['space'] = 278;
+    t['exclam'] = 333;
+    t['quotedbl'] = 474;
+    t['numbersign'] = 556;
+    t['dollar'] = 556;
+    t['percent'] = 889;
+    t['ampersand'] = 722;
+    t['quoteright'] = 278;
+    t['parenleft'] = 333;
+    t['parenright'] = 333;
+    t['asterisk'] = 389;
+    t['plus'] = 584;
+    t['comma'] = 278;
+    t['hyphen'] = 333;
+    t['period'] = 278;
+    t['slash'] = 278;
+    t['zero'] = 556;
+    t['one'] = 556;
+    t['two'] = 556;
+    t['three'] = 556;
+    t['four'] = 556;
+    t['five'] = 556;
+    t['six'] = 556;
+    t['seven'] = 556;
+    t['eight'] = 556;
+    t['nine'] = 556;
+    t['colon'] = 333;
+    t['semicolon'] = 333;
+    t['less'] = 584;
+    t['equal'] = 584;
+    t['greater'] = 584;
+    t['question'] = 611;
+    t['at'] = 975;
+    t['A'] = 722;
+    t['B'] = 722;
+    t['C'] = 722;
+    t['D'] = 722;
+    t['E'] = 667;
+    t['F'] = 611;
+    t['G'] = 778;
+    t['H'] = 722;
+    t['I'] = 278;
+    t['J'] = 556;
+    t['K'] = 722;
+    t['L'] = 611;
+    t['M'] = 833;
+    t['N'] = 722;
+    t['O'] = 778;
+    t['P'] = 667;
+    t['Q'] = 778;
+    t['R'] = 722;
+    t['S'] = 667;
+    t['T'] = 611;
+    t['U'] = 722;
+    t['V'] = 667;
+    t['W'] = 944;
+    t['X'] = 667;
+    t['Y'] = 667;
+    t['Z'] = 611;
+    t['bracketleft'] = 333;
+    t['backslash'] = 278;
+    t['bracketright'] = 333;
+    t['asciicircum'] = 584;
+    t['underscore'] = 556;
+    t['quoteleft'] = 278;
+    t['a'] = 556;
+    t['b'] = 611;
+    t['c'] = 556;
+    t['d'] = 611;
+    t['e'] = 556;
+    t['f'] = 333;
+    t['g'] = 611;
+    t['h'] = 611;
+    t['i'] = 278;
+    t['j'] = 278;
+    t['k'] = 556;
+    t['l'] = 278;
+    t['m'] = 889;
+    t['n'] = 611;
+    t['o'] = 611;
+    t['p'] = 611;
+    t['q'] = 611;
+    t['r'] = 389;
+    t['s'] = 556;
+    t['t'] = 333;
+    t['u'] = 611;
+    t['v'] = 556;
+    t['w'] = 778;
+    t['x'] = 556;
+    t['y'] = 556;
+    t['z'] = 500;
+    t['braceleft'] = 389;
+    t['bar'] = 280;
+    t['braceright'] = 389;
+    t['asciitilde'] = 584;
+    t['exclamdown'] = 333;
+    t['cent'] = 556;
+    t['sterling'] = 556;
+    t['fraction'] = 167;
+    t['yen'] = 556;
+    t['florin'] = 556;
+    t['section'] = 556;
+    t['currency'] = 556;
+    t['quotesingle'] = 238;
+    t['quotedblleft'] = 500;
+    t['guillemotleft'] = 556;
+    t['guilsinglleft'] = 333;
+    t['guilsinglright'] = 333;
+    t['fi'] = 611;
+    t['fl'] = 611;
+    t['endash'] = 556;
+    t['dagger'] = 556;
+    t['daggerdbl'] = 556;
+    t['periodcentered'] = 278;
+    t['paragraph'] = 556;
+    t['bullet'] = 350;
+    t['quotesinglbase'] = 278;
+    t['quotedblbase'] = 500;
+    t['quotedblright'] = 500;
+    t['guillemotright'] = 556;
+    t['ellipsis'] = 1000;
+    t['perthousand'] = 1000;
+    t['questiondown'] = 611;
+    t['grave'] = 333;
+    t['acute'] = 333;
+    t['circumflex'] = 333;
+    t['tilde'] = 333;
+    t['macron'] = 333;
+    t['breve'] = 333;
+    t['dotaccent'] = 333;
+    t['dieresis'] = 333;
+    t['ring'] = 333;
+    t['cedilla'] = 333;
+    t['hungarumlaut'] = 333;
+    t['ogonek'] = 333;
+    t['caron'] = 333;
+    t['emdash'] = 1000;
+    t['AE'] = 1000;
+    t['ordfeminine'] = 370;
+    t['Lslash'] = 611;
+    t['Oslash'] = 778;
+    t['OE'] = 1000;
+    t['ordmasculine'] = 365;
+    t['ae'] = 889;
+    t['dotlessi'] = 278;
+    t['lslash'] = 278;
+    t['oslash'] = 611;
+    t['oe'] = 944;
+    t['germandbls'] = 611;
+    t['Idieresis'] = 278;
+    t['eacute'] = 556;
+    t['abreve'] = 556;
+    t['uhungarumlaut'] = 611;
+    t['ecaron'] = 556;
+    t['Ydieresis'] = 667;
+    t['divide'] = 584;
+    t['Yacute'] = 667;
+    t['Acircumflex'] = 722;
+    t['aacute'] = 556;
+    t['Ucircumflex'] = 722;
+    t['yacute'] = 556;
+    t['scommaaccent'] = 556;
+    t['ecircumflex'] = 556;
+    t['Uring'] = 722;
+    t['Udieresis'] = 722;
+    t['aogonek'] = 556;
+    t['Uacute'] = 722;
+    t['uogonek'] = 611;
+    t['Edieresis'] = 667;
+    t['Dcroat'] = 722;
+    t['commaaccent'] = 250;
+    t['copyright'] = 737;
+    t['Emacron'] = 667;
+    t['ccaron'] = 556;
+    t['aring'] = 556;
+    t['Ncommaaccent'] = 722;
+    t['lacute'] = 278;
+    t['agrave'] = 556;
+    t['Tcommaaccent'] = 611;
+    t['Cacute'] = 722;
+    t['atilde'] = 556;
+    t['Edotaccent'] = 667;
+    t['scaron'] = 556;
+    t['scedilla'] = 556;
+    t['iacute'] = 278;
+    t['lozenge'] = 494;
+    t['Rcaron'] = 722;
+    t['Gcommaaccent'] = 778;
+    t['ucircumflex'] = 611;
+    t['acircumflex'] = 556;
+    t['Amacron'] = 722;
+    t['rcaron'] = 389;
+    t['ccedilla'] = 556;
+    t['Zdotaccent'] = 611;
+    t['Thorn'] = 667;
+    t['Omacron'] = 778;
+    t['Racute'] = 722;
+    t['Sacute'] = 667;
+    t['dcaron'] = 743;
+    t['Umacron'] = 722;
+    t['uring'] = 611;
+    t['threesuperior'] = 333;
+    t['Ograve'] = 778;
+    t['Agrave'] = 722;
+    t['Abreve'] = 722;
+    t['multiply'] = 584;
+    t['uacute'] = 611;
+    t['Tcaron'] = 611;
+    t['partialdiff'] = 494;
+    t['ydieresis'] = 556;
+    t['Nacute'] = 722;
+    t['icircumflex'] = 278;
+    t['Ecircumflex'] = 667;
+    t['adieresis'] = 556;
+    t['edieresis'] = 556;
+    t['cacute'] = 556;
+    t['nacute'] = 611;
+    t['umacron'] = 611;
+    t['Ncaron'] = 722;
+    t['Iacute'] = 278;
+    t['plusminus'] = 584;
+    t['brokenbar'] = 280;
+    t['registered'] = 737;
+    t['Gbreve'] = 778;
+    t['Idotaccent'] = 278;
+    t['summation'] = 600;
+    t['Egrave'] = 667;
+    t['racute'] = 389;
+    t['omacron'] = 611;
+    t['Zacute'] = 611;
+    t['Zcaron'] = 611;
+    t['greaterequal'] = 549;
+    t['Eth'] = 722;
+    t['Ccedilla'] = 722;
+    t['lcommaaccent'] = 278;
+    t['tcaron'] = 389;
+    t['eogonek'] = 556;
+    t['Uogonek'] = 722;
+    t['Aacute'] = 722;
+    t['Adieresis'] = 722;
+    t['egrave'] = 556;
+    t['zacute'] = 500;
+    t['iogonek'] = 278;
+    t['Oacute'] = 778;
+    t['oacute'] = 611;
+    t['amacron'] = 556;
+    t['sacute'] = 556;
+    t['idieresis'] = 278;
+    t['Ocircumflex'] = 778;
+    t['Ugrave'] = 722;
+    t['Delta'] = 612;
+    t['thorn'] = 611;
+    t['twosuperior'] = 333;
+    t['Odieresis'] = 778;
+    t['mu'] = 611;
+    t['igrave'] = 278;
+    t['ohungarumlaut'] = 611;
+    t['Eogonek'] = 667;
+    t['dcroat'] = 611;
+    t['threequarters'] = 834;
+    t['Scedilla'] = 667;
+    t['lcaron'] = 400;
+    t['Kcommaaccent'] = 722;
+    t['Lacute'] = 611;
+    t['trademark'] = 1000;
+    t['edotaccent'] = 556;
+    t['Igrave'] = 278;
+    t['Imacron'] = 278;
+    t['Lcaron'] = 611;
+    t['onehalf'] = 834;
+    t['lessequal'] = 549;
+    t['ocircumflex'] = 611;
+    t['ntilde'] = 611;
+    t['Uhungarumlaut'] = 722;
+    t['Eacute'] = 667;
+    t['emacron'] = 556;
+    t['gbreve'] = 611;
+    t['onequarter'] = 834;
+    t['Scaron'] = 667;
+    t['Scommaaccent'] = 667;
+    t['Ohungarumlaut'] = 778;
+    t['degree'] = 400;
+    t['ograve'] = 611;
+    t['Ccaron'] = 722;
+    t['ugrave'] = 611;
+    t['radical'] = 549;
+    t['Dcaron'] = 722;
+    t['rcommaaccent'] = 389;
+    t['Ntilde'] = 722;
+    t['otilde'] = 611;
+    t['Rcommaaccent'] = 722;
+    t['Lcommaaccent'] = 611;
+    t['Atilde'] = 722;
+    t['Aogonek'] = 722;
+    t['Aring'] = 722;
+    t['Otilde'] = 778;
+    t['zdotaccent'] = 500;
+    t['Ecaron'] = 667;
+    t['Iogonek'] = 278;
+    t['kcommaaccent'] = 556;
+    t['minus'] = 584;
+    t['Icircumflex'] = 278;
+    t['ncaron'] = 611;
+    t['tcommaaccent'] = 333;
+    t['logicalnot'] = 584;
+    t['odieresis'] = 611;
+    t['udieresis'] = 611;
+    t['notequal'] = 549;
+    t['gcommaaccent'] = 611;
+    t['eth'] = 611;
+    t['zcaron'] = 500;
+    t['ncommaaccent'] = 611;
+    t['onesuperior'] = 333;
+    t['imacron'] = 278;
+    t['Euro'] = 556;
+  });
+  t['Helvetica-Oblique'] = getLookupTableFactory(function (t) {
+    t['space'] = 278;
+    t['exclam'] = 278;
+    t['quotedbl'] = 355;
+    t['numbersign'] = 556;
+    t['dollar'] = 556;
+    t['percent'] = 889;
+    t['ampersand'] = 667;
+    t['quoteright'] = 222;
+    t['parenleft'] = 333;
+    t['parenright'] = 333;
+    t['asterisk'] = 389;
+    t['plus'] = 584;
+    t['comma'] = 278;
+    t['hyphen'] = 333;
+    t['period'] = 278;
+    t['slash'] = 278;
+    t['zero'] = 556;
+    t['one'] = 556;
+    t['two'] = 556;
+    t['three'] = 556;
+    t['four'] = 556;
+    t['five'] = 556;
+    t['six'] = 556;
+    t['seven'] = 556;
+    t['eight'] = 556;
+    t['nine'] = 556;
+    t['colon'] = 278;
+    t['semicolon'] = 278;
+    t['less'] = 584;
+    t['equal'] = 584;
+    t['greater'] = 584;
+    t['question'] = 556;
+    t['at'] = 1015;
+    t['A'] = 667;
+    t['B'] = 667;
+    t['C'] = 722;
+    t['D'] = 722;
+    t['E'] = 667;
+    t['F'] = 611;
+    t['G'] = 778;
+    t['H'] = 722;
+    t['I'] = 278;
+    t['J'] = 500;
+    t['K'] = 667;
+    t['L'] = 556;
+    t['M'] = 833;
+    t['N'] = 722;
+    t['O'] = 778;
+    t['P'] = 667;
+    t['Q'] = 778;
+    t['R'] = 722;
+    t['S'] = 667;
+    t['T'] = 611;
+    t['U'] = 722;
+    t['V'] = 667;
+    t['W'] = 944;
+    t['X'] = 667;
+    t['Y'] = 667;
+    t['Z'] = 611;
+    t['bracketleft'] = 278;
+    t['backslash'] = 278;
+    t['bracketright'] = 278;
+    t['asciicircum'] = 469;
+    t['underscore'] = 556;
+    t['quoteleft'] = 222;
+    t['a'] = 556;
+    t['b'] = 556;
+    t['c'] = 500;
+    t['d'] = 556;
+    t['e'] = 556;
+    t['f'] = 278;
+    t['g'] = 556;
+    t['h'] = 556;
+    t['i'] = 222;
+    t['j'] = 222;
+    t['k'] = 500;
+    t['l'] = 222;
+    t['m'] = 833;
+    t['n'] = 556;
+    t['o'] = 556;
+    t['p'] = 556;
+    t['q'] = 556;
+    t['r'] = 333;
+    t['s'] = 500;
+    t['t'] = 278;
+    t['u'] = 556;
+    t['v'] = 500;
+    t['w'] = 722;
+    t['x'] = 500;
+    t['y'] = 500;
+    t['z'] = 500;
+    t['braceleft'] = 334;
+    t['bar'] = 260;
+    t['braceright'] = 334;
+    t['asciitilde'] = 584;
+    t['exclamdown'] = 333;
+    t['cent'] = 556;
+    t['sterling'] = 556;
+    t['fraction'] = 167;
+    t['yen'] = 556;
+    t['florin'] = 556;
+    t['section'] = 556;
+    t['currency'] = 556;
+    t['quotesingle'] = 191;
+    t['quotedblleft'] = 333;
+    t['guillemotleft'] = 556;
+    t['guilsinglleft'] = 333;
+    t['guilsinglright'] = 333;
+    t['fi'] = 500;
+    t['fl'] = 500;
+    t['endash'] = 556;
+    t['dagger'] = 556;
+    t['daggerdbl'] = 556;
+    t['periodcentered'] = 278;
+    t['paragraph'] = 537;
+    t['bullet'] = 350;
+    t['quotesinglbase'] = 222;
+    t['quotedblbase'] = 333;
+    t['quotedblright'] = 333;
+    t['guillemotright'] = 556;
+    t['ellipsis'] = 1000;
+    t['perthousand'] = 1000;
+    t['questiondown'] = 611;
+    t['grave'] = 333;
+    t['acute'] = 333;
+    t['circumflex'] = 333;
+    t['tilde'] = 333;
+    t['macron'] = 333;
+    t['breve'] = 333;
+    t['dotaccent'] = 333;
+    t['dieresis'] = 333;
+    t['ring'] = 333;
+    t['cedilla'] = 333;
+    t['hungarumlaut'] = 333;
+    t['ogonek'] = 333;
+    t['caron'] = 333;
+    t['emdash'] = 1000;
+    t['AE'] = 1000;
+    t['ordfeminine'] = 370;
+    t['Lslash'] = 556;
+    t['Oslash'] = 778;
+    t['OE'] = 1000;
+    t['ordmasculine'] = 365;
+    t['ae'] = 889;
+    t['dotlessi'] = 278;
+    t['lslash'] = 222;
+    t['oslash'] = 611;
+    t['oe'] = 944;
+    t['germandbls'] = 611;
+    t['Idieresis'] = 278;
+    t['eacute'] = 556;
+    t['abreve'] = 556;
+    t['uhungarumlaut'] = 556;
+    t['ecaron'] = 556;
+    t['Ydieresis'] = 667;
+    t['divide'] = 584;
+    t['Yacute'] = 667;
+    t['Acircumflex'] = 667;
+    t['aacute'] = 556;
+    t['Ucircumflex'] = 722;
+    t['yacute'] = 500;
+    t['scommaaccent'] = 500;
+    t['ecircumflex'] = 556;
+    t['Uring'] = 722;
+    t['Udieresis'] = 722;
+    t['aogonek'] = 556;
+    t['Uacute'] = 722;
+    t['uogonek'] = 556;
+    t['Edieresis'] = 667;
+    t['Dcroat'] = 722;
+    t['commaaccent'] = 250;
+    t['copyright'] = 737;
+    t['Emacron'] = 667;
+    t['ccaron'] = 500;
+    t['aring'] = 556;
+    t['Ncommaaccent'] = 722;
+    t['lacute'] = 222;
+    t['agrave'] = 556;
+    t['Tcommaaccent'] = 611;
+    t['Cacute'] = 722;
+    t['atilde'] = 556;
+    t['Edotaccent'] = 667;
+    t['scaron'] = 500;
+    t['scedilla'] = 500;
+    t['iacute'] = 278;
+    t['lozenge'] = 471;
+    t['Rcaron'] = 722;
+    t['Gcommaaccent'] = 778;
+    t['ucircumflex'] = 556;
+    t['acircumflex'] = 556;
+    t['Amacron'] = 667;
+    t['rcaron'] = 333;
+    t['ccedilla'] = 500;
+    t['Zdotaccent'] = 611;
+    t['Thorn'] = 667;
+    t['Omacron'] = 778;
+    t['Racute'] = 722;
+    t['Sacute'] = 667;
+    t['dcaron'] = 643;
+    t['Umacron'] = 722;
+    t['uring'] = 556;
+    t['threesuperior'] = 333;
+    t['Ograve'] = 778;
+    t['Agrave'] = 667;
+    t['Abreve'] = 667;
+    t['multiply'] = 584;
+    t['uacute'] = 556;
+    t['Tcaron'] = 611;
+    t['partialdiff'] = 476;
+    t['ydieresis'] = 500;
+    t['Nacute'] = 722;
+    t['icircumflex'] = 278;
+    t['Ecircumflex'] = 667;
+    t['adieresis'] = 556;
+    t['edieresis'] = 556;
+    t['cacute'] = 500;
+    t['nacute'] = 556;
+    t['umacron'] = 556;
+    t['Ncaron'] = 722;
+    t['Iacute'] = 278;
+    t['plusminus'] = 584;
+    t['brokenbar'] = 260;
+    t['registered'] = 737;
+    t['Gbreve'] = 778;
+    t['Idotaccent'] = 278;
+    t['summation'] = 600;
+    t['Egrave'] = 667;
+    t['racute'] = 333;
+    t['omacron'] = 556;
+    t['Zacute'] = 611;
+    t['Zcaron'] = 611;
+    t['greaterequal'] = 549;
+    t['Eth'] = 722;
+    t['Ccedilla'] = 722;
+    t['lcommaaccent'] = 222;
+    t['tcaron'] = 317;
+    t['eogonek'] = 556;
+    t['Uogonek'] = 722;
+    t['Aacute'] = 667;
+    t['Adieresis'] = 667;
+    t['egrave'] = 556;
+    t['zacute'] = 500;
+    t['iogonek'] = 222;
+    t['Oacute'] = 778;
+    t['oacute'] = 556;
+    t['amacron'] = 556;
+    t['sacute'] = 500;
+    t['idieresis'] = 278;
+    t['Ocircumflex'] = 778;
+    t['Ugrave'] = 722;
+    t['Delta'] = 612;
+    t['thorn'] = 556;
+    t['twosuperior'] = 333;
+    t['Odieresis'] = 778;
+    t['mu'] = 556;
+    t['igrave'] = 278;
+    t['ohungarumlaut'] = 556;
+    t['Eogonek'] = 667;
+    t['dcroat'] = 556;
+    t['threequarters'] = 834;
+    t['Scedilla'] = 667;
+    t['lcaron'] = 299;
+    t['Kcommaaccent'] = 667;
+    t['Lacute'] = 556;
+    t['trademark'] = 1000;
+    t['edotaccent'] = 556;
+    t['Igrave'] = 278;
+    t['Imacron'] = 278;
+    t['Lcaron'] = 556;
+    t['onehalf'] = 834;
+    t['lessequal'] = 549;
+    t['ocircumflex'] = 556;
+    t['ntilde'] = 556;
+    t['Uhungarumlaut'] = 722;
+    t['Eacute'] = 667;
+    t['emacron'] = 556;
+    t['gbreve'] = 556;
+    t['onequarter'] = 834;
+    t['Scaron'] = 667;
+    t['Scommaaccent'] = 667;
+    t['Ohungarumlaut'] = 778;
+    t['degree'] = 400;
+    t['ograve'] = 556;
+    t['Ccaron'] = 722;
+    t['ugrave'] = 556;
+    t['radical'] = 453;
+    t['Dcaron'] = 722;
+    t['rcommaaccent'] = 333;
+    t['Ntilde'] = 722;
+    t['otilde'] = 556;
+    t['Rcommaaccent'] = 722;
+    t['Lcommaaccent'] = 556;
+    t['Atilde'] = 667;
+    t['Aogonek'] = 667;
+    t['Aring'] = 667;
+    t['Otilde'] = 778;
+    t['zdotaccent'] = 500;
+    t['Ecaron'] = 667;
+    t['Iogonek'] = 278;
+    t['kcommaaccent'] = 500;
+    t['minus'] = 584;
+    t['Icircumflex'] = 278;
+    t['ncaron'] = 556;
+    t['tcommaaccent'] = 278;
+    t['logicalnot'] = 584;
+    t['odieresis'] = 556;
+    t['udieresis'] = 556;
+    t['notequal'] = 549;
+    t['gcommaaccent'] = 556;
+    t['eth'] = 556;
+    t['zcaron'] = 500;
+    t['ncommaaccent'] = 556;
+    t['onesuperior'] = 333;
+    t['imacron'] = 278;
+    t['Euro'] = 556;
+  });
+  t['Symbol'] = getLookupTableFactory(function (t) {
+    t['space'] = 250;
+    t['exclam'] = 333;
+    t['universal'] = 713;
+    t['numbersign'] = 500;
+    t['existential'] = 549;
+    t['percent'] = 833;
+    t['ampersand'] = 778;
+    t['suchthat'] = 439;
+    t['parenleft'] = 333;
+    t['parenright'] = 333;
+    t['asteriskmath'] = 500;
+    t['plus'] = 549;
+    t['comma'] = 250;
+    t['minus'] = 549;
+    t['period'] = 250;
+    t['slash'] = 278;
+    t['zero'] = 500;
+    t['one'] = 500;
+    t['two'] = 500;
+    t['three'] = 500;
+    t['four'] = 500;
+    t['five'] = 500;
+    t['six'] = 500;
+    t['seven'] = 500;
+    t['eight'] = 500;
+    t['nine'] = 500;
+    t['colon'] = 278;
+    t['semicolon'] = 278;
+    t['less'] = 549;
+    t['equal'] = 549;
+    t['greater'] = 549;
+    t['question'] = 444;
+    t['congruent'] = 549;
+    t['Alpha'] = 722;
+    t['Beta'] = 667;
+    t['Chi'] = 722;
+    t['Delta'] = 612;
+    t['Epsilon'] = 611;
+    t['Phi'] = 763;
+    t['Gamma'] = 603;
+    t['Eta'] = 722;
+    t['Iota'] = 333;
+    t['theta1'] = 631;
+    t['Kappa'] = 722;
+    t['Lambda'] = 686;
+    t['Mu'] = 889;
+    t['Nu'] = 722;
+    t['Omicron'] = 722;
+    t['Pi'] = 768;
+    t['Theta'] = 741;
+    t['Rho'] = 556;
+    t['Sigma'] = 592;
+    t['Tau'] = 611;
+    t['Upsilon'] = 690;
+    t['sigma1'] = 439;
+    t['Omega'] = 768;
+    t['Xi'] = 645;
+    t['Psi'] = 795;
+    t['Zeta'] = 611;
+    t['bracketleft'] = 333;
+    t['therefore'] = 863;
+    t['bracketright'] = 333;
+    t['perpendicular'] = 658;
+    t['underscore'] = 500;
+    t['radicalex'] = 500;
+    t['alpha'] = 631;
+    t['beta'] = 549;
+    t['chi'] = 549;
+    t['delta'] = 494;
+    t['epsilon'] = 439;
+    t['phi'] = 521;
+    t['gamma'] = 411;
+    t['eta'] = 603;
+    t['iota'] = 329;
+    t['phi1'] = 603;
+    t['kappa'] = 549;
+    t['lambda'] = 549;
+    t['mu'] = 576;
+    t['nu'] = 521;
+    t['omicron'] = 549;
+    t['pi'] = 549;
+    t['theta'] = 521;
+    t['rho'] = 549;
+    t['sigma'] = 603;
+    t['tau'] = 439;
+    t['upsilon'] = 576;
+    t['omega1'] = 713;
+    t['omega'] = 686;
+    t['xi'] = 493;
+    t['psi'] = 686;
+    t['zeta'] = 494;
+    t['braceleft'] = 480;
+    t['bar'] = 200;
+    t['braceright'] = 480;
+    t['similar'] = 549;
+    t['Euro'] = 750;
+    t['Upsilon1'] = 620;
+    t['minute'] = 247;
+    t['lessequal'] = 549;
+    t['fraction'] = 167;
+    t['infinity'] = 713;
+    t['florin'] = 500;
+    t['club'] = 753;
+    t['diamond'] = 753;
+    t['heart'] = 753;
+    t['spade'] = 753;
+    t['arrowboth'] = 1042;
+    t['arrowleft'] = 987;
+    t['arrowup'] = 603;
+    t['arrowright'] = 987;
+    t['arrowdown'] = 603;
+    t['degree'] = 400;
+    t['plusminus'] = 549;
+    t['second'] = 411;
+    t['greaterequal'] = 549;
+    t['multiply'] = 549;
+    t['proportional'] = 713;
+    t['partialdiff'] = 494;
+    t['bullet'] = 460;
+    t['divide'] = 549;
+    t['notequal'] = 549;
+    t['equivalence'] = 549;
+    t['approxequal'] = 549;
+    t['ellipsis'] = 1000;
+    t['arrowvertex'] = 603;
+    t['arrowhorizex'] = 1000;
+    t['carriagereturn'] = 658;
+    t['aleph'] = 823;
+    t['Ifraktur'] = 686;
+    t['Rfraktur'] = 795;
+    t['weierstrass'] = 987;
+    t['circlemultiply'] = 768;
+    t['circleplus'] = 768;
+    t['emptyset'] = 823;
+    t['intersection'] = 768;
+    t['union'] = 768;
+    t['propersuperset'] = 713;
+    t['reflexsuperset'] = 713;
+    t['notsubset'] = 713;
+    t['propersubset'] = 713;
+    t['reflexsubset'] = 713;
+    t['element'] = 713;
+    t['notelement'] = 713;
+    t['angle'] = 768;
+    t['gradient'] = 713;
+    t['registerserif'] = 790;
+    t['copyrightserif'] = 790;
+    t['trademarkserif'] = 890;
+    t['product'] = 823;
+    t['radical'] = 549;
+    t['dotmath'] = 250;
+    t['logicalnot'] = 713;
+    t['logicaland'] = 603;
+    t['logicalor'] = 603;
+    t['arrowdblboth'] = 1042;
+    t['arrowdblleft'] = 987;
+    t['arrowdblup'] = 603;
+    t['arrowdblright'] = 987;
+    t['arrowdbldown'] = 603;
+    t['lozenge'] = 494;
+    t['angleleft'] = 329;
+    t['registersans'] = 790;
+    t['copyrightsans'] = 790;
+    t['trademarksans'] = 786;
+    t['summation'] = 713;
+    t['parenlefttp'] = 384;
+    t['parenleftex'] = 384;
+    t['parenleftbt'] = 384;
+    t['bracketlefttp'] = 384;
+    t['bracketleftex'] = 384;
+    t['bracketleftbt'] = 384;
+    t['bracelefttp'] = 494;
+    t['braceleftmid'] = 494;
+    t['braceleftbt'] = 494;
+    t['braceex'] = 494;
+    t['angleright'] = 329;
+    t['integral'] = 274;
+    t['integraltp'] = 686;
+    t['integralex'] = 686;
+    t['integralbt'] = 686;
+    t['parenrighttp'] = 384;
+    t['parenrightex'] = 384;
+    t['parenrightbt'] = 384;
+    t['bracketrighttp'] = 384;
+    t['bracketrightex'] = 384;
+    t['bracketrightbt'] = 384;
+    t['bracerighttp'] = 494;
+    t['bracerightmid'] = 494;
+    t['bracerightbt'] = 494;
+    t['apple'] = 790;
+  });
+  t['Times-Roman'] = getLookupTableFactory(function (t) {
+    t['space'] = 250;
+    t['exclam'] = 333;
+    t['quotedbl'] = 408;
+    t['numbersign'] = 500;
+    t['dollar'] = 500;
+    t['percent'] = 833;
+    t['ampersand'] = 778;
+    t['quoteright'] = 333;
+    t['parenleft'] = 333;
+    t['parenright'] = 333;
+    t['asterisk'] = 500;
+    t['plus'] = 564;
+    t['comma'] = 250;
+    t['hyphen'] = 333;
+    t['period'] = 250;
+    t['slash'] = 278;
+    t['zero'] = 500;
+    t['one'] = 500;
+    t['two'] = 500;
+    t['three'] = 500;
+    t['four'] = 500;
+    t['five'] = 500;
+    t['six'] = 500;
+    t['seven'] = 500;
+    t['eight'] = 500;
+    t['nine'] = 500;
+    t['colon'] = 278;
+    t['semicolon'] = 278;
+    t['less'] = 564;
+    t['equal'] = 564;
+    t['greater'] = 564;
+    t['question'] = 444;
+    t['at'] = 921;
+    t['A'] = 722;
+    t['B'] = 667;
+    t['C'] = 667;
+    t['D'] = 722;
+    t['E'] = 611;
+    t['F'] = 556;
+    t['G'] = 722;
+    t['H'] = 722;
+    t['I'] = 333;
+    t['J'] = 389;
+    t['K'] = 722;
+    t['L'] = 611;
+    t['M'] = 889;
+    t['N'] = 722;
+    t['O'] = 722;
+    t['P'] = 556;
+    t['Q'] = 722;
+    t['R'] = 667;
+    t['S'] = 556;
+    t['T'] = 611;
+    t['U'] = 722;
+    t['V'] = 722;
+    t['W'] = 944;
+    t['X'] = 722;
+    t['Y'] = 722;
+    t['Z'] = 611;
+    t['bracketleft'] = 333;
+    t['backslash'] = 278;
+    t['bracketright'] = 333;
+    t['asciicircum'] = 469;
+    t['underscore'] = 500;
+    t['quoteleft'] = 333;
+    t['a'] = 444;
+    t['b'] = 500;
+    t['c'] = 444;
+    t['d'] = 500;
+    t['e'] = 444;
+    t['f'] = 333;
+    t['g'] = 500;
+    t['h'] = 500;
+    t['i'] = 278;
+    t['j'] = 278;
+    t['k'] = 500;
+    t['l'] = 278;
+    t['m'] = 778;
+    t['n'] = 500;
+    t['o'] = 500;
+    t['p'] = 500;
+    t['q'] = 500;
+    t['r'] = 333;
+    t['s'] = 389;
+    t['t'] = 278;
+    t['u'] = 500;
+    t['v'] = 500;
+    t['w'] = 722;
+    t['x'] = 500;
+    t['y'] = 500;
+    t['z'] = 444;
+    t['braceleft'] = 480;
+    t['bar'] = 200;
+    t['braceright'] = 480;
+    t['asciitilde'] = 541;
+    t['exclamdown'] = 333;
+    t['cent'] = 500;
+    t['sterling'] = 500;
+    t['fraction'] = 167;
+    t['yen'] = 500;
+    t['florin'] = 500;
+    t['section'] = 500;
+    t['currency'] = 500;
+    t['quotesingle'] = 180;
+    t['quotedblleft'] = 444;
+    t['guillemotleft'] = 500;
+    t['guilsinglleft'] = 333;
+    t['guilsinglright'] = 333;
+    t['fi'] = 556;
+    t['fl'] = 556;
+    t['endash'] = 500;
+    t['dagger'] = 500;
+    t['daggerdbl'] = 500;
+    t['periodcentered'] = 250;
+    t['paragraph'] = 453;
+    t['bullet'] = 350;
+    t['quotesinglbase'] = 333;
+    t['quotedblbase'] = 444;
+    t['quotedblright'] = 444;
+    t['guillemotright'] = 500;
+    t['ellipsis'] = 1000;
+    t['perthousand'] = 1000;
+    t['questiondown'] = 444;
+    t['grave'] = 333;
+    t['acute'] = 333;
+    t['circumflex'] = 333;
+    t['tilde'] = 333;
+    t['macron'] = 333;
+    t['breve'] = 333;
+    t['dotaccent'] = 333;
+    t['dieresis'] = 333;
+    t['ring'] = 333;
+    t['cedilla'] = 333;
+    t['hungarumlaut'] = 333;
+    t['ogonek'] = 333;
+    t['caron'] = 333;
+    t['emdash'] = 1000;
+    t['AE'] = 889;
+    t['ordfeminine'] = 276;
+    t['Lslash'] = 611;
+    t['Oslash'] = 722;
+    t['OE'] = 889;
+    t['ordmasculine'] = 310;
+    t['ae'] = 667;
+    t['dotlessi'] = 278;
+    t['lslash'] = 278;
+    t['oslash'] = 500;
+    t['oe'] = 722;
+    t['germandbls'] = 500;
+    t['Idieresis'] = 333;
+    t['eacute'] = 444;
+    t['abreve'] = 444;
+    t['uhungarumlaut'] = 500;
+    t['ecaron'] = 444;
+    t['Ydieresis'] = 722;
+    t['divide'] = 564;
+    t['Yacute'] = 722;
+    t['Acircumflex'] = 722;
+    t['aacute'] = 444;
+    t['Ucircumflex'] = 722;
+    t['yacute'] = 500;
+    t['scommaaccent'] = 389;
+    t['ecircumflex'] = 444;
+    t['Uring'] = 722;
+    t['Udieresis'] = 722;
+    t['aogonek'] = 444;
+    t['Uacute'] = 722;
+    t['uogonek'] = 500;
+    t['Edieresis'] = 611;
+    t['Dcroat'] = 722;
+    t['commaaccent'] = 250;
+    t['copyright'] = 760;
+    t['Emacron'] = 611;
+    t['ccaron'] = 444;
+    t['aring'] = 444;
+    t['Ncommaaccent'] = 722;
+    t['lacute'] = 278;
+    t['agrave'] = 444;
+    t['Tcommaaccent'] = 611;
+    t['Cacute'] = 667;
+    t['atilde'] = 444;
+    t['Edotaccent'] = 611;
+    t['scaron'] = 389;
+    t['scedilla'] = 389;
+    t['iacute'] = 278;
+    t['lozenge'] = 471;
+    t['Rcaron'] = 667;
+    t['Gcommaaccent'] = 722;
+    t['ucircumflex'] = 500;
+    t['acircumflex'] = 444;
+    t['Amacron'] = 722;
+    t['rcaron'] = 333;
+    t['ccedilla'] = 444;
+    t['Zdotaccent'] = 611;
+    t['Thorn'] = 556;
+    t['Omacron'] = 722;
+    t['Racute'] = 667;
+    t['Sacute'] = 556;
+    t['dcaron'] = 588;
+    t['Umacron'] = 722;
+    t['uring'] = 500;
+    t['threesuperior'] = 300;
+    t['Ograve'] = 722;
+    t['Agrave'] = 722;
+    t['Abreve'] = 722;
+    t['multiply'] = 564;
+    t['uacute'] = 500;
+    t['Tcaron'] = 611;
+    t['partialdiff'] = 476;
+    t['ydieresis'] = 500;
+    t['Nacute'] = 722;
+    t['icircumflex'] = 278;
+    t['Ecircumflex'] = 611;
+    t['adieresis'] = 444;
+    t['edieresis'] = 444;
+    t['cacute'] = 444;
+    t['nacute'] = 500;
+    t['umacron'] = 500;
+    t['Ncaron'] = 722;
+    t['Iacute'] = 333;
+    t['plusminus'] = 564;
+    t['brokenbar'] = 200;
+    t['registered'] = 760;
+    t['Gbreve'] = 722;
+    t['Idotaccent'] = 333;
+    t['summation'] = 600;
+    t['Egrave'] = 611;
+    t['racute'] = 333;
+    t['omacron'] = 500;
+    t['Zacute'] = 611;
+    t['Zcaron'] = 611;
+    t['greaterequal'] = 549;
+    t['Eth'] = 722;
+    t['Ccedilla'] = 667;
+    t['lcommaaccent'] = 278;
+    t['tcaron'] = 326;
+    t['eogonek'] = 444;
+    t['Uogonek'] = 722;
+    t['Aacute'] = 722;
+    t['Adieresis'] = 722;
+    t['egrave'] = 444;
+    t['zacute'] = 444;
+    t['iogonek'] = 278;
+    t['Oacute'] = 722;
+    t['oacute'] = 500;
+    t['amacron'] = 444;
+    t['sacute'] = 389;
+    t['idieresis'] = 278;
+    t['Ocircumflex'] = 722;
+    t['Ugrave'] = 722;
+    t['Delta'] = 612;
+    t['thorn'] = 500;
+    t['twosuperior'] = 300;
+    t['Odieresis'] = 722;
+    t['mu'] = 500;
+    t['igrave'] = 278;
+    t['ohungarumlaut'] = 500;
+    t['Eogonek'] = 611;
+    t['dcroat'] = 500;
+    t['threequarters'] = 750;
+    t['Scedilla'] = 556;
+    t['lcaron'] = 344;
+    t['Kcommaaccent'] = 722;
+    t['Lacute'] = 611;
+    t['trademark'] = 980;
+    t['edotaccent'] = 444;
+    t['Igrave'] = 333;
+    t['Imacron'] = 333;
+    t['Lcaron'] = 611;
+    t['onehalf'] = 750;
+    t['lessequal'] = 549;
+    t['ocircumflex'] = 500;
+    t['ntilde'] = 500;
+    t['Uhungarumlaut'] = 722;
+    t['Eacute'] = 611;
+    t['emacron'] = 444;
+    t['gbreve'] = 500;
+    t['onequarter'] = 750;
+    t['Scaron'] = 556;
+    t['Scommaaccent'] = 556;
+    t['Ohungarumlaut'] = 722;
+    t['degree'] = 400;
+    t['ograve'] = 500;
+    t['Ccaron'] = 667;
+    t['ugrave'] = 500;
+    t['radical'] = 453;
+    t['Dcaron'] = 722;
+    t['rcommaaccent'] = 333;
+    t['Ntilde'] = 722;
+    t['otilde'] = 500;
+    t['Rcommaaccent'] = 667;
+    t['Lcommaaccent'] = 611;
+    t['Atilde'] = 722;
+    t['Aogonek'] = 722;
+    t['Aring'] = 722;
+    t['Otilde'] = 722;
+    t['zdotaccent'] = 444;
+    t['Ecaron'] = 611;
+    t['Iogonek'] = 333;
+    t['kcommaaccent'] = 500;
+    t['minus'] = 564;
+    t['Icircumflex'] = 333;
+    t['ncaron'] = 500;
+    t['tcommaaccent'] = 278;
+    t['logicalnot'] = 564;
+    t['odieresis'] = 500;
+    t['udieresis'] = 500;
+    t['notequal'] = 549;
+    t['gcommaaccent'] = 500;
+    t['eth'] = 500;
+    t['zcaron'] = 444;
+    t['ncommaaccent'] = 500;
+    t['onesuperior'] = 300;
+    t['imacron'] = 278;
+    t['Euro'] = 500;
+  });
+  t['Times-Bold'] = getLookupTableFactory(function (t) {
+    t['space'] = 250;
+    t['exclam'] = 333;
+    t['quotedbl'] = 555;
+    t['numbersign'] = 500;
+    t['dollar'] = 500;
+    t['percent'] = 1000;
+    t['ampersand'] = 833;
+    t['quoteright'] = 333;
+    t['parenleft'] = 333;
+    t['parenright'] = 333;
+    t['asterisk'] = 500;
+    t['plus'] = 570;
+    t['comma'] = 250;
+    t['hyphen'] = 333;
+    t['period'] = 250;
+    t['slash'] = 278;
+    t['zero'] = 500;
+    t['one'] = 500;
+    t['two'] = 500;
+    t['three'] = 500;
+    t['four'] = 500;
+    t['five'] = 500;
+    t['six'] = 500;
+    t['seven'] = 500;
+    t['eight'] = 500;
+    t['nine'] = 500;
+    t['colon'] = 333;
+    t['semicolon'] = 333;
+    t['less'] = 570;
+    t['equal'] = 570;
+    t['greater'] = 570;
+    t['question'] = 500;
+    t['at'] = 930;
+    t['A'] = 722;
+    t['B'] = 667;
+    t['C'] = 722;
+    t['D'] = 722;
+    t['E'] = 667;
+    t['F'] = 611;
+    t['G'] = 778;
+    t['H'] = 778;
+    t['I'] = 389;
+    t['J'] = 500;
+    t['K'] = 778;
+    t['L'] = 667;
+    t['M'] = 944;
+    t['N'] = 722;
+    t['O'] = 778;
+    t['P'] = 611;
+    t['Q'] = 778;
+    t['R'] = 722;
+    t['S'] = 556;
+    t['T'] = 667;
+    t['U'] = 722;
+    t['V'] = 722;
+    t['W'] = 1000;
+    t['X'] = 722;
+    t['Y'] = 722;
+    t['Z'] = 667;
+    t['bracketleft'] = 333;
+    t['backslash'] = 278;
+    t['bracketright'] = 333;
+    t['asciicircum'] = 581;
+    t['underscore'] = 500;
+    t['quoteleft'] = 333;
+    t['a'] = 500;
+    t['b'] = 556;
+    t['c'] = 444;
+    t['d'] = 556;
+    t['e'] = 444;
+    t['f'] = 333;
+    t['g'] = 500;
+    t['h'] = 556;
+    t['i'] = 278;
+    t['j'] = 333;
+    t['k'] = 556;
+    t['l'] = 278;
+    t['m'] = 833;
+    t['n'] = 556;
+    t['o'] = 500;
+    t['p'] = 556;
+    t['q'] = 556;
+    t['r'] = 444;
+    t['s'] = 389;
+    t['t'] = 333;
+    t['u'] = 556;
+    t['v'] = 500;
+    t['w'] = 722;
+    t['x'] = 500;
+    t['y'] = 500;
+    t['z'] = 444;
+    t['braceleft'] = 394;
+    t['bar'] = 220;
+    t['braceright'] = 394;
+    t['asciitilde'] = 520;
+    t['exclamdown'] = 333;
+    t['cent'] = 500;
+    t['sterling'] = 500;
+    t['fraction'] = 167;
+    t['yen'] = 500;
+    t['florin'] = 500;
+    t['section'] = 500;
+    t['currency'] = 500;
+    t['quotesingle'] = 278;
+    t['quotedblleft'] = 500;
+    t['guillemotleft'] = 500;
+    t['guilsinglleft'] = 333;
+    t['guilsinglright'] = 333;
+    t['fi'] = 556;
+    t['fl'] = 556;
+    t['endash'] = 500;
+    t['dagger'] = 500;
+    t['daggerdbl'] = 500;
+    t['periodcentered'] = 250;
+    t['paragraph'] = 540;
+    t['bullet'] = 350;
+    t['quotesinglbase'] = 333;
+    t['quotedblbase'] = 500;
+    t['quotedblright'] = 500;
+    t['guillemotright'] = 500;
+    t['ellipsis'] = 1000;
+    t['perthousand'] = 1000;
+    t['questiondown'] = 500;
+    t['grave'] = 333;
+    t['acute'] = 333;
+    t['circumflex'] = 333;
+    t['tilde'] = 333;
+    t['macron'] = 333;
+    t['breve'] = 333;
+    t['dotaccent'] = 333;
+    t['dieresis'] = 333;
+    t['ring'] = 333;
+    t['cedilla'] = 333;
+    t['hungarumlaut'] = 333;
+    t['ogonek'] = 333;
+    t['caron'] = 333;
+    t['emdash'] = 1000;
+    t['AE'] = 1000;
+    t['ordfeminine'] = 300;
+    t['Lslash'] = 667;
+    t['Oslash'] = 778;
+    t['OE'] = 1000;
+    t['ordmasculine'] = 330;
+    t['ae'] = 722;
+    t['dotlessi'] = 278;
+    t['lslash'] = 278;
+    t['oslash'] = 500;
+    t['oe'] = 722;
+    t['germandbls'] = 556;
+    t['Idieresis'] = 389;
+    t['eacute'] = 444;
+    t['abreve'] = 500;
+    t['uhungarumlaut'] = 556;
+    t['ecaron'] = 444;
+    t['Ydieresis'] = 722;
+    t['divide'] = 570;
+    t['Yacute'] = 722;
+    t['Acircumflex'] = 722;
+    t['aacute'] = 500;
+    t['Ucircumflex'] = 722;
+    t['yacute'] = 500;
+    t['scommaaccent'] = 389;
+    t['ecircumflex'] = 444;
+    t['Uring'] = 722;
+    t['Udieresis'] = 722;
+    t['aogonek'] = 500;
+    t['Uacute'] = 722;
+    t['uogonek'] = 556;
+    t['Edieresis'] = 667;
+    t['Dcroat'] = 722;
+    t['commaaccent'] = 250;
+    t['copyright'] = 747;
+    t['Emacron'] = 667;
+    t['ccaron'] = 444;
+    t['aring'] = 500;
+    t['Ncommaaccent'] = 722;
+    t['lacute'] = 278;
+    t['agrave'] = 500;
+    t['Tcommaaccent'] = 667;
+    t['Cacute'] = 722;
+    t['atilde'] = 500;
+    t['Edotaccent'] = 667;
+    t['scaron'] = 389;
+    t['scedilla'] = 389;
+    t['iacute'] = 278;
+    t['lozenge'] = 494;
+    t['Rcaron'] = 722;
+    t['Gcommaaccent'] = 778;
+    t['ucircumflex'] = 556;
+    t['acircumflex'] = 500;
+    t['Amacron'] = 722;
+    t['rcaron'] = 444;
+    t['ccedilla'] = 444;
+    t['Zdotaccent'] = 667;
+    t['Thorn'] = 611;
+    t['Omacron'] = 778;
+    t['Racute'] = 722;
+    t['Sacute'] = 556;
+    t['dcaron'] = 672;
+    t['Umacron'] = 722;
+    t['uring'] = 556;
+    t['threesuperior'] = 300;
+    t['Ograve'] = 778;
+    t['Agrave'] = 722;
+    t['Abreve'] = 722;
+    t['multiply'] = 570;
+    t['uacute'] = 556;
+    t['Tcaron'] = 667;
+    t['partialdiff'] = 494;
+    t['ydieresis'] = 500;
+    t['Nacute'] = 722;
+    t['icircumflex'] = 278;
+    t['Ecircumflex'] = 667;
+    t['adieresis'] = 500;
+    t['edieresis'] = 444;
+    t['cacute'] = 444;
+    t['nacute'] = 556;
+    t['umacron'] = 556;
+    t['Ncaron'] = 722;
+    t['Iacute'] = 389;
+    t['plusminus'] = 570;
+    t['brokenbar'] = 220;
+    t['registered'] = 747;
+    t['Gbreve'] = 778;
+    t['Idotaccent'] = 389;
+    t['summation'] = 600;
+    t['Egrave'] = 667;
+    t['racute'] = 444;
+    t['omacron'] = 500;
+    t['Zacute'] = 667;
+    t['Zcaron'] = 667;
+    t['greaterequal'] = 549;
+    t['Eth'] = 722;
+    t['Ccedilla'] = 722;
+    t['lcommaaccent'] = 278;
+    t['tcaron'] = 416;
+    t['eogonek'] = 444;
+    t['Uogonek'] = 722;
+    t['Aacute'] = 722;
+    t['Adieresis'] = 722;
+    t['egrave'] = 444;
+    t['zacute'] = 444;
+    t['iogonek'] = 278;
+    t['Oacute'] = 778;
+    t['oacute'] = 500;
+    t['amacron'] = 500;
+    t['sacute'] = 389;
+    t['idieresis'] = 278;
+    t['Ocircumflex'] = 778;
+    t['Ugrave'] = 722;
+    t['Delta'] = 612;
+    t['thorn'] = 556;
+    t['twosuperior'] = 300;
+    t['Odieresis'] = 778;
+    t['mu'] = 556;
+    t['igrave'] = 278;
+    t['ohungarumlaut'] = 500;
+    t['Eogonek'] = 667;
+    t['dcroat'] = 556;
+    t['threequarters'] = 750;
+    t['Scedilla'] = 556;
+    t['lcaron'] = 394;
+    t['Kcommaaccent'] = 778;
+    t['Lacute'] = 667;
+    t['trademark'] = 1000;
+    t['edotaccent'] = 444;
+    t['Igrave'] = 389;
+    t['Imacron'] = 389;
+    t['Lcaron'] = 667;
+    t['onehalf'] = 750;
+    t['lessequal'] = 549;
+    t['ocircumflex'] = 500;
+    t['ntilde'] = 556;
+    t['Uhungarumlaut'] = 722;
+    t['Eacute'] = 667;
+    t['emacron'] = 444;
+    t['gbreve'] = 500;
+    t['onequarter'] = 750;
+    t['Scaron'] = 556;
+    t['Scommaaccent'] = 556;
+    t['Ohungarumlaut'] = 778;
+    t['degree'] = 400;
+    t['ograve'] = 500;
+    t['Ccaron'] = 722;
+    t['ugrave'] = 556;
+    t['radical'] = 549;
+    t['Dcaron'] = 722;
+    t['rcommaaccent'] = 444;
+    t['Ntilde'] = 722;
+    t['otilde'] = 500;
+    t['Rcommaaccent'] = 722;
+    t['Lcommaaccent'] = 667;
+    t['Atilde'] = 722;
+    t['Aogonek'] = 722;
+    t['Aring'] = 722;
+    t['Otilde'] = 778;
+    t['zdotaccent'] = 444;
+    t['Ecaron'] = 667;
+    t['Iogonek'] = 389;
+    t['kcommaaccent'] = 556;
+    t['minus'] = 570;
+    t['Icircumflex'] = 389;
+    t['ncaron'] = 556;
+    t['tcommaaccent'] = 333;
+    t['logicalnot'] = 570;
+    t['odieresis'] = 500;
+    t['udieresis'] = 556;
+    t['notequal'] = 549;
+    t['gcommaaccent'] = 500;
+    t['eth'] = 500;
+    t['zcaron'] = 444;
+    t['ncommaaccent'] = 556;
+    t['onesuperior'] = 300;
+    t['imacron'] = 278;
+    t['Euro'] = 500;
+  });
+  t['Times-BoldItalic'] = getLookupTableFactory(function (t) {
+    t['space'] = 250;
+    t['exclam'] = 389;
+    t['quotedbl'] = 555;
+    t['numbersign'] = 500;
+    t['dollar'] = 500;
+    t['percent'] = 833;
+    t['ampersand'] = 778;
+    t['quoteright'] = 333;
+    t['parenleft'] = 333;
+    t['parenright'] = 333;
+    t['asterisk'] = 500;
+    t['plus'] = 570;
+    t['comma'] = 250;
+    t['hyphen'] = 333;
+    t['period'] = 250;
+    t['slash'] = 278;
+    t['zero'] = 500;
+    t['one'] = 500;
+    t['two'] = 500;
+    t['three'] = 500;
+    t['four'] = 500;
+    t['five'] = 500;
+    t['six'] = 500;
+    t['seven'] = 500;
+    t['eight'] = 500;
+    t['nine'] = 500;
+    t['colon'] = 333;
+    t['semicolon'] = 333;
+    t['less'] = 570;
+    t['equal'] = 570;
+    t['greater'] = 570;
+    t['question'] = 500;
+    t['at'] = 832;
+    t['A'] = 667;
+    t['B'] = 667;
+    t['C'] = 667;
+    t['D'] = 722;
+    t['E'] = 667;
+    t['F'] = 667;
+    t['G'] = 722;
+    t['H'] = 778;
+    t['I'] = 389;
+    t['J'] = 500;
+    t['K'] = 667;
+    t['L'] = 611;
+    t['M'] = 889;
+    t['N'] = 722;
+    t['O'] = 722;
+    t['P'] = 611;
+    t['Q'] = 722;
+    t['R'] = 667;
+    t['S'] = 556;
+    t['T'] = 611;
+    t['U'] = 722;
+    t['V'] = 667;
+    t['W'] = 889;
+    t['X'] = 667;
+    t['Y'] = 611;
+    t['Z'] = 611;
+    t['bracketleft'] = 333;
+    t['backslash'] = 278;
+    t['bracketright'] = 333;
+    t['asciicircum'] = 570;
+    t['underscore'] = 500;
+    t['quoteleft'] = 333;
+    t['a'] = 500;
+    t['b'] = 500;
+    t['c'] = 444;
+    t['d'] = 500;
+    t['e'] = 444;
+    t['f'] = 333;
+    t['g'] = 500;
+    t['h'] = 556;
+    t['i'] = 278;
+    t['j'] = 278;
+    t['k'] = 500;
+    t['l'] = 278;
+    t['m'] = 778;
+    t['n'] = 556;
+    t['o'] = 500;
+    t['p'] = 500;
+    t['q'] = 500;
+    t['r'] = 389;
+    t['s'] = 389;
+    t['t'] = 278;
+    t['u'] = 556;
+    t['v'] = 444;
+    t['w'] = 667;
+    t['x'] = 500;
+    t['y'] = 444;
+    t['z'] = 389;
+    t['braceleft'] = 348;
+    t['bar'] = 220;
+    t['braceright'] = 348;
+    t['asciitilde'] = 570;
+    t['exclamdown'] = 389;
+    t['cent'] = 500;
+    t['sterling'] = 500;
+    t['fraction'] = 167;
+    t['yen'] = 500;
+    t['florin'] = 500;
+    t['section'] = 500;
+    t['currency'] = 500;
+    t['quotesingle'] = 278;
+    t['quotedblleft'] = 500;
+    t['guillemotleft'] = 500;
+    t['guilsinglleft'] = 333;
+    t['guilsinglright'] = 333;
+    t['fi'] = 556;
+    t['fl'] = 556;
+    t['endash'] = 500;
+    t['dagger'] = 500;
+    t['daggerdbl'] = 500;
+    t['periodcentered'] = 250;
+    t['paragraph'] = 500;
+    t['bullet'] = 350;
+    t['quotesinglbase'] = 333;
+    t['quotedblbase'] = 500;
+    t['quotedblright'] = 500;
+    t['guillemotright'] = 500;
+    t['ellipsis'] = 1000;
+    t['perthousand'] = 1000;
+    t['questiondown'] = 500;
+    t['grave'] = 333;
+    t['acute'] = 333;
+    t['circumflex'] = 333;
+    t['tilde'] = 333;
+    t['macron'] = 333;
+    t['breve'] = 333;
+    t['dotaccent'] = 333;
+    t['dieresis'] = 333;
+    t['ring'] = 333;
+    t['cedilla'] = 333;
+    t['hungarumlaut'] = 333;
+    t['ogonek'] = 333;
+    t['caron'] = 333;
+    t['emdash'] = 1000;
+    t['AE'] = 944;
+    t['ordfeminine'] = 266;
+    t['Lslash'] = 611;
+    t['Oslash'] = 722;
+    t['OE'] = 944;
+    t['ordmasculine'] = 300;
+    t['ae'] = 722;
+    t['dotlessi'] = 278;
+    t['lslash'] = 278;
+    t['oslash'] = 500;
+    t['oe'] = 722;
+    t['germandbls'] = 500;
+    t['Idieresis'] = 389;
+    t['eacute'] = 444;
+    t['abreve'] = 500;
+    t['uhungarumlaut'] = 556;
+    t['ecaron'] = 444;
+    t['Ydieresis'] = 611;
+    t['divide'] = 570;
+    t['Yacute'] = 611;
+    t['Acircumflex'] = 667;
+    t['aacute'] = 500;
+    t['Ucircumflex'] = 722;
+    t['yacute'] = 444;
+    t['scommaaccent'] = 389;
+    t['ecircumflex'] = 444;
+    t['Uring'] = 722;
+    t['Udieresis'] = 722;
+    t['aogonek'] = 500;
+    t['Uacute'] = 722;
+    t['uogonek'] = 556;
+    t['Edieresis'] = 667;
+    t['Dcroat'] = 722;
+    t['commaaccent'] = 250;
+    t['copyright'] = 747;
+    t['Emacron'] = 667;
+    t['ccaron'] = 444;
+    t['aring'] = 500;
+    t['Ncommaaccent'] = 722;
+    t['lacute'] = 278;
+    t['agrave'] = 500;
+    t['Tcommaaccent'] = 611;
+    t['Cacute'] = 667;
+    t['atilde'] = 500;
+    t['Edotaccent'] = 667;
+    t['scaron'] = 389;
+    t['scedilla'] = 389;
+    t['iacute'] = 278;
+    t['lozenge'] = 494;
+    t['Rcaron'] = 667;
+    t['Gcommaaccent'] = 722;
+    t['ucircumflex'] = 556;
+    t['acircumflex'] = 500;
+    t['Amacron'] = 667;
+    t['rcaron'] = 389;
+    t['ccedilla'] = 444;
+    t['Zdotaccent'] = 611;
+    t['Thorn'] = 611;
+    t['Omacron'] = 722;
+    t['Racute'] = 667;
+    t['Sacute'] = 556;
+    t['dcaron'] = 608;
+    t['Umacron'] = 722;
+    t['uring'] = 556;
+    t['threesuperior'] = 300;
+    t['Ograve'] = 722;
+    t['Agrave'] = 667;
+    t['Abreve'] = 667;
+    t['multiply'] = 570;
+    t['uacute'] = 556;
+    t['Tcaron'] = 611;
+    t['partialdiff'] = 494;
+    t['ydieresis'] = 444;
+    t['Nacute'] = 722;
+    t['icircumflex'] = 278;
+    t['Ecircumflex'] = 667;
+    t['adieresis'] = 500;
+    t['edieresis'] = 444;
+    t['cacute'] = 444;
+    t['nacute'] = 556;
+    t['umacron'] = 556;
+    t['Ncaron'] = 722;
+    t['Iacute'] = 389;
+    t['plusminus'] = 570;
+    t['brokenbar'] = 220;
+    t['registered'] = 747;
+    t['Gbreve'] = 722;
+    t['Idotaccent'] = 389;
+    t['summation'] = 600;
+    t['Egrave'] = 667;
+    t['racute'] = 389;
+    t['omacron'] = 500;
+    t['Zacute'] = 611;
+    t['Zcaron'] = 611;
+    t['greaterequal'] = 549;
+    t['Eth'] = 722;
+    t['Ccedilla'] = 667;
+    t['lcommaaccent'] = 278;
+    t['tcaron'] = 366;
+    t['eogonek'] = 444;
+    t['Uogonek'] = 722;
+    t['Aacute'] = 667;
+    t['Adieresis'] = 667;
+    t['egrave'] = 444;
+    t['zacute'] = 389;
+    t['iogonek'] = 278;
+    t['Oacute'] = 722;
+    t['oacute'] = 500;
+    t['amacron'] = 500;
+    t['sacute'] = 389;
+    t['idieresis'] = 278;
+    t['Ocircumflex'] = 722;
+    t['Ugrave'] = 722;
+    t['Delta'] = 612;
+    t['thorn'] = 500;
+    t['twosuperior'] = 300;
+    t['Odieresis'] = 722;
+    t['mu'] = 576;
+    t['igrave'] = 278;
+    t['ohungarumlaut'] = 500;
+    t['Eogonek'] = 667;
+    t['dcroat'] = 500;
+    t['threequarters'] = 750;
+    t['Scedilla'] = 556;
+    t['lcaron'] = 382;
+    t['Kcommaaccent'] = 667;
+    t['Lacute'] = 611;
+    t['trademark'] = 1000;
+    t['edotaccent'] = 444;
+    t['Igrave'] = 389;
+    t['Imacron'] = 389;
+    t['Lcaron'] = 611;
+    t['onehalf'] = 750;
+    t['lessequal'] = 549;
+    t['ocircumflex'] = 500;
+    t['ntilde'] = 556;
+    t['Uhungarumlaut'] = 722;
+    t['Eacute'] = 667;
+    t['emacron'] = 444;
+    t['gbreve'] = 500;
+    t['onequarter'] = 750;
+    t['Scaron'] = 556;
+    t['Scommaaccent'] = 556;
+    t['Ohungarumlaut'] = 722;
+    t['degree'] = 400;
+    t['ograve'] = 500;
+    t['Ccaron'] = 667;
+    t['ugrave'] = 556;
+    t['radical'] = 549;
+    t['Dcaron'] = 722;
+    t['rcommaaccent'] = 389;
+    t['Ntilde'] = 722;
+    t['otilde'] = 500;
+    t['Rcommaaccent'] = 667;
+    t['Lcommaaccent'] = 611;
+    t['Atilde'] = 667;
+    t['Aogonek'] = 667;
+    t['Aring'] = 667;
+    t['Otilde'] = 722;
+    t['zdotaccent'] = 389;
+    t['Ecaron'] = 667;
+    t['Iogonek'] = 389;
+    t['kcommaaccent'] = 500;
+    t['minus'] = 606;
+    t['Icircumflex'] = 389;
+    t['ncaron'] = 556;
+    t['tcommaaccent'] = 278;
+    t['logicalnot'] = 606;
+    t['odieresis'] = 500;
+    t['udieresis'] = 556;
+    t['notequal'] = 549;
+    t['gcommaaccent'] = 500;
+    t['eth'] = 500;
+    t['zcaron'] = 389;
+    t['ncommaaccent'] = 556;
+    t['onesuperior'] = 300;
+    t['imacron'] = 278;
+    t['Euro'] = 500;
+  });
+  t['Times-Italic'] = getLookupTableFactory(function (t) {
+    t['space'] = 250;
+    t['exclam'] = 333;
+    t['quotedbl'] = 420;
+    t['numbersign'] = 500;
+    t['dollar'] = 500;
+    t['percent'] = 833;
+    t['ampersand'] = 778;
+    t['quoteright'] = 333;
+    t['parenleft'] = 333;
+    t['parenright'] = 333;
+    t['asterisk'] = 500;
+    t['plus'] = 675;
+    t['comma'] = 250;
+    t['hyphen'] = 333;
+    t['period'] = 250;
+    t['slash'] = 278;
+    t['zero'] = 500;
+    t['one'] = 500;
+    t['two'] = 500;
+    t['three'] = 500;
+    t['four'] = 500;
+    t['five'] = 500;
+    t['six'] = 500;
+    t['seven'] = 500;
+    t['eight'] = 500;
+    t['nine'] = 500;
+    t['colon'] = 333;
+    t['semicolon'] = 333;
+    t['less'] = 675;
+    t['equal'] = 675;
+    t['greater'] = 675;
+    t['question'] = 500;
+    t['at'] = 920;
+    t['A'] = 611;
+    t['B'] = 611;
+    t['C'] = 667;
+    t['D'] = 722;
+    t['E'] = 611;
+    t['F'] = 611;
+    t['G'] = 722;
+    t['H'] = 722;
+    t['I'] = 333;
+    t['J'] = 444;
+    t['K'] = 667;
+    t['L'] = 556;
+    t['M'] = 833;
+    t['N'] = 667;
+    t['O'] = 722;
+    t['P'] = 611;
+    t['Q'] = 722;
+    t['R'] = 611;
+    t['S'] = 500;
+    t['T'] = 556;
+    t['U'] = 722;
+    t['V'] = 611;
+    t['W'] = 833;
+    t['X'] = 611;
+    t['Y'] = 556;
+    t['Z'] = 556;
+    t['bracketleft'] = 389;
+    t['backslash'] = 278;
+    t['bracketright'] = 389;
+    t['asciicircum'] = 422;
+    t['underscore'] = 500;
+    t['quoteleft'] = 333;
+    t['a'] = 500;
+    t['b'] = 500;
+    t['c'] = 444;
+    t['d'] = 500;
+    t['e'] = 444;
+    t['f'] = 278;
+    t['g'] = 500;
+    t['h'] = 500;
+    t['i'] = 278;
+    t['j'] = 278;
+    t['k'] = 444;
+    t['l'] = 278;
+    t['m'] = 722;
+    t['n'] = 500;
+    t['o'] = 500;
+    t['p'] = 500;
+    t['q'] = 500;
+    t['r'] = 389;
+    t['s'] = 389;
+    t['t'] = 278;
+    t['u'] = 500;
+    t['v'] = 444;
+    t['w'] = 667;
+    t['x'] = 444;
+    t['y'] = 444;
+    t['z'] = 389;
+    t['braceleft'] = 400;
+    t['bar'] = 275;
+    t['braceright'] = 400;
+    t['asciitilde'] = 541;
+    t['exclamdown'] = 389;
+    t['cent'] = 500;
+    t['sterling'] = 500;
+    t['fraction'] = 167;
+    t['yen'] = 500;
+    t['florin'] = 500;
+    t['section'] = 500;
+    t['currency'] = 500;
+    t['quotesingle'] = 214;
+    t['quotedblleft'] = 556;
+    t['guillemotleft'] = 500;
+    t['guilsinglleft'] = 333;
+    t['guilsinglright'] = 333;
+    t['fi'] = 500;
+    t['fl'] = 500;
+    t['endash'] = 500;
+    t['dagger'] = 500;
+    t['daggerdbl'] = 500;
+    t['periodcentered'] = 250;
+    t['paragraph'] = 523;
+    t['bullet'] = 350;
+    t['quotesinglbase'] = 333;
+    t['quotedblbase'] = 556;
+    t['quotedblright'] = 556;
+    t['guillemotright'] = 500;
+    t['ellipsis'] = 889;
+    t['perthousand'] = 1000;
+    t['questiondown'] = 500;
+    t['grave'] = 333;
+    t['acute'] = 333;
+    t['circumflex'] = 333;
+    t['tilde'] = 333;
+    t['macron'] = 333;
+    t['breve'] = 333;
+    t['dotaccent'] = 333;
+    t['dieresis'] = 333;
+    t['ring'] = 333;
+    t['cedilla'] = 333;
+    t['hungarumlaut'] = 333;
+    t['ogonek'] = 333;
+    t['caron'] = 333;
+    t['emdash'] = 889;
+    t['AE'] = 889;
+    t['ordfeminine'] = 276;
+    t['Lslash'] = 556;
+    t['Oslash'] = 722;
+    t['OE'] = 944;
+    t['ordmasculine'] = 310;
+    t['ae'] = 667;
+    t['dotlessi'] = 278;
+    t['lslash'] = 278;
+    t['oslash'] = 500;
+    t['oe'] = 667;
+    t['germandbls'] = 500;
+    t['Idieresis'] = 333;
+    t['eacute'] = 444;
+    t['abreve'] = 500;
+    t['uhungarumlaut'] = 500;
+    t['ecaron'] = 444;
+    t['Ydieresis'] = 556;
+    t['divide'] = 675;
+    t['Yacute'] = 556;
+    t['Acircumflex'] = 611;
+    t['aacute'] = 500;
+    t['Ucircumflex'] = 722;
+    t['yacute'] = 444;
+    t['scommaaccent'] = 389;
+    t['ecircumflex'] = 444;
+    t['Uring'] = 722;
+    t['Udieresis'] = 722;
+    t['aogonek'] = 500;
+    t['Uacute'] = 722;
+    t['uogonek'] = 500;
+    t['Edieresis'] = 611;
+    t['Dcroat'] = 722;
+    t['commaaccent'] = 250;
+    t['copyright'] = 760;
+    t['Emacron'] = 611;
+    t['ccaron'] = 444;
+    t['aring'] = 500;
+    t['Ncommaaccent'] = 667;
+    t['lacute'] = 278;
+    t['agrave'] = 500;
+    t['Tcommaaccent'] = 556;
+    t['Cacute'] = 667;
+    t['atilde'] = 500;
+    t['Edotaccent'] = 611;
+    t['scaron'] = 389;
+    t['scedilla'] = 389;
+    t['iacute'] = 278;
+    t['lozenge'] = 471;
+    t['Rcaron'] = 611;
+    t['Gcommaaccent'] = 722;
+    t['ucircumflex'] = 500;
+    t['acircumflex'] = 500;
+    t['Amacron'] = 611;
+    t['rcaron'] = 389;
+    t['ccedilla'] = 444;
+    t['Zdotaccent'] = 556;
+    t['Thorn'] = 611;
+    t['Omacron'] = 722;
+    t['Racute'] = 611;
+    t['Sacute'] = 500;
+    t['dcaron'] = 544;
+    t['Umacron'] = 722;
+    t['uring'] = 500;
+    t['threesuperior'] = 300;
+    t['Ograve'] = 722;
+    t['Agrave'] = 611;
+    t['Abreve'] = 611;
+    t['multiply'] = 675;
+    t['uacute'] = 500;
+    t['Tcaron'] = 556;
+    t['partialdiff'] = 476;
+    t['ydieresis'] = 444;
+    t['Nacute'] = 667;
+    t['icircumflex'] = 278;
+    t['Ecircumflex'] = 611;
+    t['adieresis'] = 500;
+    t['edieresis'] = 444;
+    t['cacute'] = 444;
+    t['nacute'] = 500;
+    t['umacron'] = 500;
+    t['Ncaron'] = 667;
+    t['Iacute'] = 333;
+    t['plusminus'] = 675;
+    t['brokenbar'] = 275;
+    t['registered'] = 760;
+    t['Gbreve'] = 722;
+    t['Idotaccent'] = 333;
+    t['summation'] = 600;
+    t['Egrave'] = 611;
+    t['racute'] = 389;
+    t['omacron'] = 500;
+    t['Zacute'] = 556;
+    t['Zcaron'] = 556;
+    t['greaterequal'] = 549;
+    t['Eth'] = 722;
+    t['Ccedilla'] = 667;
+    t['lcommaaccent'] = 278;
+    t['tcaron'] = 300;
+    t['eogonek'] = 444;
+    t['Uogonek'] = 722;
+    t['Aacute'] = 611;
+    t['Adieresis'] = 611;
+    t['egrave'] = 444;
+    t['zacute'] = 389;
+    t['iogonek'] = 278;
+    t['Oacute'] = 722;
+    t['oacute'] = 500;
+    t['amacron'] = 500;
+    t['sacute'] = 389;
+    t['idieresis'] = 278;
+    t['Ocircumflex'] = 722;
+    t['Ugrave'] = 722;
+    t['Delta'] = 612;
+    t['thorn'] = 500;
+    t['twosuperior'] = 300;
+    t['Odieresis'] = 722;
+    t['mu'] = 500;
+    t['igrave'] = 278;
+    t['ohungarumlaut'] = 500;
+    t['Eogonek'] = 611;
+    t['dcroat'] = 500;
+    t['threequarters'] = 750;
+    t['Scedilla'] = 500;
+    t['lcaron'] = 300;
+    t['Kcommaaccent'] = 667;
+    t['Lacute'] = 556;
+    t['trademark'] = 980;
+    t['edotaccent'] = 444;
+    t['Igrave'] = 333;
+    t['Imacron'] = 333;
+    t['Lcaron'] = 611;
+    t['onehalf'] = 750;
+    t['lessequal'] = 549;
+    t['ocircumflex'] = 500;
+    t['ntilde'] = 500;
+    t['Uhungarumlaut'] = 722;
+    t['Eacute'] = 611;
+    t['emacron'] = 444;
+    t['gbreve'] = 500;
+    t['onequarter'] = 750;
+    t['Scaron'] = 500;
+    t['Scommaaccent'] = 500;
+    t['Ohungarumlaut'] = 722;
+    t['degree'] = 400;
+    t['ograve'] = 500;
+    t['Ccaron'] = 667;
+    t['ugrave'] = 500;
+    t['radical'] = 453;
+    t['Dcaron'] = 722;
+    t['rcommaaccent'] = 389;
+    t['Ntilde'] = 667;
+    t['otilde'] = 500;
+    t['Rcommaaccent'] = 611;
+    t['Lcommaaccent'] = 556;
+    t['Atilde'] = 611;
+    t['Aogonek'] = 611;
+    t['Aring'] = 611;
+    t['Otilde'] = 722;
+    t['zdotaccent'] = 389;
+    t['Ecaron'] = 611;
+    t['Iogonek'] = 333;
+    t['kcommaaccent'] = 444;
+    t['minus'] = 675;
+    t['Icircumflex'] = 333;
+    t['ncaron'] = 500;
+    t['tcommaaccent'] = 278;
+    t['logicalnot'] = 675;
+    t['odieresis'] = 500;
+    t['udieresis'] = 500;
+    t['notequal'] = 549;
+    t['gcommaaccent'] = 500;
+    t['eth'] = 500;
+    t['zcaron'] = 389;
+    t['ncommaaccent'] = 500;
+    t['onesuperior'] = 300;
+    t['imacron'] = 278;
+    t['Euro'] = 500;
+  });
+  t['ZapfDingbats'] = getLookupTableFactory(function (t) {
+    t['space'] = 278;
+    t['a1'] = 974;
+    t['a2'] = 961;
+    t['a202'] = 974;
+    t['a3'] = 980;
+    t['a4'] = 719;
+    t['a5'] = 789;
+    t['a119'] = 790;
+    t['a118'] = 791;
+    t['a117'] = 690;
+    t['a11'] = 960;
+    t['a12'] = 939;
+    t['a13'] = 549;
+    t['a14'] = 855;
+    t['a15'] = 911;
+    t['a16'] = 933;
+    t['a105'] = 911;
+    t['a17'] = 945;
+    t['a18'] = 974;
+    t['a19'] = 755;
+    t['a20'] = 846;
+    t['a21'] = 762;
+    t['a22'] = 761;
+    t['a23'] = 571;
+    t['a24'] = 677;
+    t['a25'] = 763;
+    t['a26'] = 760;
+    t['a27'] = 759;
+    t['a28'] = 754;
+    t['a6'] = 494;
+    t['a7'] = 552;
+    t['a8'] = 537;
+    t['a9'] = 577;
+    t['a10'] = 692;
+    t['a29'] = 786;
+    t['a30'] = 788;
+    t['a31'] = 788;
+    t['a32'] = 790;
+    t['a33'] = 793;
+    t['a34'] = 794;
+    t['a35'] = 816;
+    t['a36'] = 823;
+    t['a37'] = 789;
+    t['a38'] = 841;
+    t['a39'] = 823;
+    t['a40'] = 833;
+    t['a41'] = 816;
+    t['a42'] = 831;
+    t['a43'] = 923;
+    t['a44'] = 744;
+    t['a45'] = 723;
+    t['a46'] = 749;
+    t['a47'] = 790;
+    t['a48'] = 792;
+    t['a49'] = 695;
+    t['a50'] = 776;
+    t['a51'] = 768;
+    t['a52'] = 792;
+    t['a53'] = 759;
+    t['a54'] = 707;
+    t['a55'] = 708;
+    t['a56'] = 682;
+    t['a57'] = 701;
+    t['a58'] = 826;
+    t['a59'] = 815;
+    t['a60'] = 789;
+    t['a61'] = 789;
+    t['a62'] = 707;
+    t['a63'] = 687;
+    t['a64'] = 696;
+    t['a65'] = 689;
+    t['a66'] = 786;
+    t['a67'] = 787;
+    t['a68'] = 713;
+    t['a69'] = 791;
+    t['a70'] = 785;
+    t['a71'] = 791;
+    t['a72'] = 873;
+    t['a73'] = 761;
+    t['a74'] = 762;
+    t['a203'] = 762;
+    t['a75'] = 759;
+    t['a204'] = 759;
+    t['a76'] = 892;
+    t['a77'] = 892;
+    t['a78'] = 788;
+    t['a79'] = 784;
+    t['a81'] = 438;
+    t['a82'] = 138;
+    t['a83'] = 277;
+    t['a84'] = 415;
+    t['a97'] = 392;
+    t['a98'] = 392;
+    t['a99'] = 668;
+    t['a100'] = 668;
+    t['a89'] = 390;
+    t['a90'] = 390;
+    t['a93'] = 317;
+    t['a94'] = 317;
+    t['a91'] = 276;
+    t['a92'] = 276;
+    t['a205'] = 509;
+    t['a85'] = 509;
+    t['a206'] = 410;
+    t['a86'] = 410;
+    t['a87'] = 234;
+    t['a88'] = 234;
+    t['a95'] = 334;
+    t['a96'] = 334;
+    t['a101'] = 732;
+    t['a102'] = 544;
+    t['a103'] = 544;
+    t['a104'] = 910;
+    t['a106'] = 667;
+    t['a107'] = 760;
+    t['a108'] = 760;
+    t['a112'] = 776;
+    t['a111'] = 595;
+    t['a110'] = 694;
+    t['a109'] = 626;
+    t['a120'] = 788;
+    t['a121'] = 788;
+    t['a122'] = 788;
+    t['a123'] = 788;
+    t['a124'] = 788;
+    t['a125'] = 788;
+    t['a126'] = 788;
+    t['a127'] = 788;
+    t['a128'] = 788;
+    t['a129'] = 788;
+    t['a130'] = 788;
+    t['a131'] = 788;
+    t['a132'] = 788;
+    t['a133'] = 788;
+    t['a134'] = 788;
+    t['a135'] = 788;
+    t['a136'] = 788;
+    t['a137'] = 788;
+    t['a138'] = 788;
+    t['a139'] = 788;
+    t['a140'] = 788;
+    t['a141'] = 788;
+    t['a142'] = 788;
+    t['a143'] = 788;
+    t['a144'] = 788;
+    t['a145'] = 788;
+    t['a146'] = 788;
+    t['a147'] = 788;
+    t['a148'] = 788;
+    t['a149'] = 788;
+    t['a150'] = 788;
+    t['a151'] = 788;
+    t['a152'] = 788;
+    t['a153'] = 788;
+    t['a154'] = 788;
+    t['a155'] = 788;
+    t['a156'] = 788;
+    t['a157'] = 788;
+    t['a158'] = 788;
+    t['a159'] = 788;
+    t['a160'] = 894;
+    t['a161'] = 838;
+    t['a163'] = 1016;
+    t['a164'] = 458;
+    t['a196'] = 748;
+    t['a165'] = 924;
+    t['a192'] = 748;
+    t['a166'] = 918;
+    t['a167'] = 927;
+    t['a168'] = 928;
+    t['a169'] = 928;
+    t['a170'] = 834;
+    t['a171'] = 873;
+    t['a172'] = 828;
+    t['a173'] = 924;
+    t['a162'] = 924;
+    t['a174'] = 917;
+    t['a175'] = 930;
+    t['a176'] = 931;
+    t['a177'] = 463;
+    t['a178'] = 883;
+    t['a179'] = 836;
+    t['a193'] = 836;
+    t['a180'] = 867;
+    t['a199'] = 867;
+    t['a181'] = 696;
+    t['a200'] = 696;
+    t['a182'] = 874;
+    t['a201'] = 874;
+    t['a183'] = 760;
+    t['a184'] = 946;
+    t['a197'] = 771;
+    t['a185'] = 865;
+    t['a194'] = 771;
+    t['a198'] = 888;
+    t['a186'] = 967;
+    t['a195'] = 888;
+    t['a187'] = 831;
+    t['a188'] = 873;
+    t['a189'] = 927;
+    t['a190'] = 970;
+    t['a191'] = 918;
+  });
 });
 exports.getMetrics = getMetrics;

+ 111 - 113
lib/core/murmurhash3.js

@@ -13,122 +13,120 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var Uint32ArrayView = sharedUtil.Uint32ArrayView;
 var MurmurHash3_64 = function MurmurHash3_64Closure(seed) {
- var MASK_HIGH = 0xffff0000;
- var MASK_LOW = 0xffff;
- function MurmurHash3_64(seed) {
-  var SEED = 0xc3d2e1f0;
-  this.h1 = seed ? seed & 0xffffffff : SEED;
-  this.h2 = seed ? seed & 0xffffffff : SEED;
- }
- var alwaysUseUint32ArrayView = false;
- try {
-  new Uint32Array(new Uint8Array(5).buffer, 0, 1);
- } catch (e) {
-  alwaysUseUint32ArrayView = true;
- }
- MurmurHash3_64.prototype = {
-  update: function MurmurHash3_64_update(input) {
-   var useUint32ArrayView = alwaysUseUint32ArrayView;
-   var i;
-   if (typeof input === 'string') {
-    var data = new Uint8Array(input.length * 2);
-    var length = 0;
-    for (i = 0; i < input.length; i++) {
-     var code = input.charCodeAt(i);
-     if (code <= 0xff) {
-      data[length++] = code;
-     } else {
-      data[length++] = code >>> 8;
-      data[length++] = code & 0xff;
-     }
-    }
-   } else if (input instanceof Uint8Array) {
-    data = input;
-    length = data.length;
-   } else if (typeof input === 'object' && 'length' in input) {
-    data = input;
-    length = data.length;
-    useUint32ArrayView = true;
-   } else {
-    throw new Error('Wrong data format in MurmurHash3_64_update. ' + 'Input must be a string or array.');
-   }
-   var blockCounts = length >> 2;
-   var tailLength = length - blockCounts * 4;
-   var dataUint32 = useUint32ArrayView ? new Uint32ArrayView(data, blockCounts) : new Uint32Array(data.buffer, 0, blockCounts);
-   var k1 = 0;
-   var k2 = 0;
-   var h1 = this.h1;
-   var h2 = this.h2;
-   var C1 = 0xcc9e2d51;
-   var C2 = 0x1b873593;
-   var C1_LOW = C1 & MASK_LOW;
-   var C2_LOW = C2 & MASK_LOW;
-   for (i = 0; i < blockCounts; i++) {
-    if (i & 1) {
-     k1 = dataUint32[i];
-     k1 = k1 * C1 & MASK_HIGH | k1 * C1_LOW & MASK_LOW;
-     k1 = k1 << 15 | k1 >>> 17;
-     k1 = k1 * C2 & MASK_HIGH | k1 * C2_LOW & MASK_LOW;
-     h1 ^= k1;
-     h1 = h1 << 13 | h1 >>> 19;
-     h1 = h1 * 5 + 0xe6546b64;
-    } else {
-     k2 = dataUint32[i];
-     k2 = k2 * C1 & MASK_HIGH | k2 * C1_LOW & MASK_LOW;
-     k2 = k2 << 15 | k2 >>> 17;
-     k2 = k2 * C2 & MASK_HIGH | k2 * C2_LOW & MASK_LOW;
-     h2 ^= k2;
-     h2 = h2 << 13 | h2 >>> 19;
-     h2 = h2 * 5 + 0xe6546b64;
-    }
-   }
-   k1 = 0;
-   switch (tailLength) {
-   case 3:
-    k1 ^= data[blockCounts * 4 + 2] << 16;
-   case 2:
-    k1 ^= data[blockCounts * 4 + 1] << 8;
-   case 1:
-    k1 ^= data[blockCounts * 4];
-    k1 = k1 * C1 & MASK_HIGH | k1 * C1_LOW & MASK_LOW;
-    k1 = k1 << 15 | k1 >>> 17;
-    k1 = k1 * C2 & MASK_HIGH | k1 * C2_LOW & MASK_LOW;
-    if (blockCounts & 1) {
-     h1 ^= k1;
-    } else {
-     h2 ^= k1;
-    }
-   }
-   this.h1 = h1;
-   this.h2 = h2;
-   return this;
-  },
-  hexdigest: function MurmurHash3_64_hexdigest() {
-   var h1 = this.h1;
-   var h2 = this.h2;
-   h1 ^= h2 >>> 1;
-   h1 = h1 * 0xed558ccd & MASK_HIGH | h1 * 0x8ccd & MASK_LOW;
-   h2 = h2 * 0xff51afd7 & MASK_HIGH | ((h2 << 16 | h1 >>> 16) * 0xafd7ed55 & MASK_HIGH) >>> 16;
-   h1 ^= h2 >>> 1;
-   h1 = h1 * 0x1a85ec53 & MASK_HIGH | h1 * 0xec53 & MASK_LOW;
-   h2 = h2 * 0xc4ceb9fe & MASK_HIGH | ((h2 << 16 | h1 >>> 16) * 0xb9fe1a85 & MASK_HIGH) >>> 16;
-   h1 ^= h2 >>> 1;
-   for (var i = 0, arr = [
-      h1,
-      h2
-     ], str = ''; i < arr.length; i++) {
-    var hex = (arr[i] >>> 0).toString(16);
-    while (hex.length < 8) {
-     hex = '0' + hex;
-    }
-    str += hex;
-   }
-   return str;
+  var MASK_HIGH = 0xffff0000;
+  var MASK_LOW = 0xffff;
+  function MurmurHash3_64(seed) {
+    var SEED = 0xc3d2e1f0;
+    this.h1 = seed ? seed & 0xffffffff : SEED;
+    this.h2 = seed ? seed & 0xffffffff : SEED;
+  }
+  var alwaysUseUint32ArrayView = false;
+  try {
+    new Uint32Array(new Uint8Array(5).buffer, 0, 1);
+  } catch (e) {
+    alwaysUseUint32ArrayView = true;
   }
- };
- return MurmurHash3_64;
+  MurmurHash3_64.prototype = {
+    update: function MurmurHash3_64_update(input) {
+      var useUint32ArrayView = alwaysUseUint32ArrayView;
+      var i;
+      if (typeof input === 'string') {
+        var data = new Uint8Array(input.length * 2);
+        var length = 0;
+        for (i = 0; i < input.length; i++) {
+          var code = input.charCodeAt(i);
+          if (code <= 0xff) {
+            data[length++] = code;
+          } else {
+            data[length++] = code >>> 8;
+            data[length++] = code & 0xff;
+          }
+        }
+      } else if (input instanceof Uint8Array) {
+        data = input;
+        length = data.length;
+      } else if (typeof input === 'object' && 'length' in input) {
+        data = input;
+        length = data.length;
+        useUint32ArrayView = true;
+      } else {
+        throw new Error('Wrong data format in MurmurHash3_64_update. ' + 'Input must be a string or array.');
+      }
+      var blockCounts = length >> 2;
+      var tailLength = length - blockCounts * 4;
+      var dataUint32 = useUint32ArrayView ? new Uint32ArrayView(data, blockCounts) : new Uint32Array(data.buffer, 0, blockCounts);
+      var k1 = 0;
+      var k2 = 0;
+      var h1 = this.h1;
+      var h2 = this.h2;
+      var C1 = 0xcc9e2d51;
+      var C2 = 0x1b873593;
+      var C1_LOW = C1 & MASK_LOW;
+      var C2_LOW = C2 & MASK_LOW;
+      for (i = 0; i < blockCounts; i++) {
+        if (i & 1) {
+          k1 = dataUint32[i];
+          k1 = k1 * C1 & MASK_HIGH | k1 * C1_LOW & MASK_LOW;
+          k1 = k1 << 15 | k1 >>> 17;
+          k1 = k1 * C2 & MASK_HIGH | k1 * C2_LOW & MASK_LOW;
+          h1 ^= k1;
+          h1 = h1 << 13 | h1 >>> 19;
+          h1 = h1 * 5 + 0xe6546b64;
+        } else {
+          k2 = dataUint32[i];
+          k2 = k2 * C1 & MASK_HIGH | k2 * C1_LOW & MASK_LOW;
+          k2 = k2 << 15 | k2 >>> 17;
+          k2 = k2 * C2 & MASK_HIGH | k2 * C2_LOW & MASK_LOW;
+          h2 ^= k2;
+          h2 = h2 << 13 | h2 >>> 19;
+          h2 = h2 * 5 + 0xe6546b64;
+        }
+      }
+      k1 = 0;
+      switch (tailLength) {
+        case 3:
+          k1 ^= data[blockCounts * 4 + 2] << 16;
+        case 2:
+          k1 ^= data[blockCounts * 4 + 1] << 8;
+        case 1:
+          k1 ^= data[blockCounts * 4];
+          k1 = k1 * C1 & MASK_HIGH | k1 * C1_LOW & MASK_LOW;
+          k1 = k1 << 15 | k1 >>> 17;
+          k1 = k1 * C2 & MASK_HIGH | k1 * C2_LOW & MASK_LOW;
+          if (blockCounts & 1) {
+            h1 ^= k1;
+          } else {
+            h2 ^= k1;
+          }
+      }
+      this.h1 = h1;
+      this.h2 = h2;
+      return this;
+    },
+    hexdigest: function MurmurHash3_64_hexdigest() {
+      var h1 = this.h1;
+      var h2 = this.h2;
+      h1 ^= h2 >>> 1;
+      h1 = h1 * 0xed558ccd & MASK_HIGH | h1 * 0x8ccd & MASK_LOW;
+      h2 = h2 * 0xff51afd7 & MASK_HIGH | ((h2 << 16 | h1 >>> 16) * 0xafd7ed55 & MASK_HIGH) >>> 16;
+      h1 ^= h2 >>> 1;
+      h1 = h1 * 0x1a85ec53 & MASK_HIGH | h1 * 0xec53 & MASK_LOW;
+      h2 = h2 * 0xc4ceb9fe & MASK_HIGH | ((h2 << 16 | h1 >>> 16) * 0xb9fe1a85 & MASK_HIGH) >>> 16;
+      h1 ^= h2 >>> 1;
+      for (var i = 0, arr = [h1, h2], str = ''; i < arr.length; i++) {
+        var hex = (arr[i] >>> 0).toString(16);
+        while (hex.length < 8) {
+          hex = '0' + hex;
+        }
+        str += hex;
+      }
+      return str;
+    }
+  };
+  return MurmurHash3_64;
 }();
 exports.MurmurHash3_64 = MurmurHash3_64;

+ 450 - 449
lib/core/network.js

@@ -13,196 +13,197 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var coreWorker = require('./worker.js');
 var globalScope = sharedUtil.globalScope;
 var OK_RESPONSE = 200;
 var PARTIAL_CONTENT_RESPONSE = 206;
 function NetworkManager(url, args) {
- this.url = url;
- args = args || {};
- this.isHttp = /^https?:/i.test(url);
- this.httpHeaders = this.isHttp && args.httpHeaders || {};
- this.withCredentials = args.withCredentials || false;
- this.getXhr = args.getXhr || function NetworkManager_getXhr() {
-  return new XMLHttpRequest();
- };
- this.currXhrId = 0;
- this.pendingRequests = Object.create(null);
- this.loadedRequests = Object.create(null);
+  this.url = url;
+  args = args || {};
+  this.isHttp = /^https?:/i.test(url);
+  this.httpHeaders = this.isHttp && args.httpHeaders || {};
+  this.withCredentials = args.withCredentials || false;
+  this.getXhr = args.getXhr || function NetworkManager_getXhr() {
+    return new XMLHttpRequest();
+  };
+  this.currXhrId = 0;
+  this.pendingRequests = Object.create(null);
+  this.loadedRequests = Object.create(null);
 }
 function getArrayBuffer(xhr) {
- var data = xhr.response;
- if (typeof data !== 'string') {
-  return data;
- }
- var length = data.length;
- var array = new Uint8Array(length);
- for (var i = 0; i < length; i++) {
-  array[i] = data.charCodeAt(i) & 0xFF;
- }
- return array.buffer;
+  var data = xhr.response;
+  if (typeof data !== 'string') {
+    return data;
+  }
+  var length = data.length;
+  var array = new Uint8Array(length);
+  for (var i = 0; i < length; i++) {
+    array[i] = data.charCodeAt(i) & 0xFF;
+  }
+  return array.buffer;
 }
 var supportsMozChunked = function supportsMozChunkedClosure() {
- try {
-  var x = new XMLHttpRequest();
-  x.open('GET', globalScope.location.href);
-  x.responseType = 'moz-chunked-arraybuffer';
-  return x.responseType === 'moz-chunked-arraybuffer';
- } catch (e) {
-  return false;
- }
+  try {
+    var x = new XMLHttpRequest();
+    x.open('GET', globalScope.location.href);
+    x.responseType = 'moz-chunked-arraybuffer';
+    return x.responseType === 'moz-chunked-arraybuffer';
+  } catch (e) {
+    return false;
+  }
 }();
 NetworkManager.prototype = {
- requestRange: function NetworkManager_requestRange(begin, end, listeners) {
-  var args = {
-   begin: begin,
-   end: end
-  };
-  for (var prop in listeners) {
-   args[prop] = listeners[prop];
-  }
-  return this.request(args);
- },
- requestFull: function NetworkManager_requestFull(listeners) {
-  return this.request(listeners);
- },
- request: function NetworkManager_request(args) {
-  var xhr = this.getXhr();
-  var xhrId = this.currXhrId++;
-  var pendingRequest = this.pendingRequests[xhrId] = { xhr: xhr };
-  xhr.open('GET', this.url);
-  xhr.withCredentials = this.withCredentials;
-  for (var property in this.httpHeaders) {
-   var value = this.httpHeaders[property];
-   if (typeof value === 'undefined') {
-    continue;
-   }
-   xhr.setRequestHeader(property, value);
-  }
-  if (this.isHttp && 'begin' in args && 'end' in args) {
-   var rangeStr = args.begin + '-' + (args.end - 1);
-   xhr.setRequestHeader('Range', 'bytes=' + rangeStr);
-   pendingRequest.expectedStatus = 206;
-  } else {
-   pendingRequest.expectedStatus = 200;
-  }
-  var useMozChunkedLoading = supportsMozChunked && !!args.onProgressiveData;
-  if (useMozChunkedLoading) {
-   xhr.responseType = 'moz-chunked-arraybuffer';
-   pendingRequest.onProgressiveData = args.onProgressiveData;
-   pendingRequest.mozChunked = true;
-  } else {
-   xhr.responseType = 'arraybuffer';
-  }
-  if (args.onError) {
-   xhr.onerror = function (evt) {
-    args.onError(xhr.status);
-   };
-  }
-  xhr.onreadystatechange = this.onStateChange.bind(this, xhrId);
-  xhr.onprogress = this.onProgress.bind(this, xhrId);
-  pendingRequest.onHeadersReceived = args.onHeadersReceived;
-  pendingRequest.onDone = args.onDone;
-  pendingRequest.onError = args.onError;
-  pendingRequest.onProgress = args.onProgress;
-  xhr.send(null);
-  return xhrId;
- },
- onProgress: function NetworkManager_onProgress(xhrId, evt) {
-  var pendingRequest = this.pendingRequests[xhrId];
-  if (!pendingRequest) {
-   return;
-  }
-  if (pendingRequest.mozChunked) {
-   var chunk = getArrayBuffer(pendingRequest.xhr);
-   pendingRequest.onProgressiveData(chunk);
-  }
-  var onProgress = pendingRequest.onProgress;
-  if (onProgress) {
-   onProgress(evt);
-  }
- },
- onStateChange: function NetworkManager_onStateChange(xhrId, evt) {
-  var pendingRequest = this.pendingRequests[xhrId];
-  if (!pendingRequest) {
-   return;
-  }
-  var xhr = pendingRequest.xhr;
-  if (xhr.readyState >= 2 && pendingRequest.onHeadersReceived) {
-   pendingRequest.onHeadersReceived();
-   delete pendingRequest.onHeadersReceived;
-  }
-  if (xhr.readyState !== 4) {
-   return;
-  }
-  if (!(xhrId in this.pendingRequests)) {
-   return;
+  requestRange: function NetworkManager_requestRange(begin, end, listeners) {
+    var args = {
+      begin: begin,
+      end: end
+    };
+    for (var prop in listeners) {
+      args[prop] = listeners[prop];
+    }
+    return this.request(args);
+  },
+  requestFull: function NetworkManager_requestFull(listeners) {
+    return this.request(listeners);
+  },
+  request: function NetworkManager_request(args) {
+    var xhr = this.getXhr();
+    var xhrId = this.currXhrId++;
+    var pendingRequest = this.pendingRequests[xhrId] = { xhr: xhr };
+    xhr.open('GET', this.url);
+    xhr.withCredentials = this.withCredentials;
+    for (var property in this.httpHeaders) {
+      var value = this.httpHeaders[property];
+      if (typeof value === 'undefined') {
+        continue;
+      }
+      xhr.setRequestHeader(property, value);
+    }
+    if (this.isHttp && 'begin' in args && 'end' in args) {
+      var rangeStr = args.begin + '-' + (args.end - 1);
+      xhr.setRequestHeader('Range', 'bytes=' + rangeStr);
+      pendingRequest.expectedStatus = 206;
+    } else {
+      pendingRequest.expectedStatus = 200;
+    }
+    var useMozChunkedLoading = supportsMozChunked && !!args.onProgressiveData;
+    if (useMozChunkedLoading) {
+      xhr.responseType = 'moz-chunked-arraybuffer';
+      pendingRequest.onProgressiveData = args.onProgressiveData;
+      pendingRequest.mozChunked = true;
+    } else {
+      xhr.responseType = 'arraybuffer';
+    }
+    if (args.onError) {
+      xhr.onerror = function (evt) {
+        args.onError(xhr.status);
+      };
+    }
+    xhr.onreadystatechange = this.onStateChange.bind(this, xhrId);
+    xhr.onprogress = this.onProgress.bind(this, xhrId);
+    pendingRequest.onHeadersReceived = args.onHeadersReceived;
+    pendingRequest.onDone = args.onDone;
+    pendingRequest.onError = args.onError;
+    pendingRequest.onProgress = args.onProgress;
+    xhr.send(null);
+    return xhrId;
+  },
+  onProgress: function NetworkManager_onProgress(xhrId, evt) {
+    var pendingRequest = this.pendingRequests[xhrId];
+    if (!pendingRequest) {
+      return;
+    }
+    if (pendingRequest.mozChunked) {
+      var chunk = getArrayBuffer(pendingRequest.xhr);
+      pendingRequest.onProgressiveData(chunk);
+    }
+    var onProgress = pendingRequest.onProgress;
+    if (onProgress) {
+      onProgress(evt);
+    }
+  },
+  onStateChange: function NetworkManager_onStateChange(xhrId, evt) {
+    var pendingRequest = this.pendingRequests[xhrId];
+    if (!pendingRequest) {
+      return;
+    }
+    var xhr = pendingRequest.xhr;
+    if (xhr.readyState >= 2 && pendingRequest.onHeadersReceived) {
+      pendingRequest.onHeadersReceived();
+      delete pendingRequest.onHeadersReceived;
+    }
+    if (xhr.readyState !== 4) {
+      return;
+    }
+    if (!(xhrId in this.pendingRequests)) {
+      return;
+    }
+    delete this.pendingRequests[xhrId];
+    if (xhr.status === 0 && this.isHttp) {
+      if (pendingRequest.onError) {
+        pendingRequest.onError(xhr.status);
+      }
+      return;
+    }
+    var xhrStatus = xhr.status || OK_RESPONSE;
+    var ok_response_on_range_request = xhrStatus === OK_RESPONSE && pendingRequest.expectedStatus === PARTIAL_CONTENT_RESPONSE;
+    if (!ok_response_on_range_request && xhrStatus !== pendingRequest.expectedStatus) {
+      if (pendingRequest.onError) {
+        pendingRequest.onError(xhr.status);
+      }
+      return;
+    }
+    this.loadedRequests[xhrId] = true;
+    var chunk = getArrayBuffer(xhr);
+    if (xhrStatus === PARTIAL_CONTENT_RESPONSE) {
+      var rangeHeader = xhr.getResponseHeader('Content-Range');
+      var matches = /bytes (\d+)-(\d+)\/(\d+)/.exec(rangeHeader);
+      var begin = parseInt(matches[1], 10);
+      pendingRequest.onDone({
+        begin: begin,
+        chunk: chunk
+      });
+    } else if (pendingRequest.onProgressiveData) {
+      pendingRequest.onDone(null);
+    } else if (chunk) {
+      pendingRequest.onDone({
+        begin: 0,
+        chunk: chunk
+      });
+    } else if (pendingRequest.onError) {
+      pendingRequest.onError(xhr.status);
+    }
+  },
+  hasPendingRequests: function NetworkManager_hasPendingRequests() {
+    for (var xhrId in this.pendingRequests) {
+      return true;
+    }
+    return false;
+  },
+  getRequestXhr: function NetworkManager_getXhr(xhrId) {
+    return this.pendingRequests[xhrId].xhr;
+  },
+  isStreamingRequest: function NetworkManager_isStreamingRequest(xhrId) {
+    return !!this.pendingRequests[xhrId].onProgressiveData;
+  },
+  isPendingRequest: function NetworkManager_isPendingRequest(xhrId) {
+    return xhrId in this.pendingRequests;
+  },
+  isLoadedRequest: function NetworkManager_isLoadedRequest(xhrId) {
+    return xhrId in this.loadedRequests;
+  },
+  abortAllRequests: function NetworkManager_abortAllRequests() {
+    for (var xhrId in this.pendingRequests) {
+      this.abortRequest(xhrId | 0);
+    }
+  },
+  abortRequest: function NetworkManager_abortRequest(xhrId) {
+    var xhr = this.pendingRequests[xhrId].xhr;
+    delete this.pendingRequests[xhrId];
+    xhr.abort();
   }
-  delete this.pendingRequests[xhrId];
-  if (xhr.status === 0 && this.isHttp) {
-   if (pendingRequest.onError) {
-    pendingRequest.onError(xhr.status);
-   }
-   return;
-  }
-  var xhrStatus = xhr.status || OK_RESPONSE;
-  var ok_response_on_range_request = xhrStatus === OK_RESPONSE && pendingRequest.expectedStatus === PARTIAL_CONTENT_RESPONSE;
-  if (!ok_response_on_range_request && xhrStatus !== pendingRequest.expectedStatus) {
-   if (pendingRequest.onError) {
-    pendingRequest.onError(xhr.status);
-   }
-   return;
-  }
-  this.loadedRequests[xhrId] = true;
-  var chunk = getArrayBuffer(xhr);
-  if (xhrStatus === PARTIAL_CONTENT_RESPONSE) {
-   var rangeHeader = xhr.getResponseHeader('Content-Range');
-   var matches = /bytes (\d+)-(\d+)\/(\d+)/.exec(rangeHeader);
-   var begin = parseInt(matches[1], 10);
-   pendingRequest.onDone({
-    begin: begin,
-    chunk: chunk
-   });
-  } else if (pendingRequest.onProgressiveData) {
-   pendingRequest.onDone(null);
-  } else if (chunk) {
-   pendingRequest.onDone({
-    begin: 0,
-    chunk: chunk
-   });
-  } else if (pendingRequest.onError) {
-   pendingRequest.onError(xhr.status);
-  }
- },
- hasPendingRequests: function NetworkManager_hasPendingRequests() {
-  for (var xhrId in this.pendingRequests) {
-   return true;
-  }
-  return false;
- },
- getRequestXhr: function NetworkManager_getXhr(xhrId) {
-  return this.pendingRequests[xhrId].xhr;
- },
- isStreamingRequest: function NetworkManager_isStreamingRequest(xhrId) {
-  return !!this.pendingRequests[xhrId].onProgressiveData;
- },
- isPendingRequest: function NetworkManager_isPendingRequest(xhrId) {
-  return xhrId in this.pendingRequests;
- },
- isLoadedRequest: function NetworkManager_isLoadedRequest(xhrId) {
-  return xhrId in this.loadedRequests;
- },
- abortAllRequests: function NetworkManager_abortAllRequests() {
-  for (var xhrId in this.pendingRequests) {
-   this.abortRequest(xhrId | 0);
-  }
- },
- abortRequest: function NetworkManager_abortRequest(xhrId) {
-  var xhr = this.pendingRequests[xhrId].xhr;
-  delete this.pendingRequests[xhrId];
-  xhr.abort();
- }
 };
 var assert = sharedUtil.assert;
 var createPromiseCapability = sharedUtil.createPromiseCapability;
@@ -210,291 +211,291 @@ var isInt = sharedUtil.isInt;
 var MissingPDFException = sharedUtil.MissingPDFException;
 var UnexpectedResponseException = sharedUtil.UnexpectedResponseException;
 function PDFNetworkStream(options) {
- this._options = options;
- var source = options.source;
- this._manager = new NetworkManager(source.url, {
-  httpHeaders: source.httpHeaders,
-  withCredentials: source.withCredentials
- });
- this._rangeChunkSize = source.rangeChunkSize;
- this._fullRequestReader = null;
- this._rangeRequestReaders = [];
+  this._options = options;
+  var source = options.source;
+  this._manager = new NetworkManager(source.url, {
+    httpHeaders: source.httpHeaders,
+    withCredentials: source.withCredentials
+  });
+  this._rangeChunkSize = source.rangeChunkSize;
+  this._fullRequestReader = null;
+  this._rangeRequestReaders = [];
 }
 PDFNetworkStream.prototype = {
- _onRangeRequestReaderClosed: function PDFNetworkStream_onRangeRequestReaderClosed(reader) {
-  var i = this._rangeRequestReaders.indexOf(reader);
-  if (i >= 0) {
-   this._rangeRequestReaders.splice(i, 1);
-  }
- },
- getFullReader: function PDFNetworkStream_getFullReader() {
-  assert(!this._fullRequestReader);
-  this._fullRequestReader = new PDFNetworkStreamFullRequestReader(this._manager, this._options);
-  return this._fullRequestReader;
- },
- getRangeReader: function PDFNetworkStream_getRangeReader(begin, end) {
-  var reader = new PDFNetworkStreamRangeRequestReader(this._manager, begin, end);
-  reader.onClosed = this._onRangeRequestReaderClosed.bind(this);
-  this._rangeRequestReaders.push(reader);
-  return reader;
- },
- cancelAllRequests: function PDFNetworkStream_cancelAllRequests(reason) {
-  if (this._fullRequestReader) {
-   this._fullRequestReader.cancel(reason);
+  _onRangeRequestReaderClosed: function PDFNetworkStream_onRangeRequestReaderClosed(reader) {
+    var i = this._rangeRequestReaders.indexOf(reader);
+    if (i >= 0) {
+      this._rangeRequestReaders.splice(i, 1);
+    }
+  },
+  getFullReader: function PDFNetworkStream_getFullReader() {
+    assert(!this._fullRequestReader);
+    this._fullRequestReader = new PDFNetworkStreamFullRequestReader(this._manager, this._options);
+    return this._fullRequestReader;
+  },
+  getRangeReader: function PDFNetworkStream_getRangeReader(begin, end) {
+    var reader = new PDFNetworkStreamRangeRequestReader(this._manager, begin, end);
+    reader.onClosed = this._onRangeRequestReaderClosed.bind(this);
+    this._rangeRequestReaders.push(reader);
+    return reader;
+  },
+  cancelAllRequests: function PDFNetworkStream_cancelAllRequests(reason) {
+    if (this._fullRequestReader) {
+      this._fullRequestReader.cancel(reason);
+    }
+    var readers = this._rangeRequestReaders.slice(0);
+    readers.forEach(function (reader) {
+      reader.cancel(reason);
+    });
   }
-  var readers = this._rangeRequestReaders.slice(0);
-  readers.forEach(function (reader) {
-   reader.cancel(reason);
-  });
- }
 };
 function PDFNetworkStreamFullRequestReader(manager, options) {
- this._manager = manager;
- var source = options.source;
- var args = {
-  onHeadersReceived: this._onHeadersReceived.bind(this),
-  onProgressiveData: source.disableStream ? null : this._onProgressiveData.bind(this),
-  onDone: this._onDone.bind(this),
-  onError: this._onError.bind(this),
-  onProgress: this._onProgress.bind(this)
- };
- this._url = source.url;
- this._fullRequestId = manager.requestFull(args);
- this._headersReceivedCapability = createPromiseCapability();
- this._disableRange = options.disableRange || false;
- this._contentLength = source.length;
- this._rangeChunkSize = source.rangeChunkSize;
- if (!this._rangeChunkSize && !this._disableRange) {
-  this._disableRange = true;
- }
- this._isStreamingSupported = false;
- this._isRangeSupported = false;
- this._cachedChunks = [];
- this._requests = [];
- this._done = false;
- this._storedError = undefined;
- this.onProgress = null;
-}
-PDFNetworkStreamFullRequestReader.prototype = {
- _validateRangeRequestCapabilities: function PDFNetworkStreamFullRequestReader_validateRangeRequestCapabilities() {
-  if (this._disableRange) {
-   return false;
-  }
-  var networkManager = this._manager;
-  if (!networkManager.isHttp) {
-   return false;
-  }
-  var fullRequestXhrId = this._fullRequestId;
-  var fullRequestXhr = networkManager.getRequestXhr(fullRequestXhrId);
-  if (fullRequestXhr.getResponseHeader('Accept-Ranges') !== 'bytes') {
-   return false;
-  }
-  var contentEncoding = fullRequestXhr.getResponseHeader('Content-Encoding') || 'identity';
-  if (contentEncoding !== 'identity') {
-   return false;
-  }
-  var length = fullRequestXhr.getResponseHeader('Content-Length');
-  length = parseInt(length, 10);
-  if (!isInt(length)) {
-   return false;
-  }
-  this._contentLength = length;
-  if (length <= 2 * this._rangeChunkSize) {
-   return false;
-  }
-  return true;
- },
- _onHeadersReceived: function PDFNetworkStreamFullRequestReader_onHeadersReceived() {
-  if (this._validateRangeRequestCapabilities()) {
-   this._isRangeSupported = true;
-  }
-  var networkManager = this._manager;
-  var fullRequestXhrId = this._fullRequestId;
-  if (networkManager.isStreamingRequest(fullRequestXhrId)) {
-   this._isStreamingSupported = true;
-  } else if (this._isRangeSupported) {
-   networkManager.abortRequest(fullRequestXhrId);
-  }
-  this._headersReceivedCapability.resolve();
- },
- _onProgressiveData: function PDFNetworkStreamFullRequestReader_onProgressiveData(chunk) {
-  if (this._requests.length > 0) {
-   var requestCapability = this._requests.shift();
-   requestCapability.resolve({
-    value: chunk,
-    done: false
-   });
-  } else {
-   this._cachedChunks.push(chunk);
-  }
- },
- _onDone: function PDFNetworkStreamFullRequestReader_onDone(args) {
-  if (args) {
-   this._onProgressiveData(args.chunk);
-  }
-  this._done = true;
-  if (this._cachedChunks.length > 0) {
-   return;
-  }
-  this._requests.forEach(function (requestCapability) {
-   requestCapability.resolve({
-    value: undefined,
-    done: true
-   });
-  });
-  this._requests = [];
- },
- _onError: function PDFNetworkStreamFullRequestReader_onError(status) {
-  var url = this._url;
-  var exception;
-  if (status === 404 || status === 0 && /^file:/.test(url)) {
-   exception = new MissingPDFException('Missing PDF "' + url + '".');
-  } else {
-   exception = new UnexpectedResponseException('Unexpected server response (' + status + ') while retrieving PDF "' + url + '".', status);
-  }
-  this._storedError = exception;
-  this._headersReceivedCapability.reject(exception);
-  this._requests.forEach(function (requestCapability) {
-   requestCapability.reject(exception);
-  });
-  this._requests = [];
+  this._manager = manager;
+  var source = options.source;
+  var args = {
+    onHeadersReceived: this._onHeadersReceived.bind(this),
+    onProgressiveData: source.disableStream ? null : this._onProgressiveData.bind(this),
+    onDone: this._onDone.bind(this),
+    onError: this._onError.bind(this),
+    onProgress: this._onProgress.bind(this)
+  };
+  this._url = source.url;
+  this._fullRequestId = manager.requestFull(args);
+  this._headersReceivedCapability = createPromiseCapability();
+  this._disableRange = options.disableRange || false;
+  this._contentLength = source.length;
+  this._rangeChunkSize = source.rangeChunkSize;
+  if (!this._rangeChunkSize && !this._disableRange) {
+    this._disableRange = true;
+  }
+  this._isStreamingSupported = false;
+  this._isRangeSupported = false;
   this._cachedChunks = [];
- },
- _onProgress: function PDFNetworkStreamFullRequestReader_onProgress(data) {
-  if (this.onProgress) {
-   this.onProgress({
-    loaded: data.loaded,
-    total: data.lengthComputable ? data.total : this._contentLength
-   });
-  }
- },
- get isRangeSupported() {
-  return this._isRangeSupported;
- },
- get isStreamingSupported() {
-  return this._isStreamingSupported;
- },
- get contentLength() {
-  return this._contentLength;
- },
- get headersReady() {
-  return this._headersReceivedCapability.promise;
- },
- read: function PDFNetworkStreamFullRequestReader_read() {
-  if (this._storedError) {
-   return Promise.reject(this._storedError);
-  }
-  if (this._cachedChunks.length > 0) {
-   var chunk = this._cachedChunks.shift();
-   return Promise.resolve(chunk);
-  }
-  if (this._done) {
-   return Promise.resolve({
-    value: undefined,
-    done: true
-   });
-  }
-  var requestCapability = createPromiseCapability();
-  this._requests.push(requestCapability);
-  return requestCapability.promise;
- },
- cancel: function PDFNetworkStreamFullRequestReader_cancel(reason) {
-  this._done = true;
-  this._headersReceivedCapability.reject(reason);
-  this._requests.forEach(function (requestCapability) {
-   requestCapability.resolve({
-    value: undefined,
-    done: true
-   });
-  });
   this._requests = [];
-  if (this._manager.isPendingRequest(this._fullRequestId)) {
-   this._manager.abortRequest(this._fullRequestId);
+  this._done = false;
+  this._storedError = undefined;
+  this.onProgress = null;
+}
+PDFNetworkStreamFullRequestReader.prototype = {
+  _validateRangeRequestCapabilities: function PDFNetworkStreamFullRequestReader_validateRangeRequestCapabilities() {
+    if (this._disableRange) {
+      return false;
+    }
+    var networkManager = this._manager;
+    if (!networkManager.isHttp) {
+      return false;
+    }
+    var fullRequestXhrId = this._fullRequestId;
+    var fullRequestXhr = networkManager.getRequestXhr(fullRequestXhrId);
+    if (fullRequestXhr.getResponseHeader('Accept-Ranges') !== 'bytes') {
+      return false;
+    }
+    var contentEncoding = fullRequestXhr.getResponseHeader('Content-Encoding') || 'identity';
+    if (contentEncoding !== 'identity') {
+      return false;
+    }
+    var length = fullRequestXhr.getResponseHeader('Content-Length');
+    length = parseInt(length, 10);
+    if (!isInt(length)) {
+      return false;
+    }
+    this._contentLength = length;
+    if (length <= 2 * this._rangeChunkSize) {
+      return false;
+    }
+    return true;
+  },
+  _onHeadersReceived: function PDFNetworkStreamFullRequestReader_onHeadersReceived() {
+    if (this._validateRangeRequestCapabilities()) {
+      this._isRangeSupported = true;
+    }
+    var networkManager = this._manager;
+    var fullRequestXhrId = this._fullRequestId;
+    if (networkManager.isStreamingRequest(fullRequestXhrId)) {
+      this._isStreamingSupported = true;
+    } else if (this._isRangeSupported) {
+      networkManager.abortRequest(fullRequestXhrId);
+    }
+    this._headersReceivedCapability.resolve();
+  },
+  _onProgressiveData: function PDFNetworkStreamFullRequestReader_onProgressiveData(chunk) {
+    if (this._requests.length > 0) {
+      var requestCapability = this._requests.shift();
+      requestCapability.resolve({
+        value: chunk,
+        done: false
+      });
+    } else {
+      this._cachedChunks.push(chunk);
+    }
+  },
+  _onDone: function PDFNetworkStreamFullRequestReader_onDone(args) {
+    if (args) {
+      this._onProgressiveData(args.chunk);
+    }
+    this._done = true;
+    if (this._cachedChunks.length > 0) {
+      return;
+    }
+    this._requests.forEach(function (requestCapability) {
+      requestCapability.resolve({
+        value: undefined,
+        done: true
+      });
+    });
+    this._requests = [];
+  },
+  _onError: function PDFNetworkStreamFullRequestReader_onError(status) {
+    var url = this._url;
+    var exception;
+    if (status === 404 || status === 0 && /^file:/.test(url)) {
+      exception = new MissingPDFException('Missing PDF "' + url + '".');
+    } else {
+      exception = new UnexpectedResponseException('Unexpected server response (' + status + ') while retrieving PDF "' + url + '".', status);
+    }
+    this._storedError = exception;
+    this._headersReceivedCapability.reject(exception);
+    this._requests.forEach(function (requestCapability) {
+      requestCapability.reject(exception);
+    });
+    this._requests = [];
+    this._cachedChunks = [];
+  },
+  _onProgress: function PDFNetworkStreamFullRequestReader_onProgress(data) {
+    if (this.onProgress) {
+      this.onProgress({
+        loaded: data.loaded,
+        total: data.lengthComputable ? data.total : this._contentLength
+      });
+    }
+  },
+  get isRangeSupported() {
+    return this._isRangeSupported;
+  },
+  get isStreamingSupported() {
+    return this._isStreamingSupported;
+  },
+  get contentLength() {
+    return this._contentLength;
+  },
+  get headersReady() {
+    return this._headersReceivedCapability.promise;
+  },
+  read: function PDFNetworkStreamFullRequestReader_read() {
+    if (this._storedError) {
+      return Promise.reject(this._storedError);
+    }
+    if (this._cachedChunks.length > 0) {
+      var chunk = this._cachedChunks.shift();
+      return Promise.resolve(chunk);
+    }
+    if (this._done) {
+      return Promise.resolve({
+        value: undefined,
+        done: true
+      });
+    }
+    var requestCapability = createPromiseCapability();
+    this._requests.push(requestCapability);
+    return requestCapability.promise;
+  },
+  cancel: function PDFNetworkStreamFullRequestReader_cancel(reason) {
+    this._done = true;
+    this._headersReceivedCapability.reject(reason);
+    this._requests.forEach(function (requestCapability) {
+      requestCapability.resolve({
+        value: undefined,
+        done: true
+      });
+    });
+    this._requests = [];
+    if (this._manager.isPendingRequest(this._fullRequestId)) {
+      this._manager.abortRequest(this._fullRequestId);
+    }
+    this._fullRequestReader = null;
   }
-  this._fullRequestReader = null;
- }
 };
 function PDFNetworkStreamRangeRequestReader(manager, begin, end) {
- this._manager = manager;
- var args = {
-  onDone: this._onDone.bind(this),
-  onProgress: this._onProgress.bind(this)
- };
- this._requestId = manager.requestRange(begin, end, args);
- this._requests = [];
- this._queuedChunk = null;
- this._done = false;
- this.onProgress = null;
- this.onClosed = null;
+  this._manager = manager;
+  var args = {
+    onDone: this._onDone.bind(this),
+    onProgress: this._onProgress.bind(this)
+  };
+  this._requestId = manager.requestRange(begin, end, args);
+  this._requests = [];
+  this._queuedChunk = null;
+  this._done = false;
+  this.onProgress = null;
+  this.onClosed = null;
 }
 PDFNetworkStreamRangeRequestReader.prototype = {
- _close: function PDFNetworkStreamRangeRequestReader_close() {
-  if (this.onClosed) {
-   this.onClosed(this);
-  }
- },
- _onDone: function PDFNetworkStreamRangeRequestReader_onDone(data) {
-  var chunk = data.chunk;
-  if (this._requests.length > 0) {
-   var requestCapability = this._requests.shift();
-   requestCapability.resolve({
-    value: chunk,
-    done: false
-   });
-  } else {
-   this._queuedChunk = chunk;
-  }
-  this._done = true;
-  this._requests.forEach(function (requestCapability) {
-   requestCapability.resolve({
-    value: undefined,
-    done: true
-   });
-  });
-  this._requests = [];
-  this._close();
- },
- _onProgress: function PDFNetworkStreamRangeRequestReader_onProgress(evt) {
-  if (!this.isStreamingSupported && this.onProgress) {
-   this.onProgress({ loaded: evt.loaded });
-  }
- },
- get isStreamingSupported() {
-  return false;
- },
- read: function PDFNetworkStreamRangeRequestReader_read() {
-  if (this._queuedChunk !== null) {
-   var chunk = this._queuedChunk;
-   this._queuedChunk = null;
-   return Promise.resolve({
-    value: chunk,
-    done: false
-   });
-  }
-  if (this._done) {
-   return Promise.resolve({
-    value: undefined,
-    done: true
-   });
-  }
-  var requestCapability = createPromiseCapability();
-  this._requests.push(requestCapability);
-  return requestCapability.promise;
- },
- cancel: function PDFNetworkStreamRangeRequestReader_cancel(reason) {
-  this._done = true;
-  this._requests.forEach(function (requestCapability) {
-   requestCapability.resolve({
-    value: undefined,
-    done: true
-   });
-  });
-  this._requests = [];
-  if (this._manager.isPendingRequest(this._requestId)) {
-   this._manager.abortRequest(this._requestId);
+  _close: function PDFNetworkStreamRangeRequestReader_close() {
+    if (this.onClosed) {
+      this.onClosed(this);
+    }
+  },
+  _onDone: function PDFNetworkStreamRangeRequestReader_onDone(data) {
+    var chunk = data.chunk;
+    if (this._requests.length > 0) {
+      var requestCapability = this._requests.shift();
+      requestCapability.resolve({
+        value: chunk,
+        done: false
+      });
+    } else {
+      this._queuedChunk = chunk;
+    }
+    this._done = true;
+    this._requests.forEach(function (requestCapability) {
+      requestCapability.resolve({
+        value: undefined,
+        done: true
+      });
+    });
+    this._requests = [];
+    this._close();
+  },
+  _onProgress: function PDFNetworkStreamRangeRequestReader_onProgress(evt) {
+    if (!this.isStreamingSupported && this.onProgress) {
+      this.onProgress({ loaded: evt.loaded });
+    }
+  },
+  get isStreamingSupported() {
+    return false;
+  },
+  read: function PDFNetworkStreamRangeRequestReader_read() {
+    if (this._queuedChunk !== null) {
+      var chunk = this._queuedChunk;
+      this._queuedChunk = null;
+      return Promise.resolve({
+        value: chunk,
+        done: false
+      });
+    }
+    if (this._done) {
+      return Promise.resolve({
+        value: undefined,
+        done: true
+      });
+    }
+    var requestCapability = createPromiseCapability();
+    this._requests.push(requestCapability);
+    return requestCapability.promise;
+  },
+  cancel: function PDFNetworkStreamRangeRequestReader_cancel(reason) {
+    this._done = true;
+    this._requests.forEach(function (requestCapability) {
+      requestCapability.resolve({
+        value: undefined,
+        done: true
+      });
+    });
+    this._requests = [];
+    if (this._manager.isPendingRequest(this._requestId)) {
+      this._manager.abortRequest(this._requestId);
+    }
+    this._close();
   }
-  this._close();
- }
 };
 coreWorker.setPDFNetworkStreamClass(PDFNetworkStream);
 exports.PDFNetworkStream = PDFNetworkStream;

+ 1358 - 1375
lib/core/obj.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var coreCrypto = require('./crypto.js');
@@ -52,1413 +53,1395 @@ var Parser = coreParser.Parser;
 var ChunkedStream = coreChunkedStream.ChunkedStream;
 var ColorSpace = coreColorSpace.ColorSpace;
 var Catalog = function CatalogClosure() {
- function Catalog(pdfManager, xref, pageFactory) {
-  this.pdfManager = pdfManager;
-  this.xref = xref;
-  this.catDict = xref.getCatalogObj();
-  this.fontCache = new RefSetCache();
-  this.builtInCMapCache = Object.create(null);
-  assert(isDict(this.catDict), 'catalog object is not a dictionary');
-  this.pageFactory = pageFactory;
-  this.pagePromises = [];
- }
- Catalog.prototype = {
-  get metadata() {
-   var streamRef = this.catDict.getRaw('Metadata');
-   if (!isRef(streamRef)) {
-    return shadow(this, 'metadata', null);
-   }
-   var encryptMetadata = !this.xref.encrypt ? false : this.xref.encrypt.encryptMetadata;
-   var stream = this.xref.fetch(streamRef, !encryptMetadata);
-   var metadata;
-   if (stream && isDict(stream.dict)) {
-    var type = stream.dict.get('Type');
-    var subtype = stream.dict.get('Subtype');
-    if (isName(type, 'Metadata') && isName(subtype, 'XML')) {
-     try {
-      metadata = stringToUTF8String(bytesToString(stream.getBytes()));
-     } catch (e) {
-      if (e instanceof MissingDataException) {
-       throw e;
-      }
-      info('Skipping invalid metadata.');
-     }
-    }
-   }
-   return shadow(this, 'metadata', metadata);
-  },
-  get toplevelPagesDict() {
-   var pagesObj = this.catDict.get('Pages');
-   assert(isDict(pagesObj), 'invalid top-level pages dictionary');
-   return shadow(this, 'toplevelPagesDict', pagesObj);
-  },
-  get documentOutline() {
-   var obj = null;
-   try {
-    obj = this.readDocumentOutline();
-   } catch (ex) {
-    if (ex instanceof MissingDataException) {
-     throw ex;
-    }
-    warn('Unable to read document outline');
-   }
-   return shadow(this, 'documentOutline', obj);
-  },
-  readDocumentOutline: function Catalog_readDocumentOutline() {
-   var obj = this.catDict.get('Outlines');
-   if (!isDict(obj)) {
-    return null;
-   }
-   obj = obj.getRaw('First');
-   if (!isRef(obj)) {
-    return null;
-   }
-   var root = { items: [] };
-   var queue = [{
-     obj: obj,
-     parent: root
-    }];
-   var processed = new RefSet();
-   processed.put(obj);
-   var xref = this.xref, blackColor = new Uint8Array(3);
-   while (queue.length > 0) {
-    var i = queue.shift();
-    var outlineDict = xref.fetchIfRef(i.obj);
-    if (outlineDict === null) {
-     continue;
-    }
-    assert(outlineDict.has('Title'), 'Invalid outline item');
-    var data = {
-     url: null,
-     dest: null
-    };
-    Catalog.parseDestDictionary({
-     destDict: outlineDict,
-     resultObj: data,
-     docBaseUrl: this.pdfManager.docBaseUrl
-    });
-    var title = outlineDict.get('Title');
-    var flags = outlineDict.get('F') || 0;
-    var color = outlineDict.getArray('C'), rgbColor = blackColor;
-    if (isArray(color) && color.length === 3 && (color[0] !== 0 || color[1] !== 0 || color[2] !== 0)) {
-     rgbColor = ColorSpace.singletons.rgb.getRgb(color, 0);
-    }
-    var outlineItem = {
-     dest: data.dest,
-     url: data.url,
-     unsafeUrl: data.unsafeUrl,
-     newWindow: data.newWindow,
-     title: stringToPDFString(title),
-     color: rgbColor,
-     count: outlineDict.get('Count'),
-     bold: !!(flags & 2),
-     italic: !!(flags & 1),
-     items: []
-    };
-    i.parent.items.push(outlineItem);
-    obj = outlineDict.getRaw('First');
-    if (isRef(obj) && !processed.has(obj)) {
-     queue.push({
-      obj: obj,
-      parent: outlineItem
-     });
-     processed.put(obj);
-    }
-    obj = outlineDict.getRaw('Next');
-    if (isRef(obj) && !processed.has(obj)) {
-     queue.push({
-      obj: obj,
-      parent: i.parent
-     });
-     processed.put(obj);
-    }
-   }
-   return root.items.length > 0 ? root.items : null;
-  },
-  get numPages() {
-   var obj = this.toplevelPagesDict.get('Count');
-   assert(isInt(obj), 'page count in top level pages object is not an integer');
-   return shadow(this, 'num', obj);
-  },
-  get destinations() {
-   function fetchDestination(dest) {
-    return isDict(dest) ? dest.get('D') : dest;
-   }
-   var xref = this.xref;
-   var dests = {}, nameTreeRef, nameDictionaryRef;
-   var obj = this.catDict.get('Names');
-   if (obj && obj.has('Dests')) {
-    nameTreeRef = obj.getRaw('Dests');
-   } else if (this.catDict.has('Dests')) {
-    nameDictionaryRef = this.catDict.get('Dests');
-   }
-   if (nameDictionaryRef) {
-    obj = nameDictionaryRef;
-    obj.forEach(function catalogForEach(key, value) {
-     if (!value) {
-      return;
-     }
-     dests[key] = fetchDestination(value);
-    });
-   }
-   if (nameTreeRef) {
-    var nameTree = new NameTree(nameTreeRef, xref);
-    var names = nameTree.getAll();
-    for (var name in names) {
-     dests[name] = fetchDestination(names[name]);
-    }
-   }
-   return shadow(this, 'destinations', dests);
-  },
-  getDestination: function Catalog_getDestination(destinationId) {
-   function fetchDestination(dest) {
-    return isDict(dest) ? dest.get('D') : dest;
-   }
-   var xref = this.xref;
-   var dest = null, nameTreeRef, nameDictionaryRef;
-   var obj = this.catDict.get('Names');
-   if (obj && obj.has('Dests')) {
-    nameTreeRef = obj.getRaw('Dests');
-   } else if (this.catDict.has('Dests')) {
-    nameDictionaryRef = this.catDict.get('Dests');
-   }
-   if (nameDictionaryRef) {
-    var value = nameDictionaryRef.get(destinationId);
-    if (value) {
-     dest = fetchDestination(value);
-    }
-   }
-   if (nameTreeRef) {
-    var nameTree = new NameTree(nameTreeRef, xref);
-    dest = fetchDestination(nameTree.get(destinationId));
-   }
-   return dest;
-  },
-  get pageLabels() {
-   var obj = null;
-   try {
-    obj = this.readPageLabels();
-   } catch (ex) {
-    if (ex instanceof MissingDataException) {
-     throw ex;
-    }
-    warn('Unable to read page labels.');
-   }
-   return shadow(this, 'pageLabels', obj);
-  },
-  readPageLabels: function Catalog_readPageLabels() {
-   var obj = this.catDict.getRaw('PageLabels');
-   if (!obj) {
-    return null;
-   }
-   var pageLabels = new Array(this.numPages);
-   var style = null;
-   var prefix = '';
-   var numberTree = new NumberTree(obj, this.xref);
-   var nums = numberTree.getAll();
-   var currentLabel = '', currentIndex = 1;
-   for (var i = 0, ii = this.numPages; i < ii; i++) {
-    if (i in nums) {
-     var labelDict = nums[i];
-     assert(isDict(labelDict), 'The PageLabel is not a dictionary.');
-     var type = labelDict.get('Type');
-     assert(!type || isName(type, 'PageLabel'), 'Invalid type in PageLabel dictionary.');
-     var s = labelDict.get('S');
-     assert(!s || isName(s), 'Invalid style in PageLabel dictionary.');
-     style = s ? s.name : null;
-     var p = labelDict.get('P');
-     assert(!p || isString(p), 'Invalid prefix in PageLabel dictionary.');
-     prefix = p ? stringToPDFString(p) : '';
-     var st = labelDict.get('St');
-     assert(!st || isInt(st) && st >= 1, 'Invalid start in PageLabel dictionary.');
-     currentIndex = st || 1;
-    }
-    switch (style) {
-    case 'D':
-     currentLabel = currentIndex;
-     break;
-    case 'R':
-    case 'r':
-     currentLabel = Util.toRoman(currentIndex, style === 'r');
-     break;
-    case 'A':
-    case 'a':
-     var LIMIT = 26;
-     var A_UPPER_CASE = 0x41, A_LOWER_CASE = 0x61;
-     var baseCharCode = style === 'a' ? A_LOWER_CASE : A_UPPER_CASE;
-     var letterIndex = currentIndex - 1;
-     var character = String.fromCharCode(baseCharCode + letterIndex % LIMIT);
-     var charBuf = [];
-     for (var j = 0, jj = letterIndex / LIMIT | 0; j <= jj; j++) {
-      charBuf.push(character);
-     }
-     currentLabel = charBuf.join('');
-     break;
-    default:
-     assert(!style, 'Invalid style "' + style + '" in PageLabel dictionary.');
-    }
-    pageLabels[i] = prefix + currentLabel;
-    currentLabel = '';
-    currentIndex++;
-   }
-   return pageLabels;
-  },
-  get attachments() {
-   var xref = this.xref;
-   var attachments = null, nameTreeRef;
-   var obj = this.catDict.get('Names');
-   if (obj) {
-    nameTreeRef = obj.getRaw('EmbeddedFiles');
-   }
-   if (nameTreeRef) {
-    var nameTree = new NameTree(nameTreeRef, xref);
-    var names = nameTree.getAll();
-    for (var name in names) {
-     var fs = new FileSpec(names[name], xref);
-     if (!attachments) {
-      attachments = Object.create(null);
-     }
-     attachments[stringToPDFString(name)] = fs.serializable;
-    }
-   }
-   return shadow(this, 'attachments', attachments);
-  },
-  get javaScript() {
-   var xref = this.xref;
-   var obj = this.catDict.get('Names');
-   var javaScript = [];
-   function appendIfJavaScriptDict(jsDict) {
-    var type = jsDict.get('S');
-    if (!isName(type, 'JavaScript')) {
-     return;
-    }
-    var js = jsDict.get('JS');
-    if (isStream(js)) {
-     js = bytesToString(js.getBytes());
-    } else if (!isString(js)) {
-     return;
-    }
-    javaScript.push(stringToPDFString(js));
-   }
-   if (obj && obj.has('JavaScript')) {
-    var nameTree = new NameTree(obj.getRaw('JavaScript'), xref);
-    var names = nameTree.getAll();
-    for (var name in names) {
-     var jsDict = names[name];
-     if (isDict(jsDict)) {
-      appendIfJavaScriptDict(jsDict);
-     }
-    }
-   }
-   var openactionDict = this.catDict.get('OpenAction');
-   if (isDict(openactionDict, 'Action')) {
-    var actionType = openactionDict.get('S');
-    if (isName(actionType, 'Named')) {
-     var action = openactionDict.get('N');
-     if (isName(action, 'Print')) {
-      javaScript.push('print({});');
-     }
-    } else {
-     appendIfJavaScriptDict(openactionDict);
-    }
-   }
-   return shadow(this, 'javaScript', javaScript);
-  },
-  cleanup: function Catalog_cleanup() {
-   var promises = [];
-   this.fontCache.forEach(function (promise) {
-    promises.push(promise);
-   });
-   return Promise.all(promises).then(function (translatedFonts) {
-    for (var i = 0, ii = translatedFonts.length; i < ii; i++) {
-     var font = translatedFonts[i].dict;
-     delete font.translated;
-    }
-    this.fontCache.clear();
+  function Catalog(pdfManager, xref, pageFactory) {
+    this.pdfManager = pdfManager;
+    this.xref = xref;
+    this.catDict = xref.getCatalogObj();
+    this.fontCache = new RefSetCache();
     this.builtInCMapCache = Object.create(null);
-   }.bind(this));
-  },
-  getPage: function Catalog_getPage(pageIndex) {
-   if (!(pageIndex in this.pagePromises)) {
-    this.pagePromises[pageIndex] = this.getPageDict(pageIndex).then(function (a) {
-     var dict = a[0];
-     var ref = a[1];
-     return this.pageFactory.createPage(pageIndex, dict, ref, this.fontCache, this.builtInCMapCache);
-    }.bind(this));
-   }
-   return this.pagePromises[pageIndex];
-  },
-  getPageDict: function Catalog_getPageDict(pageIndex) {
-   var capability = createPromiseCapability();
-   var nodesToVisit = [this.catDict.getRaw('Pages')];
-   var currentPageIndex = 0;
-   var xref = this.xref;
-   function next() {
-    while (nodesToVisit.length) {
-     var currentNode = nodesToVisit.pop();
-     if (isRef(currentNode)) {
-      xref.fetchAsync(currentNode).then(function (obj) {
-       if (isDict(obj, 'Page') || isDict(obj) && !obj.has('Kids')) {
-        if (pageIndex === currentPageIndex) {
-         capability.resolve([
-          obj,
-          currentNode
-         ]);
+    assert(isDict(this.catDict), 'catalog object is not a dictionary');
+    this.pageFactory = pageFactory;
+    this.pagePromises = [];
+  }
+  Catalog.prototype = {
+    get metadata() {
+      var streamRef = this.catDict.getRaw('Metadata');
+      if (!isRef(streamRef)) {
+        return shadow(this, 'metadata', null);
+      }
+      var encryptMetadata = !this.xref.encrypt ? false : this.xref.encrypt.encryptMetadata;
+      var stream = this.xref.fetch(streamRef, !encryptMetadata);
+      var metadata;
+      if (stream && isDict(stream.dict)) {
+        var type = stream.dict.get('Type');
+        var subtype = stream.dict.get('Subtype');
+        if (isName(type, 'Metadata') && isName(subtype, 'XML')) {
+          try {
+            metadata = stringToUTF8String(bytesToString(stream.getBytes()));
+          } catch (e) {
+            if (e instanceof MissingDataException) {
+              throw e;
+            }
+            info('Skipping invalid metadata.');
+          }
+        }
+      }
+      return shadow(this, 'metadata', metadata);
+    },
+    get toplevelPagesDict() {
+      var pagesObj = this.catDict.get('Pages');
+      assert(isDict(pagesObj), 'invalid top-level pages dictionary');
+      return shadow(this, 'toplevelPagesDict', pagesObj);
+    },
+    get documentOutline() {
+      var obj = null;
+      try {
+        obj = this.readDocumentOutline();
+      } catch (ex) {
+        if (ex instanceof MissingDataException) {
+          throw ex;
+        }
+        warn('Unable to read document outline');
+      }
+      return shadow(this, 'documentOutline', obj);
+    },
+    readDocumentOutline: function Catalog_readDocumentOutline() {
+      var obj = this.catDict.get('Outlines');
+      if (!isDict(obj)) {
+        return null;
+      }
+      obj = obj.getRaw('First');
+      if (!isRef(obj)) {
+        return null;
+      }
+      var root = { items: [] };
+      var queue = [{
+        obj: obj,
+        parent: root
+      }];
+      var processed = new RefSet();
+      processed.put(obj);
+      var xref = this.xref,
+          blackColor = new Uint8Array(3);
+      while (queue.length > 0) {
+        var i = queue.shift();
+        var outlineDict = xref.fetchIfRef(i.obj);
+        if (outlineDict === null) {
+          continue;
+        }
+        assert(outlineDict.has('Title'), 'Invalid outline item');
+        var data = {
+          url: null,
+          dest: null
+        };
+        Catalog.parseDestDictionary({
+          destDict: outlineDict,
+          resultObj: data,
+          docBaseUrl: this.pdfManager.docBaseUrl
+        });
+        var title = outlineDict.get('Title');
+        var flags = outlineDict.get('F') || 0;
+        var color = outlineDict.getArray('C'),
+            rgbColor = blackColor;
+        if (isArray(color) && color.length === 3 && (color[0] !== 0 || color[1] !== 0 || color[2] !== 0)) {
+          rgbColor = ColorSpace.singletons.rgb.getRgb(color, 0);
+        }
+        var outlineItem = {
+          dest: data.dest,
+          url: data.url,
+          unsafeUrl: data.unsafeUrl,
+          newWindow: data.newWindow,
+          title: stringToPDFString(title),
+          color: rgbColor,
+          count: outlineDict.get('Count'),
+          bold: !!(flags & 2),
+          italic: !!(flags & 1),
+          items: []
+        };
+        i.parent.items.push(outlineItem);
+        obj = outlineDict.getRaw('First');
+        if (isRef(obj) && !processed.has(obj)) {
+          queue.push({
+            obj: obj,
+            parent: outlineItem
+          });
+          processed.put(obj);
+        }
+        obj = outlineDict.getRaw('Next');
+        if (isRef(obj) && !processed.has(obj)) {
+          queue.push({
+            obj: obj,
+            parent: i.parent
+          });
+          processed.put(obj);
+        }
+      }
+      return root.items.length > 0 ? root.items : null;
+    },
+    get numPages() {
+      var obj = this.toplevelPagesDict.get('Count');
+      assert(isInt(obj), 'page count in top level pages object is not an integer');
+      return shadow(this, 'num', obj);
+    },
+    get destinations() {
+      function fetchDestination(dest) {
+        return isDict(dest) ? dest.get('D') : dest;
+      }
+      var xref = this.xref;
+      var dests = {},
+          nameTreeRef,
+          nameDictionaryRef;
+      var obj = this.catDict.get('Names');
+      if (obj && obj.has('Dests')) {
+        nameTreeRef = obj.getRaw('Dests');
+      } else if (this.catDict.has('Dests')) {
+        nameDictionaryRef = this.catDict.get('Dests');
+      }
+      if (nameDictionaryRef) {
+        obj = nameDictionaryRef;
+        obj.forEach(function catalogForEach(key, value) {
+          if (!value) {
+            return;
+          }
+          dests[key] = fetchDestination(value);
+        });
+      }
+      if (nameTreeRef) {
+        var nameTree = new NameTree(nameTreeRef, xref);
+        var names = nameTree.getAll();
+        for (var name in names) {
+          dests[name] = fetchDestination(names[name]);
+        }
+      }
+      return shadow(this, 'destinations', dests);
+    },
+    getDestination: function Catalog_getDestination(destinationId) {
+      function fetchDestination(dest) {
+        return isDict(dest) ? dest.get('D') : dest;
+      }
+      var xref = this.xref;
+      var dest = null,
+          nameTreeRef,
+          nameDictionaryRef;
+      var obj = this.catDict.get('Names');
+      if (obj && obj.has('Dests')) {
+        nameTreeRef = obj.getRaw('Dests');
+      } else if (this.catDict.has('Dests')) {
+        nameDictionaryRef = this.catDict.get('Dests');
+      }
+      if (nameDictionaryRef) {
+        var value = nameDictionaryRef.get(destinationId);
+        if (value) {
+          dest = fetchDestination(value);
+        }
+      }
+      if (nameTreeRef) {
+        var nameTree = new NameTree(nameTreeRef, xref);
+        dest = fetchDestination(nameTree.get(destinationId));
+      }
+      return dest;
+    },
+    get pageLabels() {
+      var obj = null;
+      try {
+        obj = this.readPageLabels();
+      } catch (ex) {
+        if (ex instanceof MissingDataException) {
+          throw ex;
+        }
+        warn('Unable to read page labels.');
+      }
+      return shadow(this, 'pageLabels', obj);
+    },
+    readPageLabels: function Catalog_readPageLabels() {
+      var obj = this.catDict.getRaw('PageLabels');
+      if (!obj) {
+        return null;
+      }
+      var pageLabels = new Array(this.numPages);
+      var style = null;
+      var prefix = '';
+      var numberTree = new NumberTree(obj, this.xref);
+      var nums = numberTree.getAll();
+      var currentLabel = '',
+          currentIndex = 1;
+      for (var i = 0, ii = this.numPages; i < ii; i++) {
+        if (i in nums) {
+          var labelDict = nums[i];
+          assert(isDict(labelDict), 'The PageLabel is not a dictionary.');
+          var type = labelDict.get('Type');
+          assert(!type || isName(type, 'PageLabel'), 'Invalid type in PageLabel dictionary.');
+          var s = labelDict.get('S');
+          assert(!s || isName(s), 'Invalid style in PageLabel dictionary.');
+          style = s ? s.name : null;
+          var p = labelDict.get('P');
+          assert(!p || isString(p), 'Invalid prefix in PageLabel dictionary.');
+          prefix = p ? stringToPDFString(p) : '';
+          var st = labelDict.get('St');
+          assert(!st || isInt(st) && st >= 1, 'Invalid start in PageLabel dictionary.');
+          currentIndex = st || 1;
+        }
+        switch (style) {
+          case 'D':
+            currentLabel = currentIndex;
+            break;
+          case 'R':
+          case 'r':
+            currentLabel = Util.toRoman(currentIndex, style === 'r');
+            break;
+          case 'A':
+          case 'a':
+            var LIMIT = 26;
+            var A_UPPER_CASE = 0x41,
+                A_LOWER_CASE = 0x61;
+            var baseCharCode = style === 'a' ? A_LOWER_CASE : A_UPPER_CASE;
+            var letterIndex = currentIndex - 1;
+            var character = String.fromCharCode(baseCharCode + letterIndex % LIMIT);
+            var charBuf = [];
+            for (var j = 0, jj = letterIndex / LIMIT | 0; j <= jj; j++) {
+              charBuf.push(character);
+            }
+            currentLabel = charBuf.join('');
+            break;
+          default:
+            assert(!style, 'Invalid style "' + style + '" in PageLabel dictionary.');
+        }
+        pageLabels[i] = prefix + currentLabel;
+        currentLabel = '';
+        currentIndex++;
+      }
+      return pageLabels;
+    },
+    get attachments() {
+      var xref = this.xref;
+      var attachments = null,
+          nameTreeRef;
+      var obj = this.catDict.get('Names');
+      if (obj) {
+        nameTreeRef = obj.getRaw('EmbeddedFiles');
+      }
+      if (nameTreeRef) {
+        var nameTree = new NameTree(nameTreeRef, xref);
+        var names = nameTree.getAll();
+        for (var name in names) {
+          var fs = new FileSpec(names[name], xref);
+          if (!attachments) {
+            attachments = Object.create(null);
+          }
+          attachments[stringToPDFString(name)] = fs.serializable;
+        }
+      }
+      return shadow(this, 'attachments', attachments);
+    },
+    get javaScript() {
+      var xref = this.xref;
+      var obj = this.catDict.get('Names');
+      var javaScript = [];
+      function appendIfJavaScriptDict(jsDict) {
+        var type = jsDict.get('S');
+        if (!isName(type, 'JavaScript')) {
+          return;
+        }
+        var js = jsDict.get('JS');
+        if (isStream(js)) {
+          js = bytesToString(js.getBytes());
+        } else if (!isString(js)) {
+          return;
+        }
+        javaScript.push(stringToPDFString(js));
+      }
+      if (obj && obj.has('JavaScript')) {
+        var nameTree = new NameTree(obj.getRaw('JavaScript'), xref);
+        var names = nameTree.getAll();
+        for (var name in names) {
+          var jsDict = names[name];
+          if (isDict(jsDict)) {
+            appendIfJavaScriptDict(jsDict);
+          }
+        }
+      }
+      var openactionDict = this.catDict.get('OpenAction');
+      if (isDict(openactionDict, 'Action')) {
+        var actionType = openactionDict.get('S');
+        if (isName(actionType, 'Named')) {
+          var action = openactionDict.get('N');
+          if (isName(action, 'Print')) {
+            javaScript.push('print({});');
+          }
         } else {
-         currentPageIndex++;
-         next();
+          appendIfJavaScriptDict(openactionDict);
         }
-        return;
-       }
-       nodesToVisit.push(obj);
-       next();
-      }, capability.reject);
-      return;
-     }
-     assert(isDict(currentNode), 'page dictionary kid reference points to wrong type of object');
-     var count = currentNode.get('Count');
-     if (currentPageIndex + count <= pageIndex) {
-      currentPageIndex += count;
-      continue;
-     }
-     var kids = currentNode.get('Kids');
-     assert(isArray(kids), 'page dictionary kids object is not an array');
-     for (var last = kids.length - 1; last >= 0; last--) {
-      nodesToVisit.push(kids[last]);
-     }
+      }
+      return shadow(this, 'javaScript', javaScript);
+    },
+    cleanup: function Catalog_cleanup() {
+      var promises = [];
+      this.fontCache.forEach(function (promise) {
+        promises.push(promise);
+      });
+      return Promise.all(promises).then(function (translatedFonts) {
+        for (var i = 0, ii = translatedFonts.length; i < ii; i++) {
+          var font = translatedFonts[i].dict;
+          delete font.translated;
+        }
+        this.fontCache.clear();
+        this.builtInCMapCache = Object.create(null);
+      }.bind(this));
+    },
+    getPage: function Catalog_getPage(pageIndex) {
+      if (!(pageIndex in this.pagePromises)) {
+        this.pagePromises[pageIndex] = this.getPageDict(pageIndex).then(function (a) {
+          var dict = a[0];
+          var ref = a[1];
+          return this.pageFactory.createPage(pageIndex, dict, ref, this.fontCache, this.builtInCMapCache);
+        }.bind(this));
+      }
+      return this.pagePromises[pageIndex];
+    },
+    getPageDict: function Catalog_getPageDict(pageIndex) {
+      var capability = createPromiseCapability();
+      var nodesToVisit = [this.catDict.getRaw('Pages')];
+      var currentPageIndex = 0;
+      var xref = this.xref;
+      function next() {
+        while (nodesToVisit.length) {
+          var currentNode = nodesToVisit.pop();
+          if (isRef(currentNode)) {
+            xref.fetchAsync(currentNode).then(function (obj) {
+              if (isDict(obj, 'Page') || isDict(obj) && !obj.has('Kids')) {
+                if (pageIndex === currentPageIndex) {
+                  capability.resolve([obj, currentNode]);
+                } else {
+                  currentPageIndex++;
+                  next();
+                }
+                return;
+              }
+              nodesToVisit.push(obj);
+              next();
+            }, capability.reject);
+            return;
+          }
+          assert(isDict(currentNode), 'page dictionary kid reference points to wrong type of object');
+          var count = currentNode.get('Count');
+          if (currentPageIndex + count <= pageIndex) {
+            currentPageIndex += count;
+            continue;
+          }
+          var kids = currentNode.get('Kids');
+          assert(isArray(kids), 'page dictionary kids object is not an array');
+          for (var last = kids.length - 1; last >= 0; last--) {
+            nodesToVisit.push(kids[last]);
+          }
+        }
+        capability.reject('Page index ' + pageIndex + ' not found.');
+      }
+      next();
+      return capability.promise;
+    },
+    getPageIndex: function Catalog_getPageIndex(pageRef) {
+      var xref = this.xref;
+      function pagesBeforeRef(kidRef) {
+        var total = 0;
+        var parentRef;
+        return xref.fetchAsync(kidRef).then(function (node) {
+          if (isRefsEqual(kidRef, pageRef) && !isDict(node, 'Page') && !(isDict(node) && !node.has('Type') && node.has('Contents'))) {
+            throw new Error('The reference does not point to a /Page Dict.');
+          }
+          if (!node) {
+            return null;
+          }
+          assert(isDict(node), 'node must be a Dict.');
+          parentRef = node.getRaw('Parent');
+          return node.getAsync('Parent');
+        }).then(function (parent) {
+          if (!parent) {
+            return null;
+          }
+          assert(isDict(parent), 'parent must be a Dict.');
+          return parent.getAsync('Kids');
+        }).then(function (kids) {
+          if (!kids) {
+            return null;
+          }
+          var kidPromises = [];
+          var found = false;
+          for (var i = 0; i < kids.length; i++) {
+            var kid = kids[i];
+            assert(isRef(kid), 'kid must be a Ref.');
+            if (kid.num === kidRef.num) {
+              found = true;
+              break;
+            }
+            kidPromises.push(xref.fetchAsync(kid).then(function (kid) {
+              if (kid.has('Count')) {
+                var count = kid.get('Count');
+                total += count;
+              } else {
+                total++;
+              }
+            }));
+          }
+          if (!found) {
+            error('kid ref not found in parents kids');
+          }
+          return Promise.all(kidPromises).then(function () {
+            return [total, parentRef];
+          });
+        });
+      }
+      var total = 0;
+      function next(ref) {
+        return pagesBeforeRef(ref).then(function (args) {
+          if (!args) {
+            return total;
+          }
+          var count = args[0];
+          var parentRef = args[1];
+          total += count;
+          return next(parentRef);
+        });
+      }
+      return next(pageRef);
     }
-    capability.reject('Page index ' + pageIndex + ' not found.');
-   }
-   next();
-   return capability.promise;
-  },
-  getPageIndex: function Catalog_getPageIndex(pageRef) {
-   var xref = this.xref;
-   function pagesBeforeRef(kidRef) {
-    var total = 0;
-    var parentRef;
-    return xref.fetchAsync(kidRef).then(function (node) {
-     if (isRefsEqual(kidRef, pageRef) && !isDict(node, 'Page') && !(isDict(node) && !node.has('Type') && node.has('Contents'))) {
-      throw new Error('The reference does not point to a /Page Dict.');
-     }
-     if (!node) {
-      return null;
-     }
-     assert(isDict(node), 'node must be a Dict.');
-     parentRef = node.getRaw('Parent');
-     return node.getAsync('Parent');
-    }).then(function (parent) {
-     if (!parent) {
-      return null;
-     }
-     assert(isDict(parent), 'parent must be a Dict.');
-     return parent.getAsync('Kids');
-    }).then(function (kids) {
-     if (!kids) {
-      return null;
-     }
-     var kidPromises = [];
-     var found = false;
-     for (var i = 0; i < kids.length; i++) {
-      var kid = kids[i];
-      assert(isRef(kid), 'kid must be a Ref.');
-      if (kid.num === kidRef.num) {
-       found = true;
-       break;
-      }
-      kidPromises.push(xref.fetchAsync(kid).then(function (kid) {
-       if (kid.has('Count')) {
-        var count = kid.get('Count');
-        total += count;
-       } else {
-        total++;
-       }
-      }));
-     }
-     if (!found) {
-      error('kid ref not found in parents kids');
-     }
-     return Promise.all(kidPromises).then(function () {
-      return [
-       total,
-       parentRef
-      ];
-     });
-    });
-   }
-   var total = 0;
-   function next(ref) {
-    return pagesBeforeRef(ref).then(function (args) {
-     if (!args) {
-      return total;
-     }
-     var count = args[0];
-     var parentRef = args[1];
-     total += count;
-     return next(parentRef);
-    });
-   }
-   return next(pageRef);
-  }
- };
- Catalog.parseDestDictionary = function Catalog_parseDestDictionary(params) {
-  function addDefaultProtocolToUrl(url) {
-   if (url.indexOf('www.') === 0) {
-    return 'http://' + url;
-   }
-   return url;
-  }
-  function tryConvertUrlEncoding(url) {
-   try {
-    return stringToUTF8String(url);
-   } catch (e) {
-    return url;
-   }
-  }
-  var destDict = params.destDict;
-  if (!isDict(destDict)) {
-   warn('Catalog_parseDestDictionary: "destDict" must be a dictionary.');
-   return;
-  }
-  var resultObj = params.resultObj;
-  if (typeof resultObj !== 'object') {
-   warn('Catalog_parseDestDictionary: "resultObj" must be an object.');
-   return;
-  }
-  var docBaseUrl = params.docBaseUrl || null;
-  var action = destDict.get('A'), url, dest;
-  if (isDict(action)) {
-   var linkType = action.get('S').name;
-   switch (linkType) {
-   case 'URI':
-    url = action.get('URI');
-    if (isName(url)) {
-     url = '/' + url.name;
-    } else if (isString(url)) {
-     url = addDefaultProtocolToUrl(url);
+  };
+  Catalog.parseDestDictionary = function Catalog_parseDestDictionary(params) {
+    function addDefaultProtocolToUrl(url) {
+      if (url.indexOf('www.') === 0) {
+        return 'http://' + url;
+      }
+      return url;
     }
-    break;
-   case 'GoTo':
-    dest = action.get('D');
-    break;
-   case 'Launch':
-   case 'GoToR':
-    var urlDict = action.get('F');
-    if (isDict(urlDict)) {
-     url = urlDict.get('F') || null;
-    } else if (isString(urlDict)) {
-     url = urlDict;
+    function tryConvertUrlEncoding(url) {
+      try {
+        return stringToUTF8String(url);
+      } catch (e) {
+        return url;
+      }
     }
-    var remoteDest = action.get('D');
-    if (remoteDest) {
-     if (isName(remoteDest)) {
-      remoteDest = remoteDest.name;
-     }
-     if (isString(url)) {
-      var baseUrl = url.split('#')[0];
-      if (isString(remoteDest)) {
-       url = baseUrl + '#' + (/^\d+$/.test(remoteDest) ? 'nameddest=' : '') + remoteDest;
-      } else if (isArray(remoteDest)) {
-       url = baseUrl + '#' + JSON.stringify(remoteDest);
-      }
-     }
+    var destDict = params.destDict;
+    if (!isDict(destDict)) {
+      warn('Catalog_parseDestDictionary: "destDict" must be a dictionary.');
+      return;
     }
-    var newWindow = action.get('NewWindow');
-    if (isBool(newWindow)) {
-     resultObj.newWindow = newWindow;
+    var resultObj = params.resultObj;
+    if (typeof resultObj !== 'object') {
+      warn('Catalog_parseDestDictionary: "resultObj" must be an object.');
+      return;
     }
-    break;
-   case 'Named':
-    var namedAction = action.get('N');
-    if (isName(namedAction)) {
-     resultObj.action = namedAction.name;
+    var docBaseUrl = params.docBaseUrl || null;
+    var action = destDict.get('A'),
+        url,
+        dest;
+    if (isDict(action)) {
+      var linkType = action.get('S').name;
+      switch (linkType) {
+        case 'URI':
+          url = action.get('URI');
+          if (isName(url)) {
+            url = '/' + url.name;
+          } else if (isString(url)) {
+            url = addDefaultProtocolToUrl(url);
+          }
+          break;
+        case 'GoTo':
+          dest = action.get('D');
+          break;
+        case 'Launch':
+        case 'GoToR':
+          var urlDict = action.get('F');
+          if (isDict(urlDict)) {
+            url = urlDict.get('F') || null;
+          } else if (isString(urlDict)) {
+            url = urlDict;
+          }
+          var remoteDest = action.get('D');
+          if (remoteDest) {
+            if (isName(remoteDest)) {
+              remoteDest = remoteDest.name;
+            }
+            if (isString(url)) {
+              var baseUrl = url.split('#')[0];
+              if (isString(remoteDest)) {
+                url = baseUrl + '#' + (/^\d+$/.test(remoteDest) ? 'nameddest=' : '') + remoteDest;
+              } else if (isArray(remoteDest)) {
+                url = baseUrl + '#' + JSON.stringify(remoteDest);
+              }
+            }
+          }
+          var newWindow = action.get('NewWindow');
+          if (isBool(newWindow)) {
+            resultObj.newWindow = newWindow;
+          }
+          break;
+        case 'Named':
+          var namedAction = action.get('N');
+          if (isName(namedAction)) {
+            resultObj.action = namedAction.name;
+          }
+          break;
+        case 'JavaScript':
+          var jsAction = action.get('JS'),
+              js;
+          if (isStream(jsAction)) {
+            js = bytesToString(jsAction.getBytes());
+          } else if (isString(jsAction)) {
+            js = jsAction;
+          }
+          if (js) {
+            var URL_OPEN_METHODS = ['app.launchURL', 'window.open'];
+            var regex = new RegExp('^\\s*(' + URL_OPEN_METHODS.join('|').split('.').join('\\.') + ')\\((?:\'|\")([^\'\"]*)(?:\'|\")(?:,\\s*(\\w+)\\)|\\))', 'i');
+            var jsUrl = regex.exec(stringToPDFString(js));
+            if (jsUrl && jsUrl[2]) {
+              url = jsUrl[2];
+              if (jsUrl[3] === 'true' && jsUrl[1] === 'app.launchURL') {
+                resultObj.newWindow = true;
+              }
+              break;
+            }
+          }
+        default:
+          warn('Catalog_parseDestDictionary: Unrecognized link type "' + linkType + '".');
+          break;
+      }
+    } else if (destDict.has('Dest')) {
+      dest = destDict.get('Dest');
     }
-    break;
-   case 'JavaScript':
-    var jsAction = action.get('JS'), js;
-    if (isStream(jsAction)) {
-     js = bytesToString(jsAction.getBytes());
-    } else if (isString(jsAction)) {
-     js = jsAction;
+    if (isString(url)) {
+      url = tryConvertUrlEncoding(url);
+      var absoluteUrl = createValidAbsoluteUrl(url, docBaseUrl);
+      if (absoluteUrl) {
+        resultObj.url = absoluteUrl.href;
+      }
+      resultObj.unsafeUrl = url;
     }
-    if (js) {
-     var URL_OPEN_METHODS = [
-      'app.launchURL',
-      'window.open'
-     ];
-     var regex = new RegExp('^\\s*(' + URL_OPEN_METHODS.join('|').split('.').join('\\.') + ')\\((?:\'|\")([^\'\"]*)(?:\'|\")(?:,\\s*(\\w+)\\)|\\))', 'i');
-     var jsUrl = regex.exec(stringToPDFString(js));
-     if (jsUrl && jsUrl[2]) {
-      url = jsUrl[2];
-      if (jsUrl[3] === 'true' && jsUrl[1] === 'app.launchURL') {
-       resultObj.newWindow = true;
-      }
-      break;
-     }
+    if (dest) {
+      if (isName(dest)) {
+        dest = dest.name;
+      }
+      if (isString(dest) || isArray(dest)) {
+        resultObj.dest = dest;
+      }
     }
-   default:
-    warn('Catalog_parseDestDictionary: Unrecognized link type "' + linkType + '".');
-    break;
-   }
-  } else if (destDict.has('Dest')) {
-   dest = destDict.get('Dest');
-  }
-  if (isString(url)) {
-   url = tryConvertUrlEncoding(url);
-   var absoluteUrl = createValidAbsoluteUrl(url, docBaseUrl);
-   if (absoluteUrl) {
-    resultObj.url = absoluteUrl.href;
-   }
-   resultObj.unsafeUrl = url;
-  }
-  if (dest) {
-   if (isName(dest)) {
-    dest = dest.name;
-   }
-   if (isString(dest) || isArray(dest)) {
-    resultObj.dest = dest;
-   }
-  }
- };
- return Catalog;
+  };
+  return Catalog;
 }();
 var XRef = function XRefClosure() {
- function XRef(stream, pdfManager) {
-  this.stream = stream;
-  this.pdfManager = pdfManager;
-  this.entries = [];
-  this.xrefstms = Object.create(null);
-  this.cache = [];
-  this.stats = {
-   streamTypes: [],
-   fontTypes: []
-  };
- }
- XRef.prototype = {
-  setStartXRef: function XRef_setStartXRef(startXRef) {
-   this.startXRefQueue = [startXRef];
-  },
-  parse: function XRef_parse(recoveryMode) {
-   var trailerDict;
-   if (!recoveryMode) {
-    trailerDict = this.readXRef();
-   } else {
-    warn('Indexing all PDF objects');
-    trailerDict = this.indexObjects();
-   }
-   trailerDict.assignXref(this);
-   this.trailer = trailerDict;
-   var encrypt = trailerDict.get('Encrypt');
-   if (isDict(encrypt)) {
-    var ids = trailerDict.get('ID');
-    var fileId = ids && ids.length ? ids[0] : '';
-    encrypt.suppressEncryption = true;
-    this.encrypt = new CipherTransformFactory(encrypt, fileId, this.pdfManager.password);
-   }
-   if (!(this.root = trailerDict.get('Root'))) {
-    error('Invalid root reference');
-   }
-  },
-  processXRefTable: function XRef_processXRefTable(parser) {
-   if (!('tableState' in this)) {
-    this.tableState = {
-     entryNum: 0,
-     streamPos: parser.lexer.stream.pos,
-     parserBuf1: parser.buf1,
-     parserBuf2: parser.buf2
+  function XRef(stream, pdfManager) {
+    this.stream = stream;
+    this.pdfManager = pdfManager;
+    this.entries = [];
+    this.xrefstms = Object.create(null);
+    this.cache = [];
+    this.stats = {
+      streamTypes: [],
+      fontTypes: []
     };
-   }
-   var obj = this.readXRefTable(parser);
-   if (!isCmd(obj, 'trailer')) {
-    error('Invalid XRef table: could not find trailer dictionary');
-   }
-   var dict = parser.getObj();
-   if (!isDict(dict) && dict.dict) {
-    dict = dict.dict;
-   }
-   if (!isDict(dict)) {
-    error('Invalid XRef table: could not parse trailer dictionary');
-   }
-   delete this.tableState;
-   return dict;
-  },
-  readXRefTable: function XRef_readXRefTable(parser) {
-   var stream = parser.lexer.stream;
-   var tableState = this.tableState;
-   stream.pos = tableState.streamPos;
-   parser.buf1 = tableState.parserBuf1;
-   parser.buf2 = tableState.parserBuf2;
-   var obj;
-   while (true) {
-    if (!('firstEntryNum' in tableState) || !('entryCount' in tableState)) {
-     if (isCmd(obj = parser.getObj(), 'trailer')) {
-      break;
-     }
-     tableState.firstEntryNum = obj;
-     tableState.entryCount = parser.getObj();
-    }
-    var first = tableState.firstEntryNum;
-    var count = tableState.entryCount;
-    if (!isInt(first) || !isInt(count)) {
-     error('Invalid XRef table: wrong types in subsection header');
-    }
-    for (var i = tableState.entryNum; i < count; i++) {
-     tableState.streamPos = stream.pos;
-     tableState.entryNum = i;
-     tableState.parserBuf1 = parser.buf1;
-     tableState.parserBuf2 = parser.buf2;
-     var entry = {};
-     entry.offset = parser.getObj();
-     entry.gen = parser.getObj();
-     var type = parser.getObj();
-     if (isCmd(type, 'f')) {
-      entry.free = true;
-     } else if (isCmd(type, 'n')) {
-      entry.uncompressed = true;
-     }
-     if (!isInt(entry.offset) || !isInt(entry.gen) || !(entry.free || entry.uncompressed)) {
-      error('Invalid entry in XRef subsection: ' + first + ', ' + count);
-     }
-     if (i === 0 && entry.free && first === 1) {
-      first = 0;
-     }
-     if (!this.entries[i + first]) {
-      this.entries[i + first] = entry;
-     }
-    }
-    tableState.entryNum = 0;
-    tableState.streamPos = stream.pos;
-    tableState.parserBuf1 = parser.buf1;
-    tableState.parserBuf2 = parser.buf2;
-    delete tableState.firstEntryNum;
-    delete tableState.entryCount;
-   }
-   if (this.entries[0] && !this.entries[0].free) {
-    error('Invalid XRef table: unexpected first object');
-   }
-   return obj;
-  },
-  processXRefStream: function XRef_processXRefStream(stream) {
-   if (!('streamState' in this)) {
-    var streamParameters = stream.dict;
-    var byteWidths = streamParameters.get('W');
-    var range = streamParameters.get('Index');
-    if (!range) {
-     range = [
-      0,
-      streamParameters.get('Size')
-     ];
-    }
-    this.streamState = {
-     entryRanges: range,
-     byteWidths: byteWidths,
-     entryNum: 0,
-     streamPos: stream.pos
-    };
-   }
-   this.readXRefStream(stream);
-   delete this.streamState;
-   return stream.dict;
-  },
-  readXRefStream: function XRef_readXRefStream(stream) {
-   var i, j;
-   var streamState = this.streamState;
-   stream.pos = streamState.streamPos;
-   var byteWidths = streamState.byteWidths;
-   var typeFieldWidth = byteWidths[0];
-   var offsetFieldWidth = byteWidths[1];
-   var generationFieldWidth = byteWidths[2];
-   var entryRanges = streamState.entryRanges;
-   while (entryRanges.length > 0) {
-    var first = entryRanges[0];
-    var n = entryRanges[1];
-    if (!isInt(first) || !isInt(n)) {
-     error('Invalid XRef range fields: ' + first + ', ' + n);
-    }
-    if (!isInt(typeFieldWidth) || !isInt(offsetFieldWidth) || !isInt(generationFieldWidth)) {
-     error('Invalid XRef entry fields length: ' + first + ', ' + n);
-    }
-    for (i = streamState.entryNum; i < n; ++i) {
-     streamState.entryNum = i;
-     streamState.streamPos = stream.pos;
-     var type = 0, offset = 0, generation = 0;
-     for (j = 0; j < typeFieldWidth; ++j) {
-      type = type << 8 | stream.getByte();
-     }
-     if (typeFieldWidth === 0) {
-      type = 1;
-     }
-     for (j = 0; j < offsetFieldWidth; ++j) {
-      offset = offset << 8 | stream.getByte();
-     }
-     for (j = 0; j < generationFieldWidth; ++j) {
-      generation = generation << 8 | stream.getByte();
-     }
-     var entry = {};
-     entry.offset = offset;
-     entry.gen = generation;
-     switch (type) {
-     case 0:
-      entry.free = true;
-      break;
-     case 1:
-      entry.uncompressed = true;
-      break;
-     case 2:
-      break;
-     default:
-      error('Invalid XRef entry type: ' + type);
-     }
-     if (!this.entries[first + i]) {
-      this.entries[first + i] = entry;
-     }
-    }
-    streamState.entryNum = 0;
-    streamState.streamPos = stream.pos;
-    entryRanges.splice(0, 2);
-   }
-  },
-  indexObjects: function XRef_indexObjects() {
-   var TAB = 0x9, LF = 0xA, CR = 0xD, SPACE = 0x20;
-   var PERCENT = 0x25, LT = 0x3C;
-   function readToken(data, offset) {
-    var token = '', ch = data[offset];
-    while (ch !== LF && ch !== CR && ch !== LT) {
-     if (++offset >= data.length) {
-      break;
-     }
-     token += String.fromCharCode(ch);
-     ch = data[offset];
-    }
-    return token;
-   }
-   function skipUntil(data, offset, what) {
-    var length = what.length, dataLength = data.length;
-    var skipped = 0;
-    while (offset < dataLength) {
-     var i = 0;
-     while (i < length && data[offset + i] === what[i]) {
-      ++i;
-     }
-     if (i >= length) {
-      break;
-     }
-     offset++;
-     skipped++;
-    }
-    return skipped;
-   }
-   var objRegExp = /^(\d+)\s+(\d+)\s+obj\b/;
-   var trailerBytes = new Uint8Array([
-    116,
-    114,
-    97,
-    105,
-    108,
-    101,
-    114
-   ]);
-   var startxrefBytes = new Uint8Array([
-    115,
-    116,
-    97,
-    114,
-    116,
-    120,
-    114,
-    101,
-    102
-   ]);
-   var endobjBytes = new Uint8Array([
-    101,
-    110,
-    100,
-    111,
-    98,
-    106
-   ]);
-   var xrefBytes = new Uint8Array([
-    47,
-    88,
-    82,
-    101,
-    102
-   ]);
-   this.entries.length = 0;
-   var stream = this.stream;
-   stream.pos = 0;
-   var buffer = stream.getBytes();
-   var position = stream.start, length = buffer.length;
-   var trailers = [], xrefStms = [];
-   while (position < length) {
-    var ch = buffer[position];
-    if (ch === TAB || ch === LF || ch === CR || ch === SPACE) {
-     ++position;
-     continue;
-    }
-    if (ch === PERCENT) {
-     do {
-      ++position;
-      if (position >= length) {
-       break;
-      }
-      ch = buffer[position];
-     } while (ch !== LF && ch !== CR);
-     continue;
-    }
-    var token = readToken(buffer, position);
-    var m;
-    if (token.indexOf('xref') === 0 && (token.length === 4 || /\s/.test(token[4]))) {
-     position += skipUntil(buffer, position, trailerBytes);
-     trailers.push(position);
-     position += skipUntil(buffer, position, startxrefBytes);
-    } else if (m = objRegExp.exec(token)) {
-     if (typeof this.entries[m[1]] === 'undefined') {
-      this.entries[m[1]] = {
-       offset: position - stream.start,
-       gen: m[2] | 0,
-       uncompressed: true
-      };
-     }
-     var contentLength = skipUntil(buffer, position, endobjBytes) + 7;
-     var content = buffer.subarray(position, position + contentLength);
-     var xrefTagOffset = skipUntil(content, 0, xrefBytes);
-     if (xrefTagOffset < contentLength && content[xrefTagOffset + 5] < 64) {
-      xrefStms.push(position - stream.start);
-      this.xrefstms[position - stream.start] = 1;
-     }
-     position += contentLength;
-    } else if (token.indexOf('trailer') === 0 && (token.length === 7 || /\s/.test(token[7]))) {
-     trailers.push(position);
-     position += skipUntil(buffer, position, startxrefBytes);
-    } else {
-     position += token.length + 1;
-    }
-   }
-   var i, ii;
-   for (i = 0, ii = xrefStms.length; i < ii; ++i) {
-    this.startXRefQueue.push(xrefStms[i]);
-    this.readXRef(true);
-   }
-   var dict;
-   for (i = 0, ii = trailers.length; i < ii; ++i) {
-    stream.pos = trailers[i];
-    var parser = new Parser(new Lexer(stream), true, this, true);
-    var obj = parser.getObj();
-    if (!isCmd(obj, 'trailer')) {
-     continue;
-    }
-    dict = parser.getObj();
-    if (!isDict(dict)) {
-     continue;
-    }
-    if (dict.has('ID')) {
-     return dict;
-    }
-   }
-   if (dict) {
-    return dict;
-   }
-   throw new InvalidPDFException('Invalid PDF structure');
-  },
-  readXRef: function XRef_readXRef(recoveryMode) {
-   var stream = this.stream;
-   try {
-    while (this.startXRefQueue.length) {
-     var startXRef = this.startXRefQueue[0];
-     stream.pos = startXRef + stream.start;
-     var parser = new Parser(new Lexer(stream), true, this);
-     var obj = parser.getObj();
-     var dict;
-     if (isCmd(obj, 'xref')) {
-      dict = this.processXRefTable(parser);
-      if (!this.topDict) {
-       this.topDict = dict;
-      }
-      obj = dict.get('XRefStm');
-      if (isInt(obj)) {
-       var pos = obj;
-       if (!(pos in this.xrefstms)) {
-        this.xrefstms[pos] = 1;
-        this.startXRefQueue.push(pos);
-       }
-      }
-     } else if (isInt(obj)) {
-      if (!isInt(parser.getObj()) || !isCmd(parser.getObj(), 'obj') || !isStream(obj = parser.getObj())) {
-       error('Invalid XRef stream');
-      }
-      dict = this.processXRefStream(obj);
-      if (!this.topDict) {
-       this.topDict = dict;
-      }
-      if (!dict) {
-       error('Failed to read XRef stream');
-      }
-     } else {
-      error('Invalid XRef stream header');
-     }
-     obj = dict.get('Prev');
-     if (isInt(obj)) {
-      this.startXRefQueue.push(obj);
-     } else if (isRef(obj)) {
-      this.startXRefQueue.push(obj.num);
-     }
-     this.startXRefQueue.shift();
-    }
-    return this.topDict;
-   } catch (e) {
-    if (e instanceof MissingDataException) {
-     throw e;
-    }
-    info('(while reading XRef): ' + e);
-   }
-   if (recoveryMode) {
-    return;
-   }
-   throw new XRefParseException();
-  },
-  getEntry: function XRef_getEntry(i) {
-   var xrefEntry = this.entries[i];
-   if (xrefEntry && !xrefEntry.free && xrefEntry.offset) {
-    return xrefEntry;
-   }
-   return null;
-  },
-  fetchIfRef: function XRef_fetchIfRef(obj, suppressEncryption) {
-   if (!isRef(obj)) {
-    return obj;
-   }
-   return this.fetch(obj, suppressEncryption);
-  },
-  fetch: function XRef_fetch(ref, suppressEncryption) {
-   assert(isRef(ref), 'ref object is not a reference');
-   var num = ref.num;
-   if (num in this.cache) {
-    var cacheEntry = this.cache[num];
-    if (isDict(cacheEntry) && !cacheEntry.objId) {
-     cacheEntry.objId = ref.toString();
-    }
-    return cacheEntry;
-   }
-   var xrefEntry = this.getEntry(num);
-   if (xrefEntry === null) {
-    return this.cache[num] = null;
-   }
-   if (xrefEntry.uncompressed) {
-    xrefEntry = this.fetchUncompressed(ref, xrefEntry, suppressEncryption);
-   } else {
-    xrefEntry = this.fetchCompressed(xrefEntry, suppressEncryption);
-   }
-   if (isDict(xrefEntry)) {
-    xrefEntry.objId = ref.toString();
-   } else if (isStream(xrefEntry)) {
-    xrefEntry.dict.objId = ref.toString();
-   }
-   return xrefEntry;
-  },
-  fetchUncompressed: function XRef_fetchUncompressed(ref, xrefEntry, suppressEncryption) {
-   var gen = ref.gen;
-   var num = ref.num;
-   if (xrefEntry.gen !== gen) {
-    error('inconsistent generation in XRef');
-   }
-   var stream = this.stream.makeSubStream(xrefEntry.offset + this.stream.start);
-   var parser = new Parser(new Lexer(stream), true, this);
-   var obj1 = parser.getObj();
-   var obj2 = parser.getObj();
-   var obj3 = parser.getObj();
-   if (!isInt(obj1) || parseInt(obj1, 10) !== num || !isInt(obj2) || parseInt(obj2, 10) !== gen || !isCmd(obj3)) {
-    error('bad XRef entry');
-   }
-   if (!isCmd(obj3, 'obj')) {
-    if (obj3.cmd.indexOf('obj') === 0) {
-     num = parseInt(obj3.cmd.substring(3), 10);
-     if (!isNaN(num)) {
-      return num;
-     }
-    }
-    error('bad XRef entry');
-   }
-   if (this.encrypt && !suppressEncryption) {
-    xrefEntry = parser.getObj(this.encrypt.createCipherTransform(num, gen));
-   } else {
-    xrefEntry = parser.getObj();
-   }
-   if (!isStream(xrefEntry)) {
-    this.cache[num] = xrefEntry;
-   }
-   return xrefEntry;
-  },
-  fetchCompressed: function XRef_fetchCompressed(xrefEntry, suppressEncryption) {
-   var tableOffset = xrefEntry.offset;
-   var stream = this.fetch(new Ref(tableOffset, 0));
-   if (!isStream(stream)) {
-    error('bad ObjStm stream');
-   }
-   var first = stream.dict.get('First');
-   var n = stream.dict.get('N');
-   if (!isInt(first) || !isInt(n)) {
-    error('invalid first and n parameters for ObjStm stream');
-   }
-   var parser = new Parser(new Lexer(stream), false, this);
-   parser.allowStreams = true;
-   var i, entries = [], num, nums = [];
-   for (i = 0; i < n; ++i) {
-    num = parser.getObj();
-    if (!isInt(num)) {
-     error('invalid object number in the ObjStm stream: ' + num);
-    }
-    nums.push(num);
-    var offset = parser.getObj();
-    if (!isInt(offset)) {
-     error('invalid object offset in the ObjStm stream: ' + offset);
-    }
-   }
-   for (i = 0; i < n; ++i) {
-    entries.push(parser.getObj());
-    if (isCmd(parser.buf1, 'endobj')) {
-     parser.shift();
-    }
-    num = nums[i];
-    var entry = this.entries[num];
-    if (entry && entry.offset === tableOffset && entry.gen === i) {
-     this.cache[num] = entries[i];
-    }
-   }
-   xrefEntry = entries[xrefEntry.gen];
-   if (xrefEntry === undefined) {
-    error('bad XRef entry for compressed object');
-   }
-   return xrefEntry;
-  },
-  fetchIfRefAsync: function XRef_fetchIfRefAsync(obj, suppressEncryption) {
-   if (!isRef(obj)) {
-    return Promise.resolve(obj);
-   }
-   return this.fetchAsync(obj, suppressEncryption);
-  },
-  fetchAsync: function XRef_fetchAsync(ref, suppressEncryption) {
-   var streamManager = this.stream.manager;
-   var xref = this;
-   return new Promise(function tryFetch(resolve, reject) {
-    try {
-     resolve(xref.fetch(ref, suppressEncryption));
-    } catch (e) {
-     if (e instanceof MissingDataException) {
-      streamManager.requestRange(e.begin, e.end).then(function () {
-       tryFetch(resolve, reject);
-      }, reject);
-      return;
-     }
-     reject(e);
-    }
-   });
-  },
-  getCatalogObj: function XRef_getCatalogObj() {
-   return this.root;
   }
- };
- return XRef;
+  XRef.prototype = {
+    setStartXRef: function XRef_setStartXRef(startXRef) {
+      this.startXRefQueue = [startXRef];
+    },
+    parse: function XRef_parse(recoveryMode) {
+      var trailerDict;
+      if (!recoveryMode) {
+        trailerDict = this.readXRef();
+      } else {
+        warn('Indexing all PDF objects');
+        trailerDict = this.indexObjects();
+      }
+      trailerDict.assignXref(this);
+      this.trailer = trailerDict;
+      var encrypt = trailerDict.get('Encrypt');
+      if (isDict(encrypt)) {
+        var ids = trailerDict.get('ID');
+        var fileId = ids && ids.length ? ids[0] : '';
+        encrypt.suppressEncryption = true;
+        this.encrypt = new CipherTransformFactory(encrypt, fileId, this.pdfManager.password);
+      }
+      if (!(this.root = trailerDict.get('Root'))) {
+        error('Invalid root reference');
+      }
+    },
+    processXRefTable: function XRef_processXRefTable(parser) {
+      if (!('tableState' in this)) {
+        this.tableState = {
+          entryNum: 0,
+          streamPos: parser.lexer.stream.pos,
+          parserBuf1: parser.buf1,
+          parserBuf2: parser.buf2
+        };
+      }
+      var obj = this.readXRefTable(parser);
+      if (!isCmd(obj, 'trailer')) {
+        error('Invalid XRef table: could not find trailer dictionary');
+      }
+      var dict = parser.getObj();
+      if (!isDict(dict) && dict.dict) {
+        dict = dict.dict;
+      }
+      if (!isDict(dict)) {
+        error('Invalid XRef table: could not parse trailer dictionary');
+      }
+      delete this.tableState;
+      return dict;
+    },
+    readXRefTable: function XRef_readXRefTable(parser) {
+      var stream = parser.lexer.stream;
+      var tableState = this.tableState;
+      stream.pos = tableState.streamPos;
+      parser.buf1 = tableState.parserBuf1;
+      parser.buf2 = tableState.parserBuf2;
+      var obj;
+      while (true) {
+        if (!('firstEntryNum' in tableState) || !('entryCount' in tableState)) {
+          if (isCmd(obj = parser.getObj(), 'trailer')) {
+            break;
+          }
+          tableState.firstEntryNum = obj;
+          tableState.entryCount = parser.getObj();
+        }
+        var first = tableState.firstEntryNum;
+        var count = tableState.entryCount;
+        if (!isInt(first) || !isInt(count)) {
+          error('Invalid XRef table: wrong types in subsection header');
+        }
+        for (var i = tableState.entryNum; i < count; i++) {
+          tableState.streamPos = stream.pos;
+          tableState.entryNum = i;
+          tableState.parserBuf1 = parser.buf1;
+          tableState.parserBuf2 = parser.buf2;
+          var entry = {};
+          entry.offset = parser.getObj();
+          entry.gen = parser.getObj();
+          var type = parser.getObj();
+          if (isCmd(type, 'f')) {
+            entry.free = true;
+          } else if (isCmd(type, 'n')) {
+            entry.uncompressed = true;
+          }
+          if (!isInt(entry.offset) || !isInt(entry.gen) || !(entry.free || entry.uncompressed)) {
+            error('Invalid entry in XRef subsection: ' + first + ', ' + count);
+          }
+          if (i === 0 && entry.free && first === 1) {
+            first = 0;
+          }
+          if (!this.entries[i + first]) {
+            this.entries[i + first] = entry;
+          }
+        }
+        tableState.entryNum = 0;
+        tableState.streamPos = stream.pos;
+        tableState.parserBuf1 = parser.buf1;
+        tableState.parserBuf2 = parser.buf2;
+        delete tableState.firstEntryNum;
+        delete tableState.entryCount;
+      }
+      if (this.entries[0] && !this.entries[0].free) {
+        error('Invalid XRef table: unexpected first object');
+      }
+      return obj;
+    },
+    processXRefStream: function XRef_processXRefStream(stream) {
+      if (!('streamState' in this)) {
+        var streamParameters = stream.dict;
+        var byteWidths = streamParameters.get('W');
+        var range = streamParameters.get('Index');
+        if (!range) {
+          range = [0, streamParameters.get('Size')];
+        }
+        this.streamState = {
+          entryRanges: range,
+          byteWidths: byteWidths,
+          entryNum: 0,
+          streamPos: stream.pos
+        };
+      }
+      this.readXRefStream(stream);
+      delete this.streamState;
+      return stream.dict;
+    },
+    readXRefStream: function XRef_readXRefStream(stream) {
+      var i, j;
+      var streamState = this.streamState;
+      stream.pos = streamState.streamPos;
+      var byteWidths = streamState.byteWidths;
+      var typeFieldWidth = byteWidths[0];
+      var offsetFieldWidth = byteWidths[1];
+      var generationFieldWidth = byteWidths[2];
+      var entryRanges = streamState.entryRanges;
+      while (entryRanges.length > 0) {
+        var first = entryRanges[0];
+        var n = entryRanges[1];
+        if (!isInt(first) || !isInt(n)) {
+          error('Invalid XRef range fields: ' + first + ', ' + n);
+        }
+        if (!isInt(typeFieldWidth) || !isInt(offsetFieldWidth) || !isInt(generationFieldWidth)) {
+          error('Invalid XRef entry fields length: ' + first + ', ' + n);
+        }
+        for (i = streamState.entryNum; i < n; ++i) {
+          streamState.entryNum = i;
+          streamState.streamPos = stream.pos;
+          var type = 0,
+              offset = 0,
+              generation = 0;
+          for (j = 0; j < typeFieldWidth; ++j) {
+            type = type << 8 | stream.getByte();
+          }
+          if (typeFieldWidth === 0) {
+            type = 1;
+          }
+          for (j = 0; j < offsetFieldWidth; ++j) {
+            offset = offset << 8 | stream.getByte();
+          }
+          for (j = 0; j < generationFieldWidth; ++j) {
+            generation = generation << 8 | stream.getByte();
+          }
+          var entry = {};
+          entry.offset = offset;
+          entry.gen = generation;
+          switch (type) {
+            case 0:
+              entry.free = true;
+              break;
+            case 1:
+              entry.uncompressed = true;
+              break;
+            case 2:
+              break;
+            default:
+              error('Invalid XRef entry type: ' + type);
+          }
+          if (!this.entries[first + i]) {
+            this.entries[first + i] = entry;
+          }
+        }
+        streamState.entryNum = 0;
+        streamState.streamPos = stream.pos;
+        entryRanges.splice(0, 2);
+      }
+    },
+    indexObjects: function XRef_indexObjects() {
+      var TAB = 0x9,
+          LF = 0xA,
+          CR = 0xD,
+          SPACE = 0x20;
+      var PERCENT = 0x25,
+          LT = 0x3C;
+      function readToken(data, offset) {
+        var token = '',
+            ch = data[offset];
+        while (ch !== LF && ch !== CR && ch !== LT) {
+          if (++offset >= data.length) {
+            break;
+          }
+          token += String.fromCharCode(ch);
+          ch = data[offset];
+        }
+        return token;
+      }
+      function skipUntil(data, offset, what) {
+        var length = what.length,
+            dataLength = data.length;
+        var skipped = 0;
+        while (offset < dataLength) {
+          var i = 0;
+          while (i < length && data[offset + i] === what[i]) {
+            ++i;
+          }
+          if (i >= length) {
+            break;
+          }
+          offset++;
+          skipped++;
+        }
+        return skipped;
+      }
+      var objRegExp = /^(\d+)\s+(\d+)\s+obj\b/;
+      var trailerBytes = new Uint8Array([116, 114, 97, 105, 108, 101, 114]);
+      var startxrefBytes = new Uint8Array([115, 116, 97, 114, 116, 120, 114, 101, 102]);
+      var endobjBytes = new Uint8Array([101, 110, 100, 111, 98, 106]);
+      var xrefBytes = new Uint8Array([47, 88, 82, 101, 102]);
+      this.entries.length = 0;
+      var stream = this.stream;
+      stream.pos = 0;
+      var buffer = stream.getBytes();
+      var position = stream.start,
+          length = buffer.length;
+      var trailers = [],
+          xrefStms = [];
+      while (position < length) {
+        var ch = buffer[position];
+        if (ch === TAB || ch === LF || ch === CR || ch === SPACE) {
+          ++position;
+          continue;
+        }
+        if (ch === PERCENT) {
+          do {
+            ++position;
+            if (position >= length) {
+              break;
+            }
+            ch = buffer[position];
+          } while (ch !== LF && ch !== CR);
+          continue;
+        }
+        var token = readToken(buffer, position);
+        var m;
+        if (token.indexOf('xref') === 0 && (token.length === 4 || /\s/.test(token[4]))) {
+          position += skipUntil(buffer, position, trailerBytes);
+          trailers.push(position);
+          position += skipUntil(buffer, position, startxrefBytes);
+        } else if (m = objRegExp.exec(token)) {
+          if (typeof this.entries[m[1]] === 'undefined') {
+            this.entries[m[1]] = {
+              offset: position - stream.start,
+              gen: m[2] | 0,
+              uncompressed: true
+            };
+          }
+          var contentLength = skipUntil(buffer, position, endobjBytes) + 7;
+          var content = buffer.subarray(position, position + contentLength);
+          var xrefTagOffset = skipUntil(content, 0, xrefBytes);
+          if (xrefTagOffset < contentLength && content[xrefTagOffset + 5] < 64) {
+            xrefStms.push(position - stream.start);
+            this.xrefstms[position - stream.start] = 1;
+          }
+          position += contentLength;
+        } else if (token.indexOf('trailer') === 0 && (token.length === 7 || /\s/.test(token[7]))) {
+          trailers.push(position);
+          position += skipUntil(buffer, position, startxrefBytes);
+        } else {
+          position += token.length + 1;
+        }
+      }
+      var i, ii;
+      for (i = 0, ii = xrefStms.length; i < ii; ++i) {
+        this.startXRefQueue.push(xrefStms[i]);
+        this.readXRef(true);
+      }
+      var dict;
+      for (i = 0, ii = trailers.length; i < ii; ++i) {
+        stream.pos = trailers[i];
+        var parser = new Parser(new Lexer(stream), true, this, true);
+        var obj = parser.getObj();
+        if (!isCmd(obj, 'trailer')) {
+          continue;
+        }
+        dict = parser.getObj();
+        if (!isDict(dict)) {
+          continue;
+        }
+        if (dict.has('ID')) {
+          return dict;
+        }
+      }
+      if (dict) {
+        return dict;
+      }
+      throw new InvalidPDFException('Invalid PDF structure');
+    },
+    readXRef: function XRef_readXRef(recoveryMode) {
+      var stream = this.stream;
+      try {
+        while (this.startXRefQueue.length) {
+          var startXRef = this.startXRefQueue[0];
+          stream.pos = startXRef + stream.start;
+          var parser = new Parser(new Lexer(stream), true, this);
+          var obj = parser.getObj();
+          var dict;
+          if (isCmd(obj, 'xref')) {
+            dict = this.processXRefTable(parser);
+            if (!this.topDict) {
+              this.topDict = dict;
+            }
+            obj = dict.get('XRefStm');
+            if (isInt(obj)) {
+              var pos = obj;
+              if (!(pos in this.xrefstms)) {
+                this.xrefstms[pos] = 1;
+                this.startXRefQueue.push(pos);
+              }
+            }
+          } else if (isInt(obj)) {
+            if (!isInt(parser.getObj()) || !isCmd(parser.getObj(), 'obj') || !isStream(obj = parser.getObj())) {
+              error('Invalid XRef stream');
+            }
+            dict = this.processXRefStream(obj);
+            if (!this.topDict) {
+              this.topDict = dict;
+            }
+            if (!dict) {
+              error('Failed to read XRef stream');
+            }
+          } else {
+            error('Invalid XRef stream header');
+          }
+          obj = dict.get('Prev');
+          if (isInt(obj)) {
+            this.startXRefQueue.push(obj);
+          } else if (isRef(obj)) {
+            this.startXRefQueue.push(obj.num);
+          }
+          this.startXRefQueue.shift();
+        }
+        return this.topDict;
+      } catch (e) {
+        if (e instanceof MissingDataException) {
+          throw e;
+        }
+        info('(while reading XRef): ' + e);
+      }
+      if (recoveryMode) {
+        return;
+      }
+      throw new XRefParseException();
+    },
+    getEntry: function XRef_getEntry(i) {
+      var xrefEntry = this.entries[i];
+      if (xrefEntry && !xrefEntry.free && xrefEntry.offset) {
+        return xrefEntry;
+      }
+      return null;
+    },
+    fetchIfRef: function XRef_fetchIfRef(obj, suppressEncryption) {
+      if (!isRef(obj)) {
+        return obj;
+      }
+      return this.fetch(obj, suppressEncryption);
+    },
+    fetch: function XRef_fetch(ref, suppressEncryption) {
+      assert(isRef(ref), 'ref object is not a reference');
+      var num = ref.num;
+      if (num in this.cache) {
+        var cacheEntry = this.cache[num];
+        if (isDict(cacheEntry) && !cacheEntry.objId) {
+          cacheEntry.objId = ref.toString();
+        }
+        return cacheEntry;
+      }
+      var xrefEntry = this.getEntry(num);
+      if (xrefEntry === null) {
+        return this.cache[num] = null;
+      }
+      if (xrefEntry.uncompressed) {
+        xrefEntry = this.fetchUncompressed(ref, xrefEntry, suppressEncryption);
+      } else {
+        xrefEntry = this.fetchCompressed(xrefEntry, suppressEncryption);
+      }
+      if (isDict(xrefEntry)) {
+        xrefEntry.objId = ref.toString();
+      } else if (isStream(xrefEntry)) {
+        xrefEntry.dict.objId = ref.toString();
+      }
+      return xrefEntry;
+    },
+    fetchUncompressed: function XRef_fetchUncompressed(ref, xrefEntry, suppressEncryption) {
+      var gen = ref.gen;
+      var num = ref.num;
+      if (xrefEntry.gen !== gen) {
+        error('inconsistent generation in XRef');
+      }
+      var stream = this.stream.makeSubStream(xrefEntry.offset + this.stream.start);
+      var parser = new Parser(new Lexer(stream), true, this);
+      var obj1 = parser.getObj();
+      var obj2 = parser.getObj();
+      var obj3 = parser.getObj();
+      if (!isInt(obj1) || parseInt(obj1, 10) !== num || !isInt(obj2) || parseInt(obj2, 10) !== gen || !isCmd(obj3)) {
+        error('bad XRef entry');
+      }
+      if (!isCmd(obj3, 'obj')) {
+        if (obj3.cmd.indexOf('obj') === 0) {
+          num = parseInt(obj3.cmd.substring(3), 10);
+          if (!isNaN(num)) {
+            return num;
+          }
+        }
+        error('bad XRef entry');
+      }
+      if (this.encrypt && !suppressEncryption) {
+        xrefEntry = parser.getObj(this.encrypt.createCipherTransform(num, gen));
+      } else {
+        xrefEntry = parser.getObj();
+      }
+      if (!isStream(xrefEntry)) {
+        this.cache[num] = xrefEntry;
+      }
+      return xrefEntry;
+    },
+    fetchCompressed: function XRef_fetchCompressed(xrefEntry, suppressEncryption) {
+      var tableOffset = xrefEntry.offset;
+      var stream = this.fetch(new Ref(tableOffset, 0));
+      if (!isStream(stream)) {
+        error('bad ObjStm stream');
+      }
+      var first = stream.dict.get('First');
+      var n = stream.dict.get('N');
+      if (!isInt(first) || !isInt(n)) {
+        error('invalid first and n parameters for ObjStm stream');
+      }
+      var parser = new Parser(new Lexer(stream), false, this);
+      parser.allowStreams = true;
+      var i,
+          entries = [],
+          num,
+          nums = [];
+      for (i = 0; i < n; ++i) {
+        num = parser.getObj();
+        if (!isInt(num)) {
+          error('invalid object number in the ObjStm stream: ' + num);
+        }
+        nums.push(num);
+        var offset = parser.getObj();
+        if (!isInt(offset)) {
+          error('invalid object offset in the ObjStm stream: ' + offset);
+        }
+      }
+      for (i = 0; i < n; ++i) {
+        entries.push(parser.getObj());
+        if (isCmd(parser.buf1, 'endobj')) {
+          parser.shift();
+        }
+        num = nums[i];
+        var entry = this.entries[num];
+        if (entry && entry.offset === tableOffset && entry.gen === i) {
+          this.cache[num] = entries[i];
+        }
+      }
+      xrefEntry = entries[xrefEntry.gen];
+      if (xrefEntry === undefined) {
+        error('bad XRef entry for compressed object');
+      }
+      return xrefEntry;
+    },
+    fetchIfRefAsync: function XRef_fetchIfRefAsync(obj, suppressEncryption) {
+      if (!isRef(obj)) {
+        return Promise.resolve(obj);
+      }
+      return this.fetchAsync(obj, suppressEncryption);
+    },
+    fetchAsync: function XRef_fetchAsync(ref, suppressEncryption) {
+      var streamManager = this.stream.manager;
+      var xref = this;
+      return new Promise(function tryFetch(resolve, reject) {
+        try {
+          resolve(xref.fetch(ref, suppressEncryption));
+        } catch (e) {
+          if (e instanceof MissingDataException) {
+            streamManager.requestRange(e.begin, e.end).then(function () {
+              tryFetch(resolve, reject);
+            }, reject);
+            return;
+          }
+          reject(e);
+        }
+      });
+    },
+    getCatalogObj: function XRef_getCatalogObj() {
+      return this.root;
+    }
+  };
+  return XRef;
 }();
 var NameOrNumberTree = function NameOrNumberTreeClosure() {
- function NameOrNumberTree(root, xref) {
-  throw new Error('Cannot initialize NameOrNumberTree.');
- }
- NameOrNumberTree.prototype = {
-  getAll: function NameOrNumberTree_getAll() {
-   var dict = Object.create(null);
-   if (!this.root) {
-    return dict;
-   }
-   var xref = this.xref;
-   var processed = new RefSet();
-   processed.put(this.root);
-   var queue = [this.root];
-   while (queue.length > 0) {
-    var i, n;
-    var obj = xref.fetchIfRef(queue.shift());
-    if (!isDict(obj)) {
-     continue;
-    }
-    if (obj.has('Kids')) {
-     var kids = obj.get('Kids');
-     for (i = 0, n = kids.length; i < n; i++) {
-      var kid = kids[i];
-      assert(!processed.has(kid), 'Duplicate entry in "' + this._type + '" tree.');
-      queue.push(kid);
-      processed.put(kid);
-     }
-     continue;
-    }
-    var entries = obj.get(this._type);
-    if (isArray(entries)) {
-     for (i = 0, n = entries.length; i < n; i += 2) {
-      dict[xref.fetchIfRef(entries[i])] = xref.fetchIfRef(entries[i + 1]);
-     }
-    }
-   }
-   return dict;
-  },
-  get: function NameOrNumberTree_get(key) {
-   if (!this.root) {
-    return null;
-   }
-   var xref = this.xref;
-   var kidsOrEntries = xref.fetchIfRef(this.root);
-   var loopCount = 0;
-   var MAX_LEVELS = 10;
-   var l, r, m;
-   while (kidsOrEntries.has('Kids')) {
-    if (++loopCount > MAX_LEVELS) {
-     warn('Search depth limit reached for "' + this._type + '" tree.');
-     return null;
-    }
-    var kids = kidsOrEntries.get('Kids');
-    if (!isArray(kids)) {
-     return null;
-    }
-    l = 0;
-    r = kids.length - 1;
-    while (l <= r) {
-     m = l + r >> 1;
-     var kid = xref.fetchIfRef(kids[m]);
-     var limits = kid.get('Limits');
-     if (key < xref.fetchIfRef(limits[0])) {
-      r = m - 1;
-     } else if (key > xref.fetchIfRef(limits[1])) {
-      l = m + 1;
-     } else {
-      kidsOrEntries = xref.fetchIfRef(kids[m]);
-      break;
-     }
-    }
-    if (l > r) {
-     return null;
-    }
-   }
-   var entries = kidsOrEntries.get(this._type);
-   if (isArray(entries)) {
-    l = 0;
-    r = entries.length - 2;
-    while (l <= r) {
-     m = l + r & ~1;
-     var currentKey = xref.fetchIfRef(entries[m]);
-     if (key < currentKey) {
-      r = m - 2;
-     } else if (key > currentKey) {
-      l = m + 2;
-     } else {
-      return xref.fetchIfRef(entries[m + 1]);
-     }
-    }
-   }
-   return null;
+  function NameOrNumberTree(root, xref) {
+    throw new Error('Cannot initialize NameOrNumberTree.');
   }
- };
- return NameOrNumberTree;
+  NameOrNumberTree.prototype = {
+    getAll: function NameOrNumberTree_getAll() {
+      var dict = Object.create(null);
+      if (!this.root) {
+        return dict;
+      }
+      var xref = this.xref;
+      var processed = new RefSet();
+      processed.put(this.root);
+      var queue = [this.root];
+      while (queue.length > 0) {
+        var i, n;
+        var obj = xref.fetchIfRef(queue.shift());
+        if (!isDict(obj)) {
+          continue;
+        }
+        if (obj.has('Kids')) {
+          var kids = obj.get('Kids');
+          for (i = 0, n = kids.length; i < n; i++) {
+            var kid = kids[i];
+            assert(!processed.has(kid), 'Duplicate entry in "' + this._type + '" tree.');
+            queue.push(kid);
+            processed.put(kid);
+          }
+          continue;
+        }
+        var entries = obj.get(this._type);
+        if (isArray(entries)) {
+          for (i = 0, n = entries.length; i < n; i += 2) {
+            dict[xref.fetchIfRef(entries[i])] = xref.fetchIfRef(entries[i + 1]);
+          }
+        }
+      }
+      return dict;
+    },
+    get: function NameOrNumberTree_get(key) {
+      if (!this.root) {
+        return null;
+      }
+      var xref = this.xref;
+      var kidsOrEntries = xref.fetchIfRef(this.root);
+      var loopCount = 0;
+      var MAX_LEVELS = 10;
+      var l, r, m;
+      while (kidsOrEntries.has('Kids')) {
+        if (++loopCount > MAX_LEVELS) {
+          warn('Search depth limit reached for "' + this._type + '" tree.');
+          return null;
+        }
+        var kids = kidsOrEntries.get('Kids');
+        if (!isArray(kids)) {
+          return null;
+        }
+        l = 0;
+        r = kids.length - 1;
+        while (l <= r) {
+          m = l + r >> 1;
+          var kid = xref.fetchIfRef(kids[m]);
+          var limits = kid.get('Limits');
+          if (key < xref.fetchIfRef(limits[0])) {
+            r = m - 1;
+          } else if (key > xref.fetchIfRef(limits[1])) {
+            l = m + 1;
+          } else {
+            kidsOrEntries = xref.fetchIfRef(kids[m]);
+            break;
+          }
+        }
+        if (l > r) {
+          return null;
+        }
+      }
+      var entries = kidsOrEntries.get(this._type);
+      if (isArray(entries)) {
+        l = 0;
+        r = entries.length - 2;
+        while (l <= r) {
+          m = l + r & ~1;
+          var currentKey = xref.fetchIfRef(entries[m]);
+          if (key < currentKey) {
+            r = m - 2;
+          } else if (key > currentKey) {
+            l = m + 2;
+          } else {
+            return xref.fetchIfRef(entries[m + 1]);
+          }
+        }
+      }
+      return null;
+    }
+  };
+  return NameOrNumberTree;
 }();
 var NameTree = function NameTreeClosure() {
- function NameTree(root, xref) {
-  this.root = root;
-  this.xref = xref;
-  this._type = 'Names';
- }
- Util.inherit(NameTree, NameOrNumberTree, {});
- return NameTree;
+  function NameTree(root, xref) {
+    this.root = root;
+    this.xref = xref;
+    this._type = 'Names';
+  }
+  Util.inherit(NameTree, NameOrNumberTree, {});
+  return NameTree;
 }();
 var NumberTree = function NumberTreeClosure() {
- function NumberTree(root, xref) {
-  this.root = root;
-  this.xref = xref;
-  this._type = 'Nums';
- }
- Util.inherit(NumberTree, NameOrNumberTree, {});
- return NumberTree;
+  function NumberTree(root, xref) {
+    this.root = root;
+    this.xref = xref;
+    this._type = 'Nums';
+  }
+  Util.inherit(NumberTree, NameOrNumberTree, {});
+  return NumberTree;
 }();
 var FileSpec = function FileSpecClosure() {
- function FileSpec(root, xref) {
-  if (!root || !isDict(root)) {
-   return;
-  }
-  this.xref = xref;
-  this.root = root;
-  if (root.has('FS')) {
-   this.fs = root.get('FS');
-  }
-  this.description = root.has('Desc') ? stringToPDFString(root.get('Desc')) : '';
-  if (root.has('RF')) {
-   warn('Related file specifications are not supported');
-  }
-  this.contentAvailable = true;
-  if (!root.has('EF')) {
-   this.contentAvailable = false;
-   warn('Non-embedded file specifications are not supported');
-  }
- }
- function pickPlatformItem(dict) {
-  if (dict.has('UF')) {
-   return dict.get('UF');
-  } else if (dict.has('F')) {
-   return dict.get('F');
-  } else if (dict.has('Unix')) {
-   return dict.get('Unix');
-  } else if (dict.has('Mac')) {
-   return dict.get('Mac');
-  } else if (dict.has('DOS')) {
-   return dict.get('DOS');
+  function FileSpec(root, xref) {
+    if (!root || !isDict(root)) {
+      return;
+    }
+    this.xref = xref;
+    this.root = root;
+    if (root.has('FS')) {
+      this.fs = root.get('FS');
+    }
+    this.description = root.has('Desc') ? stringToPDFString(root.get('Desc')) : '';
+    if (root.has('RF')) {
+      warn('Related file specifications are not supported');
+    }
+    this.contentAvailable = true;
+    if (!root.has('EF')) {
+      this.contentAvailable = false;
+      warn('Non-embedded file specifications are not supported');
+    }
   }
-  return null;
- }
- FileSpec.prototype = {
-  get filename() {
-   if (!this._filename && this.root) {
-    var filename = pickPlatformItem(this.root) || 'unnamed';
-    this._filename = stringToPDFString(filename).replace(/\\\\/g, '\\').replace(/\\\//g, '/').replace(/\\/g, '/');
-   }
-   return this._filename;
-  },
-  get content() {
-   if (!this.contentAvailable) {
-    return null;
-   }
-   if (!this.contentRef && this.root) {
-    this.contentRef = pickPlatformItem(this.root.get('EF'));
-   }
-   var content = null;
-   if (this.contentRef) {
-    var xref = this.xref;
-    var fileObj = xref.fetchIfRef(this.contentRef);
-    if (fileObj && isStream(fileObj)) {
-     content = fileObj.getBytes();
-    } else {
-     warn('Embedded file specification points to non-existing/invalid ' + 'content');
+  function pickPlatformItem(dict) {
+    if (dict.has('UF')) {
+      return dict.get('UF');
+    } else if (dict.has('F')) {
+      return dict.get('F');
+    } else if (dict.has('Unix')) {
+      return dict.get('Unix');
+    } else if (dict.has('Mac')) {
+      return dict.get('Mac');
+    } else if (dict.has('DOS')) {
+      return dict.get('DOS');
     }
-   } else {
-    warn('Embedded file specification does not have a content');
-   }
-   return content;
-  },
-  get serializable() {
-   return {
-    filename: this.filename,
-    content: this.content
-   };
+    return null;
   }
- };
- return FileSpec;
+  FileSpec.prototype = {
+    get filename() {
+      if (!this._filename && this.root) {
+        var filename = pickPlatformItem(this.root) || 'unnamed';
+        this._filename = stringToPDFString(filename).replace(/\\\\/g, '\\').replace(/\\\//g, '/').replace(/\\/g, '/');
+      }
+      return this._filename;
+    },
+    get content() {
+      if (!this.contentAvailable) {
+        return null;
+      }
+      if (!this.contentRef && this.root) {
+        this.contentRef = pickPlatformItem(this.root.get('EF'));
+      }
+      var content = null;
+      if (this.contentRef) {
+        var xref = this.xref;
+        var fileObj = xref.fetchIfRef(this.contentRef);
+        if (fileObj && isStream(fileObj)) {
+          content = fileObj.getBytes();
+        } else {
+          warn('Embedded file specification points to non-existing/invalid ' + 'content');
+        }
+      } else {
+        warn('Embedded file specification does not have a content');
+      }
+      return content;
+    },
+    get serializable() {
+      return {
+        filename: this.filename,
+        content: this.content
+      };
+    }
+  };
+  return FileSpec;
 }();
 var ObjectLoader = function () {
- function mayHaveChildren(value) {
-  return isRef(value) || isDict(value) || isArray(value) || isStream(value);
- }
- function addChildren(node, nodesToVisit) {
-  var value;
-  if (isDict(node) || isStream(node)) {
-   var map;
-   if (isDict(node)) {
-    map = node.map;
-   } else {
-    map = node.dict.map;
-   }
-   for (var key in map) {
-    value = map[key];
-    if (mayHaveChildren(value)) {
-     nodesToVisit.push(value);
-    }
-   }
-  } else if (isArray(node)) {
-   for (var i = 0, ii = node.length; i < ii; i++) {
-    value = node[i];
-    if (mayHaveChildren(value)) {
-     nodesToVisit.push(value);
-    }
-   }
+  function mayHaveChildren(value) {
+    return isRef(value) || isDict(value) || isArray(value) || isStream(value);
   }
- }
- function ObjectLoader(obj, keys, xref) {
-  this.obj = obj;
-  this.keys = keys;
-  this.xref = xref;
-  this.refSet = null;
-  this.capability = null;
- }
- ObjectLoader.prototype = {
-  load: function ObjectLoader_load() {
-   var keys = this.keys;
-   this.capability = createPromiseCapability();
-   if (!(this.xref.stream instanceof ChunkedStream) || this.xref.stream.getMissingChunks().length === 0) {
-    this.capability.resolve();
-    return this.capability.promise;
-   }
-   this.refSet = new RefSet();
-   var nodesToVisit = [];
-   for (var i = 0; i < keys.length; i++) {
-    nodesToVisit.push(this.obj[keys[i]]);
-   }
-   this._walk(nodesToVisit);
-   return this.capability.promise;
-  },
-  _walk: function ObjectLoader_walk(nodesToVisit) {
-   var nodesToRevisit = [];
-   var pendingRequests = [];
-   while (nodesToVisit.length) {
-    var currentNode = nodesToVisit.pop();
-    if (isRef(currentNode)) {
-     if (this.refSet.has(currentNode)) {
-      continue;
-     }
-     try {
-      var ref = currentNode;
-      this.refSet.put(ref);
-      currentNode = this.xref.fetch(currentNode);
-     } catch (e) {
-      if (!(e instanceof MissingDataException)) {
-       throw e;
-      }
-      nodesToRevisit.push(currentNode);
-      pendingRequests.push({
-       begin: e.begin,
-       end: e.end
-      });
-     }
-    }
-    if (currentNode && currentNode.getBaseStreams) {
-     var baseStreams = currentNode.getBaseStreams();
-     var foundMissingData = false;
-     for (var i = 0; i < baseStreams.length; i++) {
-      var stream = baseStreams[i];
-      if (stream.getMissingChunks && stream.getMissingChunks().length) {
-       foundMissingData = true;
-       pendingRequests.push({
-        begin: stream.start,
-        end: stream.end
-       });
-      }
-     }
-     if (foundMissingData) {
-      nodesToRevisit.push(currentNode);
-     }
+  function addChildren(node, nodesToVisit) {
+    var value;
+    if (isDict(node) || isStream(node)) {
+      var map;
+      if (isDict(node)) {
+        map = node.map;
+      } else {
+        map = node.dict.map;
+      }
+      for (var key in map) {
+        value = map[key];
+        if (mayHaveChildren(value)) {
+          nodesToVisit.push(value);
+        }
+      }
+    } else if (isArray(node)) {
+      for (var i = 0, ii = node.length; i < ii; i++) {
+        value = node[i];
+        if (mayHaveChildren(value)) {
+          nodesToVisit.push(value);
+        }
+      }
     }
-    addChildren(currentNode, nodesToVisit);
-   }
-   if (pendingRequests.length) {
-    this.xref.stream.manager.requestRanges(pendingRequests).then(function pendingRequestCallback() {
-     nodesToVisit = nodesToRevisit;
-     for (var i = 0; i < nodesToRevisit.length; i++) {
-      var node = nodesToRevisit[i];
-      if (isRef(node)) {
-       this.refSet.remove(node);
-      }
-     }
-     this._walk(nodesToVisit);
-    }.bind(this), this.capability.reject);
-    return;
-   }
-   this.refSet = null;
-   this.capability.resolve();
   }
- };
- return ObjectLoader;
+  function ObjectLoader(obj, keys, xref) {
+    this.obj = obj;
+    this.keys = keys;
+    this.xref = xref;
+    this.refSet = null;
+    this.capability = null;
+  }
+  ObjectLoader.prototype = {
+    load: function ObjectLoader_load() {
+      var keys = this.keys;
+      this.capability = createPromiseCapability();
+      if (!(this.xref.stream instanceof ChunkedStream) || this.xref.stream.getMissingChunks().length === 0) {
+        this.capability.resolve();
+        return this.capability.promise;
+      }
+      this.refSet = new RefSet();
+      var nodesToVisit = [];
+      for (var i = 0; i < keys.length; i++) {
+        nodesToVisit.push(this.obj[keys[i]]);
+      }
+      this._walk(nodesToVisit);
+      return this.capability.promise;
+    },
+    _walk: function ObjectLoader_walk(nodesToVisit) {
+      var nodesToRevisit = [];
+      var pendingRequests = [];
+      while (nodesToVisit.length) {
+        var currentNode = nodesToVisit.pop();
+        if (isRef(currentNode)) {
+          if (this.refSet.has(currentNode)) {
+            continue;
+          }
+          try {
+            var ref = currentNode;
+            this.refSet.put(ref);
+            currentNode = this.xref.fetch(currentNode);
+          } catch (e) {
+            if (!(e instanceof MissingDataException)) {
+              throw e;
+            }
+            nodesToRevisit.push(currentNode);
+            pendingRequests.push({
+              begin: e.begin,
+              end: e.end
+            });
+          }
+        }
+        if (currentNode && currentNode.getBaseStreams) {
+          var baseStreams = currentNode.getBaseStreams();
+          var foundMissingData = false;
+          for (var i = 0; i < baseStreams.length; i++) {
+            var stream = baseStreams[i];
+            if (stream.getMissingChunks && stream.getMissingChunks().length) {
+              foundMissingData = true;
+              pendingRequests.push({
+                begin: stream.start,
+                end: stream.end
+              });
+            }
+          }
+          if (foundMissingData) {
+            nodesToRevisit.push(currentNode);
+          }
+        }
+        addChildren(currentNode, nodesToVisit);
+      }
+      if (pendingRequests.length) {
+        this.xref.stream.manager.requestRanges(pendingRequests).then(function pendingRequestCallback() {
+          nodesToVisit = nodesToRevisit;
+          for (var i = 0; i < nodesToRevisit.length; i++) {
+            var node = nodesToRevisit[i];
+            if (isRef(node)) {
+              this.refSet.remove(node);
+            }
+          }
+          this._walk(nodesToVisit);
+        }.bind(this), this.capability.reject);
+        return;
+      }
+      this.refSet = null;
+      this.capability.resolve();
+    }
+  };
+  return ObjectLoader;
 }();
 exports.Catalog = Catalog;
 exports.ObjectLoader = ObjectLoader;

+ 902 - 1139
lib/core/parser.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var coreStream = require('./stream.js');
@@ -48,1171 +49,933 @@ var PredictorStream = coreStream.PredictorStream;
 var RunLengthStream = coreStream.RunLengthStream;
 var MAX_LENGTH_TO_CACHE = 1000;
 var Parser = function ParserClosure() {
- function Parser(lexer, allowStreams, xref, recoveryMode) {
-  this.lexer = lexer;
-  this.allowStreams = allowStreams;
-  this.xref = xref;
-  this.recoveryMode = recoveryMode || false;
-  this.imageCache = Object.create(null);
-  this.refill();
- }
- Parser.prototype = {
-  refill: function Parser_refill() {
-   this.buf1 = this.lexer.getObj();
-   this.buf2 = this.lexer.getObj();
-  },
-  shift: function Parser_shift() {
-   if (isCmd(this.buf2, 'ID')) {
-    this.buf1 = this.buf2;
-    this.buf2 = null;
-   } else {
-    this.buf1 = this.buf2;
-    this.buf2 = this.lexer.getObj();
-   }
-  },
-  tryShift: function Parser_tryShift() {
-   try {
-    this.shift();
-    return true;
-   } catch (e) {
-    if (e instanceof MissingDataException) {
-     throw e;
-    }
-    return false;
-   }
-  },
-  getObj: function Parser_getObj(cipherTransform) {
-   var buf1 = this.buf1;
-   this.shift();
-   if (buf1 instanceof Cmd) {
-    switch (buf1.cmd) {
-    case 'BI':
-     return this.makeInlineImage(cipherTransform);
-    case '[':
-     var array = [];
-     while (!isCmd(this.buf1, ']') && !isEOF(this.buf1)) {
-      array.push(this.getObj(cipherTransform));
-     }
-     if (isEOF(this.buf1)) {
-      if (!this.recoveryMode) {
-       error('End of file inside array');
+  function Parser(lexer, allowStreams, xref, recoveryMode) {
+    this.lexer = lexer;
+    this.allowStreams = allowStreams;
+    this.xref = xref;
+    this.recoveryMode = recoveryMode || false;
+    this.imageCache = Object.create(null);
+    this.refill();
+  }
+  Parser.prototype = {
+    refill: function Parser_refill() {
+      this.buf1 = this.lexer.getObj();
+      this.buf2 = this.lexer.getObj();
+    },
+    shift: function Parser_shift() {
+      if (isCmd(this.buf2, 'ID')) {
+        this.buf1 = this.buf2;
+        this.buf2 = null;
+      } else {
+        this.buf1 = this.buf2;
+        this.buf2 = this.lexer.getObj();
       }
-      return array;
-     }
-     this.shift();
-     return array;
-    case '<<':
-     var dict = new Dict(this.xref);
-     while (!isCmd(this.buf1, '>>') && !isEOF(this.buf1)) {
-      if (!isName(this.buf1)) {
-       info('Malformed dictionary: key must be a name object');
-       this.shift();
-       continue;
+    },
+    tryShift: function Parser_tryShift() {
+      try {
+        this.shift();
+        return true;
+      } catch (e) {
+        if (e instanceof MissingDataException) {
+          throw e;
+        }
+        return false;
       }
-      var key = this.buf1.name;
+    },
+    getObj: function Parser_getObj(cipherTransform) {
+      var buf1 = this.buf1;
       this.shift();
-      if (isEOF(this.buf1)) {
-       break;
+      if (buf1 instanceof Cmd) {
+        switch (buf1.cmd) {
+          case 'BI':
+            return this.makeInlineImage(cipherTransform);
+          case '[':
+            var array = [];
+            while (!isCmd(this.buf1, ']') && !isEOF(this.buf1)) {
+              array.push(this.getObj(cipherTransform));
+            }
+            if (isEOF(this.buf1)) {
+              if (!this.recoveryMode) {
+                error('End of file inside array');
+              }
+              return array;
+            }
+            this.shift();
+            return array;
+          case '<<':
+            var dict = new Dict(this.xref);
+            while (!isCmd(this.buf1, '>>') && !isEOF(this.buf1)) {
+              if (!isName(this.buf1)) {
+                info('Malformed dictionary: key must be a name object');
+                this.shift();
+                continue;
+              }
+              var key = this.buf1.name;
+              this.shift();
+              if (isEOF(this.buf1)) {
+                break;
+              }
+              dict.set(key, this.getObj(cipherTransform));
+            }
+            if (isEOF(this.buf1)) {
+              if (!this.recoveryMode) {
+                error('End of file inside dictionary');
+              }
+              return dict;
+            }
+            if (isCmd(this.buf2, 'stream')) {
+              return this.allowStreams ? this.makeStream(dict, cipherTransform) : dict;
+            }
+            this.shift();
+            return dict;
+          default:
+            return buf1;
+        }
       }
-      dict.set(key, this.getObj(cipherTransform));
-     }
-     if (isEOF(this.buf1)) {
-      if (!this.recoveryMode) {
-       error('End of file inside dictionary');
+      if (isInt(buf1)) {
+        var num = buf1;
+        if (isInt(this.buf1) && isCmd(this.buf2, 'R')) {
+          var ref = new Ref(num, this.buf1);
+          this.shift();
+          this.shift();
+          return ref;
+        }
+        return num;
       }
-      return dict;
-     }
-     if (isCmd(this.buf2, 'stream')) {
-      return this.allowStreams ? this.makeStream(dict, cipherTransform) : dict;
-     }
-     this.shift();
-     return dict;
-    default:
-     return buf1;
-    }
-   }
-   if (isInt(buf1)) {
-    var num = buf1;
-    if (isInt(this.buf1) && isCmd(this.buf2, 'R')) {
-     var ref = new Ref(num, this.buf1);
-     this.shift();
-     this.shift();
-     return ref;
-    }
-    return num;
-   }
-   if (isString(buf1)) {
-    var str = buf1;
-    if (cipherTransform) {
-     str = cipherTransform.decryptString(str);
-    }
-    return str;
-   }
-   return buf1;
-  },
-  findDefaultInlineStreamEnd: function Parser_findDefaultInlineStreamEnd(stream) {
-   var E = 0x45, I = 0x49, SPACE = 0x20, LF = 0xA, CR = 0xD;
-   var startPos = stream.pos, state = 0, ch, i, n, followingBytes;
-   while ((ch = stream.getByte()) !== -1) {
-    if (state === 0) {
-     state = ch === E ? 1 : 0;
-    } else if (state === 1) {
-     state = ch === I ? 2 : 0;
-    } else {
-     assert(state === 2);
-     if (ch === SPACE || ch === LF || ch === CR) {
-      n = 5;
-      followingBytes = stream.peekBytes(n);
-      for (i = 0; i < n; i++) {
-       ch = followingBytes[i];
-       if (ch !== LF && ch !== CR && (ch < SPACE || ch > 0x7F)) {
-        state = 0;
-        break;
-       }
+      if (isString(buf1)) {
+        var str = buf1;
+        if (cipherTransform) {
+          str = cipherTransform.decryptString(str);
+        }
+        return str;
       }
-      if (state === 2) {
-       break;
+      return buf1;
+    },
+    findDefaultInlineStreamEnd: function Parser_findDefaultInlineStreamEnd(stream) {
+      var E = 0x45,
+          I = 0x49,
+          SPACE = 0x20,
+          LF = 0xA,
+          CR = 0xD;
+      var startPos = stream.pos,
+          state = 0,
+          ch,
+          i,
+          n,
+          followingBytes;
+      while ((ch = stream.getByte()) !== -1) {
+        if (state === 0) {
+          state = ch === E ? 1 : 0;
+        } else if (state === 1) {
+          state = ch === I ? 2 : 0;
+        } else {
+          assert(state === 2);
+          if (ch === SPACE || ch === LF || ch === CR) {
+            n = 5;
+            followingBytes = stream.peekBytes(n);
+            for (i = 0; i < n; i++) {
+              ch = followingBytes[i];
+              if (ch !== LF && ch !== CR && (ch < SPACE || ch > 0x7F)) {
+                state = 0;
+                break;
+              }
+            }
+            if (state === 2) {
+              break;
+            }
+          } else {
+            state = 0;
+          }
+        }
       }
-     } else {
-      state = 0;
-     }
-    }
-   }
-   return stream.pos - 4 - startPos;
-  },
-  findDCTDecodeInlineStreamEnd: function Parser_findDCTDecodeInlineStreamEnd(stream) {
-   var startPos = stream.pos, foundEOI = false, b, markerLength, length;
-   while ((b = stream.getByte()) !== -1) {
-    if (b !== 0xFF) {
-     continue;
-    }
-    switch (stream.getByte()) {
-    case 0x00:
-     break;
-    case 0xFF:
-     stream.skip(-1);
-     break;
-    case 0xD9:
-     foundEOI = true;
-     break;
-    case 0xC0:
-    case 0xC1:
-    case 0xC2:
-    case 0xC3:
-    case 0xC5:
-    case 0xC6:
-    case 0xC7:
-    case 0xC9:
-    case 0xCA:
-    case 0xCB:
-    case 0xCD:
-    case 0xCE:
-    case 0xCF:
-    case 0xC4:
-    case 0xCC:
-    case 0xDA:
-    case 0xDB:
-    case 0xDC:
-    case 0xDD:
-    case 0xDE:
-    case 0xDF:
-    case 0xE0:
-    case 0xE1:
-    case 0xE2:
-    case 0xE3:
-    case 0xE4:
-    case 0xE5:
-    case 0xE6:
-    case 0xE7:
-    case 0xE8:
-    case 0xE9:
-    case 0xEA:
-    case 0xEB:
-    case 0xEC:
-    case 0xED:
-    case 0xEE:
-    case 0xEF:
-    case 0xFE:
-     markerLength = stream.getUint16();
-     if (markerLength > 2) {
-      stream.skip(markerLength - 2);
-     } else {
-      stream.skip(-2);
-     }
-     break;
-    }
-    if (foundEOI) {
-     break;
-    }
-   }
-   length = stream.pos - startPos;
-   if (b === -1) {
-    warn('Inline DCTDecode image stream: ' + 'EOI marker not found, searching for /EI/ instead.');
-    stream.skip(-length);
-    return this.findDefaultInlineStreamEnd(stream);
-   }
-   this.inlineStreamSkipEI(stream);
-   return length;
-  },
-  findASCII85DecodeInlineStreamEnd: function Parser_findASCII85DecodeInlineStreamEnd(stream) {
-   var TILDE = 0x7E, GT = 0x3E;
-   var startPos = stream.pos, ch, length;
-   while ((ch = stream.getByte()) !== -1) {
-    if (ch === TILDE && stream.peekByte() === GT) {
-     stream.skip();
-     break;
-    }
-   }
-   length = stream.pos - startPos;
-   if (ch === -1) {
-    warn('Inline ASCII85Decode image stream: ' + 'EOD marker not found, searching for /EI/ instead.');
-    stream.skip(-length);
-    return this.findDefaultInlineStreamEnd(stream);
-   }
-   this.inlineStreamSkipEI(stream);
-   return length;
-  },
-  findASCIIHexDecodeInlineStreamEnd: function Parser_findASCIIHexDecodeInlineStreamEnd(stream) {
-   var GT = 0x3E;
-   var startPos = stream.pos, ch, length;
-   while ((ch = stream.getByte()) !== -1) {
-    if (ch === GT) {
-     break;
-    }
-   }
-   length = stream.pos - startPos;
-   if (ch === -1) {
-    warn('Inline ASCIIHexDecode image stream: ' + 'EOD marker not found, searching for /EI/ instead.');
-    stream.skip(-length);
-    return this.findDefaultInlineStreamEnd(stream);
-   }
-   this.inlineStreamSkipEI(stream);
-   return length;
-  },
-  inlineStreamSkipEI: function Parser_inlineStreamSkipEI(stream) {
-   var E = 0x45, I = 0x49;
-   var state = 0, ch;
-   while ((ch = stream.getByte()) !== -1) {
-    if (state === 0) {
-     state = ch === E ? 1 : 0;
-    } else if (state === 1) {
-     state = ch === I ? 2 : 0;
-    } else if (state === 2) {
-     break;
-    }
-   }
-  },
-  makeInlineImage: function Parser_makeInlineImage(cipherTransform) {
-   var lexer = this.lexer;
-   var stream = lexer.stream;
-   var dict = new Dict(this.xref);
-   while (!isCmd(this.buf1, 'ID') && !isEOF(this.buf1)) {
-    if (!isName(this.buf1)) {
-     error('Dictionary key must be a name object');
-    }
-    var key = this.buf1.name;
-    this.shift();
-    if (isEOF(this.buf1)) {
-     break;
-    }
-    dict.set(key, this.getObj(cipherTransform));
-   }
-   var filter = dict.get('Filter', 'F'), filterName;
-   if (isName(filter)) {
-    filterName = filter.name;
-   } else if (isArray(filter)) {
-    var filterZero = this.xref.fetchIfRef(filter[0]);
-    if (isName(filterZero)) {
-     filterName = filterZero.name;
-    }
-   }
-   var startPos = stream.pos, length, i, ii;
-   if (filterName === 'DCTDecode' || filterName === 'DCT') {
-    length = this.findDCTDecodeInlineStreamEnd(stream);
-   } else if (filterName === 'ASCII85Decide' || filterName === 'A85') {
-    length = this.findASCII85DecodeInlineStreamEnd(stream);
-   } else if (filterName === 'ASCIIHexDecode' || filterName === 'AHx') {
-    length = this.findASCIIHexDecodeInlineStreamEnd(stream);
-   } else {
-    length = this.findDefaultInlineStreamEnd(stream);
-   }
-   var imageStream = stream.makeSubStream(startPos, length, dict);
-   var adler32;
-   if (length < MAX_LENGTH_TO_CACHE) {
-    var imageBytes = imageStream.getBytes();
-    imageStream.reset();
-    var a = 1;
-    var b = 0;
-    for (i = 0, ii = imageBytes.length; i < ii; ++i) {
-     a += imageBytes[i] & 0xff;
-     b += a;
-    }
-    adler32 = b % 65521 << 16 | a % 65521;
-    if (this.imageCache.adler32 === adler32) {
-     this.buf2 = Cmd.get('EI');
-     this.shift();
-     this.imageCache[adler32].reset();
-     return this.imageCache[adler32];
-    }
-   }
-   if (cipherTransform) {
-    imageStream = cipherTransform.createStream(imageStream, length);
-   }
-   imageStream = this.filter(imageStream, dict, length);
-   imageStream.dict = dict;
-   if (adler32 !== undefined) {
-    imageStream.cacheKey = 'inline_' + length + '_' + adler32;
-    this.imageCache[adler32] = imageStream;
-   }
-   this.buf2 = Cmd.get('EI');
-   this.shift();
-   return imageStream;
-  },
-  makeStream: function Parser_makeStream(dict, cipherTransform) {
-   var lexer = this.lexer;
-   var stream = lexer.stream;
-   lexer.skipToNextLine();
-   var pos = stream.pos - 1;
-   var length = dict.get('Length');
-   if (!isInt(length)) {
-    info('Bad ' + length + ' attribute in stream');
-    length = 0;
-   }
-   stream.pos = pos + length;
-   lexer.nextChar();
-   if (this.tryShift() && isCmd(this.buf2, 'endstream')) {
-    this.shift();
-   } else {
-    stream.pos = pos;
-    var SCAN_BLOCK_SIZE = 2048;
-    var ENDSTREAM_SIGNATURE_LENGTH = 9;
-    var ENDSTREAM_SIGNATURE = [
-     0x65,
-     0x6E,
-     0x64,
-     0x73,
-     0x74,
-     0x72,
-     0x65,
-     0x61,
-     0x6D
-    ];
-    var skipped = 0, found = false, i, j;
-    while (stream.pos < stream.end) {
-     var scanBytes = stream.peekBytes(SCAN_BLOCK_SIZE);
-     var scanLength = scanBytes.length - ENDSTREAM_SIGNATURE_LENGTH;
-     if (scanLength <= 0) {
-      break;
-     }
-     found = false;
-     i = 0;
-     while (i < scanLength) {
-      j = 0;
-      while (j < ENDSTREAM_SIGNATURE_LENGTH && scanBytes[i + j] === ENDSTREAM_SIGNATURE[j]) {
-       j++;
+      return stream.pos - 4 - startPos;
+    },
+    findDCTDecodeInlineStreamEnd: function Parser_findDCTDecodeInlineStreamEnd(stream) {
+      var startPos = stream.pos,
+          foundEOI = false,
+          b,
+          markerLength,
+          length;
+      while ((b = stream.getByte()) !== -1) {
+        if (b !== 0xFF) {
+          continue;
+        }
+        switch (stream.getByte()) {
+          case 0x00:
+            break;
+          case 0xFF:
+            stream.skip(-1);
+            break;
+          case 0xD9:
+            foundEOI = true;
+            break;
+          case 0xC0:
+          case 0xC1:
+          case 0xC2:
+          case 0xC3:
+          case 0xC5:
+          case 0xC6:
+          case 0xC7:
+          case 0xC9:
+          case 0xCA:
+          case 0xCB:
+          case 0xCD:
+          case 0xCE:
+          case 0xCF:
+          case 0xC4:
+          case 0xCC:
+          case 0xDA:
+          case 0xDB:
+          case 0xDC:
+          case 0xDD:
+          case 0xDE:
+          case 0xDF:
+          case 0xE0:
+          case 0xE1:
+          case 0xE2:
+          case 0xE3:
+          case 0xE4:
+          case 0xE5:
+          case 0xE6:
+          case 0xE7:
+          case 0xE8:
+          case 0xE9:
+          case 0xEA:
+          case 0xEB:
+          case 0xEC:
+          case 0xED:
+          case 0xEE:
+          case 0xEF:
+          case 0xFE:
+            markerLength = stream.getUint16();
+            if (markerLength > 2) {
+              stream.skip(markerLength - 2);
+            } else {
+              stream.skip(-2);
+            }
+            break;
+        }
+        if (foundEOI) {
+          break;
+        }
       }
-      if (j >= ENDSTREAM_SIGNATURE_LENGTH) {
-       found = true;
-       break;
+      length = stream.pos - startPos;
+      if (b === -1) {
+        warn('Inline DCTDecode image stream: ' + 'EOI marker not found, searching for /EI/ instead.');
+        stream.skip(-length);
+        return this.findDefaultInlineStreamEnd(stream);
       }
-      i++;
-     }
-     if (found) {
-      skipped += i;
-      stream.pos += i;
-      break;
-     }
-     skipped += scanLength;
-     stream.pos += scanLength;
-    }
-    if (!found) {
-     error('Missing endstream');
-    }
-    length = skipped;
-    lexer.nextChar();
-    this.shift();
-    this.shift();
-   }
-   this.shift();
-   stream = stream.makeSubStream(pos, length, dict);
-   if (cipherTransform) {
-    stream = cipherTransform.createStream(stream, length);
-   }
-   stream = this.filter(stream, dict, length);
-   stream.dict = dict;
-   return stream;
-  },
-  filter: function Parser_filter(stream, dict, length) {
-   var filter = dict.get('Filter', 'F');
-   var params = dict.get('DecodeParms', 'DP');
-   if (isName(filter)) {
-    if (isArray(params)) {
-     params = this.xref.fetchIfRef(params[0]);
-    }
-    return this.makeFilter(stream, filter.name, length, params);
-   }
-   var maybeLength = length;
-   if (isArray(filter)) {
-    var filterArray = filter;
-    var paramsArray = params;
-    for (var i = 0, ii = filterArray.length; i < ii; ++i) {
-     filter = this.xref.fetchIfRef(filterArray[i]);
-     if (!isName(filter)) {
-      error('Bad filter name: ' + filter);
-     }
-     params = null;
-     if (isArray(paramsArray) && i in paramsArray) {
-      params = this.xref.fetchIfRef(paramsArray[i]);
-     }
-     stream = this.makeFilter(stream, filter.name, maybeLength, params);
-     maybeLength = null;
-    }
-   }
-   return stream;
-  },
-  makeFilter: function Parser_makeFilter(stream, name, maybeLength, params) {
-   if (maybeLength === 0) {
-    warn('Empty "' + name + '" stream.');
-    return new NullStream(stream);
-   }
-   try {
-    var xrefStreamStats = this.xref.stats.streamTypes;
-    if (name === 'FlateDecode' || name === 'Fl') {
-     xrefStreamStats[StreamType.FLATE] = true;
-     if (params) {
-      return new PredictorStream(new FlateStream(stream, maybeLength), maybeLength, params);
-     }
-     return new FlateStream(stream, maybeLength);
-    }
-    if (name === 'LZWDecode' || name === 'LZW') {
-     xrefStreamStats[StreamType.LZW] = true;
-     var earlyChange = 1;
-     if (params) {
-      if (params.has('EarlyChange')) {
-       earlyChange = params.get('EarlyChange');
+      this.inlineStreamSkipEI(stream);
+      return length;
+    },
+    findASCII85DecodeInlineStreamEnd: function Parser_findASCII85DecodeInlineStreamEnd(stream) {
+      var TILDE = 0x7E,
+          GT = 0x3E;
+      var startPos = stream.pos,
+          ch,
+          length;
+      while ((ch = stream.getByte()) !== -1) {
+        if (ch === TILDE && stream.peekByte() === GT) {
+          stream.skip();
+          break;
+        }
+      }
+      length = stream.pos - startPos;
+      if (ch === -1) {
+        warn('Inline ASCII85Decode image stream: ' + 'EOD marker not found, searching for /EI/ instead.');
+        stream.skip(-length);
+        return this.findDefaultInlineStreamEnd(stream);
+      }
+      this.inlineStreamSkipEI(stream);
+      return length;
+    },
+    findASCIIHexDecodeInlineStreamEnd: function Parser_findASCIIHexDecodeInlineStreamEnd(stream) {
+      var GT = 0x3E;
+      var startPos = stream.pos,
+          ch,
+          length;
+      while ((ch = stream.getByte()) !== -1) {
+        if (ch === GT) {
+          break;
+        }
+      }
+      length = stream.pos - startPos;
+      if (ch === -1) {
+        warn('Inline ASCIIHexDecode image stream: ' + 'EOD marker not found, searching for /EI/ instead.');
+        stream.skip(-length);
+        return this.findDefaultInlineStreamEnd(stream);
+      }
+      this.inlineStreamSkipEI(stream);
+      return length;
+    },
+    inlineStreamSkipEI: function Parser_inlineStreamSkipEI(stream) {
+      var E = 0x45,
+          I = 0x49;
+      var state = 0,
+          ch;
+      while ((ch = stream.getByte()) !== -1) {
+        if (state === 0) {
+          state = ch === E ? 1 : 0;
+        } else if (state === 1) {
+          state = ch === I ? 2 : 0;
+        } else if (state === 2) {
+          break;
+        }
+      }
+    },
+    makeInlineImage: function Parser_makeInlineImage(cipherTransform) {
+      var lexer = this.lexer;
+      var stream = lexer.stream;
+      var dict = new Dict(this.xref);
+      while (!isCmd(this.buf1, 'ID') && !isEOF(this.buf1)) {
+        if (!isName(this.buf1)) {
+          error('Dictionary key must be a name object');
+        }
+        var key = this.buf1.name;
+        this.shift();
+        if (isEOF(this.buf1)) {
+          break;
+        }
+        dict.set(key, this.getObj(cipherTransform));
+      }
+      var filter = dict.get('Filter', 'F'),
+          filterName;
+      if (isName(filter)) {
+        filterName = filter.name;
+      } else if (isArray(filter)) {
+        var filterZero = this.xref.fetchIfRef(filter[0]);
+        if (isName(filterZero)) {
+          filterName = filterZero.name;
+        }
+      }
+      var startPos = stream.pos,
+          length,
+          i,
+          ii;
+      if (filterName === 'DCTDecode' || filterName === 'DCT') {
+        length = this.findDCTDecodeInlineStreamEnd(stream);
+      } else if (filterName === 'ASCII85Decide' || filterName === 'A85') {
+        length = this.findASCII85DecodeInlineStreamEnd(stream);
+      } else if (filterName === 'ASCIIHexDecode' || filterName === 'AHx') {
+        length = this.findASCIIHexDecodeInlineStreamEnd(stream);
+      } else {
+        length = this.findDefaultInlineStreamEnd(stream);
+      }
+      var imageStream = stream.makeSubStream(startPos, length, dict);
+      var adler32;
+      if (length < MAX_LENGTH_TO_CACHE) {
+        var imageBytes = imageStream.getBytes();
+        imageStream.reset();
+        var a = 1;
+        var b = 0;
+        for (i = 0, ii = imageBytes.length; i < ii; ++i) {
+          a += imageBytes[i] & 0xff;
+          b += a;
+        }
+        adler32 = b % 65521 << 16 | a % 65521;
+        if (this.imageCache.adler32 === adler32) {
+          this.buf2 = Cmd.get('EI');
+          this.shift();
+          this.imageCache[adler32].reset();
+          return this.imageCache[adler32];
+        }
+      }
+      if (cipherTransform) {
+        imageStream = cipherTransform.createStream(imageStream, length);
+      }
+      imageStream = this.filter(imageStream, dict, length);
+      imageStream.dict = dict;
+      if (adler32 !== undefined) {
+        imageStream.cacheKey = 'inline_' + length + '_' + adler32;
+        this.imageCache[adler32] = imageStream;
+      }
+      this.buf2 = Cmd.get('EI');
+      this.shift();
+      return imageStream;
+    },
+    makeStream: function Parser_makeStream(dict, cipherTransform) {
+      var lexer = this.lexer;
+      var stream = lexer.stream;
+      lexer.skipToNextLine();
+      var pos = stream.pos - 1;
+      var length = dict.get('Length');
+      if (!isInt(length)) {
+        info('Bad ' + length + ' attribute in stream');
+        length = 0;
+      }
+      stream.pos = pos + length;
+      lexer.nextChar();
+      if (this.tryShift() && isCmd(this.buf2, 'endstream')) {
+        this.shift();
+      } else {
+        stream.pos = pos;
+        var SCAN_BLOCK_SIZE = 2048;
+        var ENDSTREAM_SIGNATURE_LENGTH = 9;
+        var ENDSTREAM_SIGNATURE = [0x65, 0x6E, 0x64, 0x73, 0x74, 0x72, 0x65, 0x61, 0x6D];
+        var skipped = 0,
+            found = false,
+            i,
+            j;
+        while (stream.pos < stream.end) {
+          var scanBytes = stream.peekBytes(SCAN_BLOCK_SIZE);
+          var scanLength = scanBytes.length - ENDSTREAM_SIGNATURE_LENGTH;
+          if (scanLength <= 0) {
+            break;
+          }
+          found = false;
+          i = 0;
+          while (i < scanLength) {
+            j = 0;
+            while (j < ENDSTREAM_SIGNATURE_LENGTH && scanBytes[i + j] === ENDSTREAM_SIGNATURE[j]) {
+              j++;
+            }
+            if (j >= ENDSTREAM_SIGNATURE_LENGTH) {
+              found = true;
+              break;
+            }
+            i++;
+          }
+          if (found) {
+            skipped += i;
+            stream.pos += i;
+            break;
+          }
+          skipped += scanLength;
+          stream.pos += scanLength;
+        }
+        if (!found) {
+          error('Missing endstream');
+        }
+        length = skipped;
+        lexer.nextChar();
+        this.shift();
+        this.shift();
+      }
+      this.shift();
+      stream = stream.makeSubStream(pos, length, dict);
+      if (cipherTransform) {
+        stream = cipherTransform.createStream(stream, length);
+      }
+      stream = this.filter(stream, dict, length);
+      stream.dict = dict;
+      return stream;
+    },
+    filter: function Parser_filter(stream, dict, length) {
+      var filter = dict.get('Filter', 'F');
+      var params = dict.get('DecodeParms', 'DP');
+      if (isName(filter)) {
+        if (isArray(params)) {
+          params = this.xref.fetchIfRef(params[0]);
+        }
+        return this.makeFilter(stream, filter.name, length, params);
+      }
+      var maybeLength = length;
+      if (isArray(filter)) {
+        var filterArray = filter;
+        var paramsArray = params;
+        for (var i = 0, ii = filterArray.length; i < ii; ++i) {
+          filter = this.xref.fetchIfRef(filterArray[i]);
+          if (!isName(filter)) {
+            error('Bad filter name: ' + filter);
+          }
+          params = null;
+          if (isArray(paramsArray) && i in paramsArray) {
+            params = this.xref.fetchIfRef(paramsArray[i]);
+          }
+          stream = this.makeFilter(stream, filter.name, maybeLength, params);
+          maybeLength = null;
+        }
+      }
+      return stream;
+    },
+    makeFilter: function Parser_makeFilter(stream, name, maybeLength, params) {
+      if (maybeLength === 0) {
+        warn('Empty "' + name + '" stream.');
+        return new NullStream(stream);
+      }
+      try {
+        var xrefStreamStats = this.xref.stats.streamTypes;
+        if (name === 'FlateDecode' || name === 'Fl') {
+          xrefStreamStats[StreamType.FLATE] = true;
+          if (params) {
+            return new PredictorStream(new FlateStream(stream, maybeLength), maybeLength, params);
+          }
+          return new FlateStream(stream, maybeLength);
+        }
+        if (name === 'LZWDecode' || name === 'LZW') {
+          xrefStreamStats[StreamType.LZW] = true;
+          var earlyChange = 1;
+          if (params) {
+            if (params.has('EarlyChange')) {
+              earlyChange = params.get('EarlyChange');
+            }
+            return new PredictorStream(new LZWStream(stream, maybeLength, earlyChange), maybeLength, params);
+          }
+          return new LZWStream(stream, maybeLength, earlyChange);
+        }
+        if (name === 'DCTDecode' || name === 'DCT') {
+          xrefStreamStats[StreamType.DCT] = true;
+          return new JpegStream(stream, maybeLength, stream.dict, params);
+        }
+        if (name === 'JPXDecode' || name === 'JPX') {
+          xrefStreamStats[StreamType.JPX] = true;
+          return new JpxStream(stream, maybeLength, stream.dict, params);
+        }
+        if (name === 'ASCII85Decode' || name === 'A85') {
+          xrefStreamStats[StreamType.A85] = true;
+          return new Ascii85Stream(stream, maybeLength);
+        }
+        if (name === 'ASCIIHexDecode' || name === 'AHx') {
+          xrefStreamStats[StreamType.AHX] = true;
+          return new AsciiHexStream(stream, maybeLength);
+        }
+        if (name === 'CCITTFaxDecode' || name === 'CCF') {
+          xrefStreamStats[StreamType.CCF] = true;
+          return new CCITTFaxStream(stream, maybeLength, params);
+        }
+        if (name === 'RunLengthDecode' || name === 'RL') {
+          xrefStreamStats[StreamType.RL] = true;
+          return new RunLengthStream(stream, maybeLength);
+        }
+        if (name === 'JBIG2Decode') {
+          xrefStreamStats[StreamType.JBIG] = true;
+          return new Jbig2Stream(stream, maybeLength, stream.dict, params);
+        }
+        warn('filter "' + name + '" not supported yet');
+        return stream;
+      } catch (ex) {
+        if (ex instanceof MissingDataException) {
+          throw ex;
+        }
+        warn('Invalid stream: \"' + ex + '\"');
+        return new NullStream(stream);
       }
-      return new PredictorStream(new LZWStream(stream, maybeLength, earlyChange), maybeLength, params);
-     }
-     return new LZWStream(stream, maybeLength, earlyChange);
-    }
-    if (name === 'DCTDecode' || name === 'DCT') {
-     xrefStreamStats[StreamType.DCT] = true;
-     return new JpegStream(stream, maybeLength, stream.dict, params);
-    }
-    if (name === 'JPXDecode' || name === 'JPX') {
-     xrefStreamStats[StreamType.JPX] = true;
-     return new JpxStream(stream, maybeLength, stream.dict, params);
-    }
-    if (name === 'ASCII85Decode' || name === 'A85') {
-     xrefStreamStats[StreamType.A85] = true;
-     return new Ascii85Stream(stream, maybeLength);
-    }
-    if (name === 'ASCIIHexDecode' || name === 'AHx') {
-     xrefStreamStats[StreamType.AHX] = true;
-     return new AsciiHexStream(stream, maybeLength);
-    }
-    if (name === 'CCITTFaxDecode' || name === 'CCF') {
-     xrefStreamStats[StreamType.CCF] = true;
-     return new CCITTFaxStream(stream, maybeLength, params);
-    }
-    if (name === 'RunLengthDecode' || name === 'RL') {
-     xrefStreamStats[StreamType.RL] = true;
-     return new RunLengthStream(stream, maybeLength);
-    }
-    if (name === 'JBIG2Decode') {
-     xrefStreamStats[StreamType.JBIG] = true;
-     return new Jbig2Stream(stream, maybeLength, stream.dict, params);
-    }
-    warn('filter "' + name + '" not supported yet');
-    return stream;
-   } catch (ex) {
-    if (ex instanceof MissingDataException) {
-     throw ex;
     }
-    warn('Invalid stream: \"' + ex + '\"');
-    return new NullStream(stream);
-   }
-  }
- };
- return Parser;
+  };
+  return Parser;
 }();
 var Lexer = function LexerClosure() {
- function Lexer(stream, knownCommands) {
-  this.stream = stream;
-  this.nextChar();
-  this.strBuf = [];
-  this.knownCommands = knownCommands;
- }
- var specialChars = [
-  1,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  1,
-  1,
-  0,
-  1,
-  1,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  1,
-  0,
-  0,
-  0,
-  0,
-  2,
-  0,
-  0,
-  2,
-  2,
-  0,
-  0,
-  0,
-  0,
-  0,
-  2,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  2,
-  0,
-  2,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  2,
-  0,
-  2,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  2,
-  0,
-  2,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0,
-  0
- ];
- function toHexDigit(ch) {
-  if (ch >= 0x30 && ch <= 0x39) {
-   return ch & 0x0F;
-  }
-  if (ch >= 0x41 && ch <= 0x46 || ch >= 0x61 && ch <= 0x66) {
-   return (ch & 0x0F) + 9;
+  function Lexer(stream, knownCommands) {
+    this.stream = stream;
+    this.nextChar();
+    this.strBuf = [];
+    this.knownCommands = knownCommands;
   }
-  return -1;
- }
- Lexer.prototype = {
-  nextChar: function Lexer_nextChar() {
-   return this.currentChar = this.stream.getByte();
-  },
-  peekChar: function Lexer_peekChar() {
-   return this.stream.peekByte();
-  },
-  getNumber: function Lexer_getNumber() {
-   var ch = this.currentChar;
-   var eNotation = false;
-   var divideBy = 0;
-   var sign = 1;
-   if (ch === 0x2D) {
-    sign = -1;
-    ch = this.nextChar();
-    if (ch === 0x2D) {
-     ch = this.nextChar();
+  var specialChars = [1, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 2, 0, 0, 2, 2, 0, 0, 0, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0];
+  function toHexDigit(ch) {
+    if (ch >= 0x30 && ch <= 0x39) {
+      return ch & 0x0F;
+    }
+    if (ch >= 0x41 && ch <= 0x46 || ch >= 0x61 && ch <= 0x66) {
+      return (ch & 0x0F) + 9;
     }
-   } else if (ch === 0x2B) {
-    ch = this.nextChar();
-   }
-   if (ch === 0x2E) {
-    divideBy = 10;
-    ch = this.nextChar();
-   }
-   if (ch < 0x30 || ch > 0x39) {
-    error('Invalid number: ' + String.fromCharCode(ch));
-    return 0;
-   }
-   var baseValue = ch - 0x30;
-   var powerValue = 0;
-   var powerValueSign = 1;
-   while ((ch = this.nextChar()) >= 0) {
-    if (0x30 <= ch && ch <= 0x39) {
-     var currentDigit = ch - 0x30;
-     if (eNotation) {
-      powerValue = powerValue * 10 + currentDigit;
-     } else {
+    return -1;
+  }
+  Lexer.prototype = {
+    nextChar: function Lexer_nextChar() {
+      return this.currentChar = this.stream.getByte();
+    },
+    peekChar: function Lexer_peekChar() {
+      return this.stream.peekByte();
+    },
+    getNumber: function Lexer_getNumber() {
+      var ch = this.currentChar;
+      var eNotation = false;
+      var divideBy = 0;
+      var sign = 1;
+      if (ch === 0x2D) {
+        sign = -1;
+        ch = this.nextChar();
+        if (ch === 0x2D) {
+          ch = this.nextChar();
+        }
+      } else if (ch === 0x2B) {
+        ch = this.nextChar();
+      }
+      if (ch === 0x2E) {
+        divideBy = 10;
+        ch = this.nextChar();
+      }
+      if (ch < 0x30 || ch > 0x39) {
+        error('Invalid number: ' + String.fromCharCode(ch));
+        return 0;
+      }
+      var baseValue = ch - 0x30;
+      var powerValue = 0;
+      var powerValueSign = 1;
+      while ((ch = this.nextChar()) >= 0) {
+        if (0x30 <= ch && ch <= 0x39) {
+          var currentDigit = ch - 0x30;
+          if (eNotation) {
+            powerValue = powerValue * 10 + currentDigit;
+          } else {
+            if (divideBy !== 0) {
+              divideBy *= 10;
+            }
+            baseValue = baseValue * 10 + currentDigit;
+          }
+        } else if (ch === 0x2E) {
+          if (divideBy === 0) {
+            divideBy = 1;
+          } else {
+            break;
+          }
+        } else if (ch === 0x2D) {
+          warn('Badly formatted number');
+        } else if (ch === 0x45 || ch === 0x65) {
+          ch = this.peekChar();
+          if (ch === 0x2B || ch === 0x2D) {
+            powerValueSign = ch === 0x2D ? -1 : 1;
+            this.nextChar();
+          } else if (ch < 0x30 || ch > 0x39) {
+            break;
+          }
+          eNotation = true;
+        } else {
+          break;
+        }
+      }
       if (divideBy !== 0) {
-       divideBy *= 10;
+        baseValue /= divideBy;
       }
-      baseValue = baseValue * 10 + currentDigit;
-     }
-    } else if (ch === 0x2E) {
-     if (divideBy === 0) {
-      divideBy = 1;
-     } else {
-      break;
-     }
-    } else if (ch === 0x2D) {
-     warn('Badly formatted number');
-    } else if (ch === 0x45 || ch === 0x65) {
-     ch = this.peekChar();
-     if (ch === 0x2B || ch === 0x2D) {
-      powerValueSign = ch === 0x2D ? -1 : 1;
-      this.nextChar();
-     } else if (ch < 0x30 || ch > 0x39) {
-      break;
-     }
-     eNotation = true;
-    } else {
-     break;
-    }
-   }
-   if (divideBy !== 0) {
-    baseValue /= divideBy;
-   }
-   if (eNotation) {
-    baseValue *= Math.pow(10, powerValueSign * powerValue);
-   }
-   return sign * baseValue;
-  },
-  getString: function Lexer_getString() {
-   var numParen = 1;
-   var done = false;
-   var strBuf = this.strBuf;
-   strBuf.length = 0;
-   var ch = this.nextChar();
-   while (true) {
-    var charBuffered = false;
-    switch (ch | 0) {
-    case -1:
-     warn('Unterminated string');
-     done = true;
-     break;
-    case 0x28:
-     ++numParen;
-     strBuf.push('(');
-     break;
-    case 0x29:
-     if (--numParen === 0) {
-      this.nextChar();
-      done = true;
-     } else {
-      strBuf.push(')');
-     }
-     break;
-    case 0x5C:
-     ch = this.nextChar();
-     switch (ch) {
-     case -1:
-      warn('Unterminated string');
-      done = true;
-      break;
-     case 0x6E:
-      strBuf.push('\n');
-      break;
-     case 0x72:
-      strBuf.push('\r');
-      break;
-     case 0x74:
-      strBuf.push('\t');
-      break;
-     case 0x62:
-      strBuf.push('\b');
-      break;
-     case 0x66:
-      strBuf.push('\f');
-      break;
-     case 0x5C:
-     case 0x28:
-     case 0x29:
-      strBuf.push(String.fromCharCode(ch));
-      break;
-     case 0x30:
-     case 0x31:
-     case 0x32:
-     case 0x33:
-     case 0x34:
-     case 0x35:
-     case 0x36:
-     case 0x37:
-      var x = ch & 0x0F;
-      ch = this.nextChar();
-      charBuffered = true;
-      if (ch >= 0x30 && ch <= 0x37) {
-       x = (x << 3) + (ch & 0x0F);
-       ch = this.nextChar();
-       if (ch >= 0x30 && ch <= 0x37) {
-        charBuffered = false;
-        x = (x << 3) + (ch & 0x0F);
-       }
+      if (eNotation) {
+        baseValue *= Math.pow(10, powerValueSign * powerValue);
       }
-      strBuf.push(String.fromCharCode(x));
-      break;
-     case 0x0D:
-      if (this.peekChar() === 0x0A) {
-       this.nextChar();
+      return sign * baseValue;
+    },
+    getString: function Lexer_getString() {
+      var numParen = 1;
+      var done = false;
+      var strBuf = this.strBuf;
+      strBuf.length = 0;
+      var ch = this.nextChar();
+      while (true) {
+        var charBuffered = false;
+        switch (ch | 0) {
+          case -1:
+            warn('Unterminated string');
+            done = true;
+            break;
+          case 0x28:
+            ++numParen;
+            strBuf.push('(');
+            break;
+          case 0x29:
+            if (--numParen === 0) {
+              this.nextChar();
+              done = true;
+            } else {
+              strBuf.push(')');
+            }
+            break;
+          case 0x5C:
+            ch = this.nextChar();
+            switch (ch) {
+              case -1:
+                warn('Unterminated string');
+                done = true;
+                break;
+              case 0x6E:
+                strBuf.push('\n');
+                break;
+              case 0x72:
+                strBuf.push('\r');
+                break;
+              case 0x74:
+                strBuf.push('\t');
+                break;
+              case 0x62:
+                strBuf.push('\b');
+                break;
+              case 0x66:
+                strBuf.push('\f');
+                break;
+              case 0x5C:
+              case 0x28:
+              case 0x29:
+                strBuf.push(String.fromCharCode(ch));
+                break;
+              case 0x30:
+              case 0x31:
+              case 0x32:
+              case 0x33:
+              case 0x34:
+              case 0x35:
+              case 0x36:
+              case 0x37:
+                var x = ch & 0x0F;
+                ch = this.nextChar();
+                charBuffered = true;
+                if (ch >= 0x30 && ch <= 0x37) {
+                  x = (x << 3) + (ch & 0x0F);
+                  ch = this.nextChar();
+                  if (ch >= 0x30 && ch <= 0x37) {
+                    charBuffered = false;
+                    x = (x << 3) + (ch & 0x0F);
+                  }
+                }
+                strBuf.push(String.fromCharCode(x));
+                break;
+              case 0x0D:
+                if (this.peekChar() === 0x0A) {
+                  this.nextChar();
+                }
+                break;
+              case 0x0A:
+                break;
+              default:
+                strBuf.push(String.fromCharCode(ch));
+                break;
+            }
+            break;
+          default:
+            strBuf.push(String.fromCharCode(ch));
+            break;
+        }
+        if (done) {
+          break;
+        }
+        if (!charBuffered) {
+          ch = this.nextChar();
+        }
       }
-      break;
-     case 0x0A:
-      break;
-     default:
-      strBuf.push(String.fromCharCode(ch));
-      break;
-     }
-     break;
-    default:
-     strBuf.push(String.fromCharCode(ch));
-     break;
-    }
-    if (done) {
-     break;
-    }
-    if (!charBuffered) {
-     ch = this.nextChar();
-    }
-   }
-   return strBuf.join('');
-  },
-  getName: function Lexer_getName() {
-   var ch, previousCh;
-   var strBuf = this.strBuf;
-   strBuf.length = 0;
-   while ((ch = this.nextChar()) >= 0 && !specialChars[ch]) {
-    if (ch === 0x23) {
-     ch = this.nextChar();
-     if (specialChars[ch]) {
-      warn('Lexer_getName: ' + 'NUMBER SIGN (#) should be followed by a hexadecimal number.');
-      strBuf.push('#');
-      break;
-     }
-     var x = toHexDigit(ch);
-     if (x !== -1) {
-      previousCh = ch;
-      ch = this.nextChar();
-      var x2 = toHexDigit(ch);
-      if (x2 === -1) {
-       warn('Lexer_getName: Illegal digit (' + String.fromCharCode(ch) + ') in hexadecimal number.');
-       strBuf.push('#', String.fromCharCode(previousCh));
-       if (specialChars[ch]) {
-        break;
-       }
-       strBuf.push(String.fromCharCode(ch));
-       continue;
+      return strBuf.join('');
+    },
+    getName: function Lexer_getName() {
+      var ch, previousCh;
+      var strBuf = this.strBuf;
+      strBuf.length = 0;
+      while ((ch = this.nextChar()) >= 0 && !specialChars[ch]) {
+        if (ch === 0x23) {
+          ch = this.nextChar();
+          if (specialChars[ch]) {
+            warn('Lexer_getName: ' + 'NUMBER SIGN (#) should be followed by a hexadecimal number.');
+            strBuf.push('#');
+            break;
+          }
+          var x = toHexDigit(ch);
+          if (x !== -1) {
+            previousCh = ch;
+            ch = this.nextChar();
+            var x2 = toHexDigit(ch);
+            if (x2 === -1) {
+              warn('Lexer_getName: Illegal digit (' + String.fromCharCode(ch) + ') in hexadecimal number.');
+              strBuf.push('#', String.fromCharCode(previousCh));
+              if (specialChars[ch]) {
+                break;
+              }
+              strBuf.push(String.fromCharCode(ch));
+              continue;
+            }
+            strBuf.push(String.fromCharCode(x << 4 | x2));
+          } else {
+            strBuf.push('#', String.fromCharCode(ch));
+          }
+        } else {
+          strBuf.push(String.fromCharCode(ch));
+        }
       }
-      strBuf.push(String.fromCharCode(x << 4 | x2));
-     } else {
-      strBuf.push('#', String.fromCharCode(ch));
-     }
-    } else {
-     strBuf.push(String.fromCharCode(ch));
-    }
-   }
-   if (strBuf.length > 127) {
-    warn('name token is longer than allowed by the spec: ' + strBuf.length);
-   }
-   return Name.get(strBuf.join(''));
-  },
-  getHexString: function Lexer_getHexString() {
-   var strBuf = this.strBuf;
-   strBuf.length = 0;
-   var ch = this.currentChar;
-   var isFirstHex = true;
-   var firstDigit;
-   var secondDigit;
-   while (true) {
-    if (ch < 0) {
-     warn('Unterminated hex string');
-     break;
-    } else if (ch === 0x3E) {
-     this.nextChar();
-     break;
-    } else if (specialChars[ch] === 1) {
-     ch = this.nextChar();
-     continue;
-    } else {
-     if (isFirstHex) {
-      firstDigit = toHexDigit(ch);
-      if (firstDigit === -1) {
-       warn('Ignoring invalid character "' + ch + '" in hex string');
-       ch = this.nextChar();
-       continue;
+      if (strBuf.length > 127) {
+        warn('name token is longer than allowed by the spec: ' + strBuf.length);
       }
-     } else {
-      secondDigit = toHexDigit(ch);
-      if (secondDigit === -1) {
-       warn('Ignoring invalid character "' + ch + '" in hex string');
-       ch = this.nextChar();
-       continue;
+      return Name.get(strBuf.join(''));
+    },
+    getHexString: function Lexer_getHexString() {
+      var strBuf = this.strBuf;
+      strBuf.length = 0;
+      var ch = this.currentChar;
+      var isFirstHex = true;
+      var firstDigit;
+      var secondDigit;
+      while (true) {
+        if (ch < 0) {
+          warn('Unterminated hex string');
+          break;
+        } else if (ch === 0x3E) {
+          this.nextChar();
+          break;
+        } else if (specialChars[ch] === 1) {
+          ch = this.nextChar();
+          continue;
+        } else {
+          if (isFirstHex) {
+            firstDigit = toHexDigit(ch);
+            if (firstDigit === -1) {
+              warn('Ignoring invalid character "' + ch + '" in hex string');
+              ch = this.nextChar();
+              continue;
+            }
+          } else {
+            secondDigit = toHexDigit(ch);
+            if (secondDigit === -1) {
+              warn('Ignoring invalid character "' + ch + '" in hex string');
+              ch = this.nextChar();
+              continue;
+            }
+            strBuf.push(String.fromCharCode(firstDigit << 4 | secondDigit));
+          }
+          isFirstHex = !isFirstHex;
+          ch = this.nextChar();
+        }
+      }
+      return strBuf.join('');
+    },
+    getObj: function Lexer_getObj() {
+      var comment = false;
+      var ch = this.currentChar;
+      while (true) {
+        if (ch < 0) {
+          return EOF;
+        }
+        if (comment) {
+          if (ch === 0x0A || ch === 0x0D) {
+            comment = false;
+          }
+        } else if (ch === 0x25) {
+          comment = true;
+        } else if (specialChars[ch] !== 1) {
+          break;
+        }
+        ch = this.nextChar();
+      }
+      switch (ch | 0) {
+        case 0x30:
+        case 0x31:
+        case 0x32:
+        case 0x33:
+        case 0x34:
+        case 0x35:
+        case 0x36:
+        case 0x37:
+        case 0x38:
+        case 0x39:
+        case 0x2B:
+        case 0x2D:
+        case 0x2E:
+          return this.getNumber();
+        case 0x28:
+          return this.getString();
+        case 0x2F:
+          return this.getName();
+        case 0x5B:
+          this.nextChar();
+          return Cmd.get('[');
+        case 0x5D:
+          this.nextChar();
+          return Cmd.get(']');
+        case 0x3C:
+          ch = this.nextChar();
+          if (ch === 0x3C) {
+            this.nextChar();
+            return Cmd.get('<<');
+          }
+          return this.getHexString();
+        case 0x3E:
+          ch = this.nextChar();
+          if (ch === 0x3E) {
+            this.nextChar();
+            return Cmd.get('>>');
+          }
+          return Cmd.get('>');
+        case 0x7B:
+          this.nextChar();
+          return Cmd.get('{');
+        case 0x7D:
+          this.nextChar();
+          return Cmd.get('}');
+        case 0x29:
+          this.nextChar();
+          error('Illegal character: ' + ch);
+          break;
+      }
+      var str = String.fromCharCode(ch);
+      var knownCommands = this.knownCommands;
+      var knownCommandFound = knownCommands && knownCommands[str] !== undefined;
+      while ((ch = this.nextChar()) >= 0 && !specialChars[ch]) {
+        var possibleCommand = str + String.fromCharCode(ch);
+        if (knownCommandFound && knownCommands[possibleCommand] === undefined) {
+          break;
+        }
+        if (str.length === 128) {
+          error('Command token too long: ' + str.length);
+        }
+        str = possibleCommand;
+        knownCommandFound = knownCommands && knownCommands[str] !== undefined;
+      }
+      if (str === 'true') {
+        return true;
+      }
+      if (str === 'false') {
+        return false;
+      }
+      if (str === 'null') {
+        return null;
+      }
+      return Cmd.get(str);
+    },
+    skipToNextLine: function Lexer_skipToNextLine() {
+      var ch = this.currentChar;
+      while (ch >= 0) {
+        if (ch === 0x0D) {
+          ch = this.nextChar();
+          if (ch === 0x0A) {
+            this.nextChar();
+          }
+          break;
+        } else if (ch === 0x0A) {
+          this.nextChar();
+          break;
+        }
+        ch = this.nextChar();
       }
-      strBuf.push(String.fromCharCode(firstDigit << 4 | secondDigit));
-     }
-     isFirstHex = !isFirstHex;
-     ch = this.nextChar();
-    }
-   }
-   return strBuf.join('');
-  },
-  getObj: function Lexer_getObj() {
-   var comment = false;
-   var ch = this.currentChar;
-   while (true) {
-    if (ch < 0) {
-     return EOF;
-    }
-    if (comment) {
-     if (ch === 0x0A || ch === 0x0D) {
-      comment = false;
-     }
-    } else if (ch === 0x25) {
-     comment = true;
-    } else if (specialChars[ch] !== 1) {
-     break;
-    }
-    ch = this.nextChar();
-   }
-   switch (ch | 0) {
-   case 0x30:
-   case 0x31:
-   case 0x32:
-   case 0x33:
-   case 0x34:
-   case 0x35:
-   case 0x36:
-   case 0x37:
-   case 0x38:
-   case 0x39:
-   case 0x2B:
-   case 0x2D:
-   case 0x2E:
-    return this.getNumber();
-   case 0x28:
-    return this.getString();
-   case 0x2F:
-    return this.getName();
-   case 0x5B:
-    this.nextChar();
-    return Cmd.get('[');
-   case 0x5D:
-    this.nextChar();
-    return Cmd.get(']');
-   case 0x3C:
-    ch = this.nextChar();
-    if (ch === 0x3C) {
-     this.nextChar();
-     return Cmd.get('<<');
-    }
-    return this.getHexString();
-   case 0x3E:
-    ch = this.nextChar();
-    if (ch === 0x3E) {
-     this.nextChar();
-     return Cmd.get('>>');
-    }
-    return Cmd.get('>');
-   case 0x7B:
-    this.nextChar();
-    return Cmd.get('{');
-   case 0x7D:
-    this.nextChar();
-    return Cmd.get('}');
-   case 0x29:
-    this.nextChar();
-    error('Illegal character: ' + ch);
-    break;
-   }
-   var str = String.fromCharCode(ch);
-   var knownCommands = this.knownCommands;
-   var knownCommandFound = knownCommands && knownCommands[str] !== undefined;
-   while ((ch = this.nextChar()) >= 0 && !specialChars[ch]) {
-    var possibleCommand = str + String.fromCharCode(ch);
-    if (knownCommandFound && knownCommands[possibleCommand] === undefined) {
-     break;
-    }
-    if (str.length === 128) {
-     error('Command token too long: ' + str.length);
-    }
-    str = possibleCommand;
-    knownCommandFound = knownCommands && knownCommands[str] !== undefined;
-   }
-   if (str === 'true') {
-    return true;
-   }
-   if (str === 'false') {
-    return false;
-   }
-   if (str === 'null') {
-    return null;
-   }
-   return Cmd.get(str);
-  },
-  skipToNextLine: function Lexer_skipToNextLine() {
-   var ch = this.currentChar;
-   while (ch >= 0) {
-    if (ch === 0x0D) {
-     ch = this.nextChar();
-     if (ch === 0x0A) {
-      this.nextChar();
-     }
-     break;
-    } else if (ch === 0x0A) {
-     this.nextChar();
-     break;
     }
-    ch = this.nextChar();
-   }
-  }
- };
- return Lexer;
+  };
+  return Lexer;
 }();
 var Linearization = {
- create: function LinearizationCreate(stream) {
-  function getInt(name, allowZeroValue) {
-   var obj = linDict.get(name);
-   if (isInt(obj) && (allowZeroValue ? obj >= 0 : obj > 0)) {
-    return obj;
-   }
-   throw new Error('The "' + name + '" parameter in the linearization ' + 'dictionary is invalid.');
-  }
-  function getHints() {
-   var hints = linDict.get('H'), hintsLength, item;
-   if (isArray(hints) && ((hintsLength = hints.length) === 2 || hintsLength === 4)) {
-    for (var index = 0; index < hintsLength; index++) {
-     if (!(isInt(item = hints[index]) && item > 0)) {
-      throw new Error('Hint (' + index + ') in the linearization dictionary is invalid.');
-     }
-    }
-    return hints;
-   }
-   throw new Error('Hint array in the linearization dictionary is invalid.');
-  }
-  var parser = new Parser(new Lexer(stream), false, null);
-  var obj1 = parser.getObj();
-  var obj2 = parser.getObj();
-  var obj3 = parser.getObj();
-  var linDict = parser.getObj();
-  var obj, length;
-  if (!(isInt(obj1) && isInt(obj2) && isCmd(obj3, 'obj') && isDict(linDict) && isNum(obj = linDict.get('Linearized')) && obj > 0)) {
-   return null;
-  } else if ((length = getInt('L')) !== stream.length) {
-   throw new Error('The "L" parameter in the linearization dictionary ' + 'does not equal the stream length.');
+  create: function LinearizationCreate(stream) {
+    function getInt(name, allowZeroValue) {
+      var obj = linDict.get(name);
+      if (isInt(obj) && (allowZeroValue ? obj >= 0 : obj > 0)) {
+        return obj;
+      }
+      throw new Error('The "' + name + '" parameter in the linearization ' + 'dictionary is invalid.');
+    }
+    function getHints() {
+      var hints = linDict.get('H'),
+          hintsLength,
+          item;
+      if (isArray(hints) && ((hintsLength = hints.length) === 2 || hintsLength === 4)) {
+        for (var index = 0; index < hintsLength; index++) {
+          if (!(isInt(item = hints[index]) && item > 0)) {
+            throw new Error('Hint (' + index + ') in the linearization dictionary is invalid.');
+          }
+        }
+        return hints;
+      }
+      throw new Error('Hint array in the linearization dictionary is invalid.');
+    }
+    var parser = new Parser(new Lexer(stream), false, null);
+    var obj1 = parser.getObj();
+    var obj2 = parser.getObj();
+    var obj3 = parser.getObj();
+    var linDict = parser.getObj();
+    var obj, length;
+    if (!(isInt(obj1) && isInt(obj2) && isCmd(obj3, 'obj') && isDict(linDict) && isNum(obj = linDict.get('Linearized')) && obj > 0)) {
+      return null;
+    } else if ((length = getInt('L')) !== stream.length) {
+      throw new Error('The "L" parameter in the linearization dictionary ' + 'does not equal the stream length.');
+    }
+    return {
+      length: length,
+      hints: getHints(),
+      objectNumberFirst: getInt('O'),
+      endFirst: getInt('E'),
+      numPages: getInt('N'),
+      mainXRefEntriesOffset: getInt('T'),
+      pageFirst: linDict.has('P') ? getInt('P', true) : 0
+    };
   }
-  return {
-   length: length,
-   hints: getHints(),
-   objectNumberFirst: getInt('O'),
-   endFirst: getInt('E'),
-   numPages: getInt('N'),
-   mainXRefEntriesOffset: getInt('T'),
-   pageFirst: linDict.has('P') ? getInt('P', true) : 0
-  };
- }
 };
 exports.Lexer = Lexer;
 exports.Linearization = Linearization;

+ 726 - 785
lib/core/pattern.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var coreFunction = require('./function.js');
@@ -28,820 +29,760 @@ var isStream = corePrimitives.isStream;
 var PDFFunction = coreFunction.PDFFunction;
 var ColorSpace = coreColorSpace.ColorSpace;
 var ShadingType = {
- FUNCTION_BASED: 1,
- AXIAL: 2,
- RADIAL: 3,
- FREE_FORM_MESH: 4,
- LATTICE_FORM_MESH: 5,
- COONS_PATCH_MESH: 6,
- TENSOR_PATCH_MESH: 7
+  FUNCTION_BASED: 1,
+  AXIAL: 2,
+  RADIAL: 3,
+  FREE_FORM_MESH: 4,
+  LATTICE_FORM_MESH: 5,
+  COONS_PATCH_MESH: 6,
+  TENSOR_PATCH_MESH: 7
 };
 var Pattern = function PatternClosure() {
- function Pattern() {
-  error('should not call Pattern constructor');
- }
- Pattern.prototype = {
-  getPattern: function Pattern_getPattern(ctx) {
-   error('Should not call Pattern.getStyle: ' + ctx);
+  function Pattern() {
+    error('should not call Pattern constructor');
   }
- };
- Pattern.parseShading = function Pattern_parseShading(shading, matrix, xref, res, handler) {
-  var dict = isStream(shading) ? shading.dict : shading;
-  var type = dict.get('ShadingType');
-  try {
-   switch (type) {
-   case ShadingType.AXIAL:
-   case ShadingType.RADIAL:
-    return new Shadings.RadialAxial(dict, matrix, xref, res);
-   case ShadingType.FREE_FORM_MESH:
-   case ShadingType.LATTICE_FORM_MESH:
-   case ShadingType.COONS_PATCH_MESH:
-   case ShadingType.TENSOR_PATCH_MESH:
-    return new Shadings.Mesh(shading, matrix, xref, res);
-   default:
-    throw new Error('Unsupported ShadingType: ' + type);
-   }
-  } catch (ex) {
-   if (ex instanceof MissingDataException) {
-    throw ex;
-   }
-   handler.send('UnsupportedFeature', { featureId: UNSUPPORTED_FEATURES.shadingPattern });
-   warn(ex);
-   return new Shadings.Dummy();
-  }
- };
- return Pattern;
+  Pattern.prototype = {
+    getPattern: function Pattern_getPattern(ctx) {
+      error('Should not call Pattern.getStyle: ' + ctx);
+    }
+  };
+  Pattern.parseShading = function Pattern_parseShading(shading, matrix, xref, res, handler) {
+    var dict = isStream(shading) ? shading.dict : shading;
+    var type = dict.get('ShadingType');
+    try {
+      switch (type) {
+        case ShadingType.AXIAL:
+        case ShadingType.RADIAL:
+          return new Shadings.RadialAxial(dict, matrix, xref, res);
+        case ShadingType.FREE_FORM_MESH:
+        case ShadingType.LATTICE_FORM_MESH:
+        case ShadingType.COONS_PATCH_MESH:
+        case ShadingType.TENSOR_PATCH_MESH:
+          return new Shadings.Mesh(shading, matrix, xref, res);
+        default:
+          throw new Error('Unsupported ShadingType: ' + type);
+      }
+    } catch (ex) {
+      if (ex instanceof MissingDataException) {
+        throw ex;
+      }
+      handler.send('UnsupportedFeature', { featureId: UNSUPPORTED_FEATURES.shadingPattern });
+      warn(ex);
+      return new Shadings.Dummy();
+    }
+  };
+  return Pattern;
 }();
 var Shadings = {};
 Shadings.SMALL_NUMBER = 1e-6;
 Shadings.RadialAxial = function RadialAxialClosure() {
- function RadialAxial(dict, matrix, xref, res) {
-  this.matrix = matrix;
-  this.coordsArr = dict.getArray('Coords');
-  this.shadingType = dict.get('ShadingType');
-  this.type = 'Pattern';
-  var cs = dict.get('ColorSpace', 'CS');
-  cs = ColorSpace.parse(cs, xref, res);
-  this.cs = cs;
-  var t0 = 0.0, t1 = 1.0;
-  if (dict.has('Domain')) {
-   var domainArr = dict.getArray('Domain');
-   t0 = domainArr[0];
-   t1 = domainArr[1];
-  }
-  var extendStart = false, extendEnd = false;
-  if (dict.has('Extend')) {
-   var extendArr = dict.getArray('Extend');
-   extendStart = extendArr[0];
-   extendEnd = extendArr[1];
-  }
-  if (this.shadingType === ShadingType.RADIAL && (!extendStart || !extendEnd)) {
-   var x1 = this.coordsArr[0];
-   var y1 = this.coordsArr[1];
-   var r1 = this.coordsArr[2];
-   var x2 = this.coordsArr[3];
-   var y2 = this.coordsArr[4];
-   var r2 = this.coordsArr[5];
-   var distance = Math.sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2));
-   if (r1 <= r2 + distance && r2 <= r1 + distance) {
-    warn('Unsupported radial gradient.');
-   }
-  }
-  this.extendStart = extendStart;
-  this.extendEnd = extendEnd;
-  var fnObj = dict.get('Function');
-  var fn = PDFFunction.parseArray(xref, fnObj);
-  var diff = t1 - t0;
-  var step = diff / 10;
-  var colorStops = this.colorStops = [];
-  if (t0 >= t1 || step <= 0) {
-   info('Bad shading domain.');
-   return;
-  }
-  var color = new Float32Array(cs.numComps), ratio = new Float32Array(1);
-  var rgbColor;
-  for (var i = t0; i <= t1; i += step) {
-   ratio[0] = i;
-   fn(ratio, 0, color, 0);
-   rgbColor = cs.getRgb(color, 0);
-   var cssColor = Util.makeCssRgb(rgbColor[0], rgbColor[1], rgbColor[2]);
-   colorStops.push([
-    (i - t0) / diff,
-    cssColor
-   ]);
-  }
-  var background = 'transparent';
-  if (dict.has('Background')) {
-   rgbColor = cs.getRgb(dict.get('Background'), 0);
-   background = Util.makeCssRgb(rgbColor[0], rgbColor[1], rgbColor[2]);
-  }
-  if (!extendStart) {
-   colorStops.unshift([
-    0,
-    background
-   ]);
-   colorStops[1][0] += Shadings.SMALL_NUMBER;
-  }
-  if (!extendEnd) {
-   colorStops[colorStops.length - 1][0] -= Shadings.SMALL_NUMBER;
-   colorStops.push([
-    1,
-    background
-   ]);
-  }
-  this.colorStops = colorStops;
- }
- RadialAxial.prototype = {
-  getIR: function RadialAxial_getIR() {
-   var coordsArr = this.coordsArr;
-   var shadingType = this.shadingType;
-   var type, p0, p1, r0, r1;
-   if (shadingType === ShadingType.AXIAL) {
-    p0 = [
-     coordsArr[0],
-     coordsArr[1]
-    ];
-    p1 = [
-     coordsArr[2],
-     coordsArr[3]
-    ];
-    r0 = null;
-    r1 = null;
-    type = 'axial';
-   } else if (shadingType === ShadingType.RADIAL) {
-    p0 = [
-     coordsArr[0],
-     coordsArr[1]
-    ];
-    p1 = [
-     coordsArr[3],
-     coordsArr[4]
-    ];
-    r0 = coordsArr[2];
-    r1 = coordsArr[5];
-    type = 'radial';
-   } else {
-    error('getPattern type unknown: ' + shadingType);
-   }
-   var matrix = this.matrix;
-   if (matrix) {
-    p0 = Util.applyTransform(p0, matrix);
-    p1 = Util.applyTransform(p1, matrix);
-    if (shadingType === ShadingType.RADIAL) {
-     var scale = Util.singularValueDecompose2dScale(matrix);
-     r0 *= scale[0];
-     r1 *= scale[1];
+  function RadialAxial(dict, matrix, xref, res) {
+    this.matrix = matrix;
+    this.coordsArr = dict.getArray('Coords');
+    this.shadingType = dict.get('ShadingType');
+    this.type = 'Pattern';
+    var cs = dict.get('ColorSpace', 'CS');
+    cs = ColorSpace.parse(cs, xref, res);
+    this.cs = cs;
+    var t0 = 0.0,
+        t1 = 1.0;
+    if (dict.has('Domain')) {
+      var domainArr = dict.getArray('Domain');
+      t0 = domainArr[0];
+      t1 = domainArr[1];
+    }
+    var extendStart = false,
+        extendEnd = false;
+    if (dict.has('Extend')) {
+      var extendArr = dict.getArray('Extend');
+      extendStart = extendArr[0];
+      extendEnd = extendArr[1];
+    }
+    if (this.shadingType === ShadingType.RADIAL && (!extendStart || !extendEnd)) {
+      var x1 = this.coordsArr[0];
+      var y1 = this.coordsArr[1];
+      var r1 = this.coordsArr[2];
+      var x2 = this.coordsArr[3];
+      var y2 = this.coordsArr[4];
+      var r2 = this.coordsArr[5];
+      var distance = Math.sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2));
+      if (r1 <= r2 + distance && r2 <= r1 + distance) {
+        warn('Unsupported radial gradient.');
+      }
+    }
+    this.extendStart = extendStart;
+    this.extendEnd = extendEnd;
+    var fnObj = dict.get('Function');
+    var fn = PDFFunction.parseArray(xref, fnObj);
+    var diff = t1 - t0;
+    var step = diff / 10;
+    var colorStops = this.colorStops = [];
+    if (t0 >= t1 || step <= 0) {
+      info('Bad shading domain.');
+      return;
+    }
+    var color = new Float32Array(cs.numComps),
+        ratio = new Float32Array(1);
+    var rgbColor;
+    for (var i = t0; i <= t1; i += step) {
+      ratio[0] = i;
+      fn(ratio, 0, color, 0);
+      rgbColor = cs.getRgb(color, 0);
+      var cssColor = Util.makeCssRgb(rgbColor[0], rgbColor[1], rgbColor[2]);
+      colorStops.push([(i - t0) / diff, cssColor]);
     }
-   }
-   return [
-    'RadialAxial',
-    type,
-    this.colorStops,
-    p0,
-    p1,
-    r0,
-    r1
-   ];
+    var background = 'transparent';
+    if (dict.has('Background')) {
+      rgbColor = cs.getRgb(dict.get('Background'), 0);
+      background = Util.makeCssRgb(rgbColor[0], rgbColor[1], rgbColor[2]);
+    }
+    if (!extendStart) {
+      colorStops.unshift([0, background]);
+      colorStops[1][0] += Shadings.SMALL_NUMBER;
+    }
+    if (!extendEnd) {
+      colorStops[colorStops.length - 1][0] -= Shadings.SMALL_NUMBER;
+      colorStops.push([1, background]);
+    }
+    this.colorStops = colorStops;
   }
- };
- return RadialAxial;
+  RadialAxial.prototype = {
+    getIR: function RadialAxial_getIR() {
+      var coordsArr = this.coordsArr;
+      var shadingType = this.shadingType;
+      var type, p0, p1, r0, r1;
+      if (shadingType === ShadingType.AXIAL) {
+        p0 = [coordsArr[0], coordsArr[1]];
+        p1 = [coordsArr[2], coordsArr[3]];
+        r0 = null;
+        r1 = null;
+        type = 'axial';
+      } else if (shadingType === ShadingType.RADIAL) {
+        p0 = [coordsArr[0], coordsArr[1]];
+        p1 = [coordsArr[3], coordsArr[4]];
+        r0 = coordsArr[2];
+        r1 = coordsArr[5];
+        type = 'radial';
+      } else {
+        error('getPattern type unknown: ' + shadingType);
+      }
+      var matrix = this.matrix;
+      if (matrix) {
+        p0 = Util.applyTransform(p0, matrix);
+        p1 = Util.applyTransform(p1, matrix);
+        if (shadingType === ShadingType.RADIAL) {
+          var scale = Util.singularValueDecompose2dScale(matrix);
+          r0 *= scale[0];
+          r1 *= scale[1];
+        }
+      }
+      return ['RadialAxial', type, this.colorStops, p0, p1, r0, r1];
+    }
+  };
+  return RadialAxial;
 }();
 Shadings.Mesh = function MeshClosure() {
- function MeshStreamReader(stream, context) {
-  this.stream = stream;
-  this.context = context;
-  this.buffer = 0;
-  this.bufferLength = 0;
-  var numComps = context.numComps;
-  this.tmpCompsBuf = new Float32Array(numComps);
-  var csNumComps = context.colorSpace.numComps;
-  this.tmpCsCompsBuf = context.colorFn ? new Float32Array(csNumComps) : this.tmpCompsBuf;
- }
- MeshStreamReader.prototype = {
-  get hasData() {
-   if (this.stream.end) {
-    return this.stream.pos < this.stream.end;
-   }
-   if (this.bufferLength > 0) {
-    return true;
-   }
-   var nextByte = this.stream.getByte();
-   if (nextByte < 0) {
-    return false;
-   }
-   this.buffer = nextByte;
-   this.bufferLength = 8;
-   return true;
-  },
-  readBits: function MeshStreamReader_readBits(n) {
-   var buffer = this.buffer;
-   var bufferLength = this.bufferLength;
-   if (n === 32) {
-    if (bufferLength === 0) {
-     return (this.stream.getByte() << 24 | this.stream.getByte() << 16 | this.stream.getByte() << 8 | this.stream.getByte()) >>> 0;
-    }
-    buffer = buffer << 24 | this.stream.getByte() << 16 | this.stream.getByte() << 8 | this.stream.getByte();
-    var nextByte = this.stream.getByte();
-    this.buffer = nextByte & (1 << bufferLength) - 1;
-    return (buffer << 8 - bufferLength | (nextByte & 0xFF) >> bufferLength) >>> 0;
-   }
-   if (n === 8 && bufferLength === 0) {
-    return this.stream.getByte();
-   }
-   while (bufferLength < n) {
-    buffer = buffer << 8 | this.stream.getByte();
-    bufferLength += 8;
-   }
-   bufferLength -= n;
-   this.bufferLength = bufferLength;
-   this.buffer = buffer & (1 << bufferLength) - 1;
-   return buffer >> bufferLength;
-  },
-  align: function MeshStreamReader_align() {
-   this.buffer = 0;
-   this.bufferLength = 0;
-  },
-  readFlag: function MeshStreamReader_readFlag() {
-   return this.readBits(this.context.bitsPerFlag);
-  },
-  readCoordinate: function MeshStreamReader_readCoordinate() {
-   var bitsPerCoordinate = this.context.bitsPerCoordinate;
-   var xi = this.readBits(bitsPerCoordinate);
-   var yi = this.readBits(bitsPerCoordinate);
-   var decode = this.context.decode;
-   var scale = bitsPerCoordinate < 32 ? 1 / ((1 << bitsPerCoordinate) - 1) : 2.3283064365386963e-10;
-   return [
-    xi * scale * (decode[1] - decode[0]) + decode[0],
-    yi * scale * (decode[3] - decode[2]) + decode[2]
-   ];
-  },
-  readComponents: function MeshStreamReader_readComponents() {
-   var numComps = this.context.numComps;
-   var bitsPerComponent = this.context.bitsPerComponent;
-   var scale = bitsPerComponent < 32 ? 1 / ((1 << bitsPerComponent) - 1) : 2.3283064365386963e-10;
-   var decode = this.context.decode;
-   var components = this.tmpCompsBuf;
-   for (var i = 0, j = 4; i < numComps; i++, j += 2) {
-    var ci = this.readBits(bitsPerComponent);
-    components[i] = ci * scale * (decode[j + 1] - decode[j]) + decode[j];
-   }
-   var color = this.tmpCsCompsBuf;
-   if (this.context.colorFn) {
-    this.context.colorFn(components, 0, color, 0);
-   }
-   return this.context.colorSpace.getRgb(color, 0);
+  function MeshStreamReader(stream, context) {
+    this.stream = stream;
+    this.context = context;
+    this.buffer = 0;
+    this.bufferLength = 0;
+    var numComps = context.numComps;
+    this.tmpCompsBuf = new Float32Array(numComps);
+    var csNumComps = context.colorSpace.numComps;
+    this.tmpCsCompsBuf = context.colorFn ? new Float32Array(csNumComps) : this.tmpCompsBuf;
   }
- };
- function decodeType4Shading(mesh, reader) {
-  var coords = mesh.coords;
-  var colors = mesh.colors;
-  var operators = [];
-  var ps = [];
-  var verticesLeft = 0;
-  while (reader.hasData) {
-   var f = reader.readFlag();
-   var coord = reader.readCoordinate();
-   var color = reader.readComponents();
-   if (verticesLeft === 0) {
-    assert(0 <= f && f <= 2, 'Unknown type4 flag');
-    switch (f) {
-    case 0:
-     verticesLeft = 3;
-     break;
-    case 1:
-     ps.push(ps[ps.length - 2], ps[ps.length - 1]);
-     verticesLeft = 1;
-     break;
-    case 2:
-     ps.push(ps[ps.length - 3], ps[ps.length - 1]);
-     verticesLeft = 1;
-     break;
+  MeshStreamReader.prototype = {
+    get hasData() {
+      if (this.stream.end) {
+        return this.stream.pos < this.stream.end;
+      }
+      if (this.bufferLength > 0) {
+        return true;
+      }
+      var nextByte = this.stream.getByte();
+      if (nextByte < 0) {
+        return false;
+      }
+      this.buffer = nextByte;
+      this.bufferLength = 8;
+      return true;
+    },
+    readBits: function MeshStreamReader_readBits(n) {
+      var buffer = this.buffer;
+      var bufferLength = this.bufferLength;
+      if (n === 32) {
+        if (bufferLength === 0) {
+          return (this.stream.getByte() << 24 | this.stream.getByte() << 16 | this.stream.getByte() << 8 | this.stream.getByte()) >>> 0;
+        }
+        buffer = buffer << 24 | this.stream.getByte() << 16 | this.stream.getByte() << 8 | this.stream.getByte();
+        var nextByte = this.stream.getByte();
+        this.buffer = nextByte & (1 << bufferLength) - 1;
+        return (buffer << 8 - bufferLength | (nextByte & 0xFF) >> bufferLength) >>> 0;
+      }
+      if (n === 8 && bufferLength === 0) {
+        return this.stream.getByte();
+      }
+      while (bufferLength < n) {
+        buffer = buffer << 8 | this.stream.getByte();
+        bufferLength += 8;
+      }
+      bufferLength -= n;
+      this.bufferLength = bufferLength;
+      this.buffer = buffer & (1 << bufferLength) - 1;
+      return buffer >> bufferLength;
+    },
+    align: function MeshStreamReader_align() {
+      this.buffer = 0;
+      this.bufferLength = 0;
+    },
+    readFlag: function MeshStreamReader_readFlag() {
+      return this.readBits(this.context.bitsPerFlag);
+    },
+    readCoordinate: function MeshStreamReader_readCoordinate() {
+      var bitsPerCoordinate = this.context.bitsPerCoordinate;
+      var xi = this.readBits(bitsPerCoordinate);
+      var yi = this.readBits(bitsPerCoordinate);
+      var decode = this.context.decode;
+      var scale = bitsPerCoordinate < 32 ? 1 / ((1 << bitsPerCoordinate) - 1) : 2.3283064365386963e-10;
+      return [xi * scale * (decode[1] - decode[0]) + decode[0], yi * scale * (decode[3] - decode[2]) + decode[2]];
+    },
+    readComponents: function MeshStreamReader_readComponents() {
+      var numComps = this.context.numComps;
+      var bitsPerComponent = this.context.bitsPerComponent;
+      var scale = bitsPerComponent < 32 ? 1 / ((1 << bitsPerComponent) - 1) : 2.3283064365386963e-10;
+      var decode = this.context.decode;
+      var components = this.tmpCompsBuf;
+      for (var i = 0, j = 4; i < numComps; i++, j += 2) {
+        var ci = this.readBits(bitsPerComponent);
+        components[i] = ci * scale * (decode[j + 1] - decode[j]) + decode[j];
+      }
+      var color = this.tmpCsCompsBuf;
+      if (this.context.colorFn) {
+        this.context.colorFn(components, 0, color, 0);
+      }
+      return this.context.colorSpace.getRgb(color, 0);
     }
-    operators.push(f);
-   }
-   ps.push(coords.length);
-   coords.push(coord);
-   colors.push(color);
-   verticesLeft--;
-   reader.align();
-  }
-  mesh.figures.push({
-   type: 'triangles',
-   coords: new Int32Array(ps),
-   colors: new Int32Array(ps)
-  });
- }
- function decodeType5Shading(mesh, reader, verticesPerRow) {
-  var coords = mesh.coords;
-  var colors = mesh.colors;
-  var ps = [];
-  while (reader.hasData) {
-   var coord = reader.readCoordinate();
-   var color = reader.readComponents();
-   ps.push(coords.length);
-   coords.push(coord);
-   colors.push(color);
-  }
-  mesh.figures.push({
-   type: 'lattice',
-   coords: new Int32Array(ps),
-   colors: new Int32Array(ps),
-   verticesPerRow: verticesPerRow
-  });
- }
- var MIN_SPLIT_PATCH_CHUNKS_AMOUNT = 3;
- var MAX_SPLIT_PATCH_CHUNKS_AMOUNT = 20;
- var TRIANGLE_DENSITY = 20;
- var getB = function getBClosure() {
-  function buildB(count) {
-   var lut = [];
-   for (var i = 0; i <= count; i++) {
-    var t = i / count, t_ = 1 - t;
-    lut.push(new Float32Array([
-     t_ * t_ * t_,
-     3 * t * t_ * t_,
-     3 * t * t * t_,
-     t * t * t
-    ]));
-   }
-   return lut;
-  }
-  var cache = [];
-  return function getB(count) {
-   if (!cache[count]) {
-    cache[count] = buildB(count);
-   }
-   return cache[count];
   };
- }();
- function buildFigureFromPatch(mesh, index) {
-  var figure = mesh.figures[index];
-  assert(figure.type === 'patch', 'Unexpected patch mesh figure');
-  var coords = mesh.coords, colors = mesh.colors;
-  var pi = figure.coords;
-  var ci = figure.colors;
-  var figureMinX = Math.min(coords[pi[0]][0], coords[pi[3]][0], coords[pi[12]][0], coords[pi[15]][0]);
-  var figureMinY = Math.min(coords[pi[0]][1], coords[pi[3]][1], coords[pi[12]][1], coords[pi[15]][1]);
-  var figureMaxX = Math.max(coords[pi[0]][0], coords[pi[3]][0], coords[pi[12]][0], coords[pi[15]][0]);
-  var figureMaxY = Math.max(coords[pi[0]][1], coords[pi[3]][1], coords[pi[12]][1], coords[pi[15]][1]);
-  var splitXBy = Math.ceil((figureMaxX - figureMinX) * TRIANGLE_DENSITY / (mesh.bounds[2] - mesh.bounds[0]));
-  splitXBy = Math.max(MIN_SPLIT_PATCH_CHUNKS_AMOUNT, Math.min(MAX_SPLIT_PATCH_CHUNKS_AMOUNT, splitXBy));
-  var splitYBy = Math.ceil((figureMaxY - figureMinY) * TRIANGLE_DENSITY / (mesh.bounds[3] - mesh.bounds[1]));
-  splitYBy = Math.max(MIN_SPLIT_PATCH_CHUNKS_AMOUNT, Math.min(MAX_SPLIT_PATCH_CHUNKS_AMOUNT, splitYBy));
-  var verticesPerRow = splitXBy + 1;
-  var figureCoords = new Int32Array((splitYBy + 1) * verticesPerRow);
-  var figureColors = new Int32Array((splitYBy + 1) * verticesPerRow);
-  var k = 0;
-  var cl = new Uint8Array(3), cr = new Uint8Array(3);
-  var c0 = colors[ci[0]], c1 = colors[ci[1]], c2 = colors[ci[2]], c3 = colors[ci[3]];
-  var bRow = getB(splitYBy), bCol = getB(splitXBy);
-  for (var row = 0; row <= splitYBy; row++) {
-   cl[0] = (c0[0] * (splitYBy - row) + c2[0] * row) / splitYBy | 0;
-   cl[1] = (c0[1] * (splitYBy - row) + c2[1] * row) / splitYBy | 0;
-   cl[2] = (c0[2] * (splitYBy - row) + c2[2] * row) / splitYBy | 0;
-   cr[0] = (c1[0] * (splitYBy - row) + c3[0] * row) / splitYBy | 0;
-   cr[1] = (c1[1] * (splitYBy - row) + c3[1] * row) / splitYBy | 0;
-   cr[2] = (c1[2] * (splitYBy - row) + c3[2] * row) / splitYBy | 0;
-   for (var col = 0; col <= splitXBy; col++, k++) {
-    if ((row === 0 || row === splitYBy) && (col === 0 || col === splitXBy)) {
-     continue;
+  function decodeType4Shading(mesh, reader) {
+    var coords = mesh.coords;
+    var colors = mesh.colors;
+    var operators = [];
+    var ps = [];
+    var verticesLeft = 0;
+    while (reader.hasData) {
+      var f = reader.readFlag();
+      var coord = reader.readCoordinate();
+      var color = reader.readComponents();
+      if (verticesLeft === 0) {
+        assert(0 <= f && f <= 2, 'Unknown type4 flag');
+        switch (f) {
+          case 0:
+            verticesLeft = 3;
+            break;
+          case 1:
+            ps.push(ps[ps.length - 2], ps[ps.length - 1]);
+            verticesLeft = 1;
+            break;
+          case 2:
+            ps.push(ps[ps.length - 3], ps[ps.length - 1]);
+            verticesLeft = 1;
+            break;
+        }
+        operators.push(f);
+      }
+      ps.push(coords.length);
+      coords.push(coord);
+      colors.push(color);
+      verticesLeft--;
+      reader.align();
     }
-    var x = 0, y = 0;
-    var q = 0;
-    for (var i = 0; i <= 3; i++) {
-     for (var j = 0; j <= 3; j++, q++) {
-      var m = bRow[row][i] * bCol[col][j];
-      x += coords[pi[q]][0] * m;
-      y += coords[pi[q]][1] * m;
-     }
+    mesh.figures.push({
+      type: 'triangles',
+      coords: new Int32Array(ps),
+      colors: new Int32Array(ps)
+    });
+  }
+  function decodeType5Shading(mesh, reader, verticesPerRow) {
+    var coords = mesh.coords;
+    var colors = mesh.colors;
+    var ps = [];
+    while (reader.hasData) {
+      var coord = reader.readCoordinate();
+      var color = reader.readComponents();
+      ps.push(coords.length);
+      coords.push(coord);
+      colors.push(color);
     }
-    figureCoords[k] = coords.length;
-    coords.push([
-     x,
-     y
-    ]);
-    figureColors[k] = colors.length;
-    var newColor = new Uint8Array(3);
-    newColor[0] = (cl[0] * (splitXBy - col) + cr[0] * col) / splitXBy | 0;
-    newColor[1] = (cl[1] * (splitXBy - col) + cr[1] * col) / splitXBy | 0;
-    newColor[2] = (cl[2] * (splitXBy - col) + cr[2] * col) / splitXBy | 0;
-    colors.push(newColor);
-   }
+    mesh.figures.push({
+      type: 'lattice',
+      coords: new Int32Array(ps),
+      colors: new Int32Array(ps),
+      verticesPerRow: verticesPerRow
+    });
   }
-  figureCoords[0] = pi[0];
-  figureColors[0] = ci[0];
-  figureCoords[splitXBy] = pi[3];
-  figureColors[splitXBy] = ci[1];
-  figureCoords[verticesPerRow * splitYBy] = pi[12];
-  figureColors[verticesPerRow * splitYBy] = ci[2];
-  figureCoords[verticesPerRow * splitYBy + splitXBy] = pi[15];
-  figureColors[verticesPerRow * splitYBy + splitXBy] = ci[3];
-  mesh.figures[index] = {
-   type: 'lattice',
-   coords: figureCoords,
-   colors: figureColors,
-   verticesPerRow: verticesPerRow
-  };
- }
- function decodeType6Shading(mesh, reader) {
-  var coords = mesh.coords;
-  var colors = mesh.colors;
-  var ps = new Int32Array(16);
-  var cs = new Int32Array(4);
-  while (reader.hasData) {
-   var f = reader.readFlag();
-   assert(0 <= f && f <= 3, 'Unknown type6 flag');
-   var i, ii;
-   var pi = coords.length;
-   for (i = 0, ii = f !== 0 ? 8 : 12; i < ii; i++) {
-    coords.push(reader.readCoordinate());
-   }
-   var ci = colors.length;
-   for (i = 0, ii = f !== 0 ? 2 : 4; i < ii; i++) {
-    colors.push(reader.readComponents());
-   }
-   var tmp1, tmp2, tmp3, tmp4;
-   switch (f) {
-   case 0:
-    ps[12] = pi + 3;
-    ps[13] = pi + 4;
-    ps[14] = pi + 5;
-    ps[15] = pi + 6;
-    ps[8] = pi + 2;
-    ps[11] = pi + 7;
-    ps[4] = pi + 1;
-    ps[7] = pi + 8;
-    ps[0] = pi;
-    ps[1] = pi + 11;
-    ps[2] = pi + 10;
-    ps[3] = pi + 9;
-    cs[2] = ci + 1;
-    cs[3] = ci + 2;
-    cs[0] = ci;
-    cs[1] = ci + 3;
-    break;
-   case 1:
-    tmp1 = ps[12];
-    tmp2 = ps[13];
-    tmp3 = ps[14];
-    tmp4 = ps[15];
-    ps[12] = tmp4;
-    ps[13] = pi + 0;
-    ps[14] = pi + 1;
-    ps[15] = pi + 2;
-    ps[8] = tmp3;
-    ps[11] = pi + 3;
-    ps[4] = tmp2;
-    ps[7] = pi + 4;
-    ps[0] = tmp1;
-    ps[1] = pi + 7;
-    ps[2] = pi + 6;
-    ps[3] = pi + 5;
-    tmp1 = cs[2];
-    tmp2 = cs[3];
-    cs[2] = tmp2;
-    cs[3] = ci;
-    cs[0] = tmp1;
-    cs[1] = ci + 1;
-    break;
-   case 2:
-    tmp1 = ps[15];
-    tmp2 = ps[11];
-    ps[12] = ps[3];
-    ps[13] = pi + 0;
-    ps[14] = pi + 1;
-    ps[15] = pi + 2;
-    ps[8] = ps[7];
-    ps[11] = pi + 3;
-    ps[4] = tmp2;
-    ps[7] = pi + 4;
-    ps[0] = tmp1;
-    ps[1] = pi + 7;
-    ps[2] = pi + 6;
-    ps[3] = pi + 5;
-    tmp1 = cs[3];
-    cs[2] = cs[1];
-    cs[3] = ci;
-    cs[0] = tmp1;
-    cs[1] = ci + 1;
-    break;
-   case 3:
-    ps[12] = ps[0];
-    ps[13] = pi + 0;
-    ps[14] = pi + 1;
-    ps[15] = pi + 2;
-    ps[8] = ps[1];
-    ps[11] = pi + 3;
-    ps[4] = ps[2];
-    ps[7] = pi + 4;
-    ps[0] = ps[3];
-    ps[1] = pi + 7;
-    ps[2] = pi + 6;
-    ps[3] = pi + 5;
-    cs[2] = cs[0];
-    cs[3] = ci;
-    cs[0] = cs[1];
-    cs[1] = ci + 1;
-    break;
-   }
-   ps[5] = coords.length;
-   coords.push([
-    (-4 * coords[ps[0]][0] - coords[ps[15]][0] + 6 * (coords[ps[4]][0] + coords[ps[1]][0]) - 2 * (coords[ps[12]][0] + coords[ps[3]][0]) + 3 * (coords[ps[13]][0] + coords[ps[7]][0])) / 9,
-    (-4 * coords[ps[0]][1] - coords[ps[15]][1] + 6 * (coords[ps[4]][1] + coords[ps[1]][1]) - 2 * (coords[ps[12]][1] + coords[ps[3]][1]) + 3 * (coords[ps[13]][1] + coords[ps[7]][1])) / 9
-   ]);
-   ps[6] = coords.length;
-   coords.push([
-    (-4 * coords[ps[3]][0] - coords[ps[12]][0] + 6 * (coords[ps[2]][0] + coords[ps[7]][0]) - 2 * (coords[ps[0]][0] + coords[ps[15]][0]) + 3 * (coords[ps[4]][0] + coords[ps[14]][0])) / 9,
-    (-4 * coords[ps[3]][1] - coords[ps[12]][1] + 6 * (coords[ps[2]][1] + coords[ps[7]][1]) - 2 * (coords[ps[0]][1] + coords[ps[15]][1]) + 3 * (coords[ps[4]][1] + coords[ps[14]][1])) / 9
-   ]);
-   ps[9] = coords.length;
-   coords.push([
-    (-4 * coords[ps[12]][0] - coords[ps[3]][0] + 6 * (coords[ps[8]][0] + coords[ps[13]][0]) - 2 * (coords[ps[0]][0] + coords[ps[15]][0]) + 3 * (coords[ps[11]][0] + coords[ps[1]][0])) / 9,
-    (-4 * coords[ps[12]][1] - coords[ps[3]][1] + 6 * (coords[ps[8]][1] + coords[ps[13]][1]) - 2 * (coords[ps[0]][1] + coords[ps[15]][1]) + 3 * (coords[ps[11]][1] + coords[ps[1]][1])) / 9
-   ]);
-   ps[10] = coords.length;
-   coords.push([
-    (-4 * coords[ps[15]][0] - coords[ps[0]][0] + 6 * (coords[ps[11]][0] + coords[ps[14]][0]) - 2 * (coords[ps[12]][0] + coords[ps[3]][0]) + 3 * (coords[ps[2]][0] + coords[ps[8]][0])) / 9,
-    (-4 * coords[ps[15]][1] - coords[ps[0]][1] + 6 * (coords[ps[11]][1] + coords[ps[14]][1]) - 2 * (coords[ps[12]][1] + coords[ps[3]][1]) + 3 * (coords[ps[2]][1] + coords[ps[8]][1])) / 9
-   ]);
-   mesh.figures.push({
-    type: 'patch',
-    coords: new Int32Array(ps),
-    colors: new Int32Array(cs)
-   });
+  var MIN_SPLIT_PATCH_CHUNKS_AMOUNT = 3;
+  var MAX_SPLIT_PATCH_CHUNKS_AMOUNT = 20;
+  var TRIANGLE_DENSITY = 20;
+  var getB = function getBClosure() {
+    function buildB(count) {
+      var lut = [];
+      for (var i = 0; i <= count; i++) {
+        var t = i / count,
+            t_ = 1 - t;
+        lut.push(new Float32Array([t_ * t_ * t_, 3 * t * t_ * t_, 3 * t * t * t_, t * t * t]));
+      }
+      return lut;
+    }
+    var cache = [];
+    return function getB(count) {
+      if (!cache[count]) {
+        cache[count] = buildB(count);
+      }
+      return cache[count];
+    };
+  }();
+  function buildFigureFromPatch(mesh, index) {
+    var figure = mesh.figures[index];
+    assert(figure.type === 'patch', 'Unexpected patch mesh figure');
+    var coords = mesh.coords,
+        colors = mesh.colors;
+    var pi = figure.coords;
+    var ci = figure.colors;
+    var figureMinX = Math.min(coords[pi[0]][0], coords[pi[3]][0], coords[pi[12]][0], coords[pi[15]][0]);
+    var figureMinY = Math.min(coords[pi[0]][1], coords[pi[3]][1], coords[pi[12]][1], coords[pi[15]][1]);
+    var figureMaxX = Math.max(coords[pi[0]][0], coords[pi[3]][0], coords[pi[12]][0], coords[pi[15]][0]);
+    var figureMaxY = Math.max(coords[pi[0]][1], coords[pi[3]][1], coords[pi[12]][1], coords[pi[15]][1]);
+    var splitXBy = Math.ceil((figureMaxX - figureMinX) * TRIANGLE_DENSITY / (mesh.bounds[2] - mesh.bounds[0]));
+    splitXBy = Math.max(MIN_SPLIT_PATCH_CHUNKS_AMOUNT, Math.min(MAX_SPLIT_PATCH_CHUNKS_AMOUNT, splitXBy));
+    var splitYBy = Math.ceil((figureMaxY - figureMinY) * TRIANGLE_DENSITY / (mesh.bounds[3] - mesh.bounds[1]));
+    splitYBy = Math.max(MIN_SPLIT_PATCH_CHUNKS_AMOUNT, Math.min(MAX_SPLIT_PATCH_CHUNKS_AMOUNT, splitYBy));
+    var verticesPerRow = splitXBy + 1;
+    var figureCoords = new Int32Array((splitYBy + 1) * verticesPerRow);
+    var figureColors = new Int32Array((splitYBy + 1) * verticesPerRow);
+    var k = 0;
+    var cl = new Uint8Array(3),
+        cr = new Uint8Array(3);
+    var c0 = colors[ci[0]],
+        c1 = colors[ci[1]],
+        c2 = colors[ci[2]],
+        c3 = colors[ci[3]];
+    var bRow = getB(splitYBy),
+        bCol = getB(splitXBy);
+    for (var row = 0; row <= splitYBy; row++) {
+      cl[0] = (c0[0] * (splitYBy - row) + c2[0] * row) / splitYBy | 0;
+      cl[1] = (c0[1] * (splitYBy - row) + c2[1] * row) / splitYBy | 0;
+      cl[2] = (c0[2] * (splitYBy - row) + c2[2] * row) / splitYBy | 0;
+      cr[0] = (c1[0] * (splitYBy - row) + c3[0] * row) / splitYBy | 0;
+      cr[1] = (c1[1] * (splitYBy - row) + c3[1] * row) / splitYBy | 0;
+      cr[2] = (c1[2] * (splitYBy - row) + c3[2] * row) / splitYBy | 0;
+      for (var col = 0; col <= splitXBy; col++, k++) {
+        if ((row === 0 || row === splitYBy) && (col === 0 || col === splitXBy)) {
+          continue;
+        }
+        var x = 0,
+            y = 0;
+        var q = 0;
+        for (var i = 0; i <= 3; i++) {
+          for (var j = 0; j <= 3; j++, q++) {
+            var m = bRow[row][i] * bCol[col][j];
+            x += coords[pi[q]][0] * m;
+            y += coords[pi[q]][1] * m;
+          }
+        }
+        figureCoords[k] = coords.length;
+        coords.push([x, y]);
+        figureColors[k] = colors.length;
+        var newColor = new Uint8Array(3);
+        newColor[0] = (cl[0] * (splitXBy - col) + cr[0] * col) / splitXBy | 0;
+        newColor[1] = (cl[1] * (splitXBy - col) + cr[1] * col) / splitXBy | 0;
+        newColor[2] = (cl[2] * (splitXBy - col) + cr[2] * col) / splitXBy | 0;
+        colors.push(newColor);
+      }
+    }
+    figureCoords[0] = pi[0];
+    figureColors[0] = ci[0];
+    figureCoords[splitXBy] = pi[3];
+    figureColors[splitXBy] = ci[1];
+    figureCoords[verticesPerRow * splitYBy] = pi[12];
+    figureColors[verticesPerRow * splitYBy] = ci[2];
+    figureCoords[verticesPerRow * splitYBy + splitXBy] = pi[15];
+    figureColors[verticesPerRow * splitYBy + splitXBy] = ci[3];
+    mesh.figures[index] = {
+      type: 'lattice',
+      coords: figureCoords,
+      colors: figureColors,
+      verticesPerRow: verticesPerRow
+    };
   }
- }
- function decodeType7Shading(mesh, reader) {
-  var coords = mesh.coords;
-  var colors = mesh.colors;
-  var ps = new Int32Array(16);
-  var cs = new Int32Array(4);
-  while (reader.hasData) {
-   var f = reader.readFlag();
-   assert(0 <= f && f <= 3, 'Unknown type7 flag');
-   var i, ii;
-   var pi = coords.length;
-   for (i = 0, ii = f !== 0 ? 12 : 16; i < ii; i++) {
-    coords.push(reader.readCoordinate());
-   }
-   var ci = colors.length;
-   for (i = 0, ii = f !== 0 ? 2 : 4; i < ii; i++) {
-    colors.push(reader.readComponents());
-   }
-   var tmp1, tmp2, tmp3, tmp4;
-   switch (f) {
-   case 0:
-    ps[12] = pi + 3;
-    ps[13] = pi + 4;
-    ps[14] = pi + 5;
-    ps[15] = pi + 6;
-    ps[8] = pi + 2;
-    ps[9] = pi + 13;
-    ps[10] = pi + 14;
-    ps[11] = pi + 7;
-    ps[4] = pi + 1;
-    ps[5] = pi + 12;
-    ps[6] = pi + 15;
-    ps[7] = pi + 8;
-    ps[0] = pi;
-    ps[1] = pi + 11;
-    ps[2] = pi + 10;
-    ps[3] = pi + 9;
-    cs[2] = ci + 1;
-    cs[3] = ci + 2;
-    cs[0] = ci;
-    cs[1] = ci + 3;
-    break;
-   case 1:
-    tmp1 = ps[12];
-    tmp2 = ps[13];
-    tmp3 = ps[14];
-    tmp4 = ps[15];
-    ps[12] = tmp4;
-    ps[13] = pi + 0;
-    ps[14] = pi + 1;
-    ps[15] = pi + 2;
-    ps[8] = tmp3;
-    ps[9] = pi + 9;
-    ps[10] = pi + 10;
-    ps[11] = pi + 3;
-    ps[4] = tmp2;
-    ps[5] = pi + 8;
-    ps[6] = pi + 11;
-    ps[7] = pi + 4;
-    ps[0] = tmp1;
-    ps[1] = pi + 7;
-    ps[2] = pi + 6;
-    ps[3] = pi + 5;
-    tmp1 = cs[2];
-    tmp2 = cs[3];
-    cs[2] = tmp2;
-    cs[3] = ci;
-    cs[0] = tmp1;
-    cs[1] = ci + 1;
-    break;
-   case 2:
-    tmp1 = ps[15];
-    tmp2 = ps[11];
-    ps[12] = ps[3];
-    ps[13] = pi + 0;
-    ps[14] = pi + 1;
-    ps[15] = pi + 2;
-    ps[8] = ps[7];
-    ps[9] = pi + 9;
-    ps[10] = pi + 10;
-    ps[11] = pi + 3;
-    ps[4] = tmp2;
-    ps[5] = pi + 8;
-    ps[6] = pi + 11;
-    ps[7] = pi + 4;
-    ps[0] = tmp1;
-    ps[1] = pi + 7;
-    ps[2] = pi + 6;
-    ps[3] = pi + 5;
-    tmp1 = cs[3];
-    cs[2] = cs[1];
-    cs[3] = ci;
-    cs[0] = tmp1;
-    cs[1] = ci + 1;
-    break;
-   case 3:
-    ps[12] = ps[0];
-    ps[13] = pi + 0;
-    ps[14] = pi + 1;
-    ps[15] = pi + 2;
-    ps[8] = ps[1];
-    ps[9] = pi + 9;
-    ps[10] = pi + 10;
-    ps[11] = pi + 3;
-    ps[4] = ps[2];
-    ps[5] = pi + 8;
-    ps[6] = pi + 11;
-    ps[7] = pi + 4;
-    ps[0] = ps[3];
-    ps[1] = pi + 7;
-    ps[2] = pi + 6;
-    ps[3] = pi + 5;
-    cs[2] = cs[0];
-    cs[3] = ci;
-    cs[0] = cs[1];
-    cs[1] = ci + 1;
-    break;
-   }
-   mesh.figures.push({
-    type: 'patch',
-    coords: new Int32Array(ps),
-    colors: new Int32Array(cs)
-   });
+  function decodeType6Shading(mesh, reader) {
+    var coords = mesh.coords;
+    var colors = mesh.colors;
+    var ps = new Int32Array(16);
+    var cs = new Int32Array(4);
+    while (reader.hasData) {
+      var f = reader.readFlag();
+      assert(0 <= f && f <= 3, 'Unknown type6 flag');
+      var i, ii;
+      var pi = coords.length;
+      for (i = 0, ii = f !== 0 ? 8 : 12; i < ii; i++) {
+        coords.push(reader.readCoordinate());
+      }
+      var ci = colors.length;
+      for (i = 0, ii = f !== 0 ? 2 : 4; i < ii; i++) {
+        colors.push(reader.readComponents());
+      }
+      var tmp1, tmp2, tmp3, tmp4;
+      switch (f) {
+        case 0:
+          ps[12] = pi + 3;
+          ps[13] = pi + 4;
+          ps[14] = pi + 5;
+          ps[15] = pi + 6;
+          ps[8] = pi + 2;
+          ps[11] = pi + 7;
+          ps[4] = pi + 1;
+          ps[7] = pi + 8;
+          ps[0] = pi;
+          ps[1] = pi + 11;
+          ps[2] = pi + 10;
+          ps[3] = pi + 9;
+          cs[2] = ci + 1;
+          cs[3] = ci + 2;
+          cs[0] = ci;
+          cs[1] = ci + 3;
+          break;
+        case 1:
+          tmp1 = ps[12];
+          tmp2 = ps[13];
+          tmp3 = ps[14];
+          tmp4 = ps[15];
+          ps[12] = tmp4;
+          ps[13] = pi + 0;
+          ps[14] = pi + 1;
+          ps[15] = pi + 2;
+          ps[8] = tmp3;
+          ps[11] = pi + 3;
+          ps[4] = tmp2;
+          ps[7] = pi + 4;
+          ps[0] = tmp1;
+          ps[1] = pi + 7;
+          ps[2] = pi + 6;
+          ps[3] = pi + 5;
+          tmp1 = cs[2];
+          tmp2 = cs[3];
+          cs[2] = tmp2;
+          cs[3] = ci;
+          cs[0] = tmp1;
+          cs[1] = ci + 1;
+          break;
+        case 2:
+          tmp1 = ps[15];
+          tmp2 = ps[11];
+          ps[12] = ps[3];
+          ps[13] = pi + 0;
+          ps[14] = pi + 1;
+          ps[15] = pi + 2;
+          ps[8] = ps[7];
+          ps[11] = pi + 3;
+          ps[4] = tmp2;
+          ps[7] = pi + 4;
+          ps[0] = tmp1;
+          ps[1] = pi + 7;
+          ps[2] = pi + 6;
+          ps[3] = pi + 5;
+          tmp1 = cs[3];
+          cs[2] = cs[1];
+          cs[3] = ci;
+          cs[0] = tmp1;
+          cs[1] = ci + 1;
+          break;
+        case 3:
+          ps[12] = ps[0];
+          ps[13] = pi + 0;
+          ps[14] = pi + 1;
+          ps[15] = pi + 2;
+          ps[8] = ps[1];
+          ps[11] = pi + 3;
+          ps[4] = ps[2];
+          ps[7] = pi + 4;
+          ps[0] = ps[3];
+          ps[1] = pi + 7;
+          ps[2] = pi + 6;
+          ps[3] = pi + 5;
+          cs[2] = cs[0];
+          cs[3] = ci;
+          cs[0] = cs[1];
+          cs[1] = ci + 1;
+          break;
+      }
+      ps[5] = coords.length;
+      coords.push([(-4 * coords[ps[0]][0] - coords[ps[15]][0] + 6 * (coords[ps[4]][0] + coords[ps[1]][0]) - 2 * (coords[ps[12]][0] + coords[ps[3]][0]) + 3 * (coords[ps[13]][0] + coords[ps[7]][0])) / 9, (-4 * coords[ps[0]][1] - coords[ps[15]][1] + 6 * (coords[ps[4]][1] + coords[ps[1]][1]) - 2 * (coords[ps[12]][1] + coords[ps[3]][1]) + 3 * (coords[ps[13]][1] + coords[ps[7]][1])) / 9]);
+      ps[6] = coords.length;
+      coords.push([(-4 * coords[ps[3]][0] - coords[ps[12]][0] + 6 * (coords[ps[2]][0] + coords[ps[7]][0]) - 2 * (coords[ps[0]][0] + coords[ps[15]][0]) + 3 * (coords[ps[4]][0] + coords[ps[14]][0])) / 9, (-4 * coords[ps[3]][1] - coords[ps[12]][1] + 6 * (coords[ps[2]][1] + coords[ps[7]][1]) - 2 * (coords[ps[0]][1] + coords[ps[15]][1]) + 3 * (coords[ps[4]][1] + coords[ps[14]][1])) / 9]);
+      ps[9] = coords.length;
+      coords.push([(-4 * coords[ps[12]][0] - coords[ps[3]][0] + 6 * (coords[ps[8]][0] + coords[ps[13]][0]) - 2 * (coords[ps[0]][0] + coords[ps[15]][0]) + 3 * (coords[ps[11]][0] + coords[ps[1]][0])) / 9, (-4 * coords[ps[12]][1] - coords[ps[3]][1] + 6 * (coords[ps[8]][1] + coords[ps[13]][1]) - 2 * (coords[ps[0]][1] + coords[ps[15]][1]) + 3 * (coords[ps[11]][1] + coords[ps[1]][1])) / 9]);
+      ps[10] = coords.length;
+      coords.push([(-4 * coords[ps[15]][0] - coords[ps[0]][0] + 6 * (coords[ps[11]][0] + coords[ps[14]][0]) - 2 * (coords[ps[12]][0] + coords[ps[3]][0]) + 3 * (coords[ps[2]][0] + coords[ps[8]][0])) / 9, (-4 * coords[ps[15]][1] - coords[ps[0]][1] + 6 * (coords[ps[11]][1] + coords[ps[14]][1]) - 2 * (coords[ps[12]][1] + coords[ps[3]][1]) + 3 * (coords[ps[2]][1] + coords[ps[8]][1])) / 9]);
+      mesh.figures.push({
+        type: 'patch',
+        coords: new Int32Array(ps),
+        colors: new Int32Array(cs)
+      });
+    }
   }
- }
- function updateBounds(mesh) {
-  var minX = mesh.coords[0][0], minY = mesh.coords[0][1], maxX = minX, maxY = minY;
-  for (var i = 1, ii = mesh.coords.length; i < ii; i++) {
-   var x = mesh.coords[i][0], y = mesh.coords[i][1];
-   minX = minX > x ? x : minX;
-   minY = minY > y ? y : minY;
-   maxX = maxX < x ? x : maxX;
-   maxY = maxY < y ? y : maxY;
+  function decodeType7Shading(mesh, reader) {
+    var coords = mesh.coords;
+    var colors = mesh.colors;
+    var ps = new Int32Array(16);
+    var cs = new Int32Array(4);
+    while (reader.hasData) {
+      var f = reader.readFlag();
+      assert(0 <= f && f <= 3, 'Unknown type7 flag');
+      var i, ii;
+      var pi = coords.length;
+      for (i = 0, ii = f !== 0 ? 12 : 16; i < ii; i++) {
+        coords.push(reader.readCoordinate());
+      }
+      var ci = colors.length;
+      for (i = 0, ii = f !== 0 ? 2 : 4; i < ii; i++) {
+        colors.push(reader.readComponents());
+      }
+      var tmp1, tmp2, tmp3, tmp4;
+      switch (f) {
+        case 0:
+          ps[12] = pi + 3;
+          ps[13] = pi + 4;
+          ps[14] = pi + 5;
+          ps[15] = pi + 6;
+          ps[8] = pi + 2;
+          ps[9] = pi + 13;
+          ps[10] = pi + 14;
+          ps[11] = pi + 7;
+          ps[4] = pi + 1;
+          ps[5] = pi + 12;
+          ps[6] = pi + 15;
+          ps[7] = pi + 8;
+          ps[0] = pi;
+          ps[1] = pi + 11;
+          ps[2] = pi + 10;
+          ps[3] = pi + 9;
+          cs[2] = ci + 1;
+          cs[3] = ci + 2;
+          cs[0] = ci;
+          cs[1] = ci + 3;
+          break;
+        case 1:
+          tmp1 = ps[12];
+          tmp2 = ps[13];
+          tmp3 = ps[14];
+          tmp4 = ps[15];
+          ps[12] = tmp4;
+          ps[13] = pi + 0;
+          ps[14] = pi + 1;
+          ps[15] = pi + 2;
+          ps[8] = tmp3;
+          ps[9] = pi + 9;
+          ps[10] = pi + 10;
+          ps[11] = pi + 3;
+          ps[4] = tmp2;
+          ps[5] = pi + 8;
+          ps[6] = pi + 11;
+          ps[7] = pi + 4;
+          ps[0] = tmp1;
+          ps[1] = pi + 7;
+          ps[2] = pi + 6;
+          ps[3] = pi + 5;
+          tmp1 = cs[2];
+          tmp2 = cs[3];
+          cs[2] = tmp2;
+          cs[3] = ci;
+          cs[0] = tmp1;
+          cs[1] = ci + 1;
+          break;
+        case 2:
+          tmp1 = ps[15];
+          tmp2 = ps[11];
+          ps[12] = ps[3];
+          ps[13] = pi + 0;
+          ps[14] = pi + 1;
+          ps[15] = pi + 2;
+          ps[8] = ps[7];
+          ps[9] = pi + 9;
+          ps[10] = pi + 10;
+          ps[11] = pi + 3;
+          ps[4] = tmp2;
+          ps[5] = pi + 8;
+          ps[6] = pi + 11;
+          ps[7] = pi + 4;
+          ps[0] = tmp1;
+          ps[1] = pi + 7;
+          ps[2] = pi + 6;
+          ps[3] = pi + 5;
+          tmp1 = cs[3];
+          cs[2] = cs[1];
+          cs[3] = ci;
+          cs[0] = tmp1;
+          cs[1] = ci + 1;
+          break;
+        case 3:
+          ps[12] = ps[0];
+          ps[13] = pi + 0;
+          ps[14] = pi + 1;
+          ps[15] = pi + 2;
+          ps[8] = ps[1];
+          ps[9] = pi + 9;
+          ps[10] = pi + 10;
+          ps[11] = pi + 3;
+          ps[4] = ps[2];
+          ps[5] = pi + 8;
+          ps[6] = pi + 11;
+          ps[7] = pi + 4;
+          ps[0] = ps[3];
+          ps[1] = pi + 7;
+          ps[2] = pi + 6;
+          ps[3] = pi + 5;
+          cs[2] = cs[0];
+          cs[3] = ci;
+          cs[0] = cs[1];
+          cs[1] = ci + 1;
+          break;
+      }
+      mesh.figures.push({
+        type: 'patch',
+        coords: new Int32Array(ps),
+        colors: new Int32Array(cs)
+      });
+    }
   }
-  mesh.bounds = [
-   minX,
-   minY,
-   maxX,
-   maxY
-  ];
- }
- function packData(mesh) {
-  var i, ii, j, jj;
-  var coords = mesh.coords;
-  var coordsPacked = new Float32Array(coords.length * 2);
-  for (i = 0, j = 0, ii = coords.length; i < ii; i++) {
-   var xy = coords[i];
-   coordsPacked[j++] = xy[0];
-   coordsPacked[j++] = xy[1];
+  function updateBounds(mesh) {
+    var minX = mesh.coords[0][0],
+        minY = mesh.coords[0][1],
+        maxX = minX,
+        maxY = minY;
+    for (var i = 1, ii = mesh.coords.length; i < ii; i++) {
+      var x = mesh.coords[i][0],
+          y = mesh.coords[i][1];
+      minX = minX > x ? x : minX;
+      minY = minY > y ? y : minY;
+      maxX = maxX < x ? x : maxX;
+      maxY = maxY < y ? y : maxY;
+    }
+    mesh.bounds = [minX, minY, maxX, maxY];
   }
-  mesh.coords = coordsPacked;
-  var colors = mesh.colors;
-  var colorsPacked = new Uint8Array(colors.length * 3);
-  for (i = 0, j = 0, ii = colors.length; i < ii; i++) {
-   var c = colors[i];
-   colorsPacked[j++] = c[0];
-   colorsPacked[j++] = c[1];
-   colorsPacked[j++] = c[2];
+  function packData(mesh) {
+    var i, ii, j, jj;
+    var coords = mesh.coords;
+    var coordsPacked = new Float32Array(coords.length * 2);
+    for (i = 0, j = 0, ii = coords.length; i < ii; i++) {
+      var xy = coords[i];
+      coordsPacked[j++] = xy[0];
+      coordsPacked[j++] = xy[1];
+    }
+    mesh.coords = coordsPacked;
+    var colors = mesh.colors;
+    var colorsPacked = new Uint8Array(colors.length * 3);
+    for (i = 0, j = 0, ii = colors.length; i < ii; i++) {
+      var c = colors[i];
+      colorsPacked[j++] = c[0];
+      colorsPacked[j++] = c[1];
+      colorsPacked[j++] = c[2];
+    }
+    mesh.colors = colorsPacked;
+    var figures = mesh.figures;
+    for (i = 0, ii = figures.length; i < ii; i++) {
+      var figure = figures[i],
+          ps = figure.coords,
+          cs = figure.colors;
+      for (j = 0, jj = ps.length; j < jj; j++) {
+        ps[j] *= 2;
+        cs[j] *= 3;
+      }
+    }
   }
-  mesh.colors = colorsPacked;
-  var figures = mesh.figures;
-  for (i = 0, ii = figures.length; i < ii; i++) {
-   var figure = figures[i], ps = figure.coords, cs = figure.colors;
-   for (j = 0, jj = ps.length; j < jj; j++) {
-    ps[j] *= 2;
-    cs[j] *= 3;
-   }
+  function Mesh(stream, matrix, xref, res) {
+    assert(isStream(stream), 'Mesh data is not a stream');
+    var dict = stream.dict;
+    this.matrix = matrix;
+    this.shadingType = dict.get('ShadingType');
+    this.type = 'Pattern';
+    this.bbox = dict.getArray('BBox');
+    var cs = dict.get('ColorSpace', 'CS');
+    cs = ColorSpace.parse(cs, xref, res);
+    this.cs = cs;
+    this.background = dict.has('Background') ? cs.getRgb(dict.get('Background'), 0) : null;
+    var fnObj = dict.get('Function');
+    var fn = fnObj ? PDFFunction.parseArray(xref, fnObj) : null;
+    this.coords = [];
+    this.colors = [];
+    this.figures = [];
+    var decodeContext = {
+      bitsPerCoordinate: dict.get('BitsPerCoordinate'),
+      bitsPerComponent: dict.get('BitsPerComponent'),
+      bitsPerFlag: dict.get('BitsPerFlag'),
+      decode: dict.getArray('Decode'),
+      colorFn: fn,
+      colorSpace: cs,
+      numComps: fn ? 1 : cs.numComps
+    };
+    var reader = new MeshStreamReader(stream, decodeContext);
+    var patchMesh = false;
+    switch (this.shadingType) {
+      case ShadingType.FREE_FORM_MESH:
+        decodeType4Shading(this, reader);
+        break;
+      case ShadingType.LATTICE_FORM_MESH:
+        var verticesPerRow = dict.get('VerticesPerRow') | 0;
+        assert(verticesPerRow >= 2, 'Invalid VerticesPerRow');
+        decodeType5Shading(this, reader, verticesPerRow);
+        break;
+      case ShadingType.COONS_PATCH_MESH:
+        decodeType6Shading(this, reader);
+        patchMesh = true;
+        break;
+      case ShadingType.TENSOR_PATCH_MESH:
+        decodeType7Shading(this, reader);
+        patchMesh = true;
+        break;
+      default:
+        error('Unsupported mesh type.');
+        break;
+    }
+    if (patchMesh) {
+      updateBounds(this);
+      for (var i = 0, ii = this.figures.length; i < ii; i++) {
+        buildFigureFromPatch(this, i);
+      }
+    }
+    updateBounds(this);
+    packData(this);
   }
- }
- function Mesh(stream, matrix, xref, res) {
-  assert(isStream(stream), 'Mesh data is not a stream');
-  var dict = stream.dict;
-  this.matrix = matrix;
-  this.shadingType = dict.get('ShadingType');
-  this.type = 'Pattern';
-  this.bbox = dict.getArray('BBox');
-  var cs = dict.get('ColorSpace', 'CS');
-  cs = ColorSpace.parse(cs, xref, res);
-  this.cs = cs;
-  this.background = dict.has('Background') ? cs.getRgb(dict.get('Background'), 0) : null;
-  var fnObj = dict.get('Function');
-  var fn = fnObj ? PDFFunction.parseArray(xref, fnObj) : null;
-  this.coords = [];
-  this.colors = [];
-  this.figures = [];
-  var decodeContext = {
-   bitsPerCoordinate: dict.get('BitsPerCoordinate'),
-   bitsPerComponent: dict.get('BitsPerComponent'),
-   bitsPerFlag: dict.get('BitsPerFlag'),
-   decode: dict.getArray('Decode'),
-   colorFn: fn,
-   colorSpace: cs,
-   numComps: fn ? 1 : cs.numComps
+  Mesh.prototype = {
+    getIR: function Mesh_getIR() {
+      return ['Mesh', this.shadingType, this.coords, this.colors, this.figures, this.bounds, this.matrix, this.bbox, this.background];
+    }
   };
-  var reader = new MeshStreamReader(stream, decodeContext);
-  var patchMesh = false;
-  switch (this.shadingType) {
-  case ShadingType.FREE_FORM_MESH:
-   decodeType4Shading(this, reader);
-   break;
-  case ShadingType.LATTICE_FORM_MESH:
-   var verticesPerRow = dict.get('VerticesPerRow') | 0;
-   assert(verticesPerRow >= 2, 'Invalid VerticesPerRow');
-   decodeType5Shading(this, reader, verticesPerRow);
-   break;
-  case ShadingType.COONS_PATCH_MESH:
-   decodeType6Shading(this, reader);
-   patchMesh = true;
-   break;
-  case ShadingType.TENSOR_PATCH_MESH:
-   decodeType7Shading(this, reader);
-   patchMesh = true;
-   break;
-  default:
-   error('Unsupported mesh type.');
-   break;
-  }
-  if (patchMesh) {
-   updateBounds(this);
-   for (var i = 0, ii = this.figures.length; i < ii; i++) {
-    buildFigureFromPatch(this, i);
-   }
-  }
-  updateBounds(this);
-  packData(this);
- }
- Mesh.prototype = {
-  getIR: function Mesh_getIR() {
-   return [
-    'Mesh',
-    this.shadingType,
-    this.coords,
-    this.colors,
-    this.figures,
-    this.bounds,
-    this.matrix,
-    this.bbox,
-    this.background
-   ];
-  }
- };
- return Mesh;
+  return Mesh;
 }();
 Shadings.Dummy = function DummyClosure() {
- function Dummy() {
-  this.type = 'Pattern';
- }
- Dummy.prototype = {
-  getIR: function Dummy_getIR() {
-   return ['Dummy'];
+  function Dummy() {
+    this.type = 'Pattern';
   }
- };
- return Dummy;
+  Dummy.prototype = {
+    getIR: function Dummy_getIR() {
+      return ['Dummy'];
+    }
+  };
+  return Dummy;
 }();
 function getTilingPatternIR(operatorList, dict, args) {
- var matrix = dict.getArray('Matrix');
- var bbox = dict.getArray('BBox');
- var xstep = dict.get('XStep');
- var ystep = dict.get('YStep');
- var paintType = dict.get('PaintType');
- var tilingType = dict.get('TilingType');
- return [
-  'TilingPattern',
-  args,
-  operatorList,
-  matrix,
-  bbox,
-  xstep,
-  ystep,
-  paintType,
-  tilingType
- ];
+  var matrix = dict.getArray('Matrix');
+  var bbox = dict.getArray('BBox');
+  var xstep = dict.get('XStep');
+  var ystep = dict.get('YStep');
+  var paintType = dict.get('PaintType');
+  var tilingType = dict.get('TilingType');
+  return ['TilingPattern', args, operatorList, matrix, bbox, xstep, ystep, paintType, tilingType];
 }
 exports.Pattern = Pattern;
 exports.getTilingPatternIR = getTilingPatternIR;

+ 153 - 154
lib/core/pdf_manager.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var coreStream = require('./stream.js');
 var coreChunkedStream = require('./chunked_stream.js');
@@ -28,167 +29,165 @@ var Stream = coreStream.Stream;
 var ChunkedStreamManager = coreChunkedStream.ChunkedStreamManager;
 var PDFDocument = coreDocument.PDFDocument;
 var BasePdfManager = function BasePdfManagerClosure() {
- function BasePdfManager() {
-  throw new Error('Cannot initialize BaseManagerManager');
- }
- BasePdfManager.prototype = {
-  get docId() {
-   return this._docId;
-  },
-  get password() {
-   return this._password;
-  },
-  get docBaseUrl() {
-   var docBaseUrl = null;
-   if (this._docBaseUrl) {
-    var absoluteUrl = createValidAbsoluteUrl(this._docBaseUrl);
-    if (absoluteUrl) {
-     docBaseUrl = absoluteUrl.href;
-    } else {
-     warn('Invalid absolute docBaseUrl: "' + this._docBaseUrl + '".');
-    }
-   }
-   return shadow(this, 'docBaseUrl', docBaseUrl);
-  },
-  onLoadedStream: function BasePdfManager_onLoadedStream() {
-   throw new NotImplementedException();
-  },
-  ensureDoc: function BasePdfManager_ensureDoc(prop, args) {
-   return this.ensure(this.pdfDocument, prop, args);
-  },
-  ensureXRef: function BasePdfManager_ensureXRef(prop, args) {
-   return this.ensure(this.pdfDocument.xref, prop, args);
-  },
-  ensureCatalog: function BasePdfManager_ensureCatalog(prop, args) {
-   return this.ensure(this.pdfDocument.catalog, prop, args);
-  },
-  getPage: function BasePdfManager_getPage(pageIndex) {
-   return this.pdfDocument.getPage(pageIndex);
-  },
-  cleanup: function BasePdfManager_cleanup() {
-   return this.pdfDocument.cleanup();
-  },
-  ensure: function BasePdfManager_ensure(obj, prop, args) {
-   return new NotImplementedException();
-  },
-  requestRange: function BasePdfManager_requestRange(begin, end) {
-   return new NotImplementedException();
-  },
-  requestLoadedStream: function BasePdfManager_requestLoadedStream() {
-   return new NotImplementedException();
-  },
-  sendProgressiveData: function BasePdfManager_sendProgressiveData(chunk) {
-   return new NotImplementedException();
-  },
-  updatePassword: function BasePdfManager_updatePassword(password) {
-   this._password = password;
-  },
-  terminate: function BasePdfManager_terminate() {
-   return new NotImplementedException();
+  function BasePdfManager() {
+    throw new Error('Cannot initialize BaseManagerManager');
   }
- };
- return BasePdfManager;
+  BasePdfManager.prototype = {
+    get docId() {
+      return this._docId;
+    },
+    get password() {
+      return this._password;
+    },
+    get docBaseUrl() {
+      var docBaseUrl = null;
+      if (this._docBaseUrl) {
+        var absoluteUrl = createValidAbsoluteUrl(this._docBaseUrl);
+        if (absoluteUrl) {
+          docBaseUrl = absoluteUrl.href;
+        } else {
+          warn('Invalid absolute docBaseUrl: "' + this._docBaseUrl + '".');
+        }
+      }
+      return shadow(this, 'docBaseUrl', docBaseUrl);
+    },
+    onLoadedStream: function BasePdfManager_onLoadedStream() {
+      throw new NotImplementedException();
+    },
+    ensureDoc: function BasePdfManager_ensureDoc(prop, args) {
+      return this.ensure(this.pdfDocument, prop, args);
+    },
+    ensureXRef: function BasePdfManager_ensureXRef(prop, args) {
+      return this.ensure(this.pdfDocument.xref, prop, args);
+    },
+    ensureCatalog: function BasePdfManager_ensureCatalog(prop, args) {
+      return this.ensure(this.pdfDocument.catalog, prop, args);
+    },
+    getPage: function BasePdfManager_getPage(pageIndex) {
+      return this.pdfDocument.getPage(pageIndex);
+    },
+    cleanup: function BasePdfManager_cleanup() {
+      return this.pdfDocument.cleanup();
+    },
+    ensure: function BasePdfManager_ensure(obj, prop, args) {
+      return new NotImplementedException();
+    },
+    requestRange: function BasePdfManager_requestRange(begin, end) {
+      return new NotImplementedException();
+    },
+    requestLoadedStream: function BasePdfManager_requestLoadedStream() {
+      return new NotImplementedException();
+    },
+    sendProgressiveData: function BasePdfManager_sendProgressiveData(chunk) {
+      return new NotImplementedException();
+    },
+    updatePassword: function BasePdfManager_updatePassword(password) {
+      this._password = password;
+    },
+    terminate: function BasePdfManager_terminate() {
+      return new NotImplementedException();
+    }
+  };
+  return BasePdfManager;
 }();
 var LocalPdfManager = function LocalPdfManagerClosure() {
- function LocalPdfManager(docId, data, password, evaluatorOptions, docBaseUrl) {
-  this._docId = docId;
-  this._password = password;
-  this._docBaseUrl = docBaseUrl;
-  this.evaluatorOptions = evaluatorOptions;
-  var stream = new Stream(data);
-  this.pdfDocument = new PDFDocument(this, stream);
-  this._loadedStreamCapability = createPromiseCapability();
-  this._loadedStreamCapability.resolve(stream);
- }
- Util.inherit(LocalPdfManager, BasePdfManager, {
-  ensure: function LocalPdfManager_ensure(obj, prop, args) {
-   return new Promise(function (resolve, reject) {
-    try {
-     var value = obj[prop];
-     var result;
-     if (typeof value === 'function') {
-      result = value.apply(obj, args);
-     } else {
-      result = value;
-     }
-     resolve(result);
-    } catch (e) {
-     reject(e);
-    }
-   });
-  },
-  requestRange: function LocalPdfManager_requestRange(begin, end) {
-   return Promise.resolve();
-  },
-  requestLoadedStream: function LocalPdfManager_requestLoadedStream() {
-  },
-  onLoadedStream: function LocalPdfManager_onLoadedStream() {
-   return this._loadedStreamCapability.promise;
-  },
-  terminate: function LocalPdfManager_terminate() {
+  function LocalPdfManager(docId, data, password, evaluatorOptions, docBaseUrl) {
+    this._docId = docId;
+    this._password = password;
+    this._docBaseUrl = docBaseUrl;
+    this.evaluatorOptions = evaluatorOptions;
+    var stream = new Stream(data);
+    this.pdfDocument = new PDFDocument(this, stream);
+    this._loadedStreamCapability = createPromiseCapability();
+    this._loadedStreamCapability.resolve(stream);
   }
- });
- return LocalPdfManager;
+  Util.inherit(LocalPdfManager, BasePdfManager, {
+    ensure: function LocalPdfManager_ensure(obj, prop, args) {
+      return new Promise(function (resolve, reject) {
+        try {
+          var value = obj[prop];
+          var result;
+          if (typeof value === 'function') {
+            result = value.apply(obj, args);
+          } else {
+            result = value;
+          }
+          resolve(result);
+        } catch (e) {
+          reject(e);
+        }
+      });
+    },
+    requestRange: function LocalPdfManager_requestRange(begin, end) {
+      return Promise.resolve();
+    },
+    requestLoadedStream: function LocalPdfManager_requestLoadedStream() {},
+    onLoadedStream: function LocalPdfManager_onLoadedStream() {
+      return this._loadedStreamCapability.promise;
+    },
+    terminate: function LocalPdfManager_terminate() {}
+  });
+  return LocalPdfManager;
 }();
 var NetworkPdfManager = function NetworkPdfManagerClosure() {
- function NetworkPdfManager(docId, pdfNetworkStream, args, evaluatorOptions, docBaseUrl) {
-  this._docId = docId;
-  this._password = args.password;
-  this._docBaseUrl = docBaseUrl;
-  this.msgHandler = args.msgHandler;
-  this.evaluatorOptions = evaluatorOptions;
-  var params = {
-   msgHandler: args.msgHandler,
-   url: args.url,
-   length: args.length,
-   disableAutoFetch: args.disableAutoFetch,
-   rangeChunkSize: args.rangeChunkSize
-  };
-  this.streamManager = new ChunkedStreamManager(pdfNetworkStream, params);
-  this.pdfDocument = new PDFDocument(this, this.streamManager.getStream());
- }
- Util.inherit(NetworkPdfManager, BasePdfManager, {
-  ensure: function NetworkPdfManager_ensure(obj, prop, args) {
-   var pdfManager = this;
-   return new Promise(function (resolve, reject) {
-    function ensureHelper() {
-     try {
-      var result;
-      var value = obj[prop];
-      if (typeof value === 'function') {
-       result = value.apply(obj, args);
-      } else {
-       result = value;
-      }
-      resolve(result);
-     } catch (e) {
-      if (!(e instanceof MissingDataException)) {
-       reject(e);
-       return;
-      }
-      pdfManager.streamManager.requestRange(e.begin, e.end).then(ensureHelper, reject);
-     }
-    }
-    ensureHelper();
-   });
-  },
-  requestRange: function NetworkPdfManager_requestRange(begin, end) {
-   return this.streamManager.requestRange(begin, end);
-  },
-  requestLoadedStream: function NetworkPdfManager_requestLoadedStream() {
-   this.streamManager.requestAllChunks();
-  },
-  sendProgressiveData: function NetworkPdfManager_sendProgressiveData(chunk) {
-   this.streamManager.onReceiveData({ chunk: chunk });
-  },
-  onLoadedStream: function NetworkPdfManager_onLoadedStream() {
-   return this.streamManager.onLoadedStream();
-  },
-  terminate: function NetworkPdfManager_terminate() {
-   this.streamManager.abort();
+  function NetworkPdfManager(docId, pdfNetworkStream, args, evaluatorOptions, docBaseUrl) {
+    this._docId = docId;
+    this._password = args.password;
+    this._docBaseUrl = docBaseUrl;
+    this.msgHandler = args.msgHandler;
+    this.evaluatorOptions = evaluatorOptions;
+    var params = {
+      msgHandler: args.msgHandler,
+      url: args.url,
+      length: args.length,
+      disableAutoFetch: args.disableAutoFetch,
+      rangeChunkSize: args.rangeChunkSize
+    };
+    this.streamManager = new ChunkedStreamManager(pdfNetworkStream, params);
+    this.pdfDocument = new PDFDocument(this, this.streamManager.getStream());
   }
- });
- return NetworkPdfManager;
+  Util.inherit(NetworkPdfManager, BasePdfManager, {
+    ensure: function NetworkPdfManager_ensure(obj, prop, args) {
+      var pdfManager = this;
+      return new Promise(function (resolve, reject) {
+        function ensureHelper() {
+          try {
+            var result;
+            var value = obj[prop];
+            if (typeof value === 'function') {
+              result = value.apply(obj, args);
+            } else {
+              result = value;
+            }
+            resolve(result);
+          } catch (e) {
+            if (!(e instanceof MissingDataException)) {
+              reject(e);
+              return;
+            }
+            pdfManager.streamManager.requestRange(e.begin, e.end).then(ensureHelper, reject);
+          }
+        }
+        ensureHelper();
+      });
+    },
+    requestRange: function NetworkPdfManager_requestRange(begin, end) {
+      return this.streamManager.requestRange(begin, end);
+    },
+    requestLoadedStream: function NetworkPdfManager_requestLoadedStream() {
+      this.streamManager.requestAllChunks();
+    },
+    sendProgressiveData: function NetworkPdfManager_sendProgressiveData(chunk) {
+      this.streamManager.onReceiveData({ chunk: chunk });
+    },
+    onLoadedStream: function NetworkPdfManager_onLoadedStream() {
+      return this.streamManager.onLoadedStream();
+    },
+    terminate: function NetworkPdfManager_terminate() {
+      this.streamManager.abort();
+    }
+  });
+  return NetworkPdfManager;
 }();
 exports.LocalPdfManager = LocalPdfManager;
 exports.NetworkPdfManager = NetworkPdfManager;

+ 179 - 175
lib/core/primitives.js

@@ -13,214 +13,218 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var isArray = sharedUtil.isArray;
 var EOF = {};
 var Name = function NameClosure() {
- function Name(name) {
-  this.name = name;
- }
- Name.prototype = {};
- var nameCache = Object.create(null);
- Name.get = function Name_get(name) {
-  var nameValue = nameCache[name];
-  return nameValue ? nameValue : nameCache[name] = new Name(name);
- };
- return Name;
+  function Name(name) {
+    this.name = name;
+  }
+  Name.prototype = {};
+  var nameCache = Object.create(null);
+  Name.get = function Name_get(name) {
+    var nameValue = nameCache[name];
+    return nameValue ? nameValue : nameCache[name] = new Name(name);
+  };
+  return Name;
 }();
 var Cmd = function CmdClosure() {
- function Cmd(cmd) {
-  this.cmd = cmd;
- }
- Cmd.prototype = {};
- var cmdCache = Object.create(null);
- Cmd.get = function Cmd_get(cmd) {
-  var cmdValue = cmdCache[cmd];
-  return cmdValue ? cmdValue : cmdCache[cmd] = new Cmd(cmd);
- };
- return Cmd;
+  function Cmd(cmd) {
+    this.cmd = cmd;
+  }
+  Cmd.prototype = {};
+  var cmdCache = Object.create(null);
+  Cmd.get = function Cmd_get(cmd) {
+    var cmdValue = cmdCache[cmd];
+    return cmdValue ? cmdValue : cmdCache[cmd] = new Cmd(cmd);
+  };
+  return Cmd;
 }();
 var Dict = function DictClosure() {
- var nonSerializable = function nonSerializableClosure() {
-  return nonSerializable;
- };
- function Dict(xref) {
-  this.map = Object.create(null);
-  this.xref = xref;
-  this.objId = null;
-  this.suppressEncryption = false;
-  this.__nonSerializable__ = nonSerializable;
- }
- Dict.prototype = {
-  assignXref: function Dict_assignXref(newXref) {
-   this.xref = newXref;
-  },
-  get: function Dict_get(key1, key2, key3) {
-   var value;
-   var xref = this.xref, suppressEncryption = this.suppressEncryption;
-   if (typeof (value = this.map[key1]) !== 'undefined' || key1 in this.map || typeof key2 === 'undefined') {
-    return xref ? xref.fetchIfRef(value, suppressEncryption) : value;
-   }
-   if (typeof (value = this.map[key2]) !== 'undefined' || key2 in this.map || typeof key3 === 'undefined') {
-    return xref ? xref.fetchIfRef(value, suppressEncryption) : value;
-   }
-   value = this.map[key3] || null;
-   return xref ? xref.fetchIfRef(value, suppressEncryption) : value;
-  },
-  getAsync: function Dict_getAsync(key1, key2, key3) {
-   var value;
-   var xref = this.xref, suppressEncryption = this.suppressEncryption;
-   if (typeof (value = this.map[key1]) !== 'undefined' || key1 in this.map || typeof key2 === 'undefined') {
-    if (xref) {
-     return xref.fetchIfRefAsync(value, suppressEncryption);
-    }
-    return Promise.resolve(value);
-   }
-   if (typeof (value = this.map[key2]) !== 'undefined' || key2 in this.map || typeof key3 === 'undefined') {
-    if (xref) {
-     return xref.fetchIfRefAsync(value, suppressEncryption);
-    }
-    return Promise.resolve(value);
-   }
-   value = this.map[key3] || null;
-   if (xref) {
-    return xref.fetchIfRefAsync(value, suppressEncryption);
-   }
-   return Promise.resolve(value);
-  },
-  getArray: function Dict_getArray(key1, key2, key3) {
-   var value = this.get(key1, key2, key3);
-   var xref = this.xref, suppressEncryption = this.suppressEncryption;
-   if (!isArray(value) || !xref) {
-    return value;
-   }
-   value = value.slice();
-   for (var i = 0, ii = value.length; i < ii; i++) {
-    if (!isRef(value[i])) {
-     continue;
-    }
-    value[i] = xref.fetch(value[i], suppressEncryption);
-   }
-   return value;
-  },
-  getRaw: function Dict_getRaw(key) {
-   return this.map[key];
-  },
-  getKeys: function Dict_getKeys() {
-   return Object.keys(this.map);
-  },
-  set: function Dict_set(key, value) {
-   this.map[key] = value;
-  },
-  has: function Dict_has(key) {
-   return key in this.map;
-  },
-  forEach: function Dict_forEach(callback) {
-   for (var key in this.map) {
-    callback(key, this.get(key));
-   }
+  var nonSerializable = function nonSerializableClosure() {
+    return nonSerializable;
+  };
+  function Dict(xref) {
+    this.map = Object.create(null);
+    this.xref = xref;
+    this.objId = null;
+    this.suppressEncryption = false;
+    this.__nonSerializable__ = nonSerializable;
   }
- };
- Dict.empty = new Dict(null);
- Dict.merge = function Dict_merge(xref, dictArray) {
-  var mergedDict = new Dict(xref);
-  for (var i = 0, ii = dictArray.length; i < ii; i++) {
-   var dict = dictArray[i];
-   if (!isDict(dict)) {
-    continue;
-   }
-   for (var keyName in dict.map) {
-    if (mergedDict.map[keyName]) {
-     continue;
+  Dict.prototype = {
+    assignXref: function Dict_assignXref(newXref) {
+      this.xref = newXref;
+    },
+    get: function Dict_get(key1, key2, key3) {
+      var value;
+      var xref = this.xref,
+          suppressEncryption = this.suppressEncryption;
+      if (typeof (value = this.map[key1]) !== 'undefined' || key1 in this.map || typeof key2 === 'undefined') {
+        return xref ? xref.fetchIfRef(value, suppressEncryption) : value;
+      }
+      if (typeof (value = this.map[key2]) !== 'undefined' || key2 in this.map || typeof key3 === 'undefined') {
+        return xref ? xref.fetchIfRef(value, suppressEncryption) : value;
+      }
+      value = this.map[key3] || null;
+      return xref ? xref.fetchIfRef(value, suppressEncryption) : value;
+    },
+    getAsync: function Dict_getAsync(key1, key2, key3) {
+      var value;
+      var xref = this.xref,
+          suppressEncryption = this.suppressEncryption;
+      if (typeof (value = this.map[key1]) !== 'undefined' || key1 in this.map || typeof key2 === 'undefined') {
+        if (xref) {
+          return xref.fetchIfRefAsync(value, suppressEncryption);
+        }
+        return Promise.resolve(value);
+      }
+      if (typeof (value = this.map[key2]) !== 'undefined' || key2 in this.map || typeof key3 === 'undefined') {
+        if (xref) {
+          return xref.fetchIfRefAsync(value, suppressEncryption);
+        }
+        return Promise.resolve(value);
+      }
+      value = this.map[key3] || null;
+      if (xref) {
+        return xref.fetchIfRefAsync(value, suppressEncryption);
+      }
+      return Promise.resolve(value);
+    },
+    getArray: function Dict_getArray(key1, key2, key3) {
+      var value = this.get(key1, key2, key3);
+      var xref = this.xref,
+          suppressEncryption = this.suppressEncryption;
+      if (!isArray(value) || !xref) {
+        return value;
+      }
+      value = value.slice();
+      for (var i = 0, ii = value.length; i < ii; i++) {
+        if (!isRef(value[i])) {
+          continue;
+        }
+        value[i] = xref.fetch(value[i], suppressEncryption);
+      }
+      return value;
+    },
+    getRaw: function Dict_getRaw(key) {
+      return this.map[key];
+    },
+    getKeys: function Dict_getKeys() {
+      return Object.keys(this.map);
+    },
+    set: function Dict_set(key, value) {
+      this.map[key] = value;
+    },
+    has: function Dict_has(key) {
+      return key in this.map;
+    },
+    forEach: function Dict_forEach(callback) {
+      for (var key in this.map) {
+        callback(key, this.get(key));
+      }
     }
-    mergedDict.map[keyName] = dict.map[keyName];
-   }
-  }
-  return mergedDict;
- };
- return Dict;
+  };
+  Dict.empty = new Dict(null);
+  Dict.merge = function Dict_merge(xref, dictArray) {
+    var mergedDict = new Dict(xref);
+    for (var i = 0, ii = dictArray.length; i < ii; i++) {
+      var dict = dictArray[i];
+      if (!isDict(dict)) {
+        continue;
+      }
+      for (var keyName in dict.map) {
+        if (mergedDict.map[keyName]) {
+          continue;
+        }
+        mergedDict.map[keyName] = dict.map[keyName];
+      }
+    }
+    return mergedDict;
+  };
+  return Dict;
 }();
 var Ref = function RefClosure() {
- function Ref(num, gen) {
-  this.num = num;
-  this.gen = gen;
- }
- Ref.prototype = {
-  toString: function Ref_toString() {
-   var str = this.num + 'R';
-   if (this.gen !== 0) {
-    str += this.gen;
-   }
-   return str;
+  function Ref(num, gen) {
+    this.num = num;
+    this.gen = gen;
   }
- };
- return Ref;
+  Ref.prototype = {
+    toString: function Ref_toString() {
+      var str = this.num + 'R';
+      if (this.gen !== 0) {
+        str += this.gen;
+      }
+      return str;
+    }
+  };
+  return Ref;
 }();
 var RefSet = function RefSetClosure() {
- function RefSet() {
-  this.dict = Object.create(null);
- }
- RefSet.prototype = {
-  has: function RefSet_has(ref) {
-   return ref.toString() in this.dict;
-  },
-  put: function RefSet_put(ref) {
-   this.dict[ref.toString()] = true;
-  },
-  remove: function RefSet_remove(ref) {
-   delete this.dict[ref.toString()];
+  function RefSet() {
+    this.dict = Object.create(null);
   }
- };
- return RefSet;
+  RefSet.prototype = {
+    has: function RefSet_has(ref) {
+      return ref.toString() in this.dict;
+    },
+    put: function RefSet_put(ref) {
+      this.dict[ref.toString()] = true;
+    },
+    remove: function RefSet_remove(ref) {
+      delete this.dict[ref.toString()];
+    }
+  };
+  return RefSet;
 }();
 var RefSetCache = function RefSetCacheClosure() {
- function RefSetCache() {
-  this.dict = Object.create(null);
- }
- RefSetCache.prototype = {
-  get: function RefSetCache_get(ref) {
-   return this.dict[ref.toString()];
-  },
-  has: function RefSetCache_has(ref) {
-   return ref.toString() in this.dict;
-  },
-  put: function RefSetCache_put(ref, obj) {
-   this.dict[ref.toString()] = obj;
-  },
-  putAlias: function RefSetCache_putAlias(ref, aliasRef) {
-   this.dict[ref.toString()] = this.get(aliasRef);
-  },
-  forEach: function RefSetCache_forEach(fn, thisArg) {
-   for (var i in this.dict) {
-    fn.call(thisArg, this.dict[i]);
-   }
-  },
-  clear: function RefSetCache_clear() {
-   this.dict = Object.create(null);
+  function RefSetCache() {
+    this.dict = Object.create(null);
   }
- };
- return RefSetCache;
+  RefSetCache.prototype = {
+    get: function RefSetCache_get(ref) {
+      return this.dict[ref.toString()];
+    },
+    has: function RefSetCache_has(ref) {
+      return ref.toString() in this.dict;
+    },
+    put: function RefSetCache_put(ref, obj) {
+      this.dict[ref.toString()] = obj;
+    },
+    putAlias: function RefSetCache_putAlias(ref, aliasRef) {
+      this.dict[ref.toString()] = this.get(aliasRef);
+    },
+    forEach: function RefSetCache_forEach(fn, thisArg) {
+      for (var i in this.dict) {
+        fn.call(thisArg, this.dict[i]);
+      }
+    },
+    clear: function RefSetCache_clear() {
+      this.dict = Object.create(null);
+    }
+  };
+  return RefSetCache;
 }();
 function isEOF(v) {
- return v === EOF;
+  return v === EOF;
 }
 function isName(v, name) {
- return v instanceof Name && (name === undefined || v.name === name);
+  return v instanceof Name && (name === undefined || v.name === name);
 }
 function isCmd(v, cmd) {
- return v instanceof Cmd && (cmd === undefined || v.cmd === cmd);
+  return v instanceof Cmd && (cmd === undefined || v.cmd === cmd);
 }
 function isDict(v, type) {
- return v instanceof Dict && (type === undefined || isName(v.get('Type'), type));
+  return v instanceof Dict && (type === undefined || isName(v.get('Type'), type));
 }
 function isRef(v) {
- return v instanceof Ref;
+  return v instanceof Ref;
 }
 function isRefsEqual(v1, v2) {
- return v1.num === v2.num && v1.gen === v2.gen;
+  return v1.num === v2.num && v1.gen === v2.gen;
 }
 function isStream(v) {
- return typeof v === 'object' && v !== null && v.getBytes !== undefined;
+  return typeof v === 'object' && v !== null && v.getBytes !== undefined;
 }
 exports.EOF = EOF;
 exports.Cmd = Cmd;

+ 174 - 173
lib/core/ps_parser.js

@@ -13,195 +13,196 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var error = sharedUtil.error;
 var isSpace = sharedUtil.isSpace;
 var EOF = corePrimitives.EOF;
 var PostScriptParser = function PostScriptParserClosure() {
- function PostScriptParser(lexer) {
-  this.lexer = lexer;
-  this.operators = [];
-  this.token = null;
-  this.prev = null;
- }
- PostScriptParser.prototype = {
-  nextToken: function PostScriptParser_nextToken() {
-   this.prev = this.token;
-   this.token = this.lexer.getToken();
-  },
-  accept: function PostScriptParser_accept(type) {
-   if (this.token.type === type) {
-    this.nextToken();
-    return true;
-   }
-   return false;
-  },
-  expect: function PostScriptParser_expect(type) {
-   if (this.accept(type)) {
-    return true;
-   }
-   error('Unexpected symbol: found ' + this.token.type + ' expected ' + type + '.');
-  },
-  parse: function PostScriptParser_parse() {
-   this.nextToken();
-   this.expect(PostScriptTokenTypes.LBRACE);
-   this.parseBlock();
-   this.expect(PostScriptTokenTypes.RBRACE);
-   return this.operators;
-  },
-  parseBlock: function PostScriptParser_parseBlock() {
-   while (true) {
-    if (this.accept(PostScriptTokenTypes.NUMBER)) {
-     this.operators.push(this.prev.value);
-    } else if (this.accept(PostScriptTokenTypes.OPERATOR)) {
-     this.operators.push(this.prev.value);
-    } else if (this.accept(PostScriptTokenTypes.LBRACE)) {
-     this.parseCondition();
-    } else {
-     return;
-    }
-   }
-  },
-  parseCondition: function PostScriptParser_parseCondition() {
-   var conditionLocation = this.operators.length;
-   this.operators.push(null, null);
-   this.parseBlock();
-   this.expect(PostScriptTokenTypes.RBRACE);
-   if (this.accept(PostScriptTokenTypes.IF)) {
-    this.operators[conditionLocation] = this.operators.length;
-    this.operators[conditionLocation + 1] = 'jz';
-   } else if (this.accept(PostScriptTokenTypes.LBRACE)) {
-    var jumpLocation = this.operators.length;
-    this.operators.push(null, null);
-    var endOfTrue = this.operators.length;
-    this.parseBlock();
-    this.expect(PostScriptTokenTypes.RBRACE);
-    this.expect(PostScriptTokenTypes.IFELSE);
-    this.operators[jumpLocation] = this.operators.length;
-    this.operators[jumpLocation + 1] = 'j';
-    this.operators[conditionLocation] = endOfTrue;
-    this.operators[conditionLocation + 1] = 'jz';
-   } else {
-    error('PS Function: error parsing conditional.');
-   }
+  function PostScriptParser(lexer) {
+    this.lexer = lexer;
+    this.operators = [];
+    this.token = null;
+    this.prev = null;
   }
- };
- return PostScriptParser;
+  PostScriptParser.prototype = {
+    nextToken: function PostScriptParser_nextToken() {
+      this.prev = this.token;
+      this.token = this.lexer.getToken();
+    },
+    accept: function PostScriptParser_accept(type) {
+      if (this.token.type === type) {
+        this.nextToken();
+        return true;
+      }
+      return false;
+    },
+    expect: function PostScriptParser_expect(type) {
+      if (this.accept(type)) {
+        return true;
+      }
+      error('Unexpected symbol: found ' + this.token.type + ' expected ' + type + '.');
+    },
+    parse: function PostScriptParser_parse() {
+      this.nextToken();
+      this.expect(PostScriptTokenTypes.LBRACE);
+      this.parseBlock();
+      this.expect(PostScriptTokenTypes.RBRACE);
+      return this.operators;
+    },
+    parseBlock: function PostScriptParser_parseBlock() {
+      while (true) {
+        if (this.accept(PostScriptTokenTypes.NUMBER)) {
+          this.operators.push(this.prev.value);
+        } else if (this.accept(PostScriptTokenTypes.OPERATOR)) {
+          this.operators.push(this.prev.value);
+        } else if (this.accept(PostScriptTokenTypes.LBRACE)) {
+          this.parseCondition();
+        } else {
+          return;
+        }
+      }
+    },
+    parseCondition: function PostScriptParser_parseCondition() {
+      var conditionLocation = this.operators.length;
+      this.operators.push(null, null);
+      this.parseBlock();
+      this.expect(PostScriptTokenTypes.RBRACE);
+      if (this.accept(PostScriptTokenTypes.IF)) {
+        this.operators[conditionLocation] = this.operators.length;
+        this.operators[conditionLocation + 1] = 'jz';
+      } else if (this.accept(PostScriptTokenTypes.LBRACE)) {
+        var jumpLocation = this.operators.length;
+        this.operators.push(null, null);
+        var endOfTrue = this.operators.length;
+        this.parseBlock();
+        this.expect(PostScriptTokenTypes.RBRACE);
+        this.expect(PostScriptTokenTypes.IFELSE);
+        this.operators[jumpLocation] = this.operators.length;
+        this.operators[jumpLocation + 1] = 'j';
+        this.operators[conditionLocation] = endOfTrue;
+        this.operators[conditionLocation + 1] = 'jz';
+      } else {
+        error('PS Function: error parsing conditional.');
+      }
+    }
+  };
+  return PostScriptParser;
 }();
 var PostScriptTokenTypes = {
- LBRACE: 0,
- RBRACE: 1,
- NUMBER: 2,
- OPERATOR: 3,
- IF: 4,
- IFELSE: 5
+  LBRACE: 0,
+  RBRACE: 1,
+  NUMBER: 2,
+  OPERATOR: 3,
+  IF: 4,
+  IFELSE: 5
 };
 var PostScriptToken = function PostScriptTokenClosure() {
- function PostScriptToken(type, value) {
-  this.type = type;
-  this.value = value;
- }
- var opCache = Object.create(null);
- PostScriptToken.getOperator = function PostScriptToken_getOperator(op) {
-  var opValue = opCache[op];
-  if (opValue) {
-   return opValue;
+  function PostScriptToken(type, value) {
+    this.type = type;
+    this.value = value;
   }
-  return opCache[op] = new PostScriptToken(PostScriptTokenTypes.OPERATOR, op);
- };
- PostScriptToken.LBRACE = new PostScriptToken(PostScriptTokenTypes.LBRACE, '{');
- PostScriptToken.RBRACE = new PostScriptToken(PostScriptTokenTypes.RBRACE, '}');
- PostScriptToken.IF = new PostScriptToken(PostScriptTokenTypes.IF, 'IF');
- PostScriptToken.IFELSE = new PostScriptToken(PostScriptTokenTypes.IFELSE, 'IFELSE');
- return PostScriptToken;
+  var opCache = Object.create(null);
+  PostScriptToken.getOperator = function PostScriptToken_getOperator(op) {
+    var opValue = opCache[op];
+    if (opValue) {
+      return opValue;
+    }
+    return opCache[op] = new PostScriptToken(PostScriptTokenTypes.OPERATOR, op);
+  };
+  PostScriptToken.LBRACE = new PostScriptToken(PostScriptTokenTypes.LBRACE, '{');
+  PostScriptToken.RBRACE = new PostScriptToken(PostScriptTokenTypes.RBRACE, '}');
+  PostScriptToken.IF = new PostScriptToken(PostScriptTokenTypes.IF, 'IF');
+  PostScriptToken.IFELSE = new PostScriptToken(PostScriptTokenTypes.IFELSE, 'IFELSE');
+  return PostScriptToken;
 }();
 var PostScriptLexer = function PostScriptLexerClosure() {
- function PostScriptLexer(stream) {
-  this.stream = stream;
-  this.nextChar();
-  this.strBuf = [];
- }
- PostScriptLexer.prototype = {
-  nextChar: function PostScriptLexer_nextChar() {
-   return this.currentChar = this.stream.getByte();
-  },
-  getToken: function PostScriptLexer_getToken() {
-   var comment = false;
-   var ch = this.currentChar;
-   while (true) {
-    if (ch < 0) {
-     return EOF;
-    }
-    if (comment) {
-     if (ch === 0x0A || ch === 0x0D) {
-      comment = false;
-     }
-    } else if (ch === 0x25) {
-     comment = true;
-    } else if (!isSpace(ch)) {
-     break;
-    }
-    ch = this.nextChar();
-   }
-   switch (ch | 0) {
-   case 0x30:
-   case 0x31:
-   case 0x32:
-   case 0x33:
-   case 0x34:
-   case 0x35:
-   case 0x36:
-   case 0x37:
-   case 0x38:
-   case 0x39:
-   case 0x2B:
-   case 0x2D:
-   case 0x2E:
-    return new PostScriptToken(PostScriptTokenTypes.NUMBER, this.getNumber());
-   case 0x7B:
-    this.nextChar();
-    return PostScriptToken.LBRACE;
-   case 0x7D:
+  function PostScriptLexer(stream) {
+    this.stream = stream;
     this.nextChar();
-    return PostScriptToken.RBRACE;
-   }
-   var strBuf = this.strBuf;
-   strBuf.length = 0;
-   strBuf[0] = String.fromCharCode(ch);
-   while ((ch = this.nextChar()) >= 0 && (ch >= 0x41 && ch <= 0x5A || ch >= 0x61 && ch <= 0x7A)) {
-    strBuf.push(String.fromCharCode(ch));
-   }
-   var str = strBuf.join('');
-   switch (str.toLowerCase()) {
-   case 'if':
-    return PostScriptToken.IF;
-   case 'ifelse':
-    return PostScriptToken.IFELSE;
-   default:
-    return PostScriptToken.getOperator(str);
-   }
-  },
-  getNumber: function PostScriptLexer_getNumber() {
-   var ch = this.currentChar;
-   var strBuf = this.strBuf;
-   strBuf.length = 0;
-   strBuf[0] = String.fromCharCode(ch);
-   while ((ch = this.nextChar()) >= 0) {
-    if (ch >= 0x30 && ch <= 0x39 || ch === 0x2D || ch === 0x2E) {
-     strBuf.push(String.fromCharCode(ch));
-    } else {
-     break;
-    }
-   }
-   var value = parseFloat(strBuf.join(''));
-   if (isNaN(value)) {
-    error('Invalid floating point number: ' + value);
-   }
-   return value;
+    this.strBuf = [];
   }
- };
- return PostScriptLexer;
+  PostScriptLexer.prototype = {
+    nextChar: function PostScriptLexer_nextChar() {
+      return this.currentChar = this.stream.getByte();
+    },
+    getToken: function PostScriptLexer_getToken() {
+      var comment = false;
+      var ch = this.currentChar;
+      while (true) {
+        if (ch < 0) {
+          return EOF;
+        }
+        if (comment) {
+          if (ch === 0x0A || ch === 0x0D) {
+            comment = false;
+          }
+        } else if (ch === 0x25) {
+          comment = true;
+        } else if (!isSpace(ch)) {
+          break;
+        }
+        ch = this.nextChar();
+      }
+      switch (ch | 0) {
+        case 0x30:
+        case 0x31:
+        case 0x32:
+        case 0x33:
+        case 0x34:
+        case 0x35:
+        case 0x36:
+        case 0x37:
+        case 0x38:
+        case 0x39:
+        case 0x2B:
+        case 0x2D:
+        case 0x2E:
+          return new PostScriptToken(PostScriptTokenTypes.NUMBER, this.getNumber());
+        case 0x7B:
+          this.nextChar();
+          return PostScriptToken.LBRACE;
+        case 0x7D:
+          this.nextChar();
+          return PostScriptToken.RBRACE;
+      }
+      var strBuf = this.strBuf;
+      strBuf.length = 0;
+      strBuf[0] = String.fromCharCode(ch);
+      while ((ch = this.nextChar()) >= 0 && (ch >= 0x41 && ch <= 0x5A || ch >= 0x61 && ch <= 0x7A)) {
+        strBuf.push(String.fromCharCode(ch));
+      }
+      var str = strBuf.join('');
+      switch (str.toLowerCase()) {
+        case 'if':
+          return PostScriptToken.IF;
+        case 'ifelse':
+          return PostScriptToken.IFELSE;
+        default:
+          return PostScriptToken.getOperator(str);
+      }
+    },
+    getNumber: function PostScriptLexer_getNumber() {
+      var ch = this.currentChar;
+      var strBuf = this.strBuf;
+      strBuf.length = 0;
+      strBuf[0] = String.fromCharCode(ch);
+      while ((ch = this.nextChar()) >= 0) {
+        if (ch >= 0x30 && ch <= 0x39 || ch === 0x2D || ch === 0x2E) {
+          strBuf.push(String.fromCharCode(ch));
+        } else {
+          break;
+        }
+      }
+      var value = parseFloat(strBuf.join(''));
+      if (isNaN(value)) {
+        error('Invalid floating point number: ' + value);
+      }
+      return value;
+    }
+  };
+  return PostScriptLexer;
 }();
 exports.PostScriptLexer = PostScriptLexer;
 exports.PostScriptParser = PostScriptParser;

+ 618 - 617
lib/core/standard_fonts.js

@@ -13,636 +13,637 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var getLookupTableFactory = sharedUtil.getLookupTableFactory;
 var getStdFontMap = getLookupTableFactory(function (t) {
- t['ArialNarrow'] = 'Helvetica';
- t['ArialNarrow-Bold'] = 'Helvetica-Bold';
- t['ArialNarrow-BoldItalic'] = 'Helvetica-BoldOblique';
- t['ArialNarrow-Italic'] = 'Helvetica-Oblique';
- t['ArialBlack'] = 'Helvetica';
- t['ArialBlack-Bold'] = 'Helvetica-Bold';
- t['ArialBlack-BoldItalic'] = 'Helvetica-BoldOblique';
- t['ArialBlack-Italic'] = 'Helvetica-Oblique';
- t['Arial-Black'] = 'Helvetica';
- t['Arial-Black-Bold'] = 'Helvetica-Bold';
- t['Arial-Black-BoldItalic'] = 'Helvetica-BoldOblique';
- t['Arial-Black-Italic'] = 'Helvetica-Oblique';
- t['Arial'] = 'Helvetica';
- t['Arial-Bold'] = 'Helvetica-Bold';
- t['Arial-BoldItalic'] = 'Helvetica-BoldOblique';
- t['Arial-Italic'] = 'Helvetica-Oblique';
- t['Arial-BoldItalicMT'] = 'Helvetica-BoldOblique';
- t['Arial-BoldMT'] = 'Helvetica-Bold';
- t['Arial-ItalicMT'] = 'Helvetica-Oblique';
- t['ArialMT'] = 'Helvetica';
- t['Courier-Bold'] = 'Courier-Bold';
- t['Courier-BoldItalic'] = 'Courier-BoldOblique';
- t['Courier-Italic'] = 'Courier-Oblique';
- t['CourierNew'] = 'Courier';
- t['CourierNew-Bold'] = 'Courier-Bold';
- t['CourierNew-BoldItalic'] = 'Courier-BoldOblique';
- t['CourierNew-Italic'] = 'Courier-Oblique';
- t['CourierNewPS-BoldItalicMT'] = 'Courier-BoldOblique';
- t['CourierNewPS-BoldMT'] = 'Courier-Bold';
- t['CourierNewPS-ItalicMT'] = 'Courier-Oblique';
- t['CourierNewPSMT'] = 'Courier';
- t['Helvetica'] = 'Helvetica';
- t['Helvetica-Bold'] = 'Helvetica-Bold';
- t['Helvetica-BoldItalic'] = 'Helvetica-BoldOblique';
- t['Helvetica-BoldOblique'] = 'Helvetica-BoldOblique';
- t['Helvetica-Italic'] = 'Helvetica-Oblique';
- t['Helvetica-Oblique'] = 'Helvetica-Oblique';
- t['Symbol-Bold'] = 'Symbol';
- t['Symbol-BoldItalic'] = 'Symbol';
- t['Symbol-Italic'] = 'Symbol';
- t['TimesNewRoman'] = 'Times-Roman';
- t['TimesNewRoman-Bold'] = 'Times-Bold';
- t['TimesNewRoman-BoldItalic'] = 'Times-BoldItalic';
- t['TimesNewRoman-Italic'] = 'Times-Italic';
- t['TimesNewRomanPS'] = 'Times-Roman';
- t['TimesNewRomanPS-Bold'] = 'Times-Bold';
- t['TimesNewRomanPS-BoldItalic'] = 'Times-BoldItalic';
- t['TimesNewRomanPS-BoldItalicMT'] = 'Times-BoldItalic';
- t['TimesNewRomanPS-BoldMT'] = 'Times-Bold';
- t['TimesNewRomanPS-Italic'] = 'Times-Italic';
- t['TimesNewRomanPS-ItalicMT'] = 'Times-Italic';
- t['TimesNewRomanPSMT'] = 'Times-Roman';
- t['TimesNewRomanPSMT-Bold'] = 'Times-Bold';
- t['TimesNewRomanPSMT-BoldItalic'] = 'Times-BoldItalic';
- t['TimesNewRomanPSMT-Italic'] = 'Times-Italic';
+  t['ArialNarrow'] = 'Helvetica';
+  t['ArialNarrow-Bold'] = 'Helvetica-Bold';
+  t['ArialNarrow-BoldItalic'] = 'Helvetica-BoldOblique';
+  t['ArialNarrow-Italic'] = 'Helvetica-Oblique';
+  t['ArialBlack'] = 'Helvetica';
+  t['ArialBlack-Bold'] = 'Helvetica-Bold';
+  t['ArialBlack-BoldItalic'] = 'Helvetica-BoldOblique';
+  t['ArialBlack-Italic'] = 'Helvetica-Oblique';
+  t['Arial-Black'] = 'Helvetica';
+  t['Arial-Black-Bold'] = 'Helvetica-Bold';
+  t['Arial-Black-BoldItalic'] = 'Helvetica-BoldOblique';
+  t['Arial-Black-Italic'] = 'Helvetica-Oblique';
+  t['Arial'] = 'Helvetica';
+  t['Arial-Bold'] = 'Helvetica-Bold';
+  t['Arial-BoldItalic'] = 'Helvetica-BoldOblique';
+  t['Arial-Italic'] = 'Helvetica-Oblique';
+  t['Arial-BoldItalicMT'] = 'Helvetica-BoldOblique';
+  t['Arial-BoldMT'] = 'Helvetica-Bold';
+  t['Arial-ItalicMT'] = 'Helvetica-Oblique';
+  t['ArialMT'] = 'Helvetica';
+  t['Courier-Bold'] = 'Courier-Bold';
+  t['Courier-BoldItalic'] = 'Courier-BoldOblique';
+  t['Courier-Italic'] = 'Courier-Oblique';
+  t['CourierNew'] = 'Courier';
+  t['CourierNew-Bold'] = 'Courier-Bold';
+  t['CourierNew-BoldItalic'] = 'Courier-BoldOblique';
+  t['CourierNew-Italic'] = 'Courier-Oblique';
+  t['CourierNewPS-BoldItalicMT'] = 'Courier-BoldOblique';
+  t['CourierNewPS-BoldMT'] = 'Courier-Bold';
+  t['CourierNewPS-ItalicMT'] = 'Courier-Oblique';
+  t['CourierNewPSMT'] = 'Courier';
+  t['Helvetica'] = 'Helvetica';
+  t['Helvetica-Bold'] = 'Helvetica-Bold';
+  t['Helvetica-BoldItalic'] = 'Helvetica-BoldOblique';
+  t['Helvetica-BoldOblique'] = 'Helvetica-BoldOblique';
+  t['Helvetica-Italic'] = 'Helvetica-Oblique';
+  t['Helvetica-Oblique'] = 'Helvetica-Oblique';
+  t['Symbol-Bold'] = 'Symbol';
+  t['Symbol-BoldItalic'] = 'Symbol';
+  t['Symbol-Italic'] = 'Symbol';
+  t['TimesNewRoman'] = 'Times-Roman';
+  t['TimesNewRoman-Bold'] = 'Times-Bold';
+  t['TimesNewRoman-BoldItalic'] = 'Times-BoldItalic';
+  t['TimesNewRoman-Italic'] = 'Times-Italic';
+  t['TimesNewRomanPS'] = 'Times-Roman';
+  t['TimesNewRomanPS-Bold'] = 'Times-Bold';
+  t['TimesNewRomanPS-BoldItalic'] = 'Times-BoldItalic';
+  t['TimesNewRomanPS-BoldItalicMT'] = 'Times-BoldItalic';
+  t['TimesNewRomanPS-BoldMT'] = 'Times-Bold';
+  t['TimesNewRomanPS-Italic'] = 'Times-Italic';
+  t['TimesNewRomanPS-ItalicMT'] = 'Times-Italic';
+  t['TimesNewRomanPSMT'] = 'Times-Roman';
+  t['TimesNewRomanPSMT-Bold'] = 'Times-Bold';
+  t['TimesNewRomanPSMT-BoldItalic'] = 'Times-BoldItalic';
+  t['TimesNewRomanPSMT-Italic'] = 'Times-Italic';
 });
 var getNonStdFontMap = getLookupTableFactory(function (t) {
- t['CenturyGothic'] = 'Helvetica';
- t['CenturyGothic-Bold'] = 'Helvetica-Bold';
- t['CenturyGothic-BoldItalic'] = 'Helvetica-BoldOblique';
- t['CenturyGothic-Italic'] = 'Helvetica-Oblique';
- t['ComicSansMS'] = 'Comic Sans MS';
- t['ComicSansMS-Bold'] = 'Comic Sans MS-Bold';
- t['ComicSansMS-BoldItalic'] = 'Comic Sans MS-BoldItalic';
- t['ComicSansMS-Italic'] = 'Comic Sans MS-Italic';
- t['LucidaConsole'] = 'Courier';
- t['LucidaConsole-Bold'] = 'Courier-Bold';
- t['LucidaConsole-BoldItalic'] = 'Courier-BoldOblique';
- t['LucidaConsole-Italic'] = 'Courier-Oblique';
- t['MS-Gothic'] = 'MS Gothic';
- t['MS-Gothic-Bold'] = 'MS Gothic-Bold';
- t['MS-Gothic-BoldItalic'] = 'MS Gothic-BoldItalic';
- t['MS-Gothic-Italic'] = 'MS Gothic-Italic';
- t['MS-Mincho'] = 'MS Mincho';
- t['MS-Mincho-Bold'] = 'MS Mincho-Bold';
- t['MS-Mincho-BoldItalic'] = 'MS Mincho-BoldItalic';
- t['MS-Mincho-Italic'] = 'MS Mincho-Italic';
- t['MS-PGothic'] = 'MS PGothic';
- t['MS-PGothic-Bold'] = 'MS PGothic-Bold';
- t['MS-PGothic-BoldItalic'] = 'MS PGothic-BoldItalic';
- t['MS-PGothic-Italic'] = 'MS PGothic-Italic';
- t['MS-PMincho'] = 'MS PMincho';
- t['MS-PMincho-Bold'] = 'MS PMincho-Bold';
- t['MS-PMincho-BoldItalic'] = 'MS PMincho-BoldItalic';
- t['MS-PMincho-Italic'] = 'MS PMincho-Italic';
- t['NuptialScript'] = 'Times-Italic';
- t['Wingdings'] = 'ZapfDingbats';
+  t['CenturyGothic'] = 'Helvetica';
+  t['CenturyGothic-Bold'] = 'Helvetica-Bold';
+  t['CenturyGothic-BoldItalic'] = 'Helvetica-BoldOblique';
+  t['CenturyGothic-Italic'] = 'Helvetica-Oblique';
+  t['ComicSansMS'] = 'Comic Sans MS';
+  t['ComicSansMS-Bold'] = 'Comic Sans MS-Bold';
+  t['ComicSansMS-BoldItalic'] = 'Comic Sans MS-BoldItalic';
+  t['ComicSansMS-Italic'] = 'Comic Sans MS-Italic';
+  t['LucidaConsole'] = 'Courier';
+  t['LucidaConsole-Bold'] = 'Courier-Bold';
+  t['LucidaConsole-BoldItalic'] = 'Courier-BoldOblique';
+  t['LucidaConsole-Italic'] = 'Courier-Oblique';
+  t['MS-Gothic'] = 'MS Gothic';
+  t['MS-Gothic-Bold'] = 'MS Gothic-Bold';
+  t['MS-Gothic-BoldItalic'] = 'MS Gothic-BoldItalic';
+  t['MS-Gothic-Italic'] = 'MS Gothic-Italic';
+  t['MS-Mincho'] = 'MS Mincho';
+  t['MS-Mincho-Bold'] = 'MS Mincho-Bold';
+  t['MS-Mincho-BoldItalic'] = 'MS Mincho-BoldItalic';
+  t['MS-Mincho-Italic'] = 'MS Mincho-Italic';
+  t['MS-PGothic'] = 'MS PGothic';
+  t['MS-PGothic-Bold'] = 'MS PGothic-Bold';
+  t['MS-PGothic-BoldItalic'] = 'MS PGothic-BoldItalic';
+  t['MS-PGothic-Italic'] = 'MS PGothic-Italic';
+  t['MS-PMincho'] = 'MS PMincho';
+  t['MS-PMincho-Bold'] = 'MS PMincho-Bold';
+  t['MS-PMincho-BoldItalic'] = 'MS PMincho-BoldItalic';
+  t['MS-PMincho-Italic'] = 'MS PMincho-Italic';
+  t['NuptialScript'] = 'Times-Italic';
+  t['Wingdings'] = 'ZapfDingbats';
 });
 var getSerifFonts = getLookupTableFactory(function (t) {
- t['Adobe Jenson'] = true;
- t['Adobe Text'] = true;
- t['Albertus'] = true;
- t['Aldus'] = true;
- t['Alexandria'] = true;
- t['Algerian'] = true;
- t['American Typewriter'] = true;
- t['Antiqua'] = true;
- t['Apex'] = true;
- t['Arno'] = true;
- t['Aster'] = true;
- t['Aurora'] = true;
- t['Baskerville'] = true;
- t['Bell'] = true;
- t['Bembo'] = true;
- t['Bembo Schoolbook'] = true;
- t['Benguiat'] = true;
- t['Berkeley Old Style'] = true;
- t['Bernhard Modern'] = true;
- t['Berthold City'] = true;
- t['Bodoni'] = true;
- t['Bauer Bodoni'] = true;
- t['Book Antiqua'] = true;
- t['Bookman'] = true;
- t['Bordeaux Roman'] = true;
- t['Californian FB'] = true;
- t['Calisto'] = true;
- t['Calvert'] = true;
- t['Capitals'] = true;
- t['Cambria'] = true;
- t['Cartier'] = true;
- t['Caslon'] = true;
- t['Catull'] = true;
- t['Centaur'] = true;
- t['Century Old Style'] = true;
- t['Century Schoolbook'] = true;
- t['Chaparral'] = true;
- t['Charis SIL'] = true;
- t['Cheltenham'] = true;
- t['Cholla Slab'] = true;
- t['Clarendon'] = true;
- t['Clearface'] = true;
- t['Cochin'] = true;
- t['Colonna'] = true;
- t['Computer Modern'] = true;
- t['Concrete Roman'] = true;
- t['Constantia'] = true;
- t['Cooper Black'] = true;
- t['Corona'] = true;
- t['Ecotype'] = true;
- t['Egyptienne'] = true;
- t['Elephant'] = true;
- t['Excelsior'] = true;
- t['Fairfield'] = true;
- t['FF Scala'] = true;
- t['Folkard'] = true;
- t['Footlight'] = true;
- t['FreeSerif'] = true;
- t['Friz Quadrata'] = true;
- t['Garamond'] = true;
- t['Gentium'] = true;
- t['Georgia'] = true;
- t['Gloucester'] = true;
- t['Goudy Old Style'] = true;
- t['Goudy Schoolbook'] = true;
- t['Goudy Pro Font'] = true;
- t['Granjon'] = true;
- t['Guardian Egyptian'] = true;
- t['Heather'] = true;
- t['Hercules'] = true;
- t['High Tower Text'] = true;
- t['Hiroshige'] = true;
- t['Hoefler Text'] = true;
- t['Humana Serif'] = true;
- t['Imprint'] = true;
- t['Ionic No. 5'] = true;
- t['Janson'] = true;
- t['Joanna'] = true;
- t['Korinna'] = true;
- t['Lexicon'] = true;
- t['Liberation Serif'] = true;
- t['Linux Libertine'] = true;
- t['Literaturnaya'] = true;
- t['Lucida'] = true;
- t['Lucida Bright'] = true;
- t['Melior'] = true;
- t['Memphis'] = true;
- t['Miller'] = true;
- t['Minion'] = true;
- t['Modern'] = true;
- t['Mona Lisa'] = true;
- t['Mrs Eaves'] = true;
- t['MS Serif'] = true;
- t['Museo Slab'] = true;
- t['New York'] = true;
- t['Nimbus Roman'] = true;
- t['NPS Rawlinson Roadway'] = true;
- t['NuptialScript'] = true;
- t['Palatino'] = true;
- t['Perpetua'] = true;
- t['Plantin'] = true;
- t['Plantin Schoolbook'] = true;
- t['Playbill'] = true;
- t['Poor Richard'] = true;
- t['Rawlinson Roadway'] = true;
- t['Renault'] = true;
- t['Requiem'] = true;
- t['Rockwell'] = true;
- t['Roman'] = true;
- t['Rotis Serif'] = true;
- t['Sabon'] = true;
- t['Scala'] = true;
- t['Seagull'] = true;
- t['Sistina'] = true;
- t['Souvenir'] = true;
- t['STIX'] = true;
- t['Stone Informal'] = true;
- t['Stone Serif'] = true;
- t['Sylfaen'] = true;
- t['Times'] = true;
- t['Trajan'] = true;
- t['Trinité'] = true;
- t['Trump Mediaeval'] = true;
- t['Utopia'] = true;
- t['Vale Type'] = true;
- t['Bitstream Vera'] = true;
- t['Vera Serif'] = true;
- t['Versailles'] = true;
- t['Wanted'] = true;
- t['Weiss'] = true;
- t['Wide Latin'] = true;
- t['Windsor'] = true;
- t['XITS'] = true;
+  t['Adobe Jenson'] = true;
+  t['Adobe Text'] = true;
+  t['Albertus'] = true;
+  t['Aldus'] = true;
+  t['Alexandria'] = true;
+  t['Algerian'] = true;
+  t['American Typewriter'] = true;
+  t['Antiqua'] = true;
+  t['Apex'] = true;
+  t['Arno'] = true;
+  t['Aster'] = true;
+  t['Aurora'] = true;
+  t['Baskerville'] = true;
+  t['Bell'] = true;
+  t['Bembo'] = true;
+  t['Bembo Schoolbook'] = true;
+  t['Benguiat'] = true;
+  t['Berkeley Old Style'] = true;
+  t['Bernhard Modern'] = true;
+  t['Berthold City'] = true;
+  t['Bodoni'] = true;
+  t['Bauer Bodoni'] = true;
+  t['Book Antiqua'] = true;
+  t['Bookman'] = true;
+  t['Bordeaux Roman'] = true;
+  t['Californian FB'] = true;
+  t['Calisto'] = true;
+  t['Calvert'] = true;
+  t['Capitals'] = true;
+  t['Cambria'] = true;
+  t['Cartier'] = true;
+  t['Caslon'] = true;
+  t['Catull'] = true;
+  t['Centaur'] = true;
+  t['Century Old Style'] = true;
+  t['Century Schoolbook'] = true;
+  t['Chaparral'] = true;
+  t['Charis SIL'] = true;
+  t['Cheltenham'] = true;
+  t['Cholla Slab'] = true;
+  t['Clarendon'] = true;
+  t['Clearface'] = true;
+  t['Cochin'] = true;
+  t['Colonna'] = true;
+  t['Computer Modern'] = true;
+  t['Concrete Roman'] = true;
+  t['Constantia'] = true;
+  t['Cooper Black'] = true;
+  t['Corona'] = true;
+  t['Ecotype'] = true;
+  t['Egyptienne'] = true;
+  t['Elephant'] = true;
+  t['Excelsior'] = true;
+  t['Fairfield'] = true;
+  t['FF Scala'] = true;
+  t['Folkard'] = true;
+  t['Footlight'] = true;
+  t['FreeSerif'] = true;
+  t['Friz Quadrata'] = true;
+  t['Garamond'] = true;
+  t['Gentium'] = true;
+  t['Georgia'] = true;
+  t['Gloucester'] = true;
+  t['Goudy Old Style'] = true;
+  t['Goudy Schoolbook'] = true;
+  t['Goudy Pro Font'] = true;
+  t['Granjon'] = true;
+  t['Guardian Egyptian'] = true;
+  t['Heather'] = true;
+  t['Hercules'] = true;
+  t['High Tower Text'] = true;
+  t['Hiroshige'] = true;
+  t['Hoefler Text'] = true;
+  t['Humana Serif'] = true;
+  t['Imprint'] = true;
+  t['Ionic No. 5'] = true;
+  t['Janson'] = true;
+  t['Joanna'] = true;
+  t['Korinna'] = true;
+  t['Lexicon'] = true;
+  t['Liberation Serif'] = true;
+  t['Linux Libertine'] = true;
+  t['Literaturnaya'] = true;
+  t['Lucida'] = true;
+  t['Lucida Bright'] = true;
+  t['Melior'] = true;
+  t['Memphis'] = true;
+  t['Miller'] = true;
+  t['Minion'] = true;
+  t['Modern'] = true;
+  t['Mona Lisa'] = true;
+  t['Mrs Eaves'] = true;
+  t['MS Serif'] = true;
+  t['Museo Slab'] = true;
+  t['New York'] = true;
+  t['Nimbus Roman'] = true;
+  t['NPS Rawlinson Roadway'] = true;
+  t['NuptialScript'] = true;
+  t['Palatino'] = true;
+  t['Perpetua'] = true;
+  t['Plantin'] = true;
+  t['Plantin Schoolbook'] = true;
+  t['Playbill'] = true;
+  t['Poor Richard'] = true;
+  t['Rawlinson Roadway'] = true;
+  t['Renault'] = true;
+  t['Requiem'] = true;
+  t['Rockwell'] = true;
+  t['Roman'] = true;
+  t['Rotis Serif'] = true;
+  t['Sabon'] = true;
+  t['Scala'] = true;
+  t['Seagull'] = true;
+  t['Sistina'] = true;
+  t['Souvenir'] = true;
+  t['STIX'] = true;
+  t['Stone Informal'] = true;
+  t['Stone Serif'] = true;
+  t['Sylfaen'] = true;
+  t['Times'] = true;
+  t['Trajan'] = true;
+  t['Trinité'] = true;
+  t['Trump Mediaeval'] = true;
+  t['Utopia'] = true;
+  t['Vale Type'] = true;
+  t['Bitstream Vera'] = true;
+  t['Vera Serif'] = true;
+  t['Versailles'] = true;
+  t['Wanted'] = true;
+  t['Weiss'] = true;
+  t['Wide Latin'] = true;
+  t['Windsor'] = true;
+  t['XITS'] = true;
 });
 var getSymbolsFonts = getLookupTableFactory(function (t) {
- t['Dingbats'] = true;
- t['Symbol'] = true;
- t['ZapfDingbats'] = true;
+  t['Dingbats'] = true;
+  t['Symbol'] = true;
+  t['ZapfDingbats'] = true;
 });
 var getGlyphMapForStandardFonts = getLookupTableFactory(function (t) {
- t[2] = 10;
- t[3] = 32;
- t[4] = 33;
- t[5] = 34;
- t[6] = 35;
- t[7] = 36;
- t[8] = 37;
- t[9] = 38;
- t[10] = 39;
- t[11] = 40;
- t[12] = 41;
- t[13] = 42;
- t[14] = 43;
- t[15] = 44;
- t[16] = 45;
- t[17] = 46;
- t[18] = 47;
- t[19] = 48;
- t[20] = 49;
- t[21] = 50;
- t[22] = 51;
- t[23] = 52;
- t[24] = 53;
- t[25] = 54;
- t[26] = 55;
- t[27] = 56;
- t[28] = 57;
- t[29] = 58;
- t[30] = 894;
- t[31] = 60;
- t[32] = 61;
- t[33] = 62;
- t[34] = 63;
- t[35] = 64;
- t[36] = 65;
- t[37] = 66;
- t[38] = 67;
- t[39] = 68;
- t[40] = 69;
- t[41] = 70;
- t[42] = 71;
- t[43] = 72;
- t[44] = 73;
- t[45] = 74;
- t[46] = 75;
- t[47] = 76;
- t[48] = 77;
- t[49] = 78;
- t[50] = 79;
- t[51] = 80;
- t[52] = 81;
- t[53] = 82;
- t[54] = 83;
- t[55] = 84;
- t[56] = 85;
- t[57] = 86;
- t[58] = 87;
- t[59] = 88;
- t[60] = 89;
- t[61] = 90;
- t[62] = 91;
- t[63] = 92;
- t[64] = 93;
- t[65] = 94;
- t[66] = 95;
- t[67] = 96;
- t[68] = 97;
- t[69] = 98;
- t[70] = 99;
- t[71] = 100;
- t[72] = 101;
- t[73] = 102;
- t[74] = 103;
- t[75] = 104;
- t[76] = 105;
- t[77] = 106;
- t[78] = 107;
- t[79] = 108;
- t[80] = 109;
- t[81] = 110;
- t[82] = 111;
- t[83] = 112;
- t[84] = 113;
- t[85] = 114;
- t[86] = 115;
- t[87] = 116;
- t[88] = 117;
- t[89] = 118;
- t[90] = 119;
- t[91] = 120;
- t[92] = 121;
- t[93] = 122;
- t[94] = 123;
- t[95] = 124;
- t[96] = 125;
- t[97] = 126;
- t[98] = 196;
- t[99] = 197;
- t[100] = 199;
- t[101] = 201;
- t[102] = 209;
- t[103] = 214;
- t[104] = 220;
- t[105] = 225;
- t[106] = 224;
- t[107] = 226;
- t[108] = 228;
- t[109] = 227;
- t[110] = 229;
- t[111] = 231;
- t[112] = 233;
- t[113] = 232;
- t[114] = 234;
- t[115] = 235;
- t[116] = 237;
- t[117] = 236;
- t[118] = 238;
- t[119] = 239;
- t[120] = 241;
- t[121] = 243;
- t[122] = 242;
- t[123] = 244;
- t[124] = 246;
- t[125] = 245;
- t[126] = 250;
- t[127] = 249;
- t[128] = 251;
- t[129] = 252;
- t[130] = 8224;
- t[131] = 176;
- t[132] = 162;
- t[133] = 163;
- t[134] = 167;
- t[135] = 8226;
- t[136] = 182;
- t[137] = 223;
- t[138] = 174;
- t[139] = 169;
- t[140] = 8482;
- t[141] = 180;
- t[142] = 168;
- t[143] = 8800;
- t[144] = 198;
- t[145] = 216;
- t[146] = 8734;
- t[147] = 177;
- t[148] = 8804;
- t[149] = 8805;
- t[150] = 165;
- t[151] = 181;
- t[152] = 8706;
- t[153] = 8721;
- t[154] = 8719;
- t[156] = 8747;
- t[157] = 170;
- t[158] = 186;
- t[159] = 8486;
- t[160] = 230;
- t[161] = 248;
- t[162] = 191;
- t[163] = 161;
- t[164] = 172;
- t[165] = 8730;
- t[166] = 402;
- t[167] = 8776;
- t[168] = 8710;
- t[169] = 171;
- t[170] = 187;
- t[171] = 8230;
- t[210] = 218;
- t[223] = 711;
- t[224] = 321;
- t[225] = 322;
- t[227] = 353;
- t[229] = 382;
- t[234] = 253;
- t[252] = 263;
- t[253] = 268;
- t[254] = 269;
- t[258] = 258;
- t[260] = 260;
- t[261] = 261;
- t[265] = 280;
- t[266] = 281;
- t[268] = 283;
- t[269] = 313;
- t[275] = 323;
- t[276] = 324;
- t[278] = 328;
- t[284] = 345;
- t[285] = 346;
- t[286] = 347;
- t[292] = 367;
- t[295] = 377;
- t[296] = 378;
- t[298] = 380;
- t[305] = 963;
- t[306] = 964;
- t[307] = 966;
- t[308] = 8215;
- t[309] = 8252;
- t[310] = 8319;
- t[311] = 8359;
- t[312] = 8592;
- t[313] = 8593;
- t[337] = 9552;
- t[493] = 1039;
- t[494] = 1040;
- t[705] = 1524;
- t[706] = 8362;
- t[710] = 64288;
- t[711] = 64298;
- t[759] = 1617;
- t[761] = 1776;
- t[763] = 1778;
- t[775] = 1652;
- t[777] = 1764;
- t[778] = 1780;
- t[779] = 1781;
- t[780] = 1782;
- t[782] = 771;
- t[783] = 64726;
- t[786] = 8363;
- t[788] = 8532;
- t[790] = 768;
- t[791] = 769;
- t[792] = 768;
- t[795] = 803;
- t[797] = 64336;
- t[798] = 64337;
- t[799] = 64342;
- t[800] = 64343;
- t[801] = 64344;
- t[802] = 64345;
- t[803] = 64362;
- t[804] = 64363;
- t[805] = 64364;
- t[2424] = 7821;
- t[2425] = 7822;
- t[2426] = 7823;
- t[2427] = 7824;
- t[2428] = 7825;
- t[2429] = 7826;
- t[2430] = 7827;
- t[2433] = 7682;
- t[2678] = 8045;
- t[2679] = 8046;
- t[2830] = 1552;
- t[2838] = 686;
- t[2840] = 751;
- t[2842] = 753;
- t[2843] = 754;
- t[2844] = 755;
- t[2846] = 757;
- t[2856] = 767;
- t[2857] = 848;
- t[2858] = 849;
- t[2862] = 853;
- t[2863] = 854;
- t[2864] = 855;
- t[2865] = 861;
- t[2866] = 862;
- t[2906] = 7460;
- t[2908] = 7462;
- t[2909] = 7463;
- t[2910] = 7464;
- t[2912] = 7466;
- t[2913] = 7467;
- t[2914] = 7468;
- t[2916] = 7470;
- t[2917] = 7471;
- t[2918] = 7472;
- t[2920] = 7474;
- t[2921] = 7475;
- t[2922] = 7476;
- t[2924] = 7478;
- t[2925] = 7479;
- t[2926] = 7480;
- t[2928] = 7482;
- t[2929] = 7483;
- t[2930] = 7484;
- t[2932] = 7486;
- t[2933] = 7487;
- t[2934] = 7488;
- t[2936] = 7490;
- t[2937] = 7491;
- t[2938] = 7492;
- t[2940] = 7494;
- t[2941] = 7495;
- t[2942] = 7496;
- t[2944] = 7498;
- t[2946] = 7500;
- t[2948] = 7502;
- t[2950] = 7504;
- t[2951] = 7505;
- t[2952] = 7506;
- t[2954] = 7508;
- t[2955] = 7509;
- t[2956] = 7510;
- t[2958] = 7512;
- t[2959] = 7513;
- t[2960] = 7514;
- t[2962] = 7516;
- t[2963] = 7517;
- t[2964] = 7518;
- t[2966] = 7520;
- t[2967] = 7521;
- t[2968] = 7522;
- t[2970] = 7524;
- t[2971] = 7525;
- t[2972] = 7526;
- t[2974] = 7528;
- t[2975] = 7529;
- t[2976] = 7530;
- t[2978] = 1537;
- t[2979] = 1538;
- t[2980] = 1539;
- t[2982] = 1549;
- t[2983] = 1551;
- t[2984] = 1552;
- t[2986] = 1554;
- t[2987] = 1555;
- t[2988] = 1556;
- t[2990] = 1623;
- t[2991] = 1624;
- t[2995] = 1775;
- t[2999] = 1791;
- t[3002] = 64290;
- t[3003] = 64291;
- t[3004] = 64292;
- t[3006] = 64294;
- t[3007] = 64295;
- t[3008] = 64296;
- t[3011] = 1900;
- t[3014] = 8223;
- t[3015] = 8244;
- t[3017] = 7532;
- t[3018] = 7533;
- t[3019] = 7534;
- t[3075] = 7590;
- t[3076] = 7591;
- t[3079] = 7594;
- t[3080] = 7595;
- t[3083] = 7598;
- t[3084] = 7599;
- t[3087] = 7602;
- t[3088] = 7603;
- t[3091] = 7606;
- t[3092] = 7607;
- t[3095] = 7610;
- t[3096] = 7611;
- t[3099] = 7614;
- t[3100] = 7615;
- t[3103] = 7618;
- t[3104] = 7619;
- t[3107] = 8337;
- t[3108] = 8338;
- t[3116] = 1884;
- t[3119] = 1885;
- t[3120] = 1885;
- t[3123] = 1886;
- t[3124] = 1886;
- t[3127] = 1887;
- t[3128] = 1887;
- t[3131] = 1888;
- t[3132] = 1888;
- t[3135] = 1889;
- t[3136] = 1889;
- t[3139] = 1890;
- t[3140] = 1890;
- t[3143] = 1891;
- t[3144] = 1891;
- t[3147] = 1892;
- t[3148] = 1892;
- t[3153] = 580;
- t[3154] = 581;
- t[3157] = 584;
- t[3158] = 585;
- t[3161] = 588;
- t[3162] = 589;
- t[3165] = 891;
- t[3166] = 892;
- t[3169] = 1274;
- t[3170] = 1275;
- t[3173] = 1278;
- t[3174] = 1279;
- t[3181] = 7622;
- t[3182] = 7623;
- t[3282] = 11799;
- t[3316] = 578;
- t[3379] = 42785;
- t[3393] = 1159;
- t[3416] = 8377;
+  t[2] = 10;
+  t[3] = 32;
+  t[4] = 33;
+  t[5] = 34;
+  t[6] = 35;
+  t[7] = 36;
+  t[8] = 37;
+  t[9] = 38;
+  t[10] = 39;
+  t[11] = 40;
+  t[12] = 41;
+  t[13] = 42;
+  t[14] = 43;
+  t[15] = 44;
+  t[16] = 45;
+  t[17] = 46;
+  t[18] = 47;
+  t[19] = 48;
+  t[20] = 49;
+  t[21] = 50;
+  t[22] = 51;
+  t[23] = 52;
+  t[24] = 53;
+  t[25] = 54;
+  t[26] = 55;
+  t[27] = 56;
+  t[28] = 57;
+  t[29] = 58;
+  t[30] = 894;
+  t[31] = 60;
+  t[32] = 61;
+  t[33] = 62;
+  t[34] = 63;
+  t[35] = 64;
+  t[36] = 65;
+  t[37] = 66;
+  t[38] = 67;
+  t[39] = 68;
+  t[40] = 69;
+  t[41] = 70;
+  t[42] = 71;
+  t[43] = 72;
+  t[44] = 73;
+  t[45] = 74;
+  t[46] = 75;
+  t[47] = 76;
+  t[48] = 77;
+  t[49] = 78;
+  t[50] = 79;
+  t[51] = 80;
+  t[52] = 81;
+  t[53] = 82;
+  t[54] = 83;
+  t[55] = 84;
+  t[56] = 85;
+  t[57] = 86;
+  t[58] = 87;
+  t[59] = 88;
+  t[60] = 89;
+  t[61] = 90;
+  t[62] = 91;
+  t[63] = 92;
+  t[64] = 93;
+  t[65] = 94;
+  t[66] = 95;
+  t[67] = 96;
+  t[68] = 97;
+  t[69] = 98;
+  t[70] = 99;
+  t[71] = 100;
+  t[72] = 101;
+  t[73] = 102;
+  t[74] = 103;
+  t[75] = 104;
+  t[76] = 105;
+  t[77] = 106;
+  t[78] = 107;
+  t[79] = 108;
+  t[80] = 109;
+  t[81] = 110;
+  t[82] = 111;
+  t[83] = 112;
+  t[84] = 113;
+  t[85] = 114;
+  t[86] = 115;
+  t[87] = 116;
+  t[88] = 117;
+  t[89] = 118;
+  t[90] = 119;
+  t[91] = 120;
+  t[92] = 121;
+  t[93] = 122;
+  t[94] = 123;
+  t[95] = 124;
+  t[96] = 125;
+  t[97] = 126;
+  t[98] = 196;
+  t[99] = 197;
+  t[100] = 199;
+  t[101] = 201;
+  t[102] = 209;
+  t[103] = 214;
+  t[104] = 220;
+  t[105] = 225;
+  t[106] = 224;
+  t[107] = 226;
+  t[108] = 228;
+  t[109] = 227;
+  t[110] = 229;
+  t[111] = 231;
+  t[112] = 233;
+  t[113] = 232;
+  t[114] = 234;
+  t[115] = 235;
+  t[116] = 237;
+  t[117] = 236;
+  t[118] = 238;
+  t[119] = 239;
+  t[120] = 241;
+  t[121] = 243;
+  t[122] = 242;
+  t[123] = 244;
+  t[124] = 246;
+  t[125] = 245;
+  t[126] = 250;
+  t[127] = 249;
+  t[128] = 251;
+  t[129] = 252;
+  t[130] = 8224;
+  t[131] = 176;
+  t[132] = 162;
+  t[133] = 163;
+  t[134] = 167;
+  t[135] = 8226;
+  t[136] = 182;
+  t[137] = 223;
+  t[138] = 174;
+  t[139] = 169;
+  t[140] = 8482;
+  t[141] = 180;
+  t[142] = 168;
+  t[143] = 8800;
+  t[144] = 198;
+  t[145] = 216;
+  t[146] = 8734;
+  t[147] = 177;
+  t[148] = 8804;
+  t[149] = 8805;
+  t[150] = 165;
+  t[151] = 181;
+  t[152] = 8706;
+  t[153] = 8721;
+  t[154] = 8719;
+  t[156] = 8747;
+  t[157] = 170;
+  t[158] = 186;
+  t[159] = 8486;
+  t[160] = 230;
+  t[161] = 248;
+  t[162] = 191;
+  t[163] = 161;
+  t[164] = 172;
+  t[165] = 8730;
+  t[166] = 402;
+  t[167] = 8776;
+  t[168] = 8710;
+  t[169] = 171;
+  t[170] = 187;
+  t[171] = 8230;
+  t[210] = 218;
+  t[223] = 711;
+  t[224] = 321;
+  t[225] = 322;
+  t[227] = 353;
+  t[229] = 382;
+  t[234] = 253;
+  t[252] = 263;
+  t[253] = 268;
+  t[254] = 269;
+  t[258] = 258;
+  t[260] = 260;
+  t[261] = 261;
+  t[265] = 280;
+  t[266] = 281;
+  t[268] = 283;
+  t[269] = 313;
+  t[275] = 323;
+  t[276] = 324;
+  t[278] = 328;
+  t[284] = 345;
+  t[285] = 346;
+  t[286] = 347;
+  t[292] = 367;
+  t[295] = 377;
+  t[296] = 378;
+  t[298] = 380;
+  t[305] = 963;
+  t[306] = 964;
+  t[307] = 966;
+  t[308] = 8215;
+  t[309] = 8252;
+  t[310] = 8319;
+  t[311] = 8359;
+  t[312] = 8592;
+  t[313] = 8593;
+  t[337] = 9552;
+  t[493] = 1039;
+  t[494] = 1040;
+  t[705] = 1524;
+  t[706] = 8362;
+  t[710] = 64288;
+  t[711] = 64298;
+  t[759] = 1617;
+  t[761] = 1776;
+  t[763] = 1778;
+  t[775] = 1652;
+  t[777] = 1764;
+  t[778] = 1780;
+  t[779] = 1781;
+  t[780] = 1782;
+  t[782] = 771;
+  t[783] = 64726;
+  t[786] = 8363;
+  t[788] = 8532;
+  t[790] = 768;
+  t[791] = 769;
+  t[792] = 768;
+  t[795] = 803;
+  t[797] = 64336;
+  t[798] = 64337;
+  t[799] = 64342;
+  t[800] = 64343;
+  t[801] = 64344;
+  t[802] = 64345;
+  t[803] = 64362;
+  t[804] = 64363;
+  t[805] = 64364;
+  t[2424] = 7821;
+  t[2425] = 7822;
+  t[2426] = 7823;
+  t[2427] = 7824;
+  t[2428] = 7825;
+  t[2429] = 7826;
+  t[2430] = 7827;
+  t[2433] = 7682;
+  t[2678] = 8045;
+  t[2679] = 8046;
+  t[2830] = 1552;
+  t[2838] = 686;
+  t[2840] = 751;
+  t[2842] = 753;
+  t[2843] = 754;
+  t[2844] = 755;
+  t[2846] = 757;
+  t[2856] = 767;
+  t[2857] = 848;
+  t[2858] = 849;
+  t[2862] = 853;
+  t[2863] = 854;
+  t[2864] = 855;
+  t[2865] = 861;
+  t[2866] = 862;
+  t[2906] = 7460;
+  t[2908] = 7462;
+  t[2909] = 7463;
+  t[2910] = 7464;
+  t[2912] = 7466;
+  t[2913] = 7467;
+  t[2914] = 7468;
+  t[2916] = 7470;
+  t[2917] = 7471;
+  t[2918] = 7472;
+  t[2920] = 7474;
+  t[2921] = 7475;
+  t[2922] = 7476;
+  t[2924] = 7478;
+  t[2925] = 7479;
+  t[2926] = 7480;
+  t[2928] = 7482;
+  t[2929] = 7483;
+  t[2930] = 7484;
+  t[2932] = 7486;
+  t[2933] = 7487;
+  t[2934] = 7488;
+  t[2936] = 7490;
+  t[2937] = 7491;
+  t[2938] = 7492;
+  t[2940] = 7494;
+  t[2941] = 7495;
+  t[2942] = 7496;
+  t[2944] = 7498;
+  t[2946] = 7500;
+  t[2948] = 7502;
+  t[2950] = 7504;
+  t[2951] = 7505;
+  t[2952] = 7506;
+  t[2954] = 7508;
+  t[2955] = 7509;
+  t[2956] = 7510;
+  t[2958] = 7512;
+  t[2959] = 7513;
+  t[2960] = 7514;
+  t[2962] = 7516;
+  t[2963] = 7517;
+  t[2964] = 7518;
+  t[2966] = 7520;
+  t[2967] = 7521;
+  t[2968] = 7522;
+  t[2970] = 7524;
+  t[2971] = 7525;
+  t[2972] = 7526;
+  t[2974] = 7528;
+  t[2975] = 7529;
+  t[2976] = 7530;
+  t[2978] = 1537;
+  t[2979] = 1538;
+  t[2980] = 1539;
+  t[2982] = 1549;
+  t[2983] = 1551;
+  t[2984] = 1552;
+  t[2986] = 1554;
+  t[2987] = 1555;
+  t[2988] = 1556;
+  t[2990] = 1623;
+  t[2991] = 1624;
+  t[2995] = 1775;
+  t[2999] = 1791;
+  t[3002] = 64290;
+  t[3003] = 64291;
+  t[3004] = 64292;
+  t[3006] = 64294;
+  t[3007] = 64295;
+  t[3008] = 64296;
+  t[3011] = 1900;
+  t[3014] = 8223;
+  t[3015] = 8244;
+  t[3017] = 7532;
+  t[3018] = 7533;
+  t[3019] = 7534;
+  t[3075] = 7590;
+  t[3076] = 7591;
+  t[3079] = 7594;
+  t[3080] = 7595;
+  t[3083] = 7598;
+  t[3084] = 7599;
+  t[3087] = 7602;
+  t[3088] = 7603;
+  t[3091] = 7606;
+  t[3092] = 7607;
+  t[3095] = 7610;
+  t[3096] = 7611;
+  t[3099] = 7614;
+  t[3100] = 7615;
+  t[3103] = 7618;
+  t[3104] = 7619;
+  t[3107] = 8337;
+  t[3108] = 8338;
+  t[3116] = 1884;
+  t[3119] = 1885;
+  t[3120] = 1885;
+  t[3123] = 1886;
+  t[3124] = 1886;
+  t[3127] = 1887;
+  t[3128] = 1887;
+  t[3131] = 1888;
+  t[3132] = 1888;
+  t[3135] = 1889;
+  t[3136] = 1889;
+  t[3139] = 1890;
+  t[3140] = 1890;
+  t[3143] = 1891;
+  t[3144] = 1891;
+  t[3147] = 1892;
+  t[3148] = 1892;
+  t[3153] = 580;
+  t[3154] = 581;
+  t[3157] = 584;
+  t[3158] = 585;
+  t[3161] = 588;
+  t[3162] = 589;
+  t[3165] = 891;
+  t[3166] = 892;
+  t[3169] = 1274;
+  t[3170] = 1275;
+  t[3173] = 1278;
+  t[3174] = 1279;
+  t[3181] = 7622;
+  t[3182] = 7623;
+  t[3282] = 11799;
+  t[3316] = 578;
+  t[3379] = 42785;
+  t[3393] = 1159;
+  t[3416] = 8377;
 });
 var getSupplementalGlyphMapForArialBlack = getLookupTableFactory(function (t) {
- t[227] = 322;
- t[264] = 261;
- t[291] = 346;
+  t[227] = 322;
+  t[264] = 261;
+  t[291] = 346;
 });
 exports.getStdFontMap = getStdFontMap;
 exports.getNonStdFontMap = getNonStdFontMap;

File diff suppressed because it is too large
+ 231 - 953
lib/core/stream.js


+ 508 - 505
lib/core/type1_parser.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var coreStream = require('./stream.js');
 var coreEncodings = require('./encodings.js');
@@ -22,533 +23,535 @@ var Stream = coreStream.Stream;
 var getEncoding = coreEncodings.getEncoding;
 var HINTING_ENABLED = false;
 var Type1CharString = function Type1CharStringClosure() {
- var COMMAND_MAP = {
-  'hstem': [1],
-  'vstem': [3],
-  'vmoveto': [4],
-  'rlineto': [5],
-  'hlineto': [6],
-  'vlineto': [7],
-  'rrcurveto': [8],
-  'callsubr': [10],
-  'flex': [
-   12,
-   35
-  ],
-  'drop': [
-   12,
-   18
-  ],
-  'endchar': [14],
-  'rmoveto': [21],
-  'hmoveto': [22],
-  'vhcurveto': [30],
-  'hvcurveto': [31]
- };
- function Type1CharString() {
-  this.width = 0;
-  this.lsb = 0;
-  this.flexing = false;
-  this.output = [];
-  this.stack = [];
- }
- Type1CharString.prototype = {
-  convert: function Type1CharString_convert(encoded, subrs, seacAnalysisEnabled) {
-   var count = encoded.length;
-   var error = false;
-   var wx, sbx, subrNumber;
-   for (var i = 0; i < count; i++) {
-    var value = encoded[i];
-    if (value < 32) {
-     if (value === 12) {
-      value = (value << 8) + encoded[++i];
-     }
-     switch (value) {
-     case 1:
-      if (!HINTING_ENABLED) {
-       this.stack = [];
-       break;
-      }
-      error = this.executeCommand(2, COMMAND_MAP.hstem);
-      break;
-     case 3:
-      if (!HINTING_ENABLED) {
-       this.stack = [];
-       break;
-      }
-      error = this.executeCommand(2, COMMAND_MAP.vstem);
-      break;
-     case 4:
-      if (this.flexing) {
-       if (this.stack.length < 1) {
-        error = true;
-        break;
-       }
-       var dy = this.stack.pop();
-       this.stack.push(0, dy);
-       break;
-      }
-      error = this.executeCommand(1, COMMAND_MAP.vmoveto);
-      break;
-     case 5:
-      error = this.executeCommand(2, COMMAND_MAP.rlineto);
-      break;
-     case 6:
-      error = this.executeCommand(1, COMMAND_MAP.hlineto);
-      break;
-     case 7:
-      error = this.executeCommand(1, COMMAND_MAP.vlineto);
-      break;
-     case 8:
-      error = this.executeCommand(6, COMMAND_MAP.rrcurveto);
-      break;
-     case 9:
-      this.stack = [];
-      break;
-     case 10:
-      if (this.stack.length < 1) {
-       error = true;
-       break;
+  var COMMAND_MAP = {
+    'hstem': [1],
+    'vstem': [3],
+    'vmoveto': [4],
+    'rlineto': [5],
+    'hlineto': [6],
+    'vlineto': [7],
+    'rrcurveto': [8],
+    'callsubr': [10],
+    'flex': [12, 35],
+    'drop': [12, 18],
+    'endchar': [14],
+    'rmoveto': [21],
+    'hmoveto': [22],
+    'vhcurveto': [30],
+    'hvcurveto': [31]
+  };
+  function Type1CharString() {
+    this.width = 0;
+    this.lsb = 0;
+    this.flexing = false;
+    this.output = [];
+    this.stack = [];
+  }
+  Type1CharString.prototype = {
+    convert: function Type1CharString_convert(encoded, subrs, seacAnalysisEnabled) {
+      var count = encoded.length;
+      var error = false;
+      var wx, sbx, subrNumber;
+      for (var i = 0; i < count; i++) {
+        var value = encoded[i];
+        if (value < 32) {
+          if (value === 12) {
+            value = (value << 8) + encoded[++i];
+          }
+          switch (value) {
+            case 1:
+              if (!HINTING_ENABLED) {
+                this.stack = [];
+                break;
+              }
+              error = this.executeCommand(2, COMMAND_MAP.hstem);
+              break;
+            case 3:
+              if (!HINTING_ENABLED) {
+                this.stack = [];
+                break;
+              }
+              error = this.executeCommand(2, COMMAND_MAP.vstem);
+              break;
+            case 4:
+              if (this.flexing) {
+                if (this.stack.length < 1) {
+                  error = true;
+                  break;
+                }
+                var dy = this.stack.pop();
+                this.stack.push(0, dy);
+                break;
+              }
+              error = this.executeCommand(1, COMMAND_MAP.vmoveto);
+              break;
+            case 5:
+              error = this.executeCommand(2, COMMAND_MAP.rlineto);
+              break;
+            case 6:
+              error = this.executeCommand(1, COMMAND_MAP.hlineto);
+              break;
+            case 7:
+              error = this.executeCommand(1, COMMAND_MAP.vlineto);
+              break;
+            case 8:
+              error = this.executeCommand(6, COMMAND_MAP.rrcurveto);
+              break;
+            case 9:
+              this.stack = [];
+              break;
+            case 10:
+              if (this.stack.length < 1) {
+                error = true;
+                break;
+              }
+              subrNumber = this.stack.pop();
+              error = this.convert(subrs[subrNumber], subrs, seacAnalysisEnabled);
+              break;
+            case 11:
+              return error;
+            case 13:
+              if (this.stack.length < 2) {
+                error = true;
+                break;
+              }
+              wx = this.stack.pop();
+              sbx = this.stack.pop();
+              this.lsb = sbx;
+              this.width = wx;
+              this.stack.push(wx, sbx);
+              error = this.executeCommand(2, COMMAND_MAP.hmoveto);
+              break;
+            case 14:
+              this.output.push(COMMAND_MAP.endchar[0]);
+              break;
+            case 21:
+              if (this.flexing) {
+                break;
+              }
+              error = this.executeCommand(2, COMMAND_MAP.rmoveto);
+              break;
+            case 22:
+              if (this.flexing) {
+                this.stack.push(0);
+                break;
+              }
+              error = this.executeCommand(1, COMMAND_MAP.hmoveto);
+              break;
+            case 30:
+              error = this.executeCommand(4, COMMAND_MAP.vhcurveto);
+              break;
+            case 31:
+              error = this.executeCommand(4, COMMAND_MAP.hvcurveto);
+              break;
+            case (12 << 8) + 0:
+              this.stack = [];
+              break;
+            case (12 << 8) + 1:
+              if (!HINTING_ENABLED) {
+                this.stack = [];
+                break;
+              }
+              error = this.executeCommand(2, COMMAND_MAP.vstem);
+              break;
+            case (12 << 8) + 2:
+              if (!HINTING_ENABLED) {
+                this.stack = [];
+                break;
+              }
+              error = this.executeCommand(2, COMMAND_MAP.hstem);
+              break;
+            case (12 << 8) + 6:
+              if (seacAnalysisEnabled) {
+                this.seac = this.stack.splice(-4, 4);
+                error = this.executeCommand(0, COMMAND_MAP.endchar);
+              } else {
+                error = this.executeCommand(4, COMMAND_MAP.endchar);
+              }
+              break;
+            case (12 << 8) + 7:
+              if (this.stack.length < 4) {
+                error = true;
+                break;
+              }
+              this.stack.pop();
+              wx = this.stack.pop();
+              var sby = this.stack.pop();
+              sbx = this.stack.pop();
+              this.lsb = sbx;
+              this.width = wx;
+              this.stack.push(wx, sbx, sby);
+              error = this.executeCommand(3, COMMAND_MAP.rmoveto);
+              break;
+            case (12 << 8) + 12:
+              if (this.stack.length < 2) {
+                error = true;
+                break;
+              }
+              var num2 = this.stack.pop();
+              var num1 = this.stack.pop();
+              this.stack.push(num1 / num2);
+              break;
+            case (12 << 8) + 16:
+              if (this.stack.length < 2) {
+                error = true;
+                break;
+              }
+              subrNumber = this.stack.pop();
+              var numArgs = this.stack.pop();
+              if (subrNumber === 0 && numArgs === 3) {
+                var flexArgs = this.stack.splice(this.stack.length - 17, 17);
+                this.stack.push(flexArgs[2] + flexArgs[0], flexArgs[3] + flexArgs[1], flexArgs[4], flexArgs[5], flexArgs[6], flexArgs[7], flexArgs[8], flexArgs[9], flexArgs[10], flexArgs[11], flexArgs[12], flexArgs[13], flexArgs[14]);
+                error = this.executeCommand(13, COMMAND_MAP.flex, true);
+                this.flexing = false;
+                this.stack.push(flexArgs[15], flexArgs[16]);
+              } else if (subrNumber === 1 && numArgs === 0) {
+                this.flexing = true;
+              }
+              break;
+            case (12 << 8) + 17:
+              break;
+            case (12 << 8) + 33:
+              this.stack = [];
+              break;
+            default:
+              warn('Unknown type 1 charstring command of "' + value + '"');
+              break;
+          }
+          if (error) {
+            break;
+          }
+          continue;
+        } else if (value <= 246) {
+          value = value - 139;
+        } else if (value <= 250) {
+          value = (value - 247) * 256 + encoded[++i] + 108;
+        } else if (value <= 254) {
+          value = -((value - 251) * 256) - encoded[++i] - 108;
+        } else {
+          value = (encoded[++i] & 0xff) << 24 | (encoded[++i] & 0xff) << 16 | (encoded[++i] & 0xff) << 8 | (encoded[++i] & 0xff) << 0;
+        }
+        this.stack.push(value);
       }
-      subrNumber = this.stack.pop();
-      error = this.convert(subrs[subrNumber], subrs, seacAnalysisEnabled);
-      break;
-     case 11:
       return error;
-     case 13:
-      if (this.stack.length < 2) {
-       error = true;
-       break;
-      }
-      wx = this.stack.pop();
-      sbx = this.stack.pop();
-      this.lsb = sbx;
-      this.width = wx;
-      this.stack.push(wx, sbx);
-      error = this.executeCommand(2, COMMAND_MAP.hmoveto);
-      break;
-     case 14:
-      this.output.push(COMMAND_MAP.endchar[0]);
-      break;
-     case 21:
-      if (this.flexing) {
-       break;
+    },
+    executeCommand: function (howManyArgs, command, keepStack) {
+      var stackLength = this.stack.length;
+      if (howManyArgs > stackLength) {
+        return true;
       }
-      error = this.executeCommand(2, COMMAND_MAP.rmoveto);
-      break;
-     case 22:
-      if (this.flexing) {
-       this.stack.push(0);
-       break;
-      }
-      error = this.executeCommand(1, COMMAND_MAP.hmoveto);
-      break;
-     case 30:
-      error = this.executeCommand(4, COMMAND_MAP.vhcurveto);
-      break;
-     case 31:
-      error = this.executeCommand(4, COMMAND_MAP.hvcurveto);
-      break;
-     case (12 << 8) + 0:
-      this.stack = [];
-      break;
-     case (12 << 8) + 1:
-      if (!HINTING_ENABLED) {
-       this.stack = [];
-       break;
-      }
-      error = this.executeCommand(2, COMMAND_MAP.vstem);
-      break;
-     case (12 << 8) + 2:
-      if (!HINTING_ENABLED) {
-       this.stack = [];
-       break;
+      var start = stackLength - howManyArgs;
+      for (var i = start; i < stackLength; i++) {
+        var value = this.stack[i];
+        if (value === (value | 0)) {
+          this.output.push(28, value >> 8 & 0xff, value & 0xff);
+        } else {
+          value = 65536 * value | 0;
+          this.output.push(255, value >> 24 & 0xFF, value >> 16 & 0xFF, value >> 8 & 0xFF, value & 0xFF);
+        }
       }
-      error = this.executeCommand(2, COMMAND_MAP.hstem);
-      break;
-     case (12 << 8) + 6:
-      if (seacAnalysisEnabled) {
-       this.seac = this.stack.splice(-4, 4);
-       error = this.executeCommand(0, COMMAND_MAP.endchar);
+      this.output.push.apply(this.output, command);
+      if (keepStack) {
+        this.stack.splice(start, howManyArgs);
       } else {
-       error = this.executeCommand(4, COMMAND_MAP.endchar);
-      }
-      break;
-     case (12 << 8) + 7:
-      if (this.stack.length < 4) {
-       error = true;
-       break;
-      }
-      this.stack.pop();
-      wx = this.stack.pop();
-      var sby = this.stack.pop();
-      sbx = this.stack.pop();
-      this.lsb = sbx;
-      this.width = wx;
-      this.stack.push(wx, sbx, sby);
-      error = this.executeCommand(3, COMMAND_MAP.rmoveto);
-      break;
-     case (12 << 8) + 12:
-      if (this.stack.length < 2) {
-       error = true;
-       break;
+        this.stack.length = 0;
       }
-      var num2 = this.stack.pop();
-      var num1 = this.stack.pop();
-      this.stack.push(num1 / num2);
-      break;
-     case (12 << 8) + 16:
-      if (this.stack.length < 2) {
-       error = true;
-       break;
-      }
-      subrNumber = this.stack.pop();
-      var numArgs = this.stack.pop();
-      if (subrNumber === 0 && numArgs === 3) {
-       var flexArgs = this.stack.splice(this.stack.length - 17, 17);
-       this.stack.push(flexArgs[2] + flexArgs[0], flexArgs[3] + flexArgs[1], flexArgs[4], flexArgs[5], flexArgs[6], flexArgs[7], flexArgs[8], flexArgs[9], flexArgs[10], flexArgs[11], flexArgs[12], flexArgs[13], flexArgs[14]);
-       error = this.executeCommand(13, COMMAND_MAP.flex, true);
-       this.flexing = false;
-       this.stack.push(flexArgs[15], flexArgs[16]);
-      } else if (subrNumber === 1 && numArgs === 0) {
-       this.flexing = true;
-      }
-      break;
-     case (12 << 8) + 17:
-      break;
-     case (12 << 8) + 33:
-      this.stack = [];
-      break;
-     default:
-      warn('Unknown type 1 charstring command of "' + value + '"');
-      break;
-     }
-     if (error) {
-      break;
-     }
-     continue;
-    } else if (value <= 246) {
-     value = value - 139;
-    } else if (value <= 250) {
-     value = (value - 247) * 256 + encoded[++i] + 108;
-    } else if (value <= 254) {
-     value = -((value - 251) * 256) - encoded[++i] - 108;
-    } else {
-     value = (encoded[++i] & 0xff) << 24 | (encoded[++i] & 0xff) << 16 | (encoded[++i] & 0xff) << 8 | (encoded[++i] & 0xff) << 0;
-    }
-    this.stack.push(value);
-   }
-   return error;
-  },
-  executeCommand: function (howManyArgs, command, keepStack) {
-   var stackLength = this.stack.length;
-   if (howManyArgs > stackLength) {
-    return true;
-   }
-   var start = stackLength - howManyArgs;
-   for (var i = start; i < stackLength; i++) {
-    var value = this.stack[i];
-    if (value === (value | 0)) {
-     this.output.push(28, value >> 8 & 0xff, value & 0xff);
-    } else {
-     value = 65536 * value | 0;
-     this.output.push(255, value >> 24 & 0xFF, value >> 16 & 0xFF, value >> 8 & 0xFF, value & 0xFF);
+      return false;
     }
-   }
-   this.output.push.apply(this.output, command);
-   if (keepStack) {
-    this.stack.splice(start, howManyArgs);
-   } else {
-    this.stack.length = 0;
-   }
-   return false;
-  }
- };
- return Type1CharString;
+  };
+  return Type1CharString;
 }();
 var Type1Parser = function Type1ParserClosure() {
- var EEXEC_ENCRYPT_KEY = 55665;
- var CHAR_STRS_ENCRYPT_KEY = 4330;
- function isHexDigit(code) {
-  return code >= 48 && code <= 57 || code >= 65 && code <= 70 || code >= 97 && code <= 102;
- }
- function decrypt(data, key, discardNumber) {
-  if (discardNumber >= data.length) {
-   return new Uint8Array(0);
-  }
-  var r = key | 0, c1 = 52845, c2 = 22719, i, j;
-  for (i = 0; i < discardNumber; i++) {
-   r = (data[i] + r) * c1 + c2 & (1 << 16) - 1;
-  }
-  var count = data.length - discardNumber;
-  var decrypted = new Uint8Array(count);
-  for (i = discardNumber, j = 0; j < count; i++, j++) {
-   var value = data[i];
-   decrypted[j] = value ^ r >> 8;
-   r = (value + r) * c1 + c2 & (1 << 16) - 1;
+  var EEXEC_ENCRYPT_KEY = 55665;
+  var CHAR_STRS_ENCRYPT_KEY = 4330;
+  function isHexDigit(code) {
+    return code >= 48 && code <= 57 || code >= 65 && code <= 70 || code >= 97 && code <= 102;
   }
-  return decrypted;
- }
- function decryptAscii(data, key, discardNumber) {
-  var r = key | 0, c1 = 52845, c2 = 22719;
-  var count = data.length, maybeLength = count >>> 1;
-  var decrypted = new Uint8Array(maybeLength);
-  var i, j;
-  for (i = 0, j = 0; i < count; i++) {
-   var digit1 = data[i];
-   if (!isHexDigit(digit1)) {
-    continue;
-   }
-   i++;
-   var digit2;
-   while (i < count && !isHexDigit(digit2 = data[i])) {
-    i++;
-   }
-   if (i < count) {
-    var value = parseInt(String.fromCharCode(digit1, digit2), 16);
-    decrypted[j++] = value ^ r >> 8;
-    r = (value + r) * c1 + c2 & (1 << 16) - 1;
-   }
-  }
-  return Array.prototype.slice.call(decrypted, discardNumber, j);
- }
- function isSpecial(c) {
-  return c === 0x2F || c === 0x5B || c === 0x5D || c === 0x7B || c === 0x7D || c === 0x28 || c === 0x29;
- }
- function Type1Parser(stream, encrypted, seacAnalysisEnabled) {
-  if (encrypted) {
-   var data = stream.getBytes();
-   var isBinary = !(isHexDigit(data[0]) && isHexDigit(data[1]) && isHexDigit(data[2]) && isHexDigit(data[3]));
-   stream = new Stream(isBinary ? decrypt(data, EEXEC_ENCRYPT_KEY, 4) : decryptAscii(data, EEXEC_ENCRYPT_KEY, 4));
-  }
-  this.seacAnalysisEnabled = !!seacAnalysisEnabled;
-  this.stream = stream;
-  this.nextChar();
- }
- Type1Parser.prototype = {
-  readNumberArray: function Type1Parser_readNumberArray() {
-   this.getToken();
-   var array = [];
-   while (true) {
-    var token = this.getToken();
-    if (token === null || token === ']' || token === '}') {
-     break;
+  function decrypt(data, key, discardNumber) {
+    if (discardNumber >= data.length) {
+      return new Uint8Array(0);
     }
-    array.push(parseFloat(token || 0));
-   }
-   return array;
-  },
-  readNumber: function Type1Parser_readNumber() {
-   var token = this.getToken();
-   return parseFloat(token || 0);
-  },
-  readInt: function Type1Parser_readInt() {
-   var token = this.getToken();
-   return parseInt(token || 0, 10) | 0;
-  },
-  readBoolean: function Type1Parser_readBoolean() {
-   var token = this.getToken();
-   return token === 'true' ? 1 : 0;
-  },
-  nextChar: function Type1_nextChar() {
-   return this.currentChar = this.stream.getByte();
-  },
-  getToken: function Type1Parser_getToken() {
-   var comment = false;
-   var ch = this.currentChar;
-   while (true) {
-    if (ch === -1) {
-     return null;
+    var r = key | 0,
+        c1 = 52845,
+        c2 = 22719,
+        i,
+        j;
+    for (i = 0; i < discardNumber; i++) {
+      r = (data[i] + r) * c1 + c2 & (1 << 16) - 1;
     }
-    if (comment) {
-     if (ch === 0x0A || ch === 0x0D) {
-      comment = false;
-     }
-    } else if (ch === 0x25) {
-     comment = true;
-    } else if (!isSpace(ch)) {
-     break;
+    var count = data.length - discardNumber;
+    var decrypted = new Uint8Array(count);
+    for (i = discardNumber, j = 0; j < count; i++, j++) {
+      var value = data[i];
+      decrypted[j] = value ^ r >> 8;
+      r = (value + r) * c1 + c2 & (1 << 16) - 1;
     }
-    ch = this.nextChar();
-   }
-   if (isSpecial(ch)) {
-    this.nextChar();
-    return String.fromCharCode(ch);
-   }
-   var token = '';
-   do {
-    token += String.fromCharCode(ch);
-    ch = this.nextChar();
-   } while (ch >= 0 && !isSpace(ch) && !isSpecial(ch));
-   return token;
-  },
-  extractFontProgram: function Type1Parser_extractFontProgram() {
-   var stream = this.stream;
-   var subrs = [], charstrings = [];
-   var privateData = Object.create(null);
-   privateData['lenIV'] = 4;
-   var program = {
-    subrs: [],
-    charstrings: [],
-    properties: { 'privateData': privateData }
-   };
-   var token, length, data, lenIV, encoded;
-   while ((token = this.getToken()) !== null) {
-    if (token !== '/') {
-     continue;
-    }
-    token = this.getToken();
-    switch (token) {
-    case 'CharStrings':
-     this.getToken();
-     this.getToken();
-     this.getToken();
-     this.getToken();
-     while (true) {
-      token = this.getToken();
-      if (token === null || token === 'end') {
-       break;
-      }
-      if (token !== '/') {
-       continue;
+    return decrypted;
+  }
+  function decryptAscii(data, key, discardNumber) {
+    var r = key | 0,
+        c1 = 52845,
+        c2 = 22719;
+    var count = data.length,
+        maybeLength = count >>> 1;
+    var decrypted = new Uint8Array(maybeLength);
+    var i, j;
+    for (i = 0, j = 0; i < count; i++) {
+      var digit1 = data[i];
+      if (!isHexDigit(digit1)) {
+        continue;
       }
-      var glyph = this.getToken();
-      length = this.readInt();
-      this.getToken();
-      data = stream.makeSubStream(stream.pos, length);
-      lenIV = program.properties.privateData['lenIV'];
-      encoded = decrypt(data.getBytes(), CHAR_STRS_ENCRYPT_KEY, lenIV);
-      stream.skip(length);
-      this.nextChar();
-      token = this.getToken();
-      if (token === 'noaccess') {
-       this.getToken();
+      i++;
+      var digit2;
+      while (i < count && !isHexDigit(digit2 = data[i])) {
+        i++;
       }
-      charstrings.push({
-       glyph: glyph,
-       encoded: encoded
-      });
-     }
-     break;
-    case 'Subrs':
-     this.readInt();
-     this.getToken();
-     while ((token = this.getToken()) === 'dup') {
-      var index = this.readInt();
-      length = this.readInt();
-      this.getToken();
-      data = stream.makeSubStream(stream.pos, length);
-      lenIV = program.properties.privateData['lenIV'];
-      encoded = decrypt(data.getBytes(), CHAR_STRS_ENCRYPT_KEY, lenIV);
-      stream.skip(length);
-      this.nextChar();
-      token = this.getToken();
-      if (token === 'noaccess') {
-       this.getToken();
+      if (i < count) {
+        var value = parseInt(String.fromCharCode(digit1, digit2), 16);
+        decrypted[j++] = value ^ r >> 8;
+        r = (value + r) * c1 + c2 & (1 << 16) - 1;
       }
-      subrs[index] = encoded;
-     }
-     break;
-    case 'BlueValues':
-    case 'OtherBlues':
-    case 'FamilyBlues':
-    case 'FamilyOtherBlues':
-     var blueArray = this.readNumberArray();
-     if (blueArray.length > 0 && blueArray.length % 2 === 0 && HINTING_ENABLED) {
-      program.properties.privateData[token] = blueArray;
-     }
-     break;
-    case 'StemSnapH':
-    case 'StemSnapV':
-     program.properties.privateData[token] = this.readNumberArray();
-     break;
-    case 'StdHW':
-    case 'StdVW':
-     program.properties.privateData[token] = this.readNumberArray()[0];
-     break;
-    case 'BlueShift':
-    case 'lenIV':
-    case 'BlueFuzz':
-    case 'BlueScale':
-    case 'LanguageGroup':
-    case 'ExpansionFactor':
-     program.properties.privateData[token] = this.readNumber();
-     break;
-    case 'ForceBold':
-     program.properties.privateData[token] = this.readBoolean();
-     break;
     }
-   }
-   for (var i = 0; i < charstrings.length; i++) {
-    glyph = charstrings[i].glyph;
-    encoded = charstrings[i].encoded;
-    var charString = new Type1CharString();
-    var error = charString.convert(encoded, subrs, this.seacAnalysisEnabled);
-    var output = charString.output;
-    if (error) {
-     output = [14];
-    }
-    program.charstrings.push({
-     glyphName: glyph,
-     charstring: output,
-     width: charString.width,
-     lsb: charString.lsb,
-     seac: charString.seac
-    });
-   }
-   return program;
-  },
-  extractFontHeader: function Type1Parser_extractFontHeader(properties) {
-   var token;
-   while ((token = this.getToken()) !== null) {
-    if (token !== '/') {
-     continue;
+    return Array.prototype.slice.call(decrypted, discardNumber, j);
+  }
+  function isSpecial(c) {
+    return c === 0x2F || c === 0x5B || c === 0x5D || c === 0x7B || c === 0x7D || c === 0x28 || c === 0x29;
+  }
+  function Type1Parser(stream, encrypted, seacAnalysisEnabled) {
+    if (encrypted) {
+      var data = stream.getBytes();
+      var isBinary = !(isHexDigit(data[0]) && isHexDigit(data[1]) && isHexDigit(data[2]) && isHexDigit(data[3]));
+      stream = new Stream(isBinary ? decrypt(data, EEXEC_ENCRYPT_KEY, 4) : decryptAscii(data, EEXEC_ENCRYPT_KEY, 4));
     }
-    token = this.getToken();
-    switch (token) {
-    case 'FontMatrix':
-     var matrix = this.readNumberArray();
-     properties.fontMatrix = matrix;
-     break;
-    case 'Encoding':
-     var encodingArg = this.getToken();
-     var encoding;
-     if (!/^\d+$/.test(encodingArg)) {
-      encoding = getEncoding(encodingArg);
-     } else {
-      encoding = [];
-      var size = parseInt(encodingArg, 10) | 0;
+    this.seacAnalysisEnabled = !!seacAnalysisEnabled;
+    this.stream = stream;
+    this.nextChar();
+  }
+  Type1Parser.prototype = {
+    readNumberArray: function Type1Parser_readNumberArray() {
       this.getToken();
-      for (var j = 0; j < size; j++) {
-       token = this.getToken();
-       while (token !== 'dup' && token !== 'def') {
+      var array = [];
+      while (true) {
+        var token = this.getToken();
+        if (token === null || token === ']' || token === '}') {
+          break;
+        }
+        array.push(parseFloat(token || 0));
+      }
+      return array;
+    },
+    readNumber: function Type1Parser_readNumber() {
+      var token = this.getToken();
+      return parseFloat(token || 0);
+    },
+    readInt: function Type1Parser_readInt() {
+      var token = this.getToken();
+      return parseInt(token || 0, 10) | 0;
+    },
+    readBoolean: function Type1Parser_readBoolean() {
+      var token = this.getToken();
+      return token === 'true' ? 1 : 0;
+    },
+    nextChar: function Type1_nextChar() {
+      return this.currentChar = this.stream.getByte();
+    },
+    getToken: function Type1Parser_getToken() {
+      var comment = false;
+      var ch = this.currentChar;
+      while (true) {
+        if (ch === -1) {
+          return null;
+        }
+        if (comment) {
+          if (ch === 0x0A || ch === 0x0D) {
+            comment = false;
+          }
+        } else if (ch === 0x25) {
+          comment = true;
+        } else if (!isSpace(ch)) {
+          break;
+        }
+        ch = this.nextChar();
+      }
+      if (isSpecial(ch)) {
+        this.nextChar();
+        return String.fromCharCode(ch);
+      }
+      var token = '';
+      do {
+        token += String.fromCharCode(ch);
+        ch = this.nextChar();
+      } while (ch >= 0 && !isSpace(ch) && !isSpecial(ch));
+      return token;
+    },
+    extractFontProgram: function Type1Parser_extractFontProgram() {
+      var stream = this.stream;
+      var subrs = [],
+          charstrings = [];
+      var privateData = Object.create(null);
+      privateData['lenIV'] = 4;
+      var program = {
+        subrs: [],
+        charstrings: [],
+        properties: { 'privateData': privateData }
+      };
+      var token, length, data, lenIV, encoded;
+      while ((token = this.getToken()) !== null) {
+        if (token !== '/') {
+          continue;
+        }
         token = this.getToken();
-        if (token === null) {
-         return;
+        switch (token) {
+          case 'CharStrings':
+            this.getToken();
+            this.getToken();
+            this.getToken();
+            this.getToken();
+            while (true) {
+              token = this.getToken();
+              if (token === null || token === 'end') {
+                break;
+              }
+              if (token !== '/') {
+                continue;
+              }
+              var glyph = this.getToken();
+              length = this.readInt();
+              this.getToken();
+              data = stream.makeSubStream(stream.pos, length);
+              lenIV = program.properties.privateData['lenIV'];
+              encoded = decrypt(data.getBytes(), CHAR_STRS_ENCRYPT_KEY, lenIV);
+              stream.skip(length);
+              this.nextChar();
+              token = this.getToken();
+              if (token === 'noaccess') {
+                this.getToken();
+              }
+              charstrings.push({
+                glyph: glyph,
+                encoded: encoded
+              });
+            }
+            break;
+          case 'Subrs':
+            this.readInt();
+            this.getToken();
+            while ((token = this.getToken()) === 'dup') {
+              var index = this.readInt();
+              length = this.readInt();
+              this.getToken();
+              data = stream.makeSubStream(stream.pos, length);
+              lenIV = program.properties.privateData['lenIV'];
+              encoded = decrypt(data.getBytes(), CHAR_STRS_ENCRYPT_KEY, lenIV);
+              stream.skip(length);
+              this.nextChar();
+              token = this.getToken();
+              if (token === 'noaccess') {
+                this.getToken();
+              }
+              subrs[index] = encoded;
+            }
+            break;
+          case 'BlueValues':
+          case 'OtherBlues':
+          case 'FamilyBlues':
+          case 'FamilyOtherBlues':
+            var blueArray = this.readNumberArray();
+            if (blueArray.length > 0 && blueArray.length % 2 === 0 && HINTING_ENABLED) {
+              program.properties.privateData[token] = blueArray;
+            }
+            break;
+          case 'StemSnapH':
+          case 'StemSnapV':
+            program.properties.privateData[token] = this.readNumberArray();
+            break;
+          case 'StdHW':
+          case 'StdVW':
+            program.properties.privateData[token] = this.readNumberArray()[0];
+            break;
+          case 'BlueShift':
+          case 'lenIV':
+          case 'BlueFuzz':
+          case 'BlueScale':
+          case 'LanguageGroup':
+          case 'ExpansionFactor':
+            program.properties.privateData[token] = this.readNumber();
+            break;
+          case 'ForceBold':
+            program.properties.privateData[token] = this.readBoolean();
+            break;
+        }
+      }
+      for (var i = 0; i < charstrings.length; i++) {
+        glyph = charstrings[i].glyph;
+        encoded = charstrings[i].encoded;
+        var charString = new Type1CharString();
+        var error = charString.convert(encoded, subrs, this.seacAnalysisEnabled);
+        var output = charString.output;
+        if (error) {
+          output = [14];
+        }
+        program.charstrings.push({
+          glyphName: glyph,
+          charstring: output,
+          width: charString.width,
+          lsb: charString.lsb,
+          seac: charString.seac
+        });
+      }
+      return program;
+    },
+    extractFontHeader: function Type1Parser_extractFontHeader(properties) {
+      var token;
+      while ((token = this.getToken()) !== null) {
+        if (token !== '/') {
+          continue;
+        }
+        token = this.getToken();
+        switch (token) {
+          case 'FontMatrix':
+            var matrix = this.readNumberArray();
+            properties.fontMatrix = matrix;
+            break;
+          case 'Encoding':
+            var encodingArg = this.getToken();
+            var encoding;
+            if (!/^\d+$/.test(encodingArg)) {
+              encoding = getEncoding(encodingArg);
+            } else {
+              encoding = [];
+              var size = parseInt(encodingArg, 10) | 0;
+              this.getToken();
+              for (var j = 0; j < size; j++) {
+                token = this.getToken();
+                while (token !== 'dup' && token !== 'def') {
+                  token = this.getToken();
+                  if (token === null) {
+                    return;
+                  }
+                }
+                if (token === 'def') {
+                  break;
+                }
+                var index = this.readInt();
+                this.getToken();
+                var glyph = this.getToken();
+                encoding[index] = glyph;
+                this.getToken();
+              }
+            }
+            properties.builtInEncoding = encoding;
+            break;
+          case 'FontBBox':
+            var fontBBox = this.readNumberArray();
+            properties.ascent = Math.max(fontBBox[3], fontBBox[1]);
+            properties.descent = Math.min(fontBBox[1], fontBBox[3]);
+            properties.ascentScaled = true;
+            break;
         }
-       }
-       if (token === 'def') {
-        break;
-       }
-       var index = this.readInt();
-       this.getToken();
-       var glyph = this.getToken();
-       encoding[index] = glyph;
-       this.getToken();
       }
-     }
-     properties.builtInEncoding = encoding;
-     break;
-    case 'FontBBox':
-     var fontBBox = this.readNumberArray();
-     properties.ascent = Math.max(fontBBox[3], fontBBox[1]);
-     properties.descent = Math.min(fontBBox[1], fontBBox[3]);
-     properties.ascentScaled = true;
-     break;
     }
-   }
-  }
- };
- return Type1Parser;
+  };
+  return Type1Parser;
 }();
 exports.Type1Parser = Type1Parser;

+ 1579 - 1701
lib/core/unicode.js

@@ -13,1971 +13,1849 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var getLookupTableFactory = sharedUtil.getLookupTableFactory;
 var getSpecialPUASymbols = getLookupTableFactory(function (t) {
- t[63721] = 0x00A9;
- t[63193] = 0x00A9;
- t[63720] = 0x00AE;
- t[63194] = 0x00AE;
- t[63722] = 0x2122;
- t[63195] = 0x2122;
- t[63729] = 0x23A7;
- t[63730] = 0x23A8;
- t[63731] = 0x23A9;
- t[63740] = 0x23AB;
- t[63741] = 0x23AC;
- t[63742] = 0x23AD;
- t[63726] = 0x23A1;
- t[63727] = 0x23A2;
- t[63728] = 0x23A3;
- t[63737] = 0x23A4;
- t[63738] = 0x23A5;
- t[63739] = 0x23A6;
- t[63723] = 0x239B;
- t[63724] = 0x239C;
- t[63725] = 0x239D;
- t[63734] = 0x239E;
- t[63735] = 0x239F;
- t[63736] = 0x23A0;
+  t[63721] = 0x00A9;
+  t[63193] = 0x00A9;
+  t[63720] = 0x00AE;
+  t[63194] = 0x00AE;
+  t[63722] = 0x2122;
+  t[63195] = 0x2122;
+  t[63729] = 0x23A7;
+  t[63730] = 0x23A8;
+  t[63731] = 0x23A9;
+  t[63740] = 0x23AB;
+  t[63741] = 0x23AC;
+  t[63742] = 0x23AD;
+  t[63726] = 0x23A1;
+  t[63727] = 0x23A2;
+  t[63728] = 0x23A3;
+  t[63737] = 0x23A4;
+  t[63738] = 0x23A5;
+  t[63739] = 0x23A6;
+  t[63723] = 0x239B;
+  t[63724] = 0x239C;
+  t[63725] = 0x239D;
+  t[63734] = 0x239E;
+  t[63735] = 0x239F;
+  t[63736] = 0x23A0;
 });
 function mapSpecialUnicodeValues(code) {
- if (code >= 0xFFF0 && code <= 0xFFFF) {
-  return 0;
- } else if (code >= 0xF600 && code <= 0xF8FF) {
-  return getSpecialPUASymbols()[code] || code;
- }
- return code;
+  if (code >= 0xFFF0 && code <= 0xFFFF) {
+    return 0;
+  } else if (code >= 0xF600 && code <= 0xF8FF) {
+    return getSpecialPUASymbols()[code] || code;
+  }
+  return code;
 }
 function getUnicodeForGlyph(name, glyphsUnicodeMap) {
- var unicode = glyphsUnicodeMap[name];
- if (unicode !== undefined) {
-  return unicode;
- }
- if (!name) {
-  return -1;
- }
- if (name[0] === 'u') {
-  var nameLen = name.length, hexStr;
-  if (nameLen === 7 && name[1] === 'n' && name[2] === 'i') {
-   hexStr = name.substr(3);
-  } else if (nameLen >= 5 && nameLen <= 7) {
-   hexStr = name.substr(1);
-  } else {
-   return -1;
-  }
-  if (hexStr === hexStr.toUpperCase()) {
-   unicode = parseInt(hexStr, 16);
-   if (unicode >= 0) {
+  var unicode = glyphsUnicodeMap[name];
+  if (unicode !== undefined) {
     return unicode;
-   }
   }
- }
- return -1;
+  if (!name) {
+    return -1;
+  }
+  if (name[0] === 'u') {
+    var nameLen = name.length,
+        hexStr;
+    if (nameLen === 7 && name[1] === 'n' && name[2] === 'i') {
+      hexStr = name.substr(3);
+    } else if (nameLen >= 5 && nameLen <= 7) {
+      hexStr = name.substr(1);
+    } else {
+      return -1;
+    }
+    if (hexStr === hexStr.toUpperCase()) {
+      unicode = parseInt(hexStr, 16);
+      if (unicode >= 0) {
+        return unicode;
+      }
+    }
+  }
+  return -1;
 }
-var UnicodeRanges = [
- {
+var UnicodeRanges = [{
   'begin': 0x0000,
   'end': 0x007F
- },
- {
+}, {
   'begin': 0x0080,
   'end': 0x00FF
- },
- {
+}, {
   'begin': 0x0100,
   'end': 0x017F
- },
- {
+}, {
   'begin': 0x0180,
   'end': 0x024F
- },
- {
+}, {
   'begin': 0x0250,
   'end': 0x02AF
- },
- {
+}, {
   'begin': 0x02B0,
   'end': 0x02FF
- },
- {
+}, {
   'begin': 0x0300,
   'end': 0x036F
- },
- {
+}, {
   'begin': 0x0370,
   'end': 0x03FF
- },
- {
+}, {
   'begin': 0x2C80,
   'end': 0x2CFF
- },
- {
+}, {
   'begin': 0x0400,
   'end': 0x04FF
- },
- {
+}, {
   'begin': 0x0530,
   'end': 0x058F
- },
- {
+}, {
   'begin': 0x0590,
   'end': 0x05FF
- },
- {
+}, {
   'begin': 0xA500,
   'end': 0xA63F
- },
- {
+}, {
   'begin': 0x0600,
   'end': 0x06FF
- },
- {
+}, {
   'begin': 0x07C0,
   'end': 0x07FF
- },
- {
+}, {
   'begin': 0x0900,
   'end': 0x097F
- },
- {
+}, {
   'begin': 0x0980,
   'end': 0x09FF
- },
- {
+}, {
   'begin': 0x0A00,
   'end': 0x0A7F
- },
- {
+}, {
   'begin': 0x0A80,
   'end': 0x0AFF
- },
- {
+}, {
   'begin': 0x0B00,
   'end': 0x0B7F
- },
- {
+}, {
   'begin': 0x0B80,
   'end': 0x0BFF
- },
- {
+}, {
   'begin': 0x0C00,
   'end': 0x0C7F
- },
- {
+}, {
   'begin': 0x0C80,
   'end': 0x0CFF
- },
- {
+}, {
   'begin': 0x0D00,
   'end': 0x0D7F
- },
- {
+}, {
   'begin': 0x0E00,
   'end': 0x0E7F
- },
- {
+}, {
   'begin': 0x0E80,
   'end': 0x0EFF
- },
- {
+}, {
   'begin': 0x10A0,
   'end': 0x10FF
- },
- {
+}, {
   'begin': 0x1B00,
   'end': 0x1B7F
- },
- {
+}, {
   'begin': 0x1100,
   'end': 0x11FF
- },
- {
+}, {
   'begin': 0x1E00,
   'end': 0x1EFF
- },
- {
+}, {
   'begin': 0x1F00,
   'end': 0x1FFF
- },
- {
+}, {
   'begin': 0x2000,
   'end': 0x206F
- },
- {
+}, {
   'begin': 0x2070,
   'end': 0x209F
- },
- {
+}, {
   'begin': 0x20A0,
   'end': 0x20CF
- },
- {
+}, {
   'begin': 0x20D0,
   'end': 0x20FF
- },
- {
+}, {
   'begin': 0x2100,
   'end': 0x214F
- },
- {
+}, {
   'begin': 0x2150,
   'end': 0x218F
- },
- {
+}, {
   'begin': 0x2190,
   'end': 0x21FF
- },
- {
+}, {
   'begin': 0x2200,
   'end': 0x22FF
- },
- {
+}, {
   'begin': 0x2300,
   'end': 0x23FF
- },
- {
+}, {
   'begin': 0x2400,
   'end': 0x243F
- },
- {
+}, {
   'begin': 0x2440,
   'end': 0x245F
- },
- {
+}, {
   'begin': 0x2460,
   'end': 0x24FF
- },
- {
+}, {
   'begin': 0x2500,
   'end': 0x257F
- },
- {
+}, {
   'begin': 0x2580,
   'end': 0x259F
- },
- {
+}, {
   'begin': 0x25A0,
   'end': 0x25FF
- },
- {
+}, {
   'begin': 0x2600,
   'end': 0x26FF
- },
- {
+}, {
   'begin': 0x2700,
   'end': 0x27BF
- },
- {
+}, {
   'begin': 0x3000,
   'end': 0x303F
- },
- {
+}, {
   'begin': 0x3040,
   'end': 0x309F
- },
- {
+}, {
   'begin': 0x30A0,
   'end': 0x30FF
- },
- {
+}, {
   'begin': 0x3100,
   'end': 0x312F
- },
- {
+}, {
   'begin': 0x3130,
   'end': 0x318F
- },
- {
+}, {
   'begin': 0xA840,
   'end': 0xA87F
- },
- {
+}, {
   'begin': 0x3200,
   'end': 0x32FF
- },
- {
+}, {
   'begin': 0x3300,
   'end': 0x33FF
- },
- {
+}, {
   'begin': 0xAC00,
   'end': 0xD7AF
- },
- {
+}, {
   'begin': 0xD800,
   'end': 0xDFFF
- },
- {
+}, {
   'begin': 0x10900,
   'end': 0x1091F
- },
- {
+}, {
   'begin': 0x4E00,
   'end': 0x9FFF
- },
- {
+}, {
   'begin': 0xE000,
   'end': 0xF8FF
- },
- {
+}, {
   'begin': 0x31C0,
   'end': 0x31EF
- },
- {
+}, {
   'begin': 0xFB00,
   'end': 0xFB4F
- },
- {
+}, {
   'begin': 0xFB50,
   'end': 0xFDFF
- },
- {
+}, {
   'begin': 0xFE20,
   'end': 0xFE2F
- },
- {
+}, {
   'begin': 0xFE10,
   'end': 0xFE1F
- },
- {
+}, {
   'begin': 0xFE50,
   'end': 0xFE6F
- },
- {
+}, {
   'begin': 0xFE70,
   'end': 0xFEFF
- },
- {
+}, {
   'begin': 0xFF00,
   'end': 0xFFEF
- },
- {
+}, {
   'begin': 0xFFF0,
   'end': 0xFFFF
- },
- {
+}, {
   'begin': 0x0F00,
   'end': 0x0FFF
- },
- {
+}, {
   'begin': 0x0700,
   'end': 0x074F
- },
- {
+}, {
   'begin': 0x0780,
   'end': 0x07BF
- },
- {
+}, {
   'begin': 0x0D80,
   'end': 0x0DFF
- },
- {
+}, {
   'begin': 0x1000,
   'end': 0x109F
- },
- {
+}, {
   'begin': 0x1200,
   'end': 0x137F
- },
- {
+}, {
   'begin': 0x13A0,
   'end': 0x13FF
- },
- {
+}, {
   'begin': 0x1400,
   'end': 0x167F
- },
- {
+}, {
   'begin': 0x1680,
   'end': 0x169F
- },
- {
+}, {
   'begin': 0x16A0,
   'end': 0x16FF
- },
- {
+}, {
   'begin': 0x1780,
   'end': 0x17FF
- },
- {
+}, {
   'begin': 0x1800,
   'end': 0x18AF
- },
- {
+}, {
   'begin': 0x2800,
   'end': 0x28FF
- },
- {
+}, {
   'begin': 0xA000,
   'end': 0xA48F
- },
- {
+}, {
   'begin': 0x1700,
   'end': 0x171F
- },
- {
+}, {
   'begin': 0x10300,
   'end': 0x1032F
- },
- {
+}, {
   'begin': 0x10330,
   'end': 0x1034F
- },
- {
+}, {
   'begin': 0x10400,
   'end': 0x1044F
- },
- {
+}, {
   'begin': 0x1D000,
   'end': 0x1D0FF
- },
- {
+}, {
   'begin': 0x1D400,
   'end': 0x1D7FF
- },
- {
+}, {
   'begin': 0xFF000,
   'end': 0xFFFFD
- },
- {
+}, {
   'begin': 0xFE00,
   'end': 0xFE0F
- },
- {
+}, {
   'begin': 0xE0000,
   'end': 0xE007F
- },
- {
+}, {
   'begin': 0x1900,
   'end': 0x194F
- },
- {
+}, {
   'begin': 0x1950,
   'end': 0x197F
- },
- {
+}, {
   'begin': 0x1980,
   'end': 0x19DF
- },
- {
+}, {
   'begin': 0x1A00,
   'end': 0x1A1F
- },
- {
+}, {
   'begin': 0x2C00,
   'end': 0x2C5F
- },
- {
+}, {
   'begin': 0x2D30,
   'end': 0x2D7F
- },
- {
+}, {
   'begin': 0x4DC0,
   'end': 0x4DFF
- },
- {
+}, {
   'begin': 0xA800,
   'end': 0xA82F
- },
- {
+}, {
   'begin': 0x10000,
   'end': 0x1007F
- },
- {
+}, {
   'begin': 0x10140,
   'end': 0x1018F
- },
- {
+}, {
   'begin': 0x10380,
   'end': 0x1039F
- },
- {
+}, {
   'begin': 0x103A0,
   'end': 0x103DF
- },
- {
+}, {
   'begin': 0x10450,
   'end': 0x1047F
- },
- {
+}, {
   'begin': 0x10480,
   'end': 0x104AF
- },
- {
+}, {
   'begin': 0x10800,
   'end': 0x1083F
- },
- {
+}, {
   'begin': 0x10A00,
   'end': 0x10A5F
- },
- {
+}, {
   'begin': 0x1D300,
   'end': 0x1D35F
- },
- {
+}, {
   'begin': 0x12000,
   'end': 0x123FF
- },
- {
+}, {
   'begin': 0x1D360,
   'end': 0x1D37F
- },
- {
+}, {
   'begin': 0x1B80,
   'end': 0x1BBF
- },
- {
+}, {
   'begin': 0x1C00,
   'end': 0x1C4F
- },
- {
+}, {
   'begin': 0x1C50,
   'end': 0x1C7F
- },
- {
+}, {
   'begin': 0xA880,
   'end': 0xA8DF
- },
- {
+}, {
   'begin': 0xA900,
   'end': 0xA92F
- },
- {
+}, {
   'begin': 0xA930,
   'end': 0xA95F
- },
- {
+}, {
   'begin': 0xAA00,
   'end': 0xAA5F
- },
- {
+}, {
   'begin': 0x10190,
   'end': 0x101CF
- },
- {
+}, {
   'begin': 0x101D0,
   'end': 0x101FF
- },
- {
+}, {
   'begin': 0x102A0,
   'end': 0x102DF
- },
- {
+}, {
   'begin': 0x1F030,
   'end': 0x1F09F
- }
-];
+}];
 function getUnicodeRangeFor(value) {
- for (var i = 0, ii = UnicodeRanges.length; i < ii; i++) {
-  var range = UnicodeRanges[i];
-  if (value >= range.begin && value < range.end) {
-   return i;
+  for (var i = 0, ii = UnicodeRanges.length; i < ii; i++) {
+    var range = UnicodeRanges[i];
+    if (value >= range.begin && value < range.end) {
+      return i;
+    }
   }
- }
- return -1;
+  return -1;
 }
 function isRTLRangeFor(value) {
- var range = UnicodeRanges[13];
- if (value >= range.begin && value < range.end) {
-  return true;
- }
- range = UnicodeRanges[11];
- if (value >= range.begin && value < range.end) {
-  return true;
- }
- return false;
+  var range = UnicodeRanges[13];
+  if (value >= range.begin && value < range.end) {
+    return true;
+  }
+  range = UnicodeRanges[11];
+  if (value >= range.begin && value < range.end) {
+    return true;
+  }
+  return false;
 }
 var getNormalizedUnicodes = getLookupTableFactory(function (t) {
- t['\u00A8'] = '\u0020\u0308';
- t['\u00AF'] = '\u0020\u0304';
- t['\u00B4'] = '\u0020\u0301';
- t['\u00B5'] = '\u03BC';
- t['\u00B8'] = '\u0020\u0327';
- t['\u0132'] = '\u0049\u004A';
- t['\u0133'] = '\u0069\u006A';
- t['\u013F'] = '\u004C\u00B7';
- t['\u0140'] = '\u006C\u00B7';
- t['\u0149'] = '\u02BC\u006E';
- t['\u017F'] = '\u0073';
- t['\u01C4'] = '\u0044\u017D';
- t['\u01C5'] = '\u0044\u017E';
- t['\u01C6'] = '\u0064\u017E';
- t['\u01C7'] = '\u004C\u004A';
- t['\u01C8'] = '\u004C\u006A';
- t['\u01C9'] = '\u006C\u006A';
- t['\u01CA'] = '\u004E\u004A';
- t['\u01CB'] = '\u004E\u006A';
- t['\u01CC'] = '\u006E\u006A';
- t['\u01F1'] = '\u0044\u005A';
- t['\u01F2'] = '\u0044\u007A';
- t['\u01F3'] = '\u0064\u007A';
- t['\u02D8'] = '\u0020\u0306';
- t['\u02D9'] = '\u0020\u0307';
- t['\u02DA'] = '\u0020\u030A';
- t['\u02DB'] = '\u0020\u0328';
- t['\u02DC'] = '\u0020\u0303';
- t['\u02DD'] = '\u0020\u030B';
- t['\u037A'] = '\u0020\u0345';
- t['\u0384'] = '\u0020\u0301';
- t['\u03D0'] = '\u03B2';
- t['\u03D1'] = '\u03B8';
- t['\u03D2'] = '\u03A5';
- t['\u03D5'] = '\u03C6';
- t['\u03D6'] = '\u03C0';
- t['\u03F0'] = '\u03BA';
- t['\u03F1'] = '\u03C1';
- t['\u03F2'] = '\u03C2';
- t['\u03F4'] = '\u0398';
- t['\u03F5'] = '\u03B5';
- t['\u03F9'] = '\u03A3';
- t['\u0587'] = '\u0565\u0582';
- t['\u0675'] = '\u0627\u0674';
- t['\u0676'] = '\u0648\u0674';
- t['\u0677'] = '\u06C7\u0674';
- t['\u0678'] = '\u064A\u0674';
- t['\u0E33'] = '\u0E4D\u0E32';
- t['\u0EB3'] = '\u0ECD\u0EB2';
- t['\u0EDC'] = '\u0EAB\u0E99';
- t['\u0EDD'] = '\u0EAB\u0EA1';
- t['\u0F77'] = '\u0FB2\u0F81';
- t['\u0F79'] = '\u0FB3\u0F81';
- t['\u1E9A'] = '\u0061\u02BE';
- t['\u1FBD'] = '\u0020\u0313';
- t['\u1FBF'] = '\u0020\u0313';
- t['\u1FC0'] = '\u0020\u0342';
- t['\u1FFE'] = '\u0020\u0314';
- t['\u2002'] = '\u0020';
- t['\u2003'] = '\u0020';
- t['\u2004'] = '\u0020';
- t['\u2005'] = '\u0020';
- t['\u2006'] = '\u0020';
- t['\u2008'] = '\u0020';
- t['\u2009'] = '\u0020';
- t['\u200A'] = '\u0020';
- t['\u2017'] = '\u0020\u0333';
- t['\u2024'] = '\u002E';
- t['\u2025'] = '\u002E\u002E';
- t['\u2026'] = '\u002E\u002E\u002E';
- t['\u2033'] = '\u2032\u2032';
- t['\u2034'] = '\u2032\u2032\u2032';
- t['\u2036'] = '\u2035\u2035';
- t['\u2037'] = '\u2035\u2035\u2035';
- t['\u203C'] = '\u0021\u0021';
- t['\u203E'] = '\u0020\u0305';
- t['\u2047'] = '\u003F\u003F';
- t['\u2048'] = '\u003F\u0021';
- t['\u2049'] = '\u0021\u003F';
- t['\u2057'] = '\u2032\u2032\u2032\u2032';
- t['\u205F'] = '\u0020';
- t['\u20A8'] = '\u0052\u0073';
- t['\u2100'] = '\u0061\u002F\u0063';
- t['\u2101'] = '\u0061\u002F\u0073';
- t['\u2103'] = '\u00B0\u0043';
- t['\u2105'] = '\u0063\u002F\u006F';
- t['\u2106'] = '\u0063\u002F\u0075';
- t['\u2107'] = '\u0190';
- t['\u2109'] = '\u00B0\u0046';
- t['\u2116'] = '\u004E\u006F';
- t['\u2121'] = '\u0054\u0045\u004C';
- t['\u2135'] = '\u05D0';
- t['\u2136'] = '\u05D1';
- t['\u2137'] = '\u05D2';
- t['\u2138'] = '\u05D3';
- t['\u213B'] = '\u0046\u0041\u0058';
- t['\u2160'] = '\u0049';
- t['\u2161'] = '\u0049\u0049';
- t['\u2162'] = '\u0049\u0049\u0049';
- t['\u2163'] = '\u0049\u0056';
- t['\u2164'] = '\u0056';
- t['\u2165'] = '\u0056\u0049';
- t['\u2166'] = '\u0056\u0049\u0049';
- t['\u2167'] = '\u0056\u0049\u0049\u0049';
- t['\u2168'] = '\u0049\u0058';
- t['\u2169'] = '\u0058';
- t['\u216A'] = '\u0058\u0049';
- t['\u216B'] = '\u0058\u0049\u0049';
- t['\u216C'] = '\u004C';
- t['\u216D'] = '\u0043';
- t['\u216E'] = '\u0044';
- t['\u216F'] = '\u004D';
- t['\u2170'] = '\u0069';
- t['\u2171'] = '\u0069\u0069';
- t['\u2172'] = '\u0069\u0069\u0069';
- t['\u2173'] = '\u0069\u0076';
- t['\u2174'] = '\u0076';
- t['\u2175'] = '\u0076\u0069';
- t['\u2176'] = '\u0076\u0069\u0069';
- t['\u2177'] = '\u0076\u0069\u0069\u0069';
- t['\u2178'] = '\u0069\u0078';
- t['\u2179'] = '\u0078';
- t['\u217A'] = '\u0078\u0069';
- t['\u217B'] = '\u0078\u0069\u0069';
- t['\u217C'] = '\u006C';
- t['\u217D'] = '\u0063';
- t['\u217E'] = '\u0064';
- t['\u217F'] = '\u006D';
- t['\u222C'] = '\u222B\u222B';
- t['\u222D'] = '\u222B\u222B\u222B';
- t['\u222F'] = '\u222E\u222E';
- t['\u2230'] = '\u222E\u222E\u222E';
- t['\u2474'] = '\u0028\u0031\u0029';
- t['\u2475'] = '\u0028\u0032\u0029';
- t['\u2476'] = '\u0028\u0033\u0029';
- t['\u2477'] = '\u0028\u0034\u0029';
- t['\u2478'] = '\u0028\u0035\u0029';
- t['\u2479'] = '\u0028\u0036\u0029';
- t['\u247A'] = '\u0028\u0037\u0029';
- t['\u247B'] = '\u0028\u0038\u0029';
- t['\u247C'] = '\u0028\u0039\u0029';
- t['\u247D'] = '\u0028\u0031\u0030\u0029';
- t['\u247E'] = '\u0028\u0031\u0031\u0029';
- t['\u247F'] = '\u0028\u0031\u0032\u0029';
- t['\u2480'] = '\u0028\u0031\u0033\u0029';
- t['\u2481'] = '\u0028\u0031\u0034\u0029';
- t['\u2482'] = '\u0028\u0031\u0035\u0029';
- t['\u2483'] = '\u0028\u0031\u0036\u0029';
- t['\u2484'] = '\u0028\u0031\u0037\u0029';
- t['\u2485'] = '\u0028\u0031\u0038\u0029';
- t['\u2486'] = '\u0028\u0031\u0039\u0029';
- t['\u2487'] = '\u0028\u0032\u0030\u0029';
- t['\u2488'] = '\u0031\u002E';
- t['\u2489'] = '\u0032\u002E';
- t['\u248A'] = '\u0033\u002E';
- t['\u248B'] = '\u0034\u002E';
- t['\u248C'] = '\u0035\u002E';
- t['\u248D'] = '\u0036\u002E';
- t['\u248E'] = '\u0037\u002E';
- t['\u248F'] = '\u0038\u002E';
- t['\u2490'] = '\u0039\u002E';
- t['\u2491'] = '\u0031\u0030\u002E';
- t['\u2492'] = '\u0031\u0031\u002E';
- t['\u2493'] = '\u0031\u0032\u002E';
- t['\u2494'] = '\u0031\u0033\u002E';
- t['\u2495'] = '\u0031\u0034\u002E';
- t['\u2496'] = '\u0031\u0035\u002E';
- t['\u2497'] = '\u0031\u0036\u002E';
- t['\u2498'] = '\u0031\u0037\u002E';
- t['\u2499'] = '\u0031\u0038\u002E';
- t['\u249A'] = '\u0031\u0039\u002E';
- t['\u249B'] = '\u0032\u0030\u002E';
- t['\u249C'] = '\u0028\u0061\u0029';
- t['\u249D'] = '\u0028\u0062\u0029';
- t['\u249E'] = '\u0028\u0063\u0029';
- t['\u249F'] = '\u0028\u0064\u0029';
- t['\u24A0'] = '\u0028\u0065\u0029';
- t['\u24A1'] = '\u0028\u0066\u0029';
- t['\u24A2'] = '\u0028\u0067\u0029';
- t['\u24A3'] = '\u0028\u0068\u0029';
- t['\u24A4'] = '\u0028\u0069\u0029';
- t['\u24A5'] = '\u0028\u006A\u0029';
- t['\u24A6'] = '\u0028\u006B\u0029';
- t['\u24A7'] = '\u0028\u006C\u0029';
- t['\u24A8'] = '\u0028\u006D\u0029';
- t['\u24A9'] = '\u0028\u006E\u0029';
- t['\u24AA'] = '\u0028\u006F\u0029';
- t['\u24AB'] = '\u0028\u0070\u0029';
- t['\u24AC'] = '\u0028\u0071\u0029';
- t['\u24AD'] = '\u0028\u0072\u0029';
- t['\u24AE'] = '\u0028\u0073\u0029';
- t['\u24AF'] = '\u0028\u0074\u0029';
- t['\u24B0'] = '\u0028\u0075\u0029';
- t['\u24B1'] = '\u0028\u0076\u0029';
- t['\u24B2'] = '\u0028\u0077\u0029';
- t['\u24B3'] = '\u0028\u0078\u0029';
- t['\u24B4'] = '\u0028\u0079\u0029';
- t['\u24B5'] = '\u0028\u007A\u0029';
- t['\u2A0C'] = '\u222B\u222B\u222B\u222B';
- t['\u2A74'] = '\u003A\u003A\u003D';
- t['\u2A75'] = '\u003D\u003D';
- t['\u2A76'] = '\u003D\u003D\u003D';
- t['\u2E9F'] = '\u6BCD';
- t['\u2EF3'] = '\u9F9F';
- t['\u2F00'] = '\u4E00';
- t['\u2F01'] = '\u4E28';
- t['\u2F02'] = '\u4E36';
- t['\u2F03'] = '\u4E3F';
- t['\u2F04'] = '\u4E59';
- t['\u2F05'] = '\u4E85';
- t['\u2F06'] = '\u4E8C';
- t['\u2F07'] = '\u4EA0';
- t['\u2F08'] = '\u4EBA';
- t['\u2F09'] = '\u513F';
- t['\u2F0A'] = '\u5165';
- t['\u2F0B'] = '\u516B';
- t['\u2F0C'] = '\u5182';
- t['\u2F0D'] = '\u5196';
- t['\u2F0E'] = '\u51AB';
- t['\u2F0F'] = '\u51E0';
- t['\u2F10'] = '\u51F5';
- t['\u2F11'] = '\u5200';
- t['\u2F12'] = '\u529B';
- t['\u2F13'] = '\u52F9';
- t['\u2F14'] = '\u5315';
- t['\u2F15'] = '\u531A';
- t['\u2F16'] = '\u5338';
- t['\u2F17'] = '\u5341';
- t['\u2F18'] = '\u535C';
- t['\u2F19'] = '\u5369';
- t['\u2F1A'] = '\u5382';
- t['\u2F1B'] = '\u53B6';
- t['\u2F1C'] = '\u53C8';
- t['\u2F1D'] = '\u53E3';
- t['\u2F1E'] = '\u56D7';
- t['\u2F1F'] = '\u571F';
- t['\u2F20'] = '\u58EB';
- t['\u2F21'] = '\u5902';
- t['\u2F22'] = '\u590A';
- t['\u2F23'] = '\u5915';
- t['\u2F24'] = '\u5927';
- t['\u2F25'] = '\u5973';
- t['\u2F26'] = '\u5B50';
- t['\u2F27'] = '\u5B80';
- t['\u2F28'] = '\u5BF8';
- t['\u2F29'] = '\u5C0F';
- t['\u2F2A'] = '\u5C22';
- t['\u2F2B'] = '\u5C38';
- t['\u2F2C'] = '\u5C6E';
- t['\u2F2D'] = '\u5C71';
- t['\u2F2E'] = '\u5DDB';
- t['\u2F2F'] = '\u5DE5';
- t['\u2F30'] = '\u5DF1';
- t['\u2F31'] = '\u5DFE';
- t['\u2F32'] = '\u5E72';
- t['\u2F33'] = '\u5E7A';
- t['\u2F34'] = '\u5E7F';
- t['\u2F35'] = '\u5EF4';
- t['\u2F36'] = '\u5EFE';
- t['\u2F37'] = '\u5F0B';
- t['\u2F38'] = '\u5F13';
- t['\u2F39'] = '\u5F50';
- t['\u2F3A'] = '\u5F61';
- t['\u2F3B'] = '\u5F73';
- t['\u2F3C'] = '\u5FC3';
- t['\u2F3D'] = '\u6208';
- t['\u2F3E'] = '\u6236';
- t['\u2F3F'] = '\u624B';
- t['\u2F40'] = '\u652F';
- t['\u2F41'] = '\u6534';
- t['\u2F42'] = '\u6587';
- t['\u2F43'] = '\u6597';
- t['\u2F44'] = '\u65A4';
- t['\u2F45'] = '\u65B9';
- t['\u2F46'] = '\u65E0';
- t['\u2F47'] = '\u65E5';
- t['\u2F48'] = '\u66F0';
- t['\u2F49'] = '\u6708';
- t['\u2F4A'] = '\u6728';
- t['\u2F4B'] = '\u6B20';
- t['\u2F4C'] = '\u6B62';
- t['\u2F4D'] = '\u6B79';
- t['\u2F4E'] = '\u6BB3';
- t['\u2F4F'] = '\u6BCB';
- t['\u2F50'] = '\u6BD4';
- t['\u2F51'] = '\u6BDB';
- t['\u2F52'] = '\u6C0F';
- t['\u2F53'] = '\u6C14';
- t['\u2F54'] = '\u6C34';
- t['\u2F55'] = '\u706B';
- t['\u2F56'] = '\u722A';
- t['\u2F57'] = '\u7236';
- t['\u2F58'] = '\u723B';
- t['\u2F59'] = '\u723F';
- t['\u2F5A'] = '\u7247';
- t['\u2F5B'] = '\u7259';
- t['\u2F5C'] = '\u725B';
- t['\u2F5D'] = '\u72AC';
- t['\u2F5E'] = '\u7384';
- t['\u2F5F'] = '\u7389';
- t['\u2F60'] = '\u74DC';
- t['\u2F61'] = '\u74E6';
- t['\u2F62'] = '\u7518';
- t['\u2F63'] = '\u751F';
- t['\u2F64'] = '\u7528';
- t['\u2F65'] = '\u7530';
- t['\u2F66'] = '\u758B';
- t['\u2F67'] = '\u7592';
- t['\u2F68'] = '\u7676';
- t['\u2F69'] = '\u767D';
- t['\u2F6A'] = '\u76AE';
- t['\u2F6B'] = '\u76BF';
- t['\u2F6C'] = '\u76EE';
- t['\u2F6D'] = '\u77DB';
- t['\u2F6E'] = '\u77E2';
- t['\u2F6F'] = '\u77F3';
- t['\u2F70'] = '\u793A';
- t['\u2F71'] = '\u79B8';
- t['\u2F72'] = '\u79BE';
- t['\u2F73'] = '\u7A74';
- t['\u2F74'] = '\u7ACB';
- t['\u2F75'] = '\u7AF9';
- t['\u2F76'] = '\u7C73';
- t['\u2F77'] = '\u7CF8';
- t['\u2F78'] = '\u7F36';
- t['\u2F79'] = '\u7F51';
- t['\u2F7A'] = '\u7F8A';
- t['\u2F7B'] = '\u7FBD';
- t['\u2F7C'] = '\u8001';
- t['\u2F7D'] = '\u800C';
- t['\u2F7E'] = '\u8012';
- t['\u2F7F'] = '\u8033';
- t['\u2F80'] = '\u807F';
- t['\u2F81'] = '\u8089';
- t['\u2F82'] = '\u81E3';
- t['\u2F83'] = '\u81EA';
- t['\u2F84'] = '\u81F3';
- t['\u2F85'] = '\u81FC';
- t['\u2F86'] = '\u820C';
- t['\u2F87'] = '\u821B';
- t['\u2F88'] = '\u821F';
- t['\u2F89'] = '\u826E';
- t['\u2F8A'] = '\u8272';
- t['\u2F8B'] = '\u8278';
- t['\u2F8C'] = '\u864D';
- t['\u2F8D'] = '\u866B';
- t['\u2F8E'] = '\u8840';
- t['\u2F8F'] = '\u884C';
- t['\u2F90'] = '\u8863';
- t['\u2F91'] = '\u897E';
- t['\u2F92'] = '\u898B';
- t['\u2F93'] = '\u89D2';
- t['\u2F94'] = '\u8A00';
- t['\u2F95'] = '\u8C37';
- t['\u2F96'] = '\u8C46';
- t['\u2F97'] = '\u8C55';
- t['\u2F98'] = '\u8C78';
- t['\u2F99'] = '\u8C9D';
- t['\u2F9A'] = '\u8D64';
- t['\u2F9B'] = '\u8D70';
- t['\u2F9C'] = '\u8DB3';
- t['\u2F9D'] = '\u8EAB';
- t['\u2F9E'] = '\u8ECA';
- t['\u2F9F'] = '\u8F9B';
- t['\u2FA0'] = '\u8FB0';
- t['\u2FA1'] = '\u8FB5';
- t['\u2FA2'] = '\u9091';
- t['\u2FA3'] = '\u9149';
- t['\u2FA4'] = '\u91C6';
- t['\u2FA5'] = '\u91CC';
- t['\u2FA6'] = '\u91D1';
- t['\u2FA7'] = '\u9577';
- t['\u2FA8'] = '\u9580';
- t['\u2FA9'] = '\u961C';
- t['\u2FAA'] = '\u96B6';
- t['\u2FAB'] = '\u96B9';
- t['\u2FAC'] = '\u96E8';
- t['\u2FAD'] = '\u9751';
- t['\u2FAE'] = '\u975E';
- t['\u2FAF'] = '\u9762';
- t['\u2FB0'] = '\u9769';
- t['\u2FB1'] = '\u97CB';
- t['\u2FB2'] = '\u97ED';
- t['\u2FB3'] = '\u97F3';
- t['\u2FB4'] = '\u9801';
- t['\u2FB5'] = '\u98A8';
- t['\u2FB6'] = '\u98DB';
- t['\u2FB7'] = '\u98DF';
- t['\u2FB8'] = '\u9996';
- t['\u2FB9'] = '\u9999';
- t['\u2FBA'] = '\u99AC';
- t['\u2FBB'] = '\u9AA8';
- t['\u2FBC'] = '\u9AD8';
- t['\u2FBD'] = '\u9ADF';
- t['\u2FBE'] = '\u9B25';
- t['\u2FBF'] = '\u9B2F';
- t['\u2FC0'] = '\u9B32';
- t['\u2FC1'] = '\u9B3C';
- t['\u2FC2'] = '\u9B5A';
- t['\u2FC3'] = '\u9CE5';
- t['\u2FC4'] = '\u9E75';
- t['\u2FC5'] = '\u9E7F';
- t['\u2FC6'] = '\u9EA5';
- t['\u2FC7'] = '\u9EBB';
- t['\u2FC8'] = '\u9EC3';
- t['\u2FC9'] = '\u9ECD';
- t['\u2FCA'] = '\u9ED1';
- t['\u2FCB'] = '\u9EF9';
- t['\u2FCC'] = '\u9EFD';
- t['\u2FCD'] = '\u9F0E';
- t['\u2FCE'] = '\u9F13';
- t['\u2FCF'] = '\u9F20';
- t['\u2FD0'] = '\u9F3B';
- t['\u2FD1'] = '\u9F4A';
- t['\u2FD2'] = '\u9F52';
- t['\u2FD3'] = '\u9F8D';
- t['\u2FD4'] = '\u9F9C';
- t['\u2FD5'] = '\u9FA0';
- t['\u3036'] = '\u3012';
- t['\u3038'] = '\u5341';
- t['\u3039'] = '\u5344';
- t['\u303A'] = '\u5345';
- t['\u309B'] = '\u0020\u3099';
- t['\u309C'] = '\u0020\u309A';
- t['\u3131'] = '\u1100';
- t['\u3132'] = '\u1101';
- t['\u3133'] = '\u11AA';
- t['\u3134'] = '\u1102';
- t['\u3135'] = '\u11AC';
- t['\u3136'] = '\u11AD';
- t['\u3137'] = '\u1103';
- t['\u3138'] = '\u1104';
- t['\u3139'] = '\u1105';
- t['\u313A'] = '\u11B0';
- t['\u313B'] = '\u11B1';
- t['\u313C'] = '\u11B2';
- t['\u313D'] = '\u11B3';
- t['\u313E'] = '\u11B4';
- t['\u313F'] = '\u11B5';
- t['\u3140'] = '\u111A';
- t['\u3141'] = '\u1106';
- t['\u3142'] = '\u1107';
- t['\u3143'] = '\u1108';
- t['\u3144'] = '\u1121';
- t['\u3145'] = '\u1109';
- t['\u3146'] = '\u110A';
- t['\u3147'] = '\u110B';
- t['\u3148'] = '\u110C';
- t['\u3149'] = '\u110D';
- t['\u314A'] = '\u110E';
- t['\u314B'] = '\u110F';
- t['\u314C'] = '\u1110';
- t['\u314D'] = '\u1111';
- t['\u314E'] = '\u1112';
- t['\u314F'] = '\u1161';
- t['\u3150'] = '\u1162';
- t['\u3151'] = '\u1163';
- t['\u3152'] = '\u1164';
- t['\u3153'] = '\u1165';
- t['\u3154'] = '\u1166';
- t['\u3155'] = '\u1167';
- t['\u3156'] = '\u1168';
- t['\u3157'] = '\u1169';
- t['\u3158'] = '\u116A';
- t['\u3159'] = '\u116B';
- t['\u315A'] = '\u116C';
- t['\u315B'] = '\u116D';
- t['\u315C'] = '\u116E';
- t['\u315D'] = '\u116F';
- t['\u315E'] = '\u1170';
- t['\u315F'] = '\u1171';
- t['\u3160'] = '\u1172';
- t['\u3161'] = '\u1173';
- t['\u3162'] = '\u1174';
- t['\u3163'] = '\u1175';
- t['\u3164'] = '\u1160';
- t['\u3165'] = '\u1114';
- t['\u3166'] = '\u1115';
- t['\u3167'] = '\u11C7';
- t['\u3168'] = '\u11C8';
- t['\u3169'] = '\u11CC';
- t['\u316A'] = '\u11CE';
- t['\u316B'] = '\u11D3';
- t['\u316C'] = '\u11D7';
- t['\u316D'] = '\u11D9';
- t['\u316E'] = '\u111C';
- t['\u316F'] = '\u11DD';
- t['\u3170'] = '\u11DF';
- t['\u3171'] = '\u111D';
- t['\u3172'] = '\u111E';
- t['\u3173'] = '\u1120';
- t['\u3174'] = '\u1122';
- t['\u3175'] = '\u1123';
- t['\u3176'] = '\u1127';
- t['\u3177'] = '\u1129';
- t['\u3178'] = '\u112B';
- t['\u3179'] = '\u112C';
- t['\u317A'] = '\u112D';
- t['\u317B'] = '\u112E';
- t['\u317C'] = '\u112F';
- t['\u317D'] = '\u1132';
- t['\u317E'] = '\u1136';
- t['\u317F'] = '\u1140';
- t['\u3180'] = '\u1147';
- t['\u3181'] = '\u114C';
- t['\u3182'] = '\u11F1';
- t['\u3183'] = '\u11F2';
- t['\u3184'] = '\u1157';
- t['\u3185'] = '\u1158';
- t['\u3186'] = '\u1159';
- t['\u3187'] = '\u1184';
- t['\u3188'] = '\u1185';
- t['\u3189'] = '\u1188';
- t['\u318A'] = '\u1191';
- t['\u318B'] = '\u1192';
- t['\u318C'] = '\u1194';
- t['\u318D'] = '\u119E';
- t['\u318E'] = '\u11A1';
- t['\u3200'] = '\u0028\u1100\u0029';
- t['\u3201'] = '\u0028\u1102\u0029';
- t['\u3202'] = '\u0028\u1103\u0029';
- t['\u3203'] = '\u0028\u1105\u0029';
- t['\u3204'] = '\u0028\u1106\u0029';
- t['\u3205'] = '\u0028\u1107\u0029';
- t['\u3206'] = '\u0028\u1109\u0029';
- t['\u3207'] = '\u0028\u110B\u0029';
- t['\u3208'] = '\u0028\u110C\u0029';
- t['\u3209'] = '\u0028\u110E\u0029';
- t['\u320A'] = '\u0028\u110F\u0029';
- t['\u320B'] = '\u0028\u1110\u0029';
- t['\u320C'] = '\u0028\u1111\u0029';
- t['\u320D'] = '\u0028\u1112\u0029';
- t['\u320E'] = '\u0028\u1100\u1161\u0029';
- t['\u320F'] = '\u0028\u1102\u1161\u0029';
- t['\u3210'] = '\u0028\u1103\u1161\u0029';
- t['\u3211'] = '\u0028\u1105\u1161\u0029';
- t['\u3212'] = '\u0028\u1106\u1161\u0029';
- t['\u3213'] = '\u0028\u1107\u1161\u0029';
- t['\u3214'] = '\u0028\u1109\u1161\u0029';
- t['\u3215'] = '\u0028\u110B\u1161\u0029';
- t['\u3216'] = '\u0028\u110C\u1161\u0029';
- t['\u3217'] = '\u0028\u110E\u1161\u0029';
- t['\u3218'] = '\u0028\u110F\u1161\u0029';
- t['\u3219'] = '\u0028\u1110\u1161\u0029';
- t['\u321A'] = '\u0028\u1111\u1161\u0029';
- t['\u321B'] = '\u0028\u1112\u1161\u0029';
- t['\u321C'] = '\u0028\u110C\u116E\u0029';
- t['\u321D'] = '\u0028\u110B\u1169\u110C\u1165\u11AB\u0029';
- t['\u321E'] = '\u0028\u110B\u1169\u1112\u116E\u0029';
- t['\u3220'] = '\u0028\u4E00\u0029';
- t['\u3221'] = '\u0028\u4E8C\u0029';
- t['\u3222'] = '\u0028\u4E09\u0029';
- t['\u3223'] = '\u0028\u56DB\u0029';
- t['\u3224'] = '\u0028\u4E94\u0029';
- t['\u3225'] = '\u0028\u516D\u0029';
- t['\u3226'] = '\u0028\u4E03\u0029';
- t['\u3227'] = '\u0028\u516B\u0029';
- t['\u3228'] = '\u0028\u4E5D\u0029';
- t['\u3229'] = '\u0028\u5341\u0029';
- t['\u322A'] = '\u0028\u6708\u0029';
- t['\u322B'] = '\u0028\u706B\u0029';
- t['\u322C'] = '\u0028\u6C34\u0029';
- t['\u322D'] = '\u0028\u6728\u0029';
- t['\u322E'] = '\u0028\u91D1\u0029';
- t['\u322F'] = '\u0028\u571F\u0029';
- t['\u3230'] = '\u0028\u65E5\u0029';
- t['\u3231'] = '\u0028\u682A\u0029';
- t['\u3232'] = '\u0028\u6709\u0029';
- t['\u3233'] = '\u0028\u793E\u0029';
- t['\u3234'] = '\u0028\u540D\u0029';
- t['\u3235'] = '\u0028\u7279\u0029';
- t['\u3236'] = '\u0028\u8CA1\u0029';
- t['\u3237'] = '\u0028\u795D\u0029';
- t['\u3238'] = '\u0028\u52B4\u0029';
- t['\u3239'] = '\u0028\u4EE3\u0029';
- t['\u323A'] = '\u0028\u547C\u0029';
- t['\u323B'] = '\u0028\u5B66\u0029';
- t['\u323C'] = '\u0028\u76E3\u0029';
- t['\u323D'] = '\u0028\u4F01\u0029';
- t['\u323E'] = '\u0028\u8CC7\u0029';
- t['\u323F'] = '\u0028\u5354\u0029';
- t['\u3240'] = '\u0028\u796D\u0029';
- t['\u3241'] = '\u0028\u4F11\u0029';
- t['\u3242'] = '\u0028\u81EA\u0029';
- t['\u3243'] = '\u0028\u81F3\u0029';
- t['\u32C0'] = '\u0031\u6708';
- t['\u32C1'] = '\u0032\u6708';
- t['\u32C2'] = '\u0033\u6708';
- t['\u32C3'] = '\u0034\u6708';
- t['\u32C4'] = '\u0035\u6708';
- t['\u32C5'] = '\u0036\u6708';
- t['\u32C6'] = '\u0037\u6708';
- t['\u32C7'] = '\u0038\u6708';
- t['\u32C8'] = '\u0039\u6708';
- t['\u32C9'] = '\u0031\u0030\u6708';
- t['\u32CA'] = '\u0031\u0031\u6708';
- t['\u32CB'] = '\u0031\u0032\u6708';
- t['\u3358'] = '\u0030\u70B9';
- t['\u3359'] = '\u0031\u70B9';
- t['\u335A'] = '\u0032\u70B9';
- t['\u335B'] = '\u0033\u70B9';
- t['\u335C'] = '\u0034\u70B9';
- t['\u335D'] = '\u0035\u70B9';
- t['\u335E'] = '\u0036\u70B9';
- t['\u335F'] = '\u0037\u70B9';
- t['\u3360'] = '\u0038\u70B9';
- t['\u3361'] = '\u0039\u70B9';
- t['\u3362'] = '\u0031\u0030\u70B9';
- t['\u3363'] = '\u0031\u0031\u70B9';
- t['\u3364'] = '\u0031\u0032\u70B9';
- t['\u3365'] = '\u0031\u0033\u70B9';
- t['\u3366'] = '\u0031\u0034\u70B9';
- t['\u3367'] = '\u0031\u0035\u70B9';
- t['\u3368'] = '\u0031\u0036\u70B9';
- t['\u3369'] = '\u0031\u0037\u70B9';
- t['\u336A'] = '\u0031\u0038\u70B9';
- t['\u336B'] = '\u0031\u0039\u70B9';
- t['\u336C'] = '\u0032\u0030\u70B9';
- t['\u336D'] = '\u0032\u0031\u70B9';
- t['\u336E'] = '\u0032\u0032\u70B9';
- t['\u336F'] = '\u0032\u0033\u70B9';
- t['\u3370'] = '\u0032\u0034\u70B9';
- t['\u33E0'] = '\u0031\u65E5';
- t['\u33E1'] = '\u0032\u65E5';
- t['\u33E2'] = '\u0033\u65E5';
- t['\u33E3'] = '\u0034\u65E5';
- t['\u33E4'] = '\u0035\u65E5';
- t['\u33E5'] = '\u0036\u65E5';
- t['\u33E6'] = '\u0037\u65E5';
- t['\u33E7'] = '\u0038\u65E5';
- t['\u33E8'] = '\u0039\u65E5';
- t['\u33E9'] = '\u0031\u0030\u65E5';
- t['\u33EA'] = '\u0031\u0031\u65E5';
- t['\u33EB'] = '\u0031\u0032\u65E5';
- t['\u33EC'] = '\u0031\u0033\u65E5';
- t['\u33ED'] = '\u0031\u0034\u65E5';
- t['\u33EE'] = '\u0031\u0035\u65E5';
- t['\u33EF'] = '\u0031\u0036\u65E5';
- t['\u33F0'] = '\u0031\u0037\u65E5';
- t['\u33F1'] = '\u0031\u0038\u65E5';
- t['\u33F2'] = '\u0031\u0039\u65E5';
- t['\u33F3'] = '\u0032\u0030\u65E5';
- t['\u33F4'] = '\u0032\u0031\u65E5';
- t['\u33F5'] = '\u0032\u0032\u65E5';
- t['\u33F6'] = '\u0032\u0033\u65E5';
- t['\u33F7'] = '\u0032\u0034\u65E5';
- t['\u33F8'] = '\u0032\u0035\u65E5';
- t['\u33F9'] = '\u0032\u0036\u65E5';
- t['\u33FA'] = '\u0032\u0037\u65E5';
- t['\u33FB'] = '\u0032\u0038\u65E5';
- t['\u33FC'] = '\u0032\u0039\u65E5';
- t['\u33FD'] = '\u0033\u0030\u65E5';
- t['\u33FE'] = '\u0033\u0031\u65E5';
- t['\uFB00'] = '\u0066\u0066';
- t['\uFB01'] = '\u0066\u0069';
- t['\uFB02'] = '\u0066\u006C';
- t['\uFB03'] = '\u0066\u0066\u0069';
- t['\uFB04'] = '\u0066\u0066\u006C';
- t['\uFB05'] = '\u017F\u0074';
- t['\uFB06'] = '\u0073\u0074';
- t['\uFB13'] = '\u0574\u0576';
- t['\uFB14'] = '\u0574\u0565';
- t['\uFB15'] = '\u0574\u056B';
- t['\uFB16'] = '\u057E\u0576';
- t['\uFB17'] = '\u0574\u056D';
- t['\uFB4F'] = '\u05D0\u05DC';
- t['\uFB50'] = '\u0671';
- t['\uFB51'] = '\u0671';
- t['\uFB52'] = '\u067B';
- t['\uFB53'] = '\u067B';
- t['\uFB54'] = '\u067B';
- t['\uFB55'] = '\u067B';
- t['\uFB56'] = '\u067E';
- t['\uFB57'] = '\u067E';
- t['\uFB58'] = '\u067E';
- t['\uFB59'] = '\u067E';
- t['\uFB5A'] = '\u0680';
- t['\uFB5B'] = '\u0680';
- t['\uFB5C'] = '\u0680';
- t['\uFB5D'] = '\u0680';
- t['\uFB5E'] = '\u067A';
- t['\uFB5F'] = '\u067A';
- t['\uFB60'] = '\u067A';
- t['\uFB61'] = '\u067A';
- t['\uFB62'] = '\u067F';
- t['\uFB63'] = '\u067F';
- t['\uFB64'] = '\u067F';
- t['\uFB65'] = '\u067F';
- t['\uFB66'] = '\u0679';
- t['\uFB67'] = '\u0679';
- t['\uFB68'] = '\u0679';
- t['\uFB69'] = '\u0679';
- t['\uFB6A'] = '\u06A4';
- t['\uFB6B'] = '\u06A4';
- t['\uFB6C'] = '\u06A4';
- t['\uFB6D'] = '\u06A4';
- t['\uFB6E'] = '\u06A6';
- t['\uFB6F'] = '\u06A6';
- t['\uFB70'] = '\u06A6';
- t['\uFB71'] = '\u06A6';
- t['\uFB72'] = '\u0684';
- t['\uFB73'] = '\u0684';
- t['\uFB74'] = '\u0684';
- t['\uFB75'] = '\u0684';
- t['\uFB76'] = '\u0683';
- t['\uFB77'] = '\u0683';
- t['\uFB78'] = '\u0683';
- t['\uFB79'] = '\u0683';
- t['\uFB7A'] = '\u0686';
- t['\uFB7B'] = '\u0686';
- t['\uFB7C'] = '\u0686';
- t['\uFB7D'] = '\u0686';
- t['\uFB7E'] = '\u0687';
- t['\uFB7F'] = '\u0687';
- t['\uFB80'] = '\u0687';
- t['\uFB81'] = '\u0687';
- t['\uFB82'] = '\u068D';
- t['\uFB83'] = '\u068D';
- t['\uFB84'] = '\u068C';
- t['\uFB85'] = '\u068C';
- t['\uFB86'] = '\u068E';
- t['\uFB87'] = '\u068E';
- t['\uFB88'] = '\u0688';
- t['\uFB89'] = '\u0688';
- t['\uFB8A'] = '\u0698';
- t['\uFB8B'] = '\u0698';
- t['\uFB8C'] = '\u0691';
- t['\uFB8D'] = '\u0691';
- t['\uFB8E'] = '\u06A9';
- t['\uFB8F'] = '\u06A9';
- t['\uFB90'] = '\u06A9';
- t['\uFB91'] = '\u06A9';
- t['\uFB92'] = '\u06AF';
- t['\uFB93'] = '\u06AF';
- t['\uFB94'] = '\u06AF';
- t['\uFB95'] = '\u06AF';
- t['\uFB96'] = '\u06B3';
- t['\uFB97'] = '\u06B3';
- t['\uFB98'] = '\u06B3';
- t['\uFB99'] = '\u06B3';
- t['\uFB9A'] = '\u06B1';
- t['\uFB9B'] = '\u06B1';
- t['\uFB9C'] = '\u06B1';
- t['\uFB9D'] = '\u06B1';
- t['\uFB9E'] = '\u06BA';
- t['\uFB9F'] = '\u06BA';
- t['\uFBA0'] = '\u06BB';
- t['\uFBA1'] = '\u06BB';
- t['\uFBA2'] = '\u06BB';
- t['\uFBA3'] = '\u06BB';
- t['\uFBA4'] = '\u06C0';
- t['\uFBA5'] = '\u06C0';
- t['\uFBA6'] = '\u06C1';
- t['\uFBA7'] = '\u06C1';
- t['\uFBA8'] = '\u06C1';
- t['\uFBA9'] = '\u06C1';
- t['\uFBAA'] = '\u06BE';
- t['\uFBAB'] = '\u06BE';
- t['\uFBAC'] = '\u06BE';
- t['\uFBAD'] = '\u06BE';
- t['\uFBAE'] = '\u06D2';
- t['\uFBAF'] = '\u06D2';
- t['\uFBB0'] = '\u06D3';
- t['\uFBB1'] = '\u06D3';
- t['\uFBD3'] = '\u06AD';
- t['\uFBD4'] = '\u06AD';
- t['\uFBD5'] = '\u06AD';
- t['\uFBD6'] = '\u06AD';
- t['\uFBD7'] = '\u06C7';
- t['\uFBD8'] = '\u06C7';
- t['\uFBD9'] = '\u06C6';
- t['\uFBDA'] = '\u06C6';
- t['\uFBDB'] = '\u06C8';
- t['\uFBDC'] = '\u06C8';
- t['\uFBDD'] = '\u0677';
- t['\uFBDE'] = '\u06CB';
- t['\uFBDF'] = '\u06CB';
- t['\uFBE0'] = '\u06C5';
- t['\uFBE1'] = '\u06C5';
- t['\uFBE2'] = '\u06C9';
- t['\uFBE3'] = '\u06C9';
- t['\uFBE4'] = '\u06D0';
- t['\uFBE5'] = '\u06D0';
- t['\uFBE6'] = '\u06D0';
- t['\uFBE7'] = '\u06D0';
- t['\uFBE8'] = '\u0649';
- t['\uFBE9'] = '\u0649';
- t['\uFBEA'] = '\u0626\u0627';
- t['\uFBEB'] = '\u0626\u0627';
- t['\uFBEC'] = '\u0626\u06D5';
- t['\uFBED'] = '\u0626\u06D5';
- t['\uFBEE'] = '\u0626\u0648';
- t['\uFBEF'] = '\u0626\u0648';
- t['\uFBF0'] = '\u0626\u06C7';
- t['\uFBF1'] = '\u0626\u06C7';
- t['\uFBF2'] = '\u0626\u06C6';
- t['\uFBF3'] = '\u0626\u06C6';
- t['\uFBF4'] = '\u0626\u06C8';
- t['\uFBF5'] = '\u0626\u06C8';
- t['\uFBF6'] = '\u0626\u06D0';
- t['\uFBF7'] = '\u0626\u06D0';
- t['\uFBF8'] = '\u0626\u06D0';
- t['\uFBF9'] = '\u0626\u0649';
- t['\uFBFA'] = '\u0626\u0649';
- t['\uFBFB'] = '\u0626\u0649';
- t['\uFBFC'] = '\u06CC';
- t['\uFBFD'] = '\u06CC';
- t['\uFBFE'] = '\u06CC';
- t['\uFBFF'] = '\u06CC';
- t['\uFC00'] = '\u0626\u062C';
- t['\uFC01'] = '\u0626\u062D';
- t['\uFC02'] = '\u0626\u0645';
- t['\uFC03'] = '\u0626\u0649';
- t['\uFC04'] = '\u0626\u064A';
- t['\uFC05'] = '\u0628\u062C';
- t['\uFC06'] = '\u0628\u062D';
- t['\uFC07'] = '\u0628\u062E';
- t['\uFC08'] = '\u0628\u0645';
- t['\uFC09'] = '\u0628\u0649';
- t['\uFC0A'] = '\u0628\u064A';
- t['\uFC0B'] = '\u062A\u062C';
- t['\uFC0C'] = '\u062A\u062D';
- t['\uFC0D'] = '\u062A\u062E';
- t['\uFC0E'] = '\u062A\u0645';
- t['\uFC0F'] = '\u062A\u0649';
- t['\uFC10'] = '\u062A\u064A';
- t['\uFC11'] = '\u062B\u062C';
- t['\uFC12'] = '\u062B\u0645';
- t['\uFC13'] = '\u062B\u0649';
- t['\uFC14'] = '\u062B\u064A';
- t['\uFC15'] = '\u062C\u062D';
- t['\uFC16'] = '\u062C\u0645';
- t['\uFC17'] = '\u062D\u062C';
- t['\uFC18'] = '\u062D\u0645';
- t['\uFC19'] = '\u062E\u062C';
- t['\uFC1A'] = '\u062E\u062D';
- t['\uFC1B'] = '\u062E\u0645';
- t['\uFC1C'] = '\u0633\u062C';
- t['\uFC1D'] = '\u0633\u062D';
- t['\uFC1E'] = '\u0633\u062E';
- t['\uFC1F'] = '\u0633\u0645';
- t['\uFC20'] = '\u0635\u062D';
- t['\uFC21'] = '\u0635\u0645';
- t['\uFC22'] = '\u0636\u062C';
- t['\uFC23'] = '\u0636\u062D';
- t['\uFC24'] = '\u0636\u062E';
- t['\uFC25'] = '\u0636\u0645';
- t['\uFC26'] = '\u0637\u062D';
- t['\uFC27'] = '\u0637\u0645';
- t['\uFC28'] = '\u0638\u0645';
- t['\uFC29'] = '\u0639\u062C';
- t['\uFC2A'] = '\u0639\u0645';
- t['\uFC2B'] = '\u063A\u062C';
- t['\uFC2C'] = '\u063A\u0645';
- t['\uFC2D'] = '\u0641\u062C';
- t['\uFC2E'] = '\u0641\u062D';
- t['\uFC2F'] = '\u0641\u062E';
- t['\uFC30'] = '\u0641\u0645';
- t['\uFC31'] = '\u0641\u0649';
- t['\uFC32'] = '\u0641\u064A';
- t['\uFC33'] = '\u0642\u062D';
- t['\uFC34'] = '\u0642\u0645';
- t['\uFC35'] = '\u0642\u0649';
- t['\uFC36'] = '\u0642\u064A';
- t['\uFC37'] = '\u0643\u0627';
- t['\uFC38'] = '\u0643\u062C';
- t['\uFC39'] = '\u0643\u062D';
- t['\uFC3A'] = '\u0643\u062E';
- t['\uFC3B'] = '\u0643\u0644';
- t['\uFC3C'] = '\u0643\u0645';
- t['\uFC3D'] = '\u0643\u0649';
- t['\uFC3E'] = '\u0643\u064A';
- t['\uFC3F'] = '\u0644\u062C';
- t['\uFC40'] = '\u0644\u062D';
- t['\uFC41'] = '\u0644\u062E';
- t['\uFC42'] = '\u0644\u0645';
- t['\uFC43'] = '\u0644\u0649';
- t['\uFC44'] = '\u0644\u064A';
- t['\uFC45'] = '\u0645\u062C';
- t['\uFC46'] = '\u0645\u062D';
- t['\uFC47'] = '\u0645\u062E';
- t['\uFC48'] = '\u0645\u0645';
- t['\uFC49'] = '\u0645\u0649';
- t['\uFC4A'] = '\u0645\u064A';
- t['\uFC4B'] = '\u0646\u062C';
- t['\uFC4C'] = '\u0646\u062D';
- t['\uFC4D'] = '\u0646\u062E';
- t['\uFC4E'] = '\u0646\u0645';
- t['\uFC4F'] = '\u0646\u0649';
- t['\uFC50'] = '\u0646\u064A';
- t['\uFC51'] = '\u0647\u062C';
- t['\uFC52'] = '\u0647\u0645';
- t['\uFC53'] = '\u0647\u0649';
- t['\uFC54'] = '\u0647\u064A';
- t['\uFC55'] = '\u064A\u062C';
- t['\uFC56'] = '\u064A\u062D';
- t['\uFC57'] = '\u064A\u062E';
- t['\uFC58'] = '\u064A\u0645';
- t['\uFC59'] = '\u064A\u0649';
- t['\uFC5A'] = '\u064A\u064A';
- t['\uFC5B'] = '\u0630\u0670';
- t['\uFC5C'] = '\u0631\u0670';
- t['\uFC5D'] = '\u0649\u0670';
- t['\uFC5E'] = '\u0020\u064C\u0651';
- t['\uFC5F'] = '\u0020\u064D\u0651';
- t['\uFC60'] = '\u0020\u064E\u0651';
- t['\uFC61'] = '\u0020\u064F\u0651';
- t['\uFC62'] = '\u0020\u0650\u0651';
- t['\uFC63'] = '\u0020\u0651\u0670';
- t['\uFC64'] = '\u0626\u0631';
- t['\uFC65'] = '\u0626\u0632';
- t['\uFC66'] = '\u0626\u0645';
- t['\uFC67'] = '\u0626\u0646';
- t['\uFC68'] = '\u0626\u0649';
- t['\uFC69'] = '\u0626\u064A';
- t['\uFC6A'] = '\u0628\u0631';
- t['\uFC6B'] = '\u0628\u0632';
- t['\uFC6C'] = '\u0628\u0645';
- t['\uFC6D'] = '\u0628\u0646';
- t['\uFC6E'] = '\u0628\u0649';
- t['\uFC6F'] = '\u0628\u064A';
- t['\uFC70'] = '\u062A\u0631';
- t['\uFC71'] = '\u062A\u0632';
- t['\uFC72'] = '\u062A\u0645';
- t['\uFC73'] = '\u062A\u0646';
- t['\uFC74'] = '\u062A\u0649';
- t['\uFC75'] = '\u062A\u064A';
- t['\uFC76'] = '\u062B\u0631';
- t['\uFC77'] = '\u062B\u0632';
- t['\uFC78'] = '\u062B\u0645';
- t['\uFC79'] = '\u062B\u0646';
- t['\uFC7A'] = '\u062B\u0649';
- t['\uFC7B'] = '\u062B\u064A';
- t['\uFC7C'] = '\u0641\u0649';
- t['\uFC7D'] = '\u0641\u064A';
- t['\uFC7E'] = '\u0642\u0649';
- t['\uFC7F'] = '\u0642\u064A';
- t['\uFC80'] = '\u0643\u0627';
- t['\uFC81'] = '\u0643\u0644';
- t['\uFC82'] = '\u0643\u0645';
- t['\uFC83'] = '\u0643\u0649';
- t['\uFC84'] = '\u0643\u064A';
- t['\uFC85'] = '\u0644\u0645';
- t['\uFC86'] = '\u0644\u0649';
- t['\uFC87'] = '\u0644\u064A';
- t['\uFC88'] = '\u0645\u0627';
- t['\uFC89'] = '\u0645\u0645';
- t['\uFC8A'] = '\u0646\u0631';
- t['\uFC8B'] = '\u0646\u0632';
- t['\uFC8C'] = '\u0646\u0645';
- t['\uFC8D'] = '\u0646\u0646';
- t['\uFC8E'] = '\u0646\u0649';
- t['\uFC8F'] = '\u0646\u064A';
- t['\uFC90'] = '\u0649\u0670';
- t['\uFC91'] = '\u064A\u0631';
- t['\uFC92'] = '\u064A\u0632';
- t['\uFC93'] = '\u064A\u0645';
- t['\uFC94'] = '\u064A\u0646';
- t['\uFC95'] = '\u064A\u0649';
- t['\uFC96'] = '\u064A\u064A';
- t['\uFC97'] = '\u0626\u062C';
- t['\uFC98'] = '\u0626\u062D';
- t['\uFC99'] = '\u0626\u062E';
- t['\uFC9A'] = '\u0626\u0645';
- t['\uFC9B'] = '\u0626\u0647';
- t['\uFC9C'] = '\u0628\u062C';
- t['\uFC9D'] = '\u0628\u062D';
- t['\uFC9E'] = '\u0628\u062E';
- t['\uFC9F'] = '\u0628\u0645';
- t['\uFCA0'] = '\u0628\u0647';
- t['\uFCA1'] = '\u062A\u062C';
- t['\uFCA2'] = '\u062A\u062D';
- t['\uFCA3'] = '\u062A\u062E';
- t['\uFCA4'] = '\u062A\u0645';
- t['\uFCA5'] = '\u062A\u0647';
- t['\uFCA6'] = '\u062B\u0645';
- t['\uFCA7'] = '\u062C\u062D';
- t['\uFCA8'] = '\u062C\u0645';
- t['\uFCA9'] = '\u062D\u062C';
- t['\uFCAA'] = '\u062D\u0645';
- t['\uFCAB'] = '\u062E\u062C';
- t['\uFCAC'] = '\u062E\u0645';
- t['\uFCAD'] = '\u0633\u062C';
- t['\uFCAE'] = '\u0633\u062D';
- t['\uFCAF'] = '\u0633\u062E';
- t['\uFCB0'] = '\u0633\u0645';
- t['\uFCB1'] = '\u0635\u062D';
- t['\uFCB2'] = '\u0635\u062E';
- t['\uFCB3'] = '\u0635\u0645';
- t['\uFCB4'] = '\u0636\u062C';
- t['\uFCB5'] = '\u0636\u062D';
- t['\uFCB6'] = '\u0636\u062E';
- t['\uFCB7'] = '\u0636\u0645';
- t['\uFCB8'] = '\u0637\u062D';
- t['\uFCB9'] = '\u0638\u0645';
- t['\uFCBA'] = '\u0639\u062C';
- t['\uFCBB'] = '\u0639\u0645';
- t['\uFCBC'] = '\u063A\u062C';
- t['\uFCBD'] = '\u063A\u0645';
- t['\uFCBE'] = '\u0641\u062C';
- t['\uFCBF'] = '\u0641\u062D';
- t['\uFCC0'] = '\u0641\u062E';
- t['\uFCC1'] = '\u0641\u0645';
- t['\uFCC2'] = '\u0642\u062D';
- t['\uFCC3'] = '\u0642\u0645';
- t['\uFCC4'] = '\u0643\u062C';
- t['\uFCC5'] = '\u0643\u062D';
- t['\uFCC6'] = '\u0643\u062E';
- t['\uFCC7'] = '\u0643\u0644';
- t['\uFCC8'] = '\u0643\u0645';
- t['\uFCC9'] = '\u0644\u062C';
- t['\uFCCA'] = '\u0644\u062D';
- t['\uFCCB'] = '\u0644\u062E';
- t['\uFCCC'] = '\u0644\u0645';
- t['\uFCCD'] = '\u0644\u0647';
- t['\uFCCE'] = '\u0645\u062C';
- t['\uFCCF'] = '\u0645\u062D';
- t['\uFCD0'] = '\u0645\u062E';
- t['\uFCD1'] = '\u0645\u0645';
- t['\uFCD2'] = '\u0646\u062C';
- t['\uFCD3'] = '\u0646\u062D';
- t['\uFCD4'] = '\u0646\u062E';
- t['\uFCD5'] = '\u0646\u0645';
- t['\uFCD6'] = '\u0646\u0647';
- t['\uFCD7'] = '\u0647\u062C';
- t['\uFCD8'] = '\u0647\u0645';
- t['\uFCD9'] = '\u0647\u0670';
- t['\uFCDA'] = '\u064A\u062C';
- t['\uFCDB'] = '\u064A\u062D';
- t['\uFCDC'] = '\u064A\u062E';
- t['\uFCDD'] = '\u064A\u0645';
- t['\uFCDE'] = '\u064A\u0647';
- t['\uFCDF'] = '\u0626\u0645';
- t['\uFCE0'] = '\u0626\u0647';
- t['\uFCE1'] = '\u0628\u0645';
- t['\uFCE2'] = '\u0628\u0647';
- t['\uFCE3'] = '\u062A\u0645';
- t['\uFCE4'] = '\u062A\u0647';
- t['\uFCE5'] = '\u062B\u0645';
- t['\uFCE6'] = '\u062B\u0647';
- t['\uFCE7'] = '\u0633\u0645';
- t['\uFCE8'] = '\u0633\u0647';
- t['\uFCE9'] = '\u0634\u0645';
- t['\uFCEA'] = '\u0634\u0647';
- t['\uFCEB'] = '\u0643\u0644';
- t['\uFCEC'] = '\u0643\u0645';
- t['\uFCED'] = '\u0644\u0645';
- t['\uFCEE'] = '\u0646\u0645';
- t['\uFCEF'] = '\u0646\u0647';
- t['\uFCF0'] = '\u064A\u0645';
- t['\uFCF1'] = '\u064A\u0647';
- t['\uFCF2'] = '\u0640\u064E\u0651';
- t['\uFCF3'] = '\u0640\u064F\u0651';
- t['\uFCF4'] = '\u0640\u0650\u0651';
- t['\uFCF5'] = '\u0637\u0649';
- t['\uFCF6'] = '\u0637\u064A';
- t['\uFCF7'] = '\u0639\u0649';
- t['\uFCF8'] = '\u0639\u064A';
- t['\uFCF9'] = '\u063A\u0649';
- t['\uFCFA'] = '\u063A\u064A';
- t['\uFCFB'] = '\u0633\u0649';
- t['\uFCFC'] = '\u0633\u064A';
- t['\uFCFD'] = '\u0634\u0649';
- t['\uFCFE'] = '\u0634\u064A';
- t['\uFCFF'] = '\u062D\u0649';
- t['\uFD00'] = '\u062D\u064A';
- t['\uFD01'] = '\u062C\u0649';
- t['\uFD02'] = '\u062C\u064A';
- t['\uFD03'] = '\u062E\u0649';
- t['\uFD04'] = '\u062E\u064A';
- t['\uFD05'] = '\u0635\u0649';
- t['\uFD06'] = '\u0635\u064A';
- t['\uFD07'] = '\u0636\u0649';
- t['\uFD08'] = '\u0636\u064A';
- t['\uFD09'] = '\u0634\u062C';
- t['\uFD0A'] = '\u0634\u062D';
- t['\uFD0B'] = '\u0634\u062E';
- t['\uFD0C'] = '\u0634\u0645';
- t['\uFD0D'] = '\u0634\u0631';
- t['\uFD0E'] = '\u0633\u0631';
- t['\uFD0F'] = '\u0635\u0631';
- t['\uFD10'] = '\u0636\u0631';
- t['\uFD11'] = '\u0637\u0649';
- t['\uFD12'] = '\u0637\u064A';
- t['\uFD13'] = '\u0639\u0649';
- t['\uFD14'] = '\u0639\u064A';
- t['\uFD15'] = '\u063A\u0649';
- t['\uFD16'] = '\u063A\u064A';
- t['\uFD17'] = '\u0633\u0649';
- t['\uFD18'] = '\u0633\u064A';
- t['\uFD19'] = '\u0634\u0649';
- t['\uFD1A'] = '\u0634\u064A';
- t['\uFD1B'] = '\u062D\u0649';
- t['\uFD1C'] = '\u062D\u064A';
- t['\uFD1D'] = '\u062C\u0649';
- t['\uFD1E'] = '\u062C\u064A';
- t['\uFD1F'] = '\u062E\u0649';
- t['\uFD20'] = '\u062E\u064A';
- t['\uFD21'] = '\u0635\u0649';
- t['\uFD22'] = '\u0635\u064A';
- t['\uFD23'] = '\u0636\u0649';
- t['\uFD24'] = '\u0636\u064A';
- t['\uFD25'] = '\u0634\u062C';
- t['\uFD26'] = '\u0634\u062D';
- t['\uFD27'] = '\u0634\u062E';
- t['\uFD28'] = '\u0634\u0645';
- t['\uFD29'] = '\u0634\u0631';
- t['\uFD2A'] = '\u0633\u0631';
- t['\uFD2B'] = '\u0635\u0631';
- t['\uFD2C'] = '\u0636\u0631';
- t['\uFD2D'] = '\u0634\u062C';
- t['\uFD2E'] = '\u0634\u062D';
- t['\uFD2F'] = '\u0634\u062E';
- t['\uFD30'] = '\u0634\u0645';
- t['\uFD31'] = '\u0633\u0647';
- t['\uFD32'] = '\u0634\u0647';
- t['\uFD33'] = '\u0637\u0645';
- t['\uFD34'] = '\u0633\u062C';
- t['\uFD35'] = '\u0633\u062D';
- t['\uFD36'] = '\u0633\u062E';
- t['\uFD37'] = '\u0634\u062C';
- t['\uFD38'] = '\u0634\u062D';
- t['\uFD39'] = '\u0634\u062E';
- t['\uFD3A'] = '\u0637\u0645';
- t['\uFD3B'] = '\u0638\u0645';
- t['\uFD3C'] = '\u0627\u064B';
- t['\uFD3D'] = '\u0627\u064B';
- t['\uFD50'] = '\u062A\u062C\u0645';
- t['\uFD51'] = '\u062A\u062D\u062C';
- t['\uFD52'] = '\u062A\u062D\u062C';
- t['\uFD53'] = '\u062A\u062D\u0645';
- t['\uFD54'] = '\u062A\u062E\u0645';
- t['\uFD55'] = '\u062A\u0645\u062C';
- t['\uFD56'] = '\u062A\u0645\u062D';
- t['\uFD57'] = '\u062A\u0645\u062E';
- t['\uFD58'] = '\u062C\u0645\u062D';
- t['\uFD59'] = '\u062C\u0645\u062D';
- t['\uFD5A'] = '\u062D\u0645\u064A';
- t['\uFD5B'] = '\u062D\u0645\u0649';
- t['\uFD5C'] = '\u0633\u062D\u062C';
- t['\uFD5D'] = '\u0633\u062C\u062D';
- t['\uFD5E'] = '\u0633\u062C\u0649';
- t['\uFD5F'] = '\u0633\u0645\u062D';
- t['\uFD60'] = '\u0633\u0645\u062D';
- t['\uFD61'] = '\u0633\u0645\u062C';
- t['\uFD62'] = '\u0633\u0645\u0645';
- t['\uFD63'] = '\u0633\u0645\u0645';
- t['\uFD64'] = '\u0635\u062D\u062D';
- t['\uFD65'] = '\u0635\u062D\u062D';
- t['\uFD66'] = '\u0635\u0645\u0645';
- t['\uFD67'] = '\u0634\u062D\u0645';
- t['\uFD68'] = '\u0634\u062D\u0645';
- t['\uFD69'] = '\u0634\u062C\u064A';
- t['\uFD6A'] = '\u0634\u0645\u062E';
- t['\uFD6B'] = '\u0634\u0645\u062E';
- t['\uFD6C'] = '\u0634\u0645\u0645';
- t['\uFD6D'] = '\u0634\u0645\u0645';
- t['\uFD6E'] = '\u0636\u062D\u0649';
- t['\uFD6F'] = '\u0636\u062E\u0645';
- t['\uFD70'] = '\u0636\u062E\u0645';
- t['\uFD71'] = '\u0637\u0645\u062D';
- t['\uFD72'] = '\u0637\u0645\u062D';
- t['\uFD73'] = '\u0637\u0645\u0645';
- t['\uFD74'] = '\u0637\u0645\u064A';
- t['\uFD75'] = '\u0639\u062C\u0645';
- t['\uFD76'] = '\u0639\u0645\u0645';
- t['\uFD77'] = '\u0639\u0645\u0645';
- t['\uFD78'] = '\u0639\u0645\u0649';
- t['\uFD79'] = '\u063A\u0645\u0645';
- t['\uFD7A'] = '\u063A\u0645\u064A';
- t['\uFD7B'] = '\u063A\u0645\u0649';
- t['\uFD7C'] = '\u0641\u062E\u0645';
- t['\uFD7D'] = '\u0641\u062E\u0645';
- t['\uFD7E'] = '\u0642\u0645\u062D';
- t['\uFD7F'] = '\u0642\u0645\u0645';
- t['\uFD80'] = '\u0644\u062D\u0645';
- t['\uFD81'] = '\u0644\u062D\u064A';
- t['\uFD82'] = '\u0644\u062D\u0649';
- t['\uFD83'] = '\u0644\u062C\u062C';
- t['\uFD84'] = '\u0644\u062C\u062C';
- t['\uFD85'] = '\u0644\u062E\u0645';
- t['\uFD86'] = '\u0644\u062E\u0645';
- t['\uFD87'] = '\u0644\u0645\u062D';
- t['\uFD88'] = '\u0644\u0645\u062D';
- t['\uFD89'] = '\u0645\u062D\u062C';
- t['\uFD8A'] = '\u0645\u062D\u0645';
- t['\uFD8B'] = '\u0645\u062D\u064A';
- t['\uFD8C'] = '\u0645\u062C\u062D';
- t['\uFD8D'] = '\u0645\u062C\u0645';
- t['\uFD8E'] = '\u0645\u062E\u062C';
- t['\uFD8F'] = '\u0645\u062E\u0645';
- t['\uFD92'] = '\u0645\u062C\u062E';
- t['\uFD93'] = '\u0647\u0645\u062C';
- t['\uFD94'] = '\u0647\u0645\u0645';
- t['\uFD95'] = '\u0646\u062D\u0645';
- t['\uFD96'] = '\u0646\u062D\u0649';
- t['\uFD97'] = '\u0646\u062C\u0645';
- t['\uFD98'] = '\u0646\u062C\u0645';
- t['\uFD99'] = '\u0646\u062C\u0649';
- t['\uFD9A'] = '\u0646\u0645\u064A';
- t['\uFD9B'] = '\u0646\u0645\u0649';
- t['\uFD9C'] = '\u064A\u0645\u0645';
- t['\uFD9D'] = '\u064A\u0645\u0645';
- t['\uFD9E'] = '\u0628\u062E\u064A';
- t['\uFD9F'] = '\u062A\u062C\u064A';
- t['\uFDA0'] = '\u062A\u062C\u0649';
- t['\uFDA1'] = '\u062A\u062E\u064A';
- t['\uFDA2'] = '\u062A\u062E\u0649';
- t['\uFDA3'] = '\u062A\u0645\u064A';
- t['\uFDA4'] = '\u062A\u0645\u0649';
- t['\uFDA5'] = '\u062C\u0645\u064A';
- t['\uFDA6'] = '\u062C\u062D\u0649';
- t['\uFDA7'] = '\u062C\u0645\u0649';
- t['\uFDA8'] = '\u0633\u062E\u0649';
- t['\uFDA9'] = '\u0635\u062D\u064A';
- t['\uFDAA'] = '\u0634\u062D\u064A';
- t['\uFDAB'] = '\u0636\u062D\u064A';
- t['\uFDAC'] = '\u0644\u062C\u064A';
- t['\uFDAD'] = '\u0644\u0645\u064A';
- t['\uFDAE'] = '\u064A\u062D\u064A';
- t['\uFDAF'] = '\u064A\u062C\u064A';
- t['\uFDB0'] = '\u064A\u0645\u064A';
- t['\uFDB1'] = '\u0645\u0645\u064A';
- t['\uFDB2'] = '\u0642\u0645\u064A';
- t['\uFDB3'] = '\u0646\u062D\u064A';
- t['\uFDB4'] = '\u0642\u0645\u062D';
- t['\uFDB5'] = '\u0644\u062D\u0645';
- t['\uFDB6'] = '\u0639\u0645\u064A';
- t['\uFDB7'] = '\u0643\u0645\u064A';
- t['\uFDB8'] = '\u0646\u062C\u062D';
- t['\uFDB9'] = '\u0645\u062E\u064A';
- t['\uFDBA'] = '\u0644\u062C\u0645';
- t['\uFDBB'] = '\u0643\u0645\u0645';
- t['\uFDBC'] = '\u0644\u062C\u0645';
- t['\uFDBD'] = '\u0646\u062C\u062D';
- t['\uFDBE'] = '\u062C\u062D\u064A';
- t['\uFDBF'] = '\u062D\u062C\u064A';
- t['\uFDC0'] = '\u0645\u062C\u064A';
- t['\uFDC1'] = '\u0641\u0645\u064A';
- t['\uFDC2'] = '\u0628\u062D\u064A';
- t['\uFDC3'] = '\u0643\u0645\u0645';
- t['\uFDC4'] = '\u0639\u062C\u0645';
- t['\uFDC5'] = '\u0635\u0645\u0645';
- t['\uFDC6'] = '\u0633\u062E\u064A';
- t['\uFDC7'] = '\u0646\u062C\u064A';
- t['\uFE49'] = '\u203E';
- t['\uFE4A'] = '\u203E';
- t['\uFE4B'] = '\u203E';
- t['\uFE4C'] = '\u203E';
- t['\uFE4D'] = '\u005F';
- t['\uFE4E'] = '\u005F';
- t['\uFE4F'] = '\u005F';
- t['\uFE80'] = '\u0621';
- t['\uFE81'] = '\u0622';
- t['\uFE82'] = '\u0622';
- t['\uFE83'] = '\u0623';
- t['\uFE84'] = '\u0623';
- t['\uFE85'] = '\u0624';
- t['\uFE86'] = '\u0624';
- t['\uFE87'] = '\u0625';
- t['\uFE88'] = '\u0625';
- t['\uFE89'] = '\u0626';
- t['\uFE8A'] = '\u0626';
- t['\uFE8B'] = '\u0626';
- t['\uFE8C'] = '\u0626';
- t['\uFE8D'] = '\u0627';
- t['\uFE8E'] = '\u0627';
- t['\uFE8F'] = '\u0628';
- t['\uFE90'] = '\u0628';
- t['\uFE91'] = '\u0628';
- t['\uFE92'] = '\u0628';
- t['\uFE93'] = '\u0629';
- t['\uFE94'] = '\u0629';
- t['\uFE95'] = '\u062A';
- t['\uFE96'] = '\u062A';
- t['\uFE97'] = '\u062A';
- t['\uFE98'] = '\u062A';
- t['\uFE99'] = '\u062B';
- t['\uFE9A'] = '\u062B';
- t['\uFE9B'] = '\u062B';
- t['\uFE9C'] = '\u062B';
- t['\uFE9D'] = '\u062C';
- t['\uFE9E'] = '\u062C';
- t['\uFE9F'] = '\u062C';
- t['\uFEA0'] = '\u062C';
- t['\uFEA1'] = '\u062D';
- t['\uFEA2'] = '\u062D';
- t['\uFEA3'] = '\u062D';
- t['\uFEA4'] = '\u062D';
- t['\uFEA5'] = '\u062E';
- t['\uFEA6'] = '\u062E';
- t['\uFEA7'] = '\u062E';
- t['\uFEA8'] = '\u062E';
- t['\uFEA9'] = '\u062F';
- t['\uFEAA'] = '\u062F';
- t['\uFEAB'] = '\u0630';
- t['\uFEAC'] = '\u0630';
- t['\uFEAD'] = '\u0631';
- t['\uFEAE'] = '\u0631';
- t['\uFEAF'] = '\u0632';
- t['\uFEB0'] = '\u0632';
- t['\uFEB1'] = '\u0633';
- t['\uFEB2'] = '\u0633';
- t['\uFEB3'] = '\u0633';
- t['\uFEB4'] = '\u0633';
- t['\uFEB5'] = '\u0634';
- t['\uFEB6'] = '\u0634';
- t['\uFEB7'] = '\u0634';
- t['\uFEB8'] = '\u0634';
- t['\uFEB9'] = '\u0635';
- t['\uFEBA'] = '\u0635';
- t['\uFEBB'] = '\u0635';
- t['\uFEBC'] = '\u0635';
- t['\uFEBD'] = '\u0636';
- t['\uFEBE'] = '\u0636';
- t['\uFEBF'] = '\u0636';
- t['\uFEC0'] = '\u0636';
- t['\uFEC1'] = '\u0637';
- t['\uFEC2'] = '\u0637';
- t['\uFEC3'] = '\u0637';
- t['\uFEC4'] = '\u0637';
- t['\uFEC5'] = '\u0638';
- t['\uFEC6'] = '\u0638';
- t['\uFEC7'] = '\u0638';
- t['\uFEC8'] = '\u0638';
- t['\uFEC9'] = '\u0639';
- t['\uFECA'] = '\u0639';
- t['\uFECB'] = '\u0639';
- t['\uFECC'] = '\u0639';
- t['\uFECD'] = '\u063A';
- t['\uFECE'] = '\u063A';
- t['\uFECF'] = '\u063A';
- t['\uFED0'] = '\u063A';
- t['\uFED1'] = '\u0641';
- t['\uFED2'] = '\u0641';
- t['\uFED3'] = '\u0641';
- t['\uFED4'] = '\u0641';
- t['\uFED5'] = '\u0642';
- t['\uFED6'] = '\u0642';
- t['\uFED7'] = '\u0642';
- t['\uFED8'] = '\u0642';
- t['\uFED9'] = '\u0643';
- t['\uFEDA'] = '\u0643';
- t['\uFEDB'] = '\u0643';
- t['\uFEDC'] = '\u0643';
- t['\uFEDD'] = '\u0644';
- t['\uFEDE'] = '\u0644';
- t['\uFEDF'] = '\u0644';
- t['\uFEE0'] = '\u0644';
- t['\uFEE1'] = '\u0645';
- t['\uFEE2'] = '\u0645';
- t['\uFEE3'] = '\u0645';
- t['\uFEE4'] = '\u0645';
- t['\uFEE5'] = '\u0646';
- t['\uFEE6'] = '\u0646';
- t['\uFEE7'] = '\u0646';
- t['\uFEE8'] = '\u0646';
- t['\uFEE9'] = '\u0647';
- t['\uFEEA'] = '\u0647';
- t['\uFEEB'] = '\u0647';
- t['\uFEEC'] = '\u0647';
- t['\uFEED'] = '\u0648';
- t['\uFEEE'] = '\u0648';
- t['\uFEEF'] = '\u0649';
- t['\uFEF0'] = '\u0649';
- t['\uFEF1'] = '\u064A';
- t['\uFEF2'] = '\u064A';
- t['\uFEF3'] = '\u064A';
- t['\uFEF4'] = '\u064A';
- t['\uFEF5'] = '\u0644\u0622';
- t['\uFEF6'] = '\u0644\u0622';
- t['\uFEF7'] = '\u0644\u0623';
- t['\uFEF8'] = '\u0644\u0623';
- t['\uFEF9'] = '\u0644\u0625';
- t['\uFEFA'] = '\u0644\u0625';
- t['\uFEFB'] = '\u0644\u0627';
- t['\uFEFC'] = '\u0644\u0627';
+  t['\u00A8'] = '\u0020\u0308';
+  t['\u00AF'] = '\u0020\u0304';
+  t['\u00B4'] = '\u0020\u0301';
+  t['\u00B5'] = '\u03BC';
+  t['\u00B8'] = '\u0020\u0327';
+  t['\u0132'] = '\u0049\u004A';
+  t['\u0133'] = '\u0069\u006A';
+  t['\u013F'] = '\u004C\u00B7';
+  t['\u0140'] = '\u006C\u00B7';
+  t['\u0149'] = '\u02BC\u006E';
+  t['\u017F'] = '\u0073';
+  t['\u01C4'] = '\u0044\u017D';
+  t['\u01C5'] = '\u0044\u017E';
+  t['\u01C6'] = '\u0064\u017E';
+  t['\u01C7'] = '\u004C\u004A';
+  t['\u01C8'] = '\u004C\u006A';
+  t['\u01C9'] = '\u006C\u006A';
+  t['\u01CA'] = '\u004E\u004A';
+  t['\u01CB'] = '\u004E\u006A';
+  t['\u01CC'] = '\u006E\u006A';
+  t['\u01F1'] = '\u0044\u005A';
+  t['\u01F2'] = '\u0044\u007A';
+  t['\u01F3'] = '\u0064\u007A';
+  t['\u02D8'] = '\u0020\u0306';
+  t['\u02D9'] = '\u0020\u0307';
+  t['\u02DA'] = '\u0020\u030A';
+  t['\u02DB'] = '\u0020\u0328';
+  t['\u02DC'] = '\u0020\u0303';
+  t['\u02DD'] = '\u0020\u030B';
+  t['\u037A'] = '\u0020\u0345';
+  t['\u0384'] = '\u0020\u0301';
+  t['\u03D0'] = '\u03B2';
+  t['\u03D1'] = '\u03B8';
+  t['\u03D2'] = '\u03A5';
+  t['\u03D5'] = '\u03C6';
+  t['\u03D6'] = '\u03C0';
+  t['\u03F0'] = '\u03BA';
+  t['\u03F1'] = '\u03C1';
+  t['\u03F2'] = '\u03C2';
+  t['\u03F4'] = '\u0398';
+  t['\u03F5'] = '\u03B5';
+  t['\u03F9'] = '\u03A3';
+  t['\u0587'] = '\u0565\u0582';
+  t['\u0675'] = '\u0627\u0674';
+  t['\u0676'] = '\u0648\u0674';
+  t['\u0677'] = '\u06C7\u0674';
+  t['\u0678'] = '\u064A\u0674';
+  t['\u0E33'] = '\u0E4D\u0E32';
+  t['\u0EB3'] = '\u0ECD\u0EB2';
+  t['\u0EDC'] = '\u0EAB\u0E99';
+  t['\u0EDD'] = '\u0EAB\u0EA1';
+  t['\u0F77'] = '\u0FB2\u0F81';
+  t['\u0F79'] = '\u0FB3\u0F81';
+  t['\u1E9A'] = '\u0061\u02BE';
+  t['\u1FBD'] = '\u0020\u0313';
+  t['\u1FBF'] = '\u0020\u0313';
+  t['\u1FC0'] = '\u0020\u0342';
+  t['\u1FFE'] = '\u0020\u0314';
+  t['\u2002'] = '\u0020';
+  t['\u2003'] = '\u0020';
+  t['\u2004'] = '\u0020';
+  t['\u2005'] = '\u0020';
+  t['\u2006'] = '\u0020';
+  t['\u2008'] = '\u0020';
+  t['\u2009'] = '\u0020';
+  t['\u200A'] = '\u0020';
+  t['\u2017'] = '\u0020\u0333';
+  t['\u2024'] = '\u002E';
+  t['\u2025'] = '\u002E\u002E';
+  t['\u2026'] = '\u002E\u002E\u002E';
+  t['\u2033'] = '\u2032\u2032';
+  t['\u2034'] = '\u2032\u2032\u2032';
+  t['\u2036'] = '\u2035\u2035';
+  t['\u2037'] = '\u2035\u2035\u2035';
+  t['\u203C'] = '\u0021\u0021';
+  t['\u203E'] = '\u0020\u0305';
+  t['\u2047'] = '\u003F\u003F';
+  t['\u2048'] = '\u003F\u0021';
+  t['\u2049'] = '\u0021\u003F';
+  t['\u2057'] = '\u2032\u2032\u2032\u2032';
+  t['\u205F'] = '\u0020';
+  t['\u20A8'] = '\u0052\u0073';
+  t['\u2100'] = '\u0061\u002F\u0063';
+  t['\u2101'] = '\u0061\u002F\u0073';
+  t['\u2103'] = '\u00B0\u0043';
+  t['\u2105'] = '\u0063\u002F\u006F';
+  t['\u2106'] = '\u0063\u002F\u0075';
+  t['\u2107'] = '\u0190';
+  t['\u2109'] = '\u00B0\u0046';
+  t['\u2116'] = '\u004E\u006F';
+  t['\u2121'] = '\u0054\u0045\u004C';
+  t['\u2135'] = '\u05D0';
+  t['\u2136'] = '\u05D1';
+  t['\u2137'] = '\u05D2';
+  t['\u2138'] = '\u05D3';
+  t['\u213B'] = '\u0046\u0041\u0058';
+  t['\u2160'] = '\u0049';
+  t['\u2161'] = '\u0049\u0049';
+  t['\u2162'] = '\u0049\u0049\u0049';
+  t['\u2163'] = '\u0049\u0056';
+  t['\u2164'] = '\u0056';
+  t['\u2165'] = '\u0056\u0049';
+  t['\u2166'] = '\u0056\u0049\u0049';
+  t['\u2167'] = '\u0056\u0049\u0049\u0049';
+  t['\u2168'] = '\u0049\u0058';
+  t['\u2169'] = '\u0058';
+  t['\u216A'] = '\u0058\u0049';
+  t['\u216B'] = '\u0058\u0049\u0049';
+  t['\u216C'] = '\u004C';
+  t['\u216D'] = '\u0043';
+  t['\u216E'] = '\u0044';
+  t['\u216F'] = '\u004D';
+  t['\u2170'] = '\u0069';
+  t['\u2171'] = '\u0069\u0069';
+  t['\u2172'] = '\u0069\u0069\u0069';
+  t['\u2173'] = '\u0069\u0076';
+  t['\u2174'] = '\u0076';
+  t['\u2175'] = '\u0076\u0069';
+  t['\u2176'] = '\u0076\u0069\u0069';
+  t['\u2177'] = '\u0076\u0069\u0069\u0069';
+  t['\u2178'] = '\u0069\u0078';
+  t['\u2179'] = '\u0078';
+  t['\u217A'] = '\u0078\u0069';
+  t['\u217B'] = '\u0078\u0069\u0069';
+  t['\u217C'] = '\u006C';
+  t['\u217D'] = '\u0063';
+  t['\u217E'] = '\u0064';
+  t['\u217F'] = '\u006D';
+  t['\u222C'] = '\u222B\u222B';
+  t['\u222D'] = '\u222B\u222B\u222B';
+  t['\u222F'] = '\u222E\u222E';
+  t['\u2230'] = '\u222E\u222E\u222E';
+  t['\u2474'] = '\u0028\u0031\u0029';
+  t['\u2475'] = '\u0028\u0032\u0029';
+  t['\u2476'] = '\u0028\u0033\u0029';
+  t['\u2477'] = '\u0028\u0034\u0029';
+  t['\u2478'] = '\u0028\u0035\u0029';
+  t['\u2479'] = '\u0028\u0036\u0029';
+  t['\u247A'] = '\u0028\u0037\u0029';
+  t['\u247B'] = '\u0028\u0038\u0029';
+  t['\u247C'] = '\u0028\u0039\u0029';
+  t['\u247D'] = '\u0028\u0031\u0030\u0029';
+  t['\u247E'] = '\u0028\u0031\u0031\u0029';
+  t['\u247F'] = '\u0028\u0031\u0032\u0029';
+  t['\u2480'] = '\u0028\u0031\u0033\u0029';
+  t['\u2481'] = '\u0028\u0031\u0034\u0029';
+  t['\u2482'] = '\u0028\u0031\u0035\u0029';
+  t['\u2483'] = '\u0028\u0031\u0036\u0029';
+  t['\u2484'] = '\u0028\u0031\u0037\u0029';
+  t['\u2485'] = '\u0028\u0031\u0038\u0029';
+  t['\u2486'] = '\u0028\u0031\u0039\u0029';
+  t['\u2487'] = '\u0028\u0032\u0030\u0029';
+  t['\u2488'] = '\u0031\u002E';
+  t['\u2489'] = '\u0032\u002E';
+  t['\u248A'] = '\u0033\u002E';
+  t['\u248B'] = '\u0034\u002E';
+  t['\u248C'] = '\u0035\u002E';
+  t['\u248D'] = '\u0036\u002E';
+  t['\u248E'] = '\u0037\u002E';
+  t['\u248F'] = '\u0038\u002E';
+  t['\u2490'] = '\u0039\u002E';
+  t['\u2491'] = '\u0031\u0030\u002E';
+  t['\u2492'] = '\u0031\u0031\u002E';
+  t['\u2493'] = '\u0031\u0032\u002E';
+  t['\u2494'] = '\u0031\u0033\u002E';
+  t['\u2495'] = '\u0031\u0034\u002E';
+  t['\u2496'] = '\u0031\u0035\u002E';
+  t['\u2497'] = '\u0031\u0036\u002E';
+  t['\u2498'] = '\u0031\u0037\u002E';
+  t['\u2499'] = '\u0031\u0038\u002E';
+  t['\u249A'] = '\u0031\u0039\u002E';
+  t['\u249B'] = '\u0032\u0030\u002E';
+  t['\u249C'] = '\u0028\u0061\u0029';
+  t['\u249D'] = '\u0028\u0062\u0029';
+  t['\u249E'] = '\u0028\u0063\u0029';
+  t['\u249F'] = '\u0028\u0064\u0029';
+  t['\u24A0'] = '\u0028\u0065\u0029';
+  t['\u24A1'] = '\u0028\u0066\u0029';
+  t['\u24A2'] = '\u0028\u0067\u0029';
+  t['\u24A3'] = '\u0028\u0068\u0029';
+  t['\u24A4'] = '\u0028\u0069\u0029';
+  t['\u24A5'] = '\u0028\u006A\u0029';
+  t['\u24A6'] = '\u0028\u006B\u0029';
+  t['\u24A7'] = '\u0028\u006C\u0029';
+  t['\u24A8'] = '\u0028\u006D\u0029';
+  t['\u24A9'] = '\u0028\u006E\u0029';
+  t['\u24AA'] = '\u0028\u006F\u0029';
+  t['\u24AB'] = '\u0028\u0070\u0029';
+  t['\u24AC'] = '\u0028\u0071\u0029';
+  t['\u24AD'] = '\u0028\u0072\u0029';
+  t['\u24AE'] = '\u0028\u0073\u0029';
+  t['\u24AF'] = '\u0028\u0074\u0029';
+  t['\u24B0'] = '\u0028\u0075\u0029';
+  t['\u24B1'] = '\u0028\u0076\u0029';
+  t['\u24B2'] = '\u0028\u0077\u0029';
+  t['\u24B3'] = '\u0028\u0078\u0029';
+  t['\u24B4'] = '\u0028\u0079\u0029';
+  t['\u24B5'] = '\u0028\u007A\u0029';
+  t['\u2A0C'] = '\u222B\u222B\u222B\u222B';
+  t['\u2A74'] = '\u003A\u003A\u003D';
+  t['\u2A75'] = '\u003D\u003D';
+  t['\u2A76'] = '\u003D\u003D\u003D';
+  t['\u2E9F'] = '\u6BCD';
+  t['\u2EF3'] = '\u9F9F';
+  t['\u2F00'] = '\u4E00';
+  t['\u2F01'] = '\u4E28';
+  t['\u2F02'] = '\u4E36';
+  t['\u2F03'] = '\u4E3F';
+  t['\u2F04'] = '\u4E59';
+  t['\u2F05'] = '\u4E85';
+  t['\u2F06'] = '\u4E8C';
+  t['\u2F07'] = '\u4EA0';
+  t['\u2F08'] = '\u4EBA';
+  t['\u2F09'] = '\u513F';
+  t['\u2F0A'] = '\u5165';
+  t['\u2F0B'] = '\u516B';
+  t['\u2F0C'] = '\u5182';
+  t['\u2F0D'] = '\u5196';
+  t['\u2F0E'] = '\u51AB';
+  t['\u2F0F'] = '\u51E0';
+  t['\u2F10'] = '\u51F5';
+  t['\u2F11'] = '\u5200';
+  t['\u2F12'] = '\u529B';
+  t['\u2F13'] = '\u52F9';
+  t['\u2F14'] = '\u5315';
+  t['\u2F15'] = '\u531A';
+  t['\u2F16'] = '\u5338';
+  t['\u2F17'] = '\u5341';
+  t['\u2F18'] = '\u535C';
+  t['\u2F19'] = '\u5369';
+  t['\u2F1A'] = '\u5382';
+  t['\u2F1B'] = '\u53B6';
+  t['\u2F1C'] = '\u53C8';
+  t['\u2F1D'] = '\u53E3';
+  t['\u2F1E'] = '\u56D7';
+  t['\u2F1F'] = '\u571F';
+  t['\u2F20'] = '\u58EB';
+  t['\u2F21'] = '\u5902';
+  t['\u2F22'] = '\u590A';
+  t['\u2F23'] = '\u5915';
+  t['\u2F24'] = '\u5927';
+  t['\u2F25'] = '\u5973';
+  t['\u2F26'] = '\u5B50';
+  t['\u2F27'] = '\u5B80';
+  t['\u2F28'] = '\u5BF8';
+  t['\u2F29'] = '\u5C0F';
+  t['\u2F2A'] = '\u5C22';
+  t['\u2F2B'] = '\u5C38';
+  t['\u2F2C'] = '\u5C6E';
+  t['\u2F2D'] = '\u5C71';
+  t['\u2F2E'] = '\u5DDB';
+  t['\u2F2F'] = '\u5DE5';
+  t['\u2F30'] = '\u5DF1';
+  t['\u2F31'] = '\u5DFE';
+  t['\u2F32'] = '\u5E72';
+  t['\u2F33'] = '\u5E7A';
+  t['\u2F34'] = '\u5E7F';
+  t['\u2F35'] = '\u5EF4';
+  t['\u2F36'] = '\u5EFE';
+  t['\u2F37'] = '\u5F0B';
+  t['\u2F38'] = '\u5F13';
+  t['\u2F39'] = '\u5F50';
+  t['\u2F3A'] = '\u5F61';
+  t['\u2F3B'] = '\u5F73';
+  t['\u2F3C'] = '\u5FC3';
+  t['\u2F3D'] = '\u6208';
+  t['\u2F3E'] = '\u6236';
+  t['\u2F3F'] = '\u624B';
+  t['\u2F40'] = '\u652F';
+  t['\u2F41'] = '\u6534';
+  t['\u2F42'] = '\u6587';
+  t['\u2F43'] = '\u6597';
+  t['\u2F44'] = '\u65A4';
+  t['\u2F45'] = '\u65B9';
+  t['\u2F46'] = '\u65E0';
+  t['\u2F47'] = '\u65E5';
+  t['\u2F48'] = '\u66F0';
+  t['\u2F49'] = '\u6708';
+  t['\u2F4A'] = '\u6728';
+  t['\u2F4B'] = '\u6B20';
+  t['\u2F4C'] = '\u6B62';
+  t['\u2F4D'] = '\u6B79';
+  t['\u2F4E'] = '\u6BB3';
+  t['\u2F4F'] = '\u6BCB';
+  t['\u2F50'] = '\u6BD4';
+  t['\u2F51'] = '\u6BDB';
+  t['\u2F52'] = '\u6C0F';
+  t['\u2F53'] = '\u6C14';
+  t['\u2F54'] = '\u6C34';
+  t['\u2F55'] = '\u706B';
+  t['\u2F56'] = '\u722A';
+  t['\u2F57'] = '\u7236';
+  t['\u2F58'] = '\u723B';
+  t['\u2F59'] = '\u723F';
+  t['\u2F5A'] = '\u7247';
+  t['\u2F5B'] = '\u7259';
+  t['\u2F5C'] = '\u725B';
+  t['\u2F5D'] = '\u72AC';
+  t['\u2F5E'] = '\u7384';
+  t['\u2F5F'] = '\u7389';
+  t['\u2F60'] = '\u74DC';
+  t['\u2F61'] = '\u74E6';
+  t['\u2F62'] = '\u7518';
+  t['\u2F63'] = '\u751F';
+  t['\u2F64'] = '\u7528';
+  t['\u2F65'] = '\u7530';
+  t['\u2F66'] = '\u758B';
+  t['\u2F67'] = '\u7592';
+  t['\u2F68'] = '\u7676';
+  t['\u2F69'] = '\u767D';
+  t['\u2F6A'] = '\u76AE';
+  t['\u2F6B'] = '\u76BF';
+  t['\u2F6C'] = '\u76EE';
+  t['\u2F6D'] = '\u77DB';
+  t['\u2F6E'] = '\u77E2';
+  t['\u2F6F'] = '\u77F3';
+  t['\u2F70'] = '\u793A';
+  t['\u2F71'] = '\u79B8';
+  t['\u2F72'] = '\u79BE';
+  t['\u2F73'] = '\u7A74';
+  t['\u2F74'] = '\u7ACB';
+  t['\u2F75'] = '\u7AF9';
+  t['\u2F76'] = '\u7C73';
+  t['\u2F77'] = '\u7CF8';
+  t['\u2F78'] = '\u7F36';
+  t['\u2F79'] = '\u7F51';
+  t['\u2F7A'] = '\u7F8A';
+  t['\u2F7B'] = '\u7FBD';
+  t['\u2F7C'] = '\u8001';
+  t['\u2F7D'] = '\u800C';
+  t['\u2F7E'] = '\u8012';
+  t['\u2F7F'] = '\u8033';
+  t['\u2F80'] = '\u807F';
+  t['\u2F81'] = '\u8089';
+  t['\u2F82'] = '\u81E3';
+  t['\u2F83'] = '\u81EA';
+  t['\u2F84'] = '\u81F3';
+  t['\u2F85'] = '\u81FC';
+  t['\u2F86'] = '\u820C';
+  t['\u2F87'] = '\u821B';
+  t['\u2F88'] = '\u821F';
+  t['\u2F89'] = '\u826E';
+  t['\u2F8A'] = '\u8272';
+  t['\u2F8B'] = '\u8278';
+  t['\u2F8C'] = '\u864D';
+  t['\u2F8D'] = '\u866B';
+  t['\u2F8E'] = '\u8840';
+  t['\u2F8F'] = '\u884C';
+  t['\u2F90'] = '\u8863';
+  t['\u2F91'] = '\u897E';
+  t['\u2F92'] = '\u898B';
+  t['\u2F93'] = '\u89D2';
+  t['\u2F94'] = '\u8A00';
+  t['\u2F95'] = '\u8C37';
+  t['\u2F96'] = '\u8C46';
+  t['\u2F97'] = '\u8C55';
+  t['\u2F98'] = '\u8C78';
+  t['\u2F99'] = '\u8C9D';
+  t['\u2F9A'] = '\u8D64';
+  t['\u2F9B'] = '\u8D70';
+  t['\u2F9C'] = '\u8DB3';
+  t['\u2F9D'] = '\u8EAB';
+  t['\u2F9E'] = '\u8ECA';
+  t['\u2F9F'] = '\u8F9B';
+  t['\u2FA0'] = '\u8FB0';
+  t['\u2FA1'] = '\u8FB5';
+  t['\u2FA2'] = '\u9091';
+  t['\u2FA3'] = '\u9149';
+  t['\u2FA4'] = '\u91C6';
+  t['\u2FA5'] = '\u91CC';
+  t['\u2FA6'] = '\u91D1';
+  t['\u2FA7'] = '\u9577';
+  t['\u2FA8'] = '\u9580';
+  t['\u2FA9'] = '\u961C';
+  t['\u2FAA'] = '\u96B6';
+  t['\u2FAB'] = '\u96B9';
+  t['\u2FAC'] = '\u96E8';
+  t['\u2FAD'] = '\u9751';
+  t['\u2FAE'] = '\u975E';
+  t['\u2FAF'] = '\u9762';
+  t['\u2FB0'] = '\u9769';
+  t['\u2FB1'] = '\u97CB';
+  t['\u2FB2'] = '\u97ED';
+  t['\u2FB3'] = '\u97F3';
+  t['\u2FB4'] = '\u9801';
+  t['\u2FB5'] = '\u98A8';
+  t['\u2FB6'] = '\u98DB';
+  t['\u2FB7'] = '\u98DF';
+  t['\u2FB8'] = '\u9996';
+  t['\u2FB9'] = '\u9999';
+  t['\u2FBA'] = '\u99AC';
+  t['\u2FBB'] = '\u9AA8';
+  t['\u2FBC'] = '\u9AD8';
+  t['\u2FBD'] = '\u9ADF';
+  t['\u2FBE'] = '\u9B25';
+  t['\u2FBF'] = '\u9B2F';
+  t['\u2FC0'] = '\u9B32';
+  t['\u2FC1'] = '\u9B3C';
+  t['\u2FC2'] = '\u9B5A';
+  t['\u2FC3'] = '\u9CE5';
+  t['\u2FC4'] = '\u9E75';
+  t['\u2FC5'] = '\u9E7F';
+  t['\u2FC6'] = '\u9EA5';
+  t['\u2FC7'] = '\u9EBB';
+  t['\u2FC8'] = '\u9EC3';
+  t['\u2FC9'] = '\u9ECD';
+  t['\u2FCA'] = '\u9ED1';
+  t['\u2FCB'] = '\u9EF9';
+  t['\u2FCC'] = '\u9EFD';
+  t['\u2FCD'] = '\u9F0E';
+  t['\u2FCE'] = '\u9F13';
+  t['\u2FCF'] = '\u9F20';
+  t['\u2FD0'] = '\u9F3B';
+  t['\u2FD1'] = '\u9F4A';
+  t['\u2FD2'] = '\u9F52';
+  t['\u2FD3'] = '\u9F8D';
+  t['\u2FD4'] = '\u9F9C';
+  t['\u2FD5'] = '\u9FA0';
+  t['\u3036'] = '\u3012';
+  t['\u3038'] = '\u5341';
+  t['\u3039'] = '\u5344';
+  t['\u303A'] = '\u5345';
+  t['\u309B'] = '\u0020\u3099';
+  t['\u309C'] = '\u0020\u309A';
+  t['\u3131'] = '\u1100';
+  t['\u3132'] = '\u1101';
+  t['\u3133'] = '\u11AA';
+  t['\u3134'] = '\u1102';
+  t['\u3135'] = '\u11AC';
+  t['\u3136'] = '\u11AD';
+  t['\u3137'] = '\u1103';
+  t['\u3138'] = '\u1104';
+  t['\u3139'] = '\u1105';
+  t['\u313A'] = '\u11B0';
+  t['\u313B'] = '\u11B1';
+  t['\u313C'] = '\u11B2';
+  t['\u313D'] = '\u11B3';
+  t['\u313E'] = '\u11B4';
+  t['\u313F'] = '\u11B5';
+  t['\u3140'] = '\u111A';
+  t['\u3141'] = '\u1106';
+  t['\u3142'] = '\u1107';
+  t['\u3143'] = '\u1108';
+  t['\u3144'] = '\u1121';
+  t['\u3145'] = '\u1109';
+  t['\u3146'] = '\u110A';
+  t['\u3147'] = '\u110B';
+  t['\u3148'] = '\u110C';
+  t['\u3149'] = '\u110D';
+  t['\u314A'] = '\u110E';
+  t['\u314B'] = '\u110F';
+  t['\u314C'] = '\u1110';
+  t['\u314D'] = '\u1111';
+  t['\u314E'] = '\u1112';
+  t['\u314F'] = '\u1161';
+  t['\u3150'] = '\u1162';
+  t['\u3151'] = '\u1163';
+  t['\u3152'] = '\u1164';
+  t['\u3153'] = '\u1165';
+  t['\u3154'] = '\u1166';
+  t['\u3155'] = '\u1167';
+  t['\u3156'] = '\u1168';
+  t['\u3157'] = '\u1169';
+  t['\u3158'] = '\u116A';
+  t['\u3159'] = '\u116B';
+  t['\u315A'] = '\u116C';
+  t['\u315B'] = '\u116D';
+  t['\u315C'] = '\u116E';
+  t['\u315D'] = '\u116F';
+  t['\u315E'] = '\u1170';
+  t['\u315F'] = '\u1171';
+  t['\u3160'] = '\u1172';
+  t['\u3161'] = '\u1173';
+  t['\u3162'] = '\u1174';
+  t['\u3163'] = '\u1175';
+  t['\u3164'] = '\u1160';
+  t['\u3165'] = '\u1114';
+  t['\u3166'] = '\u1115';
+  t['\u3167'] = '\u11C7';
+  t['\u3168'] = '\u11C8';
+  t['\u3169'] = '\u11CC';
+  t['\u316A'] = '\u11CE';
+  t['\u316B'] = '\u11D3';
+  t['\u316C'] = '\u11D7';
+  t['\u316D'] = '\u11D9';
+  t['\u316E'] = '\u111C';
+  t['\u316F'] = '\u11DD';
+  t['\u3170'] = '\u11DF';
+  t['\u3171'] = '\u111D';
+  t['\u3172'] = '\u111E';
+  t['\u3173'] = '\u1120';
+  t['\u3174'] = '\u1122';
+  t['\u3175'] = '\u1123';
+  t['\u3176'] = '\u1127';
+  t['\u3177'] = '\u1129';
+  t['\u3178'] = '\u112B';
+  t['\u3179'] = '\u112C';
+  t['\u317A'] = '\u112D';
+  t['\u317B'] = '\u112E';
+  t['\u317C'] = '\u112F';
+  t['\u317D'] = '\u1132';
+  t['\u317E'] = '\u1136';
+  t['\u317F'] = '\u1140';
+  t['\u3180'] = '\u1147';
+  t['\u3181'] = '\u114C';
+  t['\u3182'] = '\u11F1';
+  t['\u3183'] = '\u11F2';
+  t['\u3184'] = '\u1157';
+  t['\u3185'] = '\u1158';
+  t['\u3186'] = '\u1159';
+  t['\u3187'] = '\u1184';
+  t['\u3188'] = '\u1185';
+  t['\u3189'] = '\u1188';
+  t['\u318A'] = '\u1191';
+  t['\u318B'] = '\u1192';
+  t['\u318C'] = '\u1194';
+  t['\u318D'] = '\u119E';
+  t['\u318E'] = '\u11A1';
+  t['\u3200'] = '\u0028\u1100\u0029';
+  t['\u3201'] = '\u0028\u1102\u0029';
+  t['\u3202'] = '\u0028\u1103\u0029';
+  t['\u3203'] = '\u0028\u1105\u0029';
+  t['\u3204'] = '\u0028\u1106\u0029';
+  t['\u3205'] = '\u0028\u1107\u0029';
+  t['\u3206'] = '\u0028\u1109\u0029';
+  t['\u3207'] = '\u0028\u110B\u0029';
+  t['\u3208'] = '\u0028\u110C\u0029';
+  t['\u3209'] = '\u0028\u110E\u0029';
+  t['\u320A'] = '\u0028\u110F\u0029';
+  t['\u320B'] = '\u0028\u1110\u0029';
+  t['\u320C'] = '\u0028\u1111\u0029';
+  t['\u320D'] = '\u0028\u1112\u0029';
+  t['\u320E'] = '\u0028\u1100\u1161\u0029';
+  t['\u320F'] = '\u0028\u1102\u1161\u0029';
+  t['\u3210'] = '\u0028\u1103\u1161\u0029';
+  t['\u3211'] = '\u0028\u1105\u1161\u0029';
+  t['\u3212'] = '\u0028\u1106\u1161\u0029';
+  t['\u3213'] = '\u0028\u1107\u1161\u0029';
+  t['\u3214'] = '\u0028\u1109\u1161\u0029';
+  t['\u3215'] = '\u0028\u110B\u1161\u0029';
+  t['\u3216'] = '\u0028\u110C\u1161\u0029';
+  t['\u3217'] = '\u0028\u110E\u1161\u0029';
+  t['\u3218'] = '\u0028\u110F\u1161\u0029';
+  t['\u3219'] = '\u0028\u1110\u1161\u0029';
+  t['\u321A'] = '\u0028\u1111\u1161\u0029';
+  t['\u321B'] = '\u0028\u1112\u1161\u0029';
+  t['\u321C'] = '\u0028\u110C\u116E\u0029';
+  t['\u321D'] = '\u0028\u110B\u1169\u110C\u1165\u11AB\u0029';
+  t['\u321E'] = '\u0028\u110B\u1169\u1112\u116E\u0029';
+  t['\u3220'] = '\u0028\u4E00\u0029';
+  t['\u3221'] = '\u0028\u4E8C\u0029';
+  t['\u3222'] = '\u0028\u4E09\u0029';
+  t['\u3223'] = '\u0028\u56DB\u0029';
+  t['\u3224'] = '\u0028\u4E94\u0029';
+  t['\u3225'] = '\u0028\u516D\u0029';
+  t['\u3226'] = '\u0028\u4E03\u0029';
+  t['\u3227'] = '\u0028\u516B\u0029';
+  t['\u3228'] = '\u0028\u4E5D\u0029';
+  t['\u3229'] = '\u0028\u5341\u0029';
+  t['\u322A'] = '\u0028\u6708\u0029';
+  t['\u322B'] = '\u0028\u706B\u0029';
+  t['\u322C'] = '\u0028\u6C34\u0029';
+  t['\u322D'] = '\u0028\u6728\u0029';
+  t['\u322E'] = '\u0028\u91D1\u0029';
+  t['\u322F'] = '\u0028\u571F\u0029';
+  t['\u3230'] = '\u0028\u65E5\u0029';
+  t['\u3231'] = '\u0028\u682A\u0029';
+  t['\u3232'] = '\u0028\u6709\u0029';
+  t['\u3233'] = '\u0028\u793E\u0029';
+  t['\u3234'] = '\u0028\u540D\u0029';
+  t['\u3235'] = '\u0028\u7279\u0029';
+  t['\u3236'] = '\u0028\u8CA1\u0029';
+  t['\u3237'] = '\u0028\u795D\u0029';
+  t['\u3238'] = '\u0028\u52B4\u0029';
+  t['\u3239'] = '\u0028\u4EE3\u0029';
+  t['\u323A'] = '\u0028\u547C\u0029';
+  t['\u323B'] = '\u0028\u5B66\u0029';
+  t['\u323C'] = '\u0028\u76E3\u0029';
+  t['\u323D'] = '\u0028\u4F01\u0029';
+  t['\u323E'] = '\u0028\u8CC7\u0029';
+  t['\u323F'] = '\u0028\u5354\u0029';
+  t['\u3240'] = '\u0028\u796D\u0029';
+  t['\u3241'] = '\u0028\u4F11\u0029';
+  t['\u3242'] = '\u0028\u81EA\u0029';
+  t['\u3243'] = '\u0028\u81F3\u0029';
+  t['\u32C0'] = '\u0031\u6708';
+  t['\u32C1'] = '\u0032\u6708';
+  t['\u32C2'] = '\u0033\u6708';
+  t['\u32C3'] = '\u0034\u6708';
+  t['\u32C4'] = '\u0035\u6708';
+  t['\u32C5'] = '\u0036\u6708';
+  t['\u32C6'] = '\u0037\u6708';
+  t['\u32C7'] = '\u0038\u6708';
+  t['\u32C8'] = '\u0039\u6708';
+  t['\u32C9'] = '\u0031\u0030\u6708';
+  t['\u32CA'] = '\u0031\u0031\u6708';
+  t['\u32CB'] = '\u0031\u0032\u6708';
+  t['\u3358'] = '\u0030\u70B9';
+  t['\u3359'] = '\u0031\u70B9';
+  t['\u335A'] = '\u0032\u70B9';
+  t['\u335B'] = '\u0033\u70B9';
+  t['\u335C'] = '\u0034\u70B9';
+  t['\u335D'] = '\u0035\u70B9';
+  t['\u335E'] = '\u0036\u70B9';
+  t['\u335F'] = '\u0037\u70B9';
+  t['\u3360'] = '\u0038\u70B9';
+  t['\u3361'] = '\u0039\u70B9';
+  t['\u3362'] = '\u0031\u0030\u70B9';
+  t['\u3363'] = '\u0031\u0031\u70B9';
+  t['\u3364'] = '\u0031\u0032\u70B9';
+  t['\u3365'] = '\u0031\u0033\u70B9';
+  t['\u3366'] = '\u0031\u0034\u70B9';
+  t['\u3367'] = '\u0031\u0035\u70B9';
+  t['\u3368'] = '\u0031\u0036\u70B9';
+  t['\u3369'] = '\u0031\u0037\u70B9';
+  t['\u336A'] = '\u0031\u0038\u70B9';
+  t['\u336B'] = '\u0031\u0039\u70B9';
+  t['\u336C'] = '\u0032\u0030\u70B9';
+  t['\u336D'] = '\u0032\u0031\u70B9';
+  t['\u336E'] = '\u0032\u0032\u70B9';
+  t['\u336F'] = '\u0032\u0033\u70B9';
+  t['\u3370'] = '\u0032\u0034\u70B9';
+  t['\u33E0'] = '\u0031\u65E5';
+  t['\u33E1'] = '\u0032\u65E5';
+  t['\u33E2'] = '\u0033\u65E5';
+  t['\u33E3'] = '\u0034\u65E5';
+  t['\u33E4'] = '\u0035\u65E5';
+  t['\u33E5'] = '\u0036\u65E5';
+  t['\u33E6'] = '\u0037\u65E5';
+  t['\u33E7'] = '\u0038\u65E5';
+  t['\u33E8'] = '\u0039\u65E5';
+  t['\u33E9'] = '\u0031\u0030\u65E5';
+  t['\u33EA'] = '\u0031\u0031\u65E5';
+  t['\u33EB'] = '\u0031\u0032\u65E5';
+  t['\u33EC'] = '\u0031\u0033\u65E5';
+  t['\u33ED'] = '\u0031\u0034\u65E5';
+  t['\u33EE'] = '\u0031\u0035\u65E5';
+  t['\u33EF'] = '\u0031\u0036\u65E5';
+  t['\u33F0'] = '\u0031\u0037\u65E5';
+  t['\u33F1'] = '\u0031\u0038\u65E5';
+  t['\u33F2'] = '\u0031\u0039\u65E5';
+  t['\u33F3'] = '\u0032\u0030\u65E5';
+  t['\u33F4'] = '\u0032\u0031\u65E5';
+  t['\u33F5'] = '\u0032\u0032\u65E5';
+  t['\u33F6'] = '\u0032\u0033\u65E5';
+  t['\u33F7'] = '\u0032\u0034\u65E5';
+  t['\u33F8'] = '\u0032\u0035\u65E5';
+  t['\u33F9'] = '\u0032\u0036\u65E5';
+  t['\u33FA'] = '\u0032\u0037\u65E5';
+  t['\u33FB'] = '\u0032\u0038\u65E5';
+  t['\u33FC'] = '\u0032\u0039\u65E5';
+  t['\u33FD'] = '\u0033\u0030\u65E5';
+  t['\u33FE'] = '\u0033\u0031\u65E5';
+  t['\uFB00'] = '\u0066\u0066';
+  t['\uFB01'] = '\u0066\u0069';
+  t['\uFB02'] = '\u0066\u006C';
+  t['\uFB03'] = '\u0066\u0066\u0069';
+  t['\uFB04'] = '\u0066\u0066\u006C';
+  t['\uFB05'] = '\u017F\u0074';
+  t['\uFB06'] = '\u0073\u0074';
+  t['\uFB13'] = '\u0574\u0576';
+  t['\uFB14'] = '\u0574\u0565';
+  t['\uFB15'] = '\u0574\u056B';
+  t['\uFB16'] = '\u057E\u0576';
+  t['\uFB17'] = '\u0574\u056D';
+  t['\uFB4F'] = '\u05D0\u05DC';
+  t['\uFB50'] = '\u0671';
+  t['\uFB51'] = '\u0671';
+  t['\uFB52'] = '\u067B';
+  t['\uFB53'] = '\u067B';
+  t['\uFB54'] = '\u067B';
+  t['\uFB55'] = '\u067B';
+  t['\uFB56'] = '\u067E';
+  t['\uFB57'] = '\u067E';
+  t['\uFB58'] = '\u067E';
+  t['\uFB59'] = '\u067E';
+  t['\uFB5A'] = '\u0680';
+  t['\uFB5B'] = '\u0680';
+  t['\uFB5C'] = '\u0680';
+  t['\uFB5D'] = '\u0680';
+  t['\uFB5E'] = '\u067A';
+  t['\uFB5F'] = '\u067A';
+  t['\uFB60'] = '\u067A';
+  t['\uFB61'] = '\u067A';
+  t['\uFB62'] = '\u067F';
+  t['\uFB63'] = '\u067F';
+  t['\uFB64'] = '\u067F';
+  t['\uFB65'] = '\u067F';
+  t['\uFB66'] = '\u0679';
+  t['\uFB67'] = '\u0679';
+  t['\uFB68'] = '\u0679';
+  t['\uFB69'] = '\u0679';
+  t['\uFB6A'] = '\u06A4';
+  t['\uFB6B'] = '\u06A4';
+  t['\uFB6C'] = '\u06A4';
+  t['\uFB6D'] = '\u06A4';
+  t['\uFB6E'] = '\u06A6';
+  t['\uFB6F'] = '\u06A6';
+  t['\uFB70'] = '\u06A6';
+  t['\uFB71'] = '\u06A6';
+  t['\uFB72'] = '\u0684';
+  t['\uFB73'] = '\u0684';
+  t['\uFB74'] = '\u0684';
+  t['\uFB75'] = '\u0684';
+  t['\uFB76'] = '\u0683';
+  t['\uFB77'] = '\u0683';
+  t['\uFB78'] = '\u0683';
+  t['\uFB79'] = '\u0683';
+  t['\uFB7A'] = '\u0686';
+  t['\uFB7B'] = '\u0686';
+  t['\uFB7C'] = '\u0686';
+  t['\uFB7D'] = '\u0686';
+  t['\uFB7E'] = '\u0687';
+  t['\uFB7F'] = '\u0687';
+  t['\uFB80'] = '\u0687';
+  t['\uFB81'] = '\u0687';
+  t['\uFB82'] = '\u068D';
+  t['\uFB83'] = '\u068D';
+  t['\uFB84'] = '\u068C';
+  t['\uFB85'] = '\u068C';
+  t['\uFB86'] = '\u068E';
+  t['\uFB87'] = '\u068E';
+  t['\uFB88'] = '\u0688';
+  t['\uFB89'] = '\u0688';
+  t['\uFB8A'] = '\u0698';
+  t['\uFB8B'] = '\u0698';
+  t['\uFB8C'] = '\u0691';
+  t['\uFB8D'] = '\u0691';
+  t['\uFB8E'] = '\u06A9';
+  t['\uFB8F'] = '\u06A9';
+  t['\uFB90'] = '\u06A9';
+  t['\uFB91'] = '\u06A9';
+  t['\uFB92'] = '\u06AF';
+  t['\uFB93'] = '\u06AF';
+  t['\uFB94'] = '\u06AF';
+  t['\uFB95'] = '\u06AF';
+  t['\uFB96'] = '\u06B3';
+  t['\uFB97'] = '\u06B3';
+  t['\uFB98'] = '\u06B3';
+  t['\uFB99'] = '\u06B3';
+  t['\uFB9A'] = '\u06B1';
+  t['\uFB9B'] = '\u06B1';
+  t['\uFB9C'] = '\u06B1';
+  t['\uFB9D'] = '\u06B1';
+  t['\uFB9E'] = '\u06BA';
+  t['\uFB9F'] = '\u06BA';
+  t['\uFBA0'] = '\u06BB';
+  t['\uFBA1'] = '\u06BB';
+  t['\uFBA2'] = '\u06BB';
+  t['\uFBA3'] = '\u06BB';
+  t['\uFBA4'] = '\u06C0';
+  t['\uFBA5'] = '\u06C0';
+  t['\uFBA6'] = '\u06C1';
+  t['\uFBA7'] = '\u06C1';
+  t['\uFBA8'] = '\u06C1';
+  t['\uFBA9'] = '\u06C1';
+  t['\uFBAA'] = '\u06BE';
+  t['\uFBAB'] = '\u06BE';
+  t['\uFBAC'] = '\u06BE';
+  t['\uFBAD'] = '\u06BE';
+  t['\uFBAE'] = '\u06D2';
+  t['\uFBAF'] = '\u06D2';
+  t['\uFBB0'] = '\u06D3';
+  t['\uFBB1'] = '\u06D3';
+  t['\uFBD3'] = '\u06AD';
+  t['\uFBD4'] = '\u06AD';
+  t['\uFBD5'] = '\u06AD';
+  t['\uFBD6'] = '\u06AD';
+  t['\uFBD7'] = '\u06C7';
+  t['\uFBD8'] = '\u06C7';
+  t['\uFBD9'] = '\u06C6';
+  t['\uFBDA'] = '\u06C6';
+  t['\uFBDB'] = '\u06C8';
+  t['\uFBDC'] = '\u06C8';
+  t['\uFBDD'] = '\u0677';
+  t['\uFBDE'] = '\u06CB';
+  t['\uFBDF'] = '\u06CB';
+  t['\uFBE0'] = '\u06C5';
+  t['\uFBE1'] = '\u06C5';
+  t['\uFBE2'] = '\u06C9';
+  t['\uFBE3'] = '\u06C9';
+  t['\uFBE4'] = '\u06D0';
+  t['\uFBE5'] = '\u06D0';
+  t['\uFBE6'] = '\u06D0';
+  t['\uFBE7'] = '\u06D0';
+  t['\uFBE8'] = '\u0649';
+  t['\uFBE9'] = '\u0649';
+  t['\uFBEA'] = '\u0626\u0627';
+  t['\uFBEB'] = '\u0626\u0627';
+  t['\uFBEC'] = '\u0626\u06D5';
+  t['\uFBED'] = '\u0626\u06D5';
+  t['\uFBEE'] = '\u0626\u0648';
+  t['\uFBEF'] = '\u0626\u0648';
+  t['\uFBF0'] = '\u0626\u06C7';
+  t['\uFBF1'] = '\u0626\u06C7';
+  t['\uFBF2'] = '\u0626\u06C6';
+  t['\uFBF3'] = '\u0626\u06C6';
+  t['\uFBF4'] = '\u0626\u06C8';
+  t['\uFBF5'] = '\u0626\u06C8';
+  t['\uFBF6'] = '\u0626\u06D0';
+  t['\uFBF7'] = '\u0626\u06D0';
+  t['\uFBF8'] = '\u0626\u06D0';
+  t['\uFBF9'] = '\u0626\u0649';
+  t['\uFBFA'] = '\u0626\u0649';
+  t['\uFBFB'] = '\u0626\u0649';
+  t['\uFBFC'] = '\u06CC';
+  t['\uFBFD'] = '\u06CC';
+  t['\uFBFE'] = '\u06CC';
+  t['\uFBFF'] = '\u06CC';
+  t['\uFC00'] = '\u0626\u062C';
+  t['\uFC01'] = '\u0626\u062D';
+  t['\uFC02'] = '\u0626\u0645';
+  t['\uFC03'] = '\u0626\u0649';
+  t['\uFC04'] = '\u0626\u064A';
+  t['\uFC05'] = '\u0628\u062C';
+  t['\uFC06'] = '\u0628\u062D';
+  t['\uFC07'] = '\u0628\u062E';
+  t['\uFC08'] = '\u0628\u0645';
+  t['\uFC09'] = '\u0628\u0649';
+  t['\uFC0A'] = '\u0628\u064A';
+  t['\uFC0B'] = '\u062A\u062C';
+  t['\uFC0C'] = '\u062A\u062D';
+  t['\uFC0D'] = '\u062A\u062E';
+  t['\uFC0E'] = '\u062A\u0645';
+  t['\uFC0F'] = '\u062A\u0649';
+  t['\uFC10'] = '\u062A\u064A';
+  t['\uFC11'] = '\u062B\u062C';
+  t['\uFC12'] = '\u062B\u0645';
+  t['\uFC13'] = '\u062B\u0649';
+  t['\uFC14'] = '\u062B\u064A';
+  t['\uFC15'] = '\u062C\u062D';
+  t['\uFC16'] = '\u062C\u0645';
+  t['\uFC17'] = '\u062D\u062C';
+  t['\uFC18'] = '\u062D\u0645';
+  t['\uFC19'] = '\u062E\u062C';
+  t['\uFC1A'] = '\u062E\u062D';
+  t['\uFC1B'] = '\u062E\u0645';
+  t['\uFC1C'] = '\u0633\u062C';
+  t['\uFC1D'] = '\u0633\u062D';
+  t['\uFC1E'] = '\u0633\u062E';
+  t['\uFC1F'] = '\u0633\u0645';
+  t['\uFC20'] = '\u0635\u062D';
+  t['\uFC21'] = '\u0635\u0645';
+  t['\uFC22'] = '\u0636\u062C';
+  t['\uFC23'] = '\u0636\u062D';
+  t['\uFC24'] = '\u0636\u062E';
+  t['\uFC25'] = '\u0636\u0645';
+  t['\uFC26'] = '\u0637\u062D';
+  t['\uFC27'] = '\u0637\u0645';
+  t['\uFC28'] = '\u0638\u0645';
+  t['\uFC29'] = '\u0639\u062C';
+  t['\uFC2A'] = '\u0639\u0645';
+  t['\uFC2B'] = '\u063A\u062C';
+  t['\uFC2C'] = '\u063A\u0645';
+  t['\uFC2D'] = '\u0641\u062C';
+  t['\uFC2E'] = '\u0641\u062D';
+  t['\uFC2F'] = '\u0641\u062E';
+  t['\uFC30'] = '\u0641\u0645';
+  t['\uFC31'] = '\u0641\u0649';
+  t['\uFC32'] = '\u0641\u064A';
+  t['\uFC33'] = '\u0642\u062D';
+  t['\uFC34'] = '\u0642\u0645';
+  t['\uFC35'] = '\u0642\u0649';
+  t['\uFC36'] = '\u0642\u064A';
+  t['\uFC37'] = '\u0643\u0627';
+  t['\uFC38'] = '\u0643\u062C';
+  t['\uFC39'] = '\u0643\u062D';
+  t['\uFC3A'] = '\u0643\u062E';
+  t['\uFC3B'] = '\u0643\u0644';
+  t['\uFC3C'] = '\u0643\u0645';
+  t['\uFC3D'] = '\u0643\u0649';
+  t['\uFC3E'] = '\u0643\u064A';
+  t['\uFC3F'] = '\u0644\u062C';
+  t['\uFC40'] = '\u0644\u062D';
+  t['\uFC41'] = '\u0644\u062E';
+  t['\uFC42'] = '\u0644\u0645';
+  t['\uFC43'] = '\u0644\u0649';
+  t['\uFC44'] = '\u0644\u064A';
+  t['\uFC45'] = '\u0645\u062C';
+  t['\uFC46'] = '\u0645\u062D';
+  t['\uFC47'] = '\u0645\u062E';
+  t['\uFC48'] = '\u0645\u0645';
+  t['\uFC49'] = '\u0645\u0649';
+  t['\uFC4A'] = '\u0645\u064A';
+  t['\uFC4B'] = '\u0646\u062C';
+  t['\uFC4C'] = '\u0646\u062D';
+  t['\uFC4D'] = '\u0646\u062E';
+  t['\uFC4E'] = '\u0646\u0645';
+  t['\uFC4F'] = '\u0646\u0649';
+  t['\uFC50'] = '\u0646\u064A';
+  t['\uFC51'] = '\u0647\u062C';
+  t['\uFC52'] = '\u0647\u0645';
+  t['\uFC53'] = '\u0647\u0649';
+  t['\uFC54'] = '\u0647\u064A';
+  t['\uFC55'] = '\u064A\u062C';
+  t['\uFC56'] = '\u064A\u062D';
+  t['\uFC57'] = '\u064A\u062E';
+  t['\uFC58'] = '\u064A\u0645';
+  t['\uFC59'] = '\u064A\u0649';
+  t['\uFC5A'] = '\u064A\u064A';
+  t['\uFC5B'] = '\u0630\u0670';
+  t['\uFC5C'] = '\u0631\u0670';
+  t['\uFC5D'] = '\u0649\u0670';
+  t['\uFC5E'] = '\u0020\u064C\u0651';
+  t['\uFC5F'] = '\u0020\u064D\u0651';
+  t['\uFC60'] = '\u0020\u064E\u0651';
+  t['\uFC61'] = '\u0020\u064F\u0651';
+  t['\uFC62'] = '\u0020\u0650\u0651';
+  t['\uFC63'] = '\u0020\u0651\u0670';
+  t['\uFC64'] = '\u0626\u0631';
+  t['\uFC65'] = '\u0626\u0632';
+  t['\uFC66'] = '\u0626\u0645';
+  t['\uFC67'] = '\u0626\u0646';
+  t['\uFC68'] = '\u0626\u0649';
+  t['\uFC69'] = '\u0626\u064A';
+  t['\uFC6A'] = '\u0628\u0631';
+  t['\uFC6B'] = '\u0628\u0632';
+  t['\uFC6C'] = '\u0628\u0645';
+  t['\uFC6D'] = '\u0628\u0646';
+  t['\uFC6E'] = '\u0628\u0649';
+  t['\uFC6F'] = '\u0628\u064A';
+  t['\uFC70'] = '\u062A\u0631';
+  t['\uFC71'] = '\u062A\u0632';
+  t['\uFC72'] = '\u062A\u0645';
+  t['\uFC73'] = '\u062A\u0646';
+  t['\uFC74'] = '\u062A\u0649';
+  t['\uFC75'] = '\u062A\u064A';
+  t['\uFC76'] = '\u062B\u0631';
+  t['\uFC77'] = '\u062B\u0632';
+  t['\uFC78'] = '\u062B\u0645';
+  t['\uFC79'] = '\u062B\u0646';
+  t['\uFC7A'] = '\u062B\u0649';
+  t['\uFC7B'] = '\u062B\u064A';
+  t['\uFC7C'] = '\u0641\u0649';
+  t['\uFC7D'] = '\u0641\u064A';
+  t['\uFC7E'] = '\u0642\u0649';
+  t['\uFC7F'] = '\u0642\u064A';
+  t['\uFC80'] = '\u0643\u0627';
+  t['\uFC81'] = '\u0643\u0644';
+  t['\uFC82'] = '\u0643\u0645';
+  t['\uFC83'] = '\u0643\u0649';
+  t['\uFC84'] = '\u0643\u064A';
+  t['\uFC85'] = '\u0644\u0645';
+  t['\uFC86'] = '\u0644\u0649';
+  t['\uFC87'] = '\u0644\u064A';
+  t['\uFC88'] = '\u0645\u0627';
+  t['\uFC89'] = '\u0645\u0645';
+  t['\uFC8A'] = '\u0646\u0631';
+  t['\uFC8B'] = '\u0646\u0632';
+  t['\uFC8C'] = '\u0646\u0645';
+  t['\uFC8D'] = '\u0646\u0646';
+  t['\uFC8E'] = '\u0646\u0649';
+  t['\uFC8F'] = '\u0646\u064A';
+  t['\uFC90'] = '\u0649\u0670';
+  t['\uFC91'] = '\u064A\u0631';
+  t['\uFC92'] = '\u064A\u0632';
+  t['\uFC93'] = '\u064A\u0645';
+  t['\uFC94'] = '\u064A\u0646';
+  t['\uFC95'] = '\u064A\u0649';
+  t['\uFC96'] = '\u064A\u064A';
+  t['\uFC97'] = '\u0626\u062C';
+  t['\uFC98'] = '\u0626\u062D';
+  t['\uFC99'] = '\u0626\u062E';
+  t['\uFC9A'] = '\u0626\u0645';
+  t['\uFC9B'] = '\u0626\u0647';
+  t['\uFC9C'] = '\u0628\u062C';
+  t['\uFC9D'] = '\u0628\u062D';
+  t['\uFC9E'] = '\u0628\u062E';
+  t['\uFC9F'] = '\u0628\u0645';
+  t['\uFCA0'] = '\u0628\u0647';
+  t['\uFCA1'] = '\u062A\u062C';
+  t['\uFCA2'] = '\u062A\u062D';
+  t['\uFCA3'] = '\u062A\u062E';
+  t['\uFCA4'] = '\u062A\u0645';
+  t['\uFCA5'] = '\u062A\u0647';
+  t['\uFCA6'] = '\u062B\u0645';
+  t['\uFCA7'] = '\u062C\u062D';
+  t['\uFCA8'] = '\u062C\u0645';
+  t['\uFCA9'] = '\u062D\u062C';
+  t['\uFCAA'] = '\u062D\u0645';
+  t['\uFCAB'] = '\u062E\u062C';
+  t['\uFCAC'] = '\u062E\u0645';
+  t['\uFCAD'] = '\u0633\u062C';
+  t['\uFCAE'] = '\u0633\u062D';
+  t['\uFCAF'] = '\u0633\u062E';
+  t['\uFCB0'] = '\u0633\u0645';
+  t['\uFCB1'] = '\u0635\u062D';
+  t['\uFCB2'] = '\u0635\u062E';
+  t['\uFCB3'] = '\u0635\u0645';
+  t['\uFCB4'] = '\u0636\u062C';
+  t['\uFCB5'] = '\u0636\u062D';
+  t['\uFCB6'] = '\u0636\u062E';
+  t['\uFCB7'] = '\u0636\u0645';
+  t['\uFCB8'] = '\u0637\u062D';
+  t['\uFCB9'] = '\u0638\u0645';
+  t['\uFCBA'] = '\u0639\u062C';
+  t['\uFCBB'] = '\u0639\u0645';
+  t['\uFCBC'] = '\u063A\u062C';
+  t['\uFCBD'] = '\u063A\u0645';
+  t['\uFCBE'] = '\u0641\u062C';
+  t['\uFCBF'] = '\u0641\u062D';
+  t['\uFCC0'] = '\u0641\u062E';
+  t['\uFCC1'] = '\u0641\u0645';
+  t['\uFCC2'] = '\u0642\u062D';
+  t['\uFCC3'] = '\u0642\u0645';
+  t['\uFCC4'] = '\u0643\u062C';
+  t['\uFCC5'] = '\u0643\u062D';
+  t['\uFCC6'] = '\u0643\u062E';
+  t['\uFCC7'] = '\u0643\u0644';
+  t['\uFCC8'] = '\u0643\u0645';
+  t['\uFCC9'] = '\u0644\u062C';
+  t['\uFCCA'] = '\u0644\u062D';
+  t['\uFCCB'] = '\u0644\u062E';
+  t['\uFCCC'] = '\u0644\u0645';
+  t['\uFCCD'] = '\u0644\u0647';
+  t['\uFCCE'] = '\u0645\u062C';
+  t['\uFCCF'] = '\u0645\u062D';
+  t['\uFCD0'] = '\u0645\u062E';
+  t['\uFCD1'] = '\u0645\u0645';
+  t['\uFCD2'] = '\u0646\u062C';
+  t['\uFCD3'] = '\u0646\u062D';
+  t['\uFCD4'] = '\u0646\u062E';
+  t['\uFCD5'] = '\u0646\u0645';
+  t['\uFCD6'] = '\u0646\u0647';
+  t['\uFCD7'] = '\u0647\u062C';
+  t['\uFCD8'] = '\u0647\u0645';
+  t['\uFCD9'] = '\u0647\u0670';
+  t['\uFCDA'] = '\u064A\u062C';
+  t['\uFCDB'] = '\u064A\u062D';
+  t['\uFCDC'] = '\u064A\u062E';
+  t['\uFCDD'] = '\u064A\u0645';
+  t['\uFCDE'] = '\u064A\u0647';
+  t['\uFCDF'] = '\u0626\u0645';
+  t['\uFCE0'] = '\u0626\u0647';
+  t['\uFCE1'] = '\u0628\u0645';
+  t['\uFCE2'] = '\u0628\u0647';
+  t['\uFCE3'] = '\u062A\u0645';
+  t['\uFCE4'] = '\u062A\u0647';
+  t['\uFCE5'] = '\u062B\u0645';
+  t['\uFCE6'] = '\u062B\u0647';
+  t['\uFCE7'] = '\u0633\u0645';
+  t['\uFCE8'] = '\u0633\u0647';
+  t['\uFCE9'] = '\u0634\u0645';
+  t['\uFCEA'] = '\u0634\u0647';
+  t['\uFCEB'] = '\u0643\u0644';
+  t['\uFCEC'] = '\u0643\u0645';
+  t['\uFCED'] = '\u0644\u0645';
+  t['\uFCEE'] = '\u0646\u0645';
+  t['\uFCEF'] = '\u0646\u0647';
+  t['\uFCF0'] = '\u064A\u0645';
+  t['\uFCF1'] = '\u064A\u0647';
+  t['\uFCF2'] = '\u0640\u064E\u0651';
+  t['\uFCF3'] = '\u0640\u064F\u0651';
+  t['\uFCF4'] = '\u0640\u0650\u0651';
+  t['\uFCF5'] = '\u0637\u0649';
+  t['\uFCF6'] = '\u0637\u064A';
+  t['\uFCF7'] = '\u0639\u0649';
+  t['\uFCF8'] = '\u0639\u064A';
+  t['\uFCF9'] = '\u063A\u0649';
+  t['\uFCFA'] = '\u063A\u064A';
+  t['\uFCFB'] = '\u0633\u0649';
+  t['\uFCFC'] = '\u0633\u064A';
+  t['\uFCFD'] = '\u0634\u0649';
+  t['\uFCFE'] = '\u0634\u064A';
+  t['\uFCFF'] = '\u062D\u0649';
+  t['\uFD00'] = '\u062D\u064A';
+  t['\uFD01'] = '\u062C\u0649';
+  t['\uFD02'] = '\u062C\u064A';
+  t['\uFD03'] = '\u062E\u0649';
+  t['\uFD04'] = '\u062E\u064A';
+  t['\uFD05'] = '\u0635\u0649';
+  t['\uFD06'] = '\u0635\u064A';
+  t['\uFD07'] = '\u0636\u0649';
+  t['\uFD08'] = '\u0636\u064A';
+  t['\uFD09'] = '\u0634\u062C';
+  t['\uFD0A'] = '\u0634\u062D';
+  t['\uFD0B'] = '\u0634\u062E';
+  t['\uFD0C'] = '\u0634\u0645';
+  t['\uFD0D'] = '\u0634\u0631';
+  t['\uFD0E'] = '\u0633\u0631';
+  t['\uFD0F'] = '\u0635\u0631';
+  t['\uFD10'] = '\u0636\u0631';
+  t['\uFD11'] = '\u0637\u0649';
+  t['\uFD12'] = '\u0637\u064A';
+  t['\uFD13'] = '\u0639\u0649';
+  t['\uFD14'] = '\u0639\u064A';
+  t['\uFD15'] = '\u063A\u0649';
+  t['\uFD16'] = '\u063A\u064A';
+  t['\uFD17'] = '\u0633\u0649';
+  t['\uFD18'] = '\u0633\u064A';
+  t['\uFD19'] = '\u0634\u0649';
+  t['\uFD1A'] = '\u0634\u064A';
+  t['\uFD1B'] = '\u062D\u0649';
+  t['\uFD1C'] = '\u062D\u064A';
+  t['\uFD1D'] = '\u062C\u0649';
+  t['\uFD1E'] = '\u062C\u064A';
+  t['\uFD1F'] = '\u062E\u0649';
+  t['\uFD20'] = '\u062E\u064A';
+  t['\uFD21'] = '\u0635\u0649';
+  t['\uFD22'] = '\u0635\u064A';
+  t['\uFD23'] = '\u0636\u0649';
+  t['\uFD24'] = '\u0636\u064A';
+  t['\uFD25'] = '\u0634\u062C';
+  t['\uFD26'] = '\u0634\u062D';
+  t['\uFD27'] = '\u0634\u062E';
+  t['\uFD28'] = '\u0634\u0645';
+  t['\uFD29'] = '\u0634\u0631';
+  t['\uFD2A'] = '\u0633\u0631';
+  t['\uFD2B'] = '\u0635\u0631';
+  t['\uFD2C'] = '\u0636\u0631';
+  t['\uFD2D'] = '\u0634\u062C';
+  t['\uFD2E'] = '\u0634\u062D';
+  t['\uFD2F'] = '\u0634\u062E';
+  t['\uFD30'] = '\u0634\u0645';
+  t['\uFD31'] = '\u0633\u0647';
+  t['\uFD32'] = '\u0634\u0647';
+  t['\uFD33'] = '\u0637\u0645';
+  t['\uFD34'] = '\u0633\u062C';
+  t['\uFD35'] = '\u0633\u062D';
+  t['\uFD36'] = '\u0633\u062E';
+  t['\uFD37'] = '\u0634\u062C';
+  t['\uFD38'] = '\u0634\u062D';
+  t['\uFD39'] = '\u0634\u062E';
+  t['\uFD3A'] = '\u0637\u0645';
+  t['\uFD3B'] = '\u0638\u0645';
+  t['\uFD3C'] = '\u0627\u064B';
+  t['\uFD3D'] = '\u0627\u064B';
+  t['\uFD50'] = '\u062A\u062C\u0645';
+  t['\uFD51'] = '\u062A\u062D\u062C';
+  t['\uFD52'] = '\u062A\u062D\u062C';
+  t['\uFD53'] = '\u062A\u062D\u0645';
+  t['\uFD54'] = '\u062A\u062E\u0645';
+  t['\uFD55'] = '\u062A\u0645\u062C';
+  t['\uFD56'] = '\u062A\u0645\u062D';
+  t['\uFD57'] = '\u062A\u0645\u062E';
+  t['\uFD58'] = '\u062C\u0645\u062D';
+  t['\uFD59'] = '\u062C\u0645\u062D';
+  t['\uFD5A'] = '\u062D\u0645\u064A';
+  t['\uFD5B'] = '\u062D\u0645\u0649';
+  t['\uFD5C'] = '\u0633\u062D\u062C';
+  t['\uFD5D'] = '\u0633\u062C\u062D';
+  t['\uFD5E'] = '\u0633\u062C\u0649';
+  t['\uFD5F'] = '\u0633\u0645\u062D';
+  t['\uFD60'] = '\u0633\u0645\u062D';
+  t['\uFD61'] = '\u0633\u0645\u062C';
+  t['\uFD62'] = '\u0633\u0645\u0645';
+  t['\uFD63'] = '\u0633\u0645\u0645';
+  t['\uFD64'] = '\u0635\u062D\u062D';
+  t['\uFD65'] = '\u0635\u062D\u062D';
+  t['\uFD66'] = '\u0635\u0645\u0645';
+  t['\uFD67'] = '\u0634\u062D\u0645';
+  t['\uFD68'] = '\u0634\u062D\u0645';
+  t['\uFD69'] = '\u0634\u062C\u064A';
+  t['\uFD6A'] = '\u0634\u0645\u062E';
+  t['\uFD6B'] = '\u0634\u0645\u062E';
+  t['\uFD6C'] = '\u0634\u0645\u0645';
+  t['\uFD6D'] = '\u0634\u0645\u0645';
+  t['\uFD6E'] = '\u0636\u062D\u0649';
+  t['\uFD6F'] = '\u0636\u062E\u0645';
+  t['\uFD70'] = '\u0636\u062E\u0645';
+  t['\uFD71'] = '\u0637\u0645\u062D';
+  t['\uFD72'] = '\u0637\u0645\u062D';
+  t['\uFD73'] = '\u0637\u0645\u0645';
+  t['\uFD74'] = '\u0637\u0645\u064A';
+  t['\uFD75'] = '\u0639\u062C\u0645';
+  t['\uFD76'] = '\u0639\u0645\u0645';
+  t['\uFD77'] = '\u0639\u0645\u0645';
+  t['\uFD78'] = '\u0639\u0645\u0649';
+  t['\uFD79'] = '\u063A\u0645\u0645';
+  t['\uFD7A'] = '\u063A\u0645\u064A';
+  t['\uFD7B'] = '\u063A\u0645\u0649';
+  t['\uFD7C'] = '\u0641\u062E\u0645';
+  t['\uFD7D'] = '\u0641\u062E\u0645';
+  t['\uFD7E'] = '\u0642\u0645\u062D';
+  t['\uFD7F'] = '\u0642\u0645\u0645';
+  t['\uFD80'] = '\u0644\u062D\u0645';
+  t['\uFD81'] = '\u0644\u062D\u064A';
+  t['\uFD82'] = '\u0644\u062D\u0649';
+  t['\uFD83'] = '\u0644\u062C\u062C';
+  t['\uFD84'] = '\u0644\u062C\u062C';
+  t['\uFD85'] = '\u0644\u062E\u0645';
+  t['\uFD86'] = '\u0644\u062E\u0645';
+  t['\uFD87'] = '\u0644\u0645\u062D';
+  t['\uFD88'] = '\u0644\u0645\u062D';
+  t['\uFD89'] = '\u0645\u062D\u062C';
+  t['\uFD8A'] = '\u0645\u062D\u0645';
+  t['\uFD8B'] = '\u0645\u062D\u064A';
+  t['\uFD8C'] = '\u0645\u062C\u062D';
+  t['\uFD8D'] = '\u0645\u062C\u0645';
+  t['\uFD8E'] = '\u0645\u062E\u062C';
+  t['\uFD8F'] = '\u0645\u062E\u0645';
+  t['\uFD92'] = '\u0645\u062C\u062E';
+  t['\uFD93'] = '\u0647\u0645\u062C';
+  t['\uFD94'] = '\u0647\u0645\u0645';
+  t['\uFD95'] = '\u0646\u062D\u0645';
+  t['\uFD96'] = '\u0646\u062D\u0649';
+  t['\uFD97'] = '\u0646\u062C\u0645';
+  t['\uFD98'] = '\u0646\u062C\u0645';
+  t['\uFD99'] = '\u0646\u062C\u0649';
+  t['\uFD9A'] = '\u0646\u0645\u064A';
+  t['\uFD9B'] = '\u0646\u0645\u0649';
+  t['\uFD9C'] = '\u064A\u0645\u0645';
+  t['\uFD9D'] = '\u064A\u0645\u0645';
+  t['\uFD9E'] = '\u0628\u062E\u064A';
+  t['\uFD9F'] = '\u062A\u062C\u064A';
+  t['\uFDA0'] = '\u062A\u062C\u0649';
+  t['\uFDA1'] = '\u062A\u062E\u064A';
+  t['\uFDA2'] = '\u062A\u062E\u0649';
+  t['\uFDA3'] = '\u062A\u0645\u064A';
+  t['\uFDA4'] = '\u062A\u0645\u0649';
+  t['\uFDA5'] = '\u062C\u0645\u064A';
+  t['\uFDA6'] = '\u062C\u062D\u0649';
+  t['\uFDA7'] = '\u062C\u0645\u0649';
+  t['\uFDA8'] = '\u0633\u062E\u0649';
+  t['\uFDA9'] = '\u0635\u062D\u064A';
+  t['\uFDAA'] = '\u0634\u062D\u064A';
+  t['\uFDAB'] = '\u0636\u062D\u064A';
+  t['\uFDAC'] = '\u0644\u062C\u064A';
+  t['\uFDAD'] = '\u0644\u0645\u064A';
+  t['\uFDAE'] = '\u064A\u062D\u064A';
+  t['\uFDAF'] = '\u064A\u062C\u064A';
+  t['\uFDB0'] = '\u064A\u0645\u064A';
+  t['\uFDB1'] = '\u0645\u0645\u064A';
+  t['\uFDB2'] = '\u0642\u0645\u064A';
+  t['\uFDB3'] = '\u0646\u062D\u064A';
+  t['\uFDB4'] = '\u0642\u0645\u062D';
+  t['\uFDB5'] = '\u0644\u062D\u0645';
+  t['\uFDB6'] = '\u0639\u0645\u064A';
+  t['\uFDB7'] = '\u0643\u0645\u064A';
+  t['\uFDB8'] = '\u0646\u062C\u062D';
+  t['\uFDB9'] = '\u0645\u062E\u064A';
+  t['\uFDBA'] = '\u0644\u062C\u0645';
+  t['\uFDBB'] = '\u0643\u0645\u0645';
+  t['\uFDBC'] = '\u0644\u062C\u0645';
+  t['\uFDBD'] = '\u0646\u062C\u062D';
+  t['\uFDBE'] = '\u062C\u062D\u064A';
+  t['\uFDBF'] = '\u062D\u062C\u064A';
+  t['\uFDC0'] = '\u0645\u062C\u064A';
+  t['\uFDC1'] = '\u0641\u0645\u064A';
+  t['\uFDC2'] = '\u0628\u062D\u064A';
+  t['\uFDC3'] = '\u0643\u0645\u0645';
+  t['\uFDC4'] = '\u0639\u062C\u0645';
+  t['\uFDC5'] = '\u0635\u0645\u0645';
+  t['\uFDC6'] = '\u0633\u062E\u064A';
+  t['\uFDC7'] = '\u0646\u062C\u064A';
+  t['\uFE49'] = '\u203E';
+  t['\uFE4A'] = '\u203E';
+  t['\uFE4B'] = '\u203E';
+  t['\uFE4C'] = '\u203E';
+  t['\uFE4D'] = '\u005F';
+  t['\uFE4E'] = '\u005F';
+  t['\uFE4F'] = '\u005F';
+  t['\uFE80'] = '\u0621';
+  t['\uFE81'] = '\u0622';
+  t['\uFE82'] = '\u0622';
+  t['\uFE83'] = '\u0623';
+  t['\uFE84'] = '\u0623';
+  t['\uFE85'] = '\u0624';
+  t['\uFE86'] = '\u0624';
+  t['\uFE87'] = '\u0625';
+  t['\uFE88'] = '\u0625';
+  t['\uFE89'] = '\u0626';
+  t['\uFE8A'] = '\u0626';
+  t['\uFE8B'] = '\u0626';
+  t['\uFE8C'] = '\u0626';
+  t['\uFE8D'] = '\u0627';
+  t['\uFE8E'] = '\u0627';
+  t['\uFE8F'] = '\u0628';
+  t['\uFE90'] = '\u0628';
+  t['\uFE91'] = '\u0628';
+  t['\uFE92'] = '\u0628';
+  t['\uFE93'] = '\u0629';
+  t['\uFE94'] = '\u0629';
+  t['\uFE95'] = '\u062A';
+  t['\uFE96'] = '\u062A';
+  t['\uFE97'] = '\u062A';
+  t['\uFE98'] = '\u062A';
+  t['\uFE99'] = '\u062B';
+  t['\uFE9A'] = '\u062B';
+  t['\uFE9B'] = '\u062B';
+  t['\uFE9C'] = '\u062B';
+  t['\uFE9D'] = '\u062C';
+  t['\uFE9E'] = '\u062C';
+  t['\uFE9F'] = '\u062C';
+  t['\uFEA0'] = '\u062C';
+  t['\uFEA1'] = '\u062D';
+  t['\uFEA2'] = '\u062D';
+  t['\uFEA3'] = '\u062D';
+  t['\uFEA4'] = '\u062D';
+  t['\uFEA5'] = '\u062E';
+  t['\uFEA6'] = '\u062E';
+  t['\uFEA7'] = '\u062E';
+  t['\uFEA8'] = '\u062E';
+  t['\uFEA9'] = '\u062F';
+  t['\uFEAA'] = '\u062F';
+  t['\uFEAB'] = '\u0630';
+  t['\uFEAC'] = '\u0630';
+  t['\uFEAD'] = '\u0631';
+  t['\uFEAE'] = '\u0631';
+  t['\uFEAF'] = '\u0632';
+  t['\uFEB0'] = '\u0632';
+  t['\uFEB1'] = '\u0633';
+  t['\uFEB2'] = '\u0633';
+  t['\uFEB3'] = '\u0633';
+  t['\uFEB4'] = '\u0633';
+  t['\uFEB5'] = '\u0634';
+  t['\uFEB6'] = '\u0634';
+  t['\uFEB7'] = '\u0634';
+  t['\uFEB8'] = '\u0634';
+  t['\uFEB9'] = '\u0635';
+  t['\uFEBA'] = '\u0635';
+  t['\uFEBB'] = '\u0635';
+  t['\uFEBC'] = '\u0635';
+  t['\uFEBD'] = '\u0636';
+  t['\uFEBE'] = '\u0636';
+  t['\uFEBF'] = '\u0636';
+  t['\uFEC0'] = '\u0636';
+  t['\uFEC1'] = '\u0637';
+  t['\uFEC2'] = '\u0637';
+  t['\uFEC3'] = '\u0637';
+  t['\uFEC4'] = '\u0637';
+  t['\uFEC5'] = '\u0638';
+  t['\uFEC6'] = '\u0638';
+  t['\uFEC7'] = '\u0638';
+  t['\uFEC8'] = '\u0638';
+  t['\uFEC9'] = '\u0639';
+  t['\uFECA'] = '\u0639';
+  t['\uFECB'] = '\u0639';
+  t['\uFECC'] = '\u0639';
+  t['\uFECD'] = '\u063A';
+  t['\uFECE'] = '\u063A';
+  t['\uFECF'] = '\u063A';
+  t['\uFED0'] = '\u063A';
+  t['\uFED1'] = '\u0641';
+  t['\uFED2'] = '\u0641';
+  t['\uFED3'] = '\u0641';
+  t['\uFED4'] = '\u0641';
+  t['\uFED5'] = '\u0642';
+  t['\uFED6'] = '\u0642';
+  t['\uFED7'] = '\u0642';
+  t['\uFED8'] = '\u0642';
+  t['\uFED9'] = '\u0643';
+  t['\uFEDA'] = '\u0643';
+  t['\uFEDB'] = '\u0643';
+  t['\uFEDC'] = '\u0643';
+  t['\uFEDD'] = '\u0644';
+  t['\uFEDE'] = '\u0644';
+  t['\uFEDF'] = '\u0644';
+  t['\uFEE0'] = '\u0644';
+  t['\uFEE1'] = '\u0645';
+  t['\uFEE2'] = '\u0645';
+  t['\uFEE3'] = '\u0645';
+  t['\uFEE4'] = '\u0645';
+  t['\uFEE5'] = '\u0646';
+  t['\uFEE6'] = '\u0646';
+  t['\uFEE7'] = '\u0646';
+  t['\uFEE8'] = '\u0646';
+  t['\uFEE9'] = '\u0647';
+  t['\uFEEA'] = '\u0647';
+  t['\uFEEB'] = '\u0647';
+  t['\uFEEC'] = '\u0647';
+  t['\uFEED'] = '\u0648';
+  t['\uFEEE'] = '\u0648';
+  t['\uFEEF'] = '\u0649';
+  t['\uFEF0'] = '\u0649';
+  t['\uFEF1'] = '\u064A';
+  t['\uFEF2'] = '\u064A';
+  t['\uFEF3'] = '\u064A';
+  t['\uFEF4'] = '\u064A';
+  t['\uFEF5'] = '\u0644\u0622';
+  t['\uFEF6'] = '\u0644\u0622';
+  t['\uFEF7'] = '\u0644\u0623';
+  t['\uFEF8'] = '\u0644\u0623';
+  t['\uFEF9'] = '\u0644\u0625';
+  t['\uFEFA'] = '\u0644\u0625';
+  t['\uFEFB'] = '\u0644\u0627';
+  t['\uFEFC'] = '\u0644\u0627';
 });
 function reverseIfRtl(chars) {
- var charsLength = chars.length;
- if (charsLength <= 1 || !isRTLRangeFor(chars.charCodeAt(0))) {
-  return chars;
- }
- var s = '';
- for (var ii = charsLength - 1; ii >= 0; ii--) {
-  s += chars[ii];
- }
- return s;
+  var charsLength = chars.length;
+  if (charsLength <= 1 || !isRTLRangeFor(chars.charCodeAt(0))) {
+    return chars;
+  }
+  var s = '';
+  for (var ii = charsLength - 1; ii >= 0; ii--) {
+    s += chars[ii];
+  }
+  return s;
 }
 exports.mapSpecialUnicodeValues = mapSpecialUnicodeValues;
 exports.reverseIfRtl = reverseIfRtl;

+ 613 - 623
lib/core/worker.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var corePrimitives = require('./primitives.js');
 var corePdfManager = require('./pdf_manager.js');
@@ -36,660 +37,649 @@ var Ref = corePrimitives.Ref;
 var LocalPdfManager = corePdfManager.LocalPdfManager;
 var NetworkPdfManager = corePdfManager.NetworkPdfManager;
 var WorkerTask = function WorkerTaskClosure() {
- function WorkerTask(name) {
-  this.name = name;
-  this.terminated = false;
-  this._capability = createPromiseCapability();
- }
- WorkerTask.prototype = {
-  get finished() {
-   return this._capability.promise;
-  },
-  finish: function () {
-   this._capability.resolve();
-  },
-  terminate: function () {
-   this.terminated = true;
-  },
-  ensureNotTerminated: function () {
-   if (this.terminated) {
-    throw new Error('Worker task was terminated');
-   }
+  function WorkerTask(name) {
+    this.name = name;
+    this.terminated = false;
+    this._capability = createPromiseCapability();
   }
- };
- return WorkerTask;
+  WorkerTask.prototype = {
+    get finished() {
+      return this._capability.promise;
+    },
+    finish: function () {
+      this._capability.resolve();
+    },
+    terminate: function () {
+      this.terminated = true;
+    },
+    ensureNotTerminated: function () {
+      if (this.terminated) {
+        throw new Error('Worker task was terminated');
+      }
+    }
+  };
+  return WorkerTask;
 }();
 var PDFWorkerStream = function PDFWorkerStreamClosure() {
- function PDFWorkerStream(params, msgHandler) {
-  this._queuedChunks = [];
-  var initialData = params.initialData;
-  if (initialData && initialData.length > 0) {
-   this._queuedChunks.push(initialData);
-  }
-  this._msgHandler = msgHandler;
-  this._isRangeSupported = !params.disableRange;
-  this._isStreamingSupported = !params.disableStream;
-  this._contentLength = params.length;
-  this._fullRequestReader = null;
-  this._rangeReaders = [];
-  msgHandler.on('OnDataRange', this._onReceiveData.bind(this));
-  msgHandler.on('OnDataProgress', this._onProgress.bind(this));
- }
- PDFWorkerStream.prototype = {
-  _onReceiveData: function PDFWorkerStream_onReceiveData(args) {
-   if (args.begin === undefined) {
-    if (this._fullRequestReader) {
-     this._fullRequestReader._enqueue(args.chunk);
-    } else {
-     this._queuedChunks.push(args.chunk);
+  function PDFWorkerStream(params, msgHandler) {
+    this._queuedChunks = [];
+    var initialData = params.initialData;
+    if (initialData && initialData.length > 0) {
+      this._queuedChunks.push(initialData);
     }
-   } else {
-    var found = this._rangeReaders.some(function (rangeReader) {
-     if (rangeReader._begin !== args.begin) {
-      return false;
-     }
-     rangeReader._enqueue(args.chunk);
-     return true;
-    });
-    assert(found);
-   }
-  },
-  _onProgress: function PDFWorkerStream_onProgress(evt) {
-   if (this._rangeReaders.length > 0) {
-    var firstReader = this._rangeReaders[0];
-    if (firstReader.onProgress) {
-     firstReader.onProgress({ loaded: evt.loaded });
-    }
-   }
-  },
-  _removeRangeReader: function PDFWorkerStream_removeRangeReader(reader) {
-   var i = this._rangeReaders.indexOf(reader);
-   if (i >= 0) {
-    this._rangeReaders.splice(i, 1);
-   }
-  },
-  getFullReader: function PDFWorkerStream_getFullReader() {
-   assert(!this._fullRequestReader);
-   var queuedChunks = this._queuedChunks;
-   this._queuedChunks = null;
-   return new PDFWorkerStreamReader(this, queuedChunks);
-  },
-  getRangeReader: function PDFWorkerStream_getRangeReader(begin, end) {
-   var reader = new PDFWorkerStreamRangeReader(this, begin, end);
-   this._msgHandler.send('RequestDataRange', {
-    begin: begin,
-    end: end
-   });
-   this._rangeReaders.push(reader);
-   return reader;
-  },
-  cancelAllRequests: function PDFWorkerStream_cancelAllRequests(reason) {
-   if (this._fullRequestReader) {
-    this._fullRequestReader.cancel(reason);
-   }
-   var readers = this._rangeReaders.slice(0);
-   readers.forEach(function (rangeReader) {
-    rangeReader.cancel(reason);
-   });
+    this._msgHandler = msgHandler;
+    this._isRangeSupported = !params.disableRange;
+    this._isStreamingSupported = !params.disableStream;
+    this._contentLength = params.length;
+    this._fullRequestReader = null;
+    this._rangeReaders = [];
+    msgHandler.on('OnDataRange', this._onReceiveData.bind(this));
+    msgHandler.on('OnDataProgress', this._onProgress.bind(this));
   }
- };
- function PDFWorkerStreamReader(stream, queuedChunks) {
-  this._stream = stream;
-  this._done = false;
-  this._queuedChunks = queuedChunks || [];
-  this._requests = [];
-  this._headersReady = Promise.resolve();
-  stream._fullRequestReader = this;
-  this.onProgress = null;
- }
- PDFWorkerStreamReader.prototype = {
-  _enqueue: function PDFWorkerStreamReader_enqueue(chunk) {
-   if (this._done) {
-    return;
-   }
-   if (this._requests.length > 0) {
-    var requestCapability = this._requests.shift();
-    requestCapability.resolve({
-     value: chunk,
-     done: false
-    });
-    return;
-   }
-   this._queuedChunks.push(chunk);
-  },
-  get headersReady() {
-   return this._headersReady;
-  },
-  get isRangeSupported() {
-   return this._stream._isRangeSupported;
-  },
-  get isStreamingSupported() {
-   return this._stream._isStreamingSupported;
-  },
-  get contentLength() {
-   return this._stream._contentLength;
-  },
-  read: function PDFWorkerStreamReader_read() {
-   if (this._queuedChunks.length > 0) {
-    var chunk = this._queuedChunks.shift();
-    return Promise.resolve({
-     value: chunk,
-     done: false
-    });
-   }
-   if (this._done) {
-    return Promise.resolve({
-     value: undefined,
-     done: true
-    });
-   }
-   var requestCapability = createPromiseCapability();
-   this._requests.push(requestCapability);
-   return requestCapability.promise;
-  },
-  cancel: function PDFWorkerStreamReader_cancel(reason) {
-   this._done = true;
-   this._requests.forEach(function (requestCapability) {
-    requestCapability.resolve({
-     value: undefined,
-     done: true
-    });
-   });
-   this._requests = [];
+  PDFWorkerStream.prototype = {
+    _onReceiveData: function PDFWorkerStream_onReceiveData(args) {
+      if (args.begin === undefined) {
+        if (this._fullRequestReader) {
+          this._fullRequestReader._enqueue(args.chunk);
+        } else {
+          this._queuedChunks.push(args.chunk);
+        }
+      } else {
+        var found = this._rangeReaders.some(function (rangeReader) {
+          if (rangeReader._begin !== args.begin) {
+            return false;
+          }
+          rangeReader._enqueue(args.chunk);
+          return true;
+        });
+        assert(found);
+      }
+    },
+    _onProgress: function PDFWorkerStream_onProgress(evt) {
+      if (this._rangeReaders.length > 0) {
+        var firstReader = this._rangeReaders[0];
+        if (firstReader.onProgress) {
+          firstReader.onProgress({ loaded: evt.loaded });
+        }
+      }
+    },
+    _removeRangeReader: function PDFWorkerStream_removeRangeReader(reader) {
+      var i = this._rangeReaders.indexOf(reader);
+      if (i >= 0) {
+        this._rangeReaders.splice(i, 1);
+      }
+    },
+    getFullReader: function PDFWorkerStream_getFullReader() {
+      assert(!this._fullRequestReader);
+      var queuedChunks = this._queuedChunks;
+      this._queuedChunks = null;
+      return new PDFWorkerStreamReader(this, queuedChunks);
+    },
+    getRangeReader: function PDFWorkerStream_getRangeReader(begin, end) {
+      var reader = new PDFWorkerStreamRangeReader(this, begin, end);
+      this._msgHandler.send('RequestDataRange', {
+        begin: begin,
+        end: end
+      });
+      this._rangeReaders.push(reader);
+      return reader;
+    },
+    cancelAllRequests: function PDFWorkerStream_cancelAllRequests(reason) {
+      if (this._fullRequestReader) {
+        this._fullRequestReader.cancel(reason);
+      }
+      var readers = this._rangeReaders.slice(0);
+      readers.forEach(function (rangeReader) {
+        rangeReader.cancel(reason);
+      });
+    }
+  };
+  function PDFWorkerStreamReader(stream, queuedChunks) {
+    this._stream = stream;
+    this._done = false;
+    this._queuedChunks = queuedChunks || [];
+    this._requests = [];
+    this._headersReady = Promise.resolve();
+    stream._fullRequestReader = this;
+    this.onProgress = null;
   }
- };
- function PDFWorkerStreamRangeReader(stream, begin, end) {
-  this._stream = stream;
-  this._begin = begin;
-  this._end = end;
-  this._queuedChunk = null;
-  this._requests = [];
-  this._done = false;
-  this.onProgress = null;
- }
- PDFWorkerStreamRangeReader.prototype = {
-  _enqueue: function PDFWorkerStreamRangeReader_enqueue(chunk) {
-   if (this._done) {
-    return;
-   }
-   if (this._requests.length === 0) {
-    this._queuedChunk = chunk;
-   } else {
-    var requestsCapability = this._requests.shift();
-    requestsCapability.resolve({
-     value: chunk,
-     done: false
-    });
-    this._requests.forEach(function (requestCapability) {
-     requestCapability.resolve({
-      value: undefined,
-      done: true
-     });
-    });
+  PDFWorkerStreamReader.prototype = {
+    _enqueue: function PDFWorkerStreamReader_enqueue(chunk) {
+      if (this._done) {
+        return;
+      }
+      if (this._requests.length > 0) {
+        var requestCapability = this._requests.shift();
+        requestCapability.resolve({
+          value: chunk,
+          done: false
+        });
+        return;
+      }
+      this._queuedChunks.push(chunk);
+    },
+    get headersReady() {
+      return this._headersReady;
+    },
+    get isRangeSupported() {
+      return this._stream._isRangeSupported;
+    },
+    get isStreamingSupported() {
+      return this._stream._isStreamingSupported;
+    },
+    get contentLength() {
+      return this._stream._contentLength;
+    },
+    read: function PDFWorkerStreamReader_read() {
+      if (this._queuedChunks.length > 0) {
+        var chunk = this._queuedChunks.shift();
+        return Promise.resolve({
+          value: chunk,
+          done: false
+        });
+      }
+      if (this._done) {
+        return Promise.resolve({
+          value: undefined,
+          done: true
+        });
+      }
+      var requestCapability = createPromiseCapability();
+      this._requests.push(requestCapability);
+      return requestCapability.promise;
+    },
+    cancel: function PDFWorkerStreamReader_cancel(reason) {
+      this._done = true;
+      this._requests.forEach(function (requestCapability) {
+        requestCapability.resolve({
+          value: undefined,
+          done: true
+        });
+      });
+      this._requests = [];
+    }
+  };
+  function PDFWorkerStreamRangeReader(stream, begin, end) {
+    this._stream = stream;
+    this._begin = begin;
+    this._end = end;
+    this._queuedChunk = null;
     this._requests = [];
-   }
-   this._done = true;
-   this._stream._removeRangeReader(this);
-  },
-  get isStreamingSupported() {
-   return false;
-  },
-  read: function PDFWorkerStreamRangeReader_read() {
-   if (this._queuedChunk) {
-    return Promise.resolve({
-     value: this._queuedChunk,
-     done: false
-    });
-   }
-   if (this._done) {
-    return Promise.resolve({
-     value: undefined,
-     done: true
-    });
-   }
-   var requestCapability = createPromiseCapability();
-   this._requests.push(requestCapability);
-   return requestCapability.promise;
-  },
-  cancel: function PDFWorkerStreamRangeReader_cancel(reason) {
-   this._done = true;
-   this._requests.forEach(function (requestCapability) {
-    requestCapability.resolve({
-     value: undefined,
-     done: true
-    });
-   });
-   this._requests = [];
-   this._stream._removeRangeReader(this);
+    this._done = false;
+    this.onProgress = null;
   }
- };
- return PDFWorkerStream;
+  PDFWorkerStreamRangeReader.prototype = {
+    _enqueue: function PDFWorkerStreamRangeReader_enqueue(chunk) {
+      if (this._done) {
+        return;
+      }
+      if (this._requests.length === 0) {
+        this._queuedChunk = chunk;
+      } else {
+        var requestsCapability = this._requests.shift();
+        requestsCapability.resolve({
+          value: chunk,
+          done: false
+        });
+        this._requests.forEach(function (requestCapability) {
+          requestCapability.resolve({
+            value: undefined,
+            done: true
+          });
+        });
+        this._requests = [];
+      }
+      this._done = true;
+      this._stream._removeRangeReader(this);
+    },
+    get isStreamingSupported() {
+      return false;
+    },
+    read: function PDFWorkerStreamRangeReader_read() {
+      if (this._queuedChunk) {
+        return Promise.resolve({
+          value: this._queuedChunk,
+          done: false
+        });
+      }
+      if (this._done) {
+        return Promise.resolve({
+          value: undefined,
+          done: true
+        });
+      }
+      var requestCapability = createPromiseCapability();
+      this._requests.push(requestCapability);
+      return requestCapability.promise;
+    },
+    cancel: function PDFWorkerStreamRangeReader_cancel(reason) {
+      this._done = true;
+      this._requests.forEach(function (requestCapability) {
+        requestCapability.resolve({
+          value: undefined,
+          done: true
+        });
+      });
+      this._requests = [];
+      this._stream._removeRangeReader(this);
+    }
+  };
+  return PDFWorkerStream;
 }();
 var PDFNetworkStream;
 function setPDFNetworkStreamClass(cls) {
- PDFNetworkStream = cls;
+  PDFNetworkStream = cls;
 }
 var WorkerMessageHandler = {
- setup: function wphSetup(handler, port) {
-  var testMessageProcessed = false;
-  handler.on('test', function wphSetupTest(data) {
-   if (testMessageProcessed) {
-    return;
-   }
-   testMessageProcessed = true;
-   if (!(data instanceof Uint8Array)) {
-    handler.send('test', 'main', false);
-    return;
-   }
-   var supportTransfers = data[0] === 255;
-   handler.postMessageTransfers = supportTransfers;
-   var xhr = new XMLHttpRequest();
-   var responseExists = 'response' in xhr;
-   try {
-    xhr.responseType;
-   } catch (e) {
-    responseExists = false;
-   }
-   if (!responseExists) {
-    handler.send('test', false);
-    return;
-   }
-   handler.send('test', {
-    supportTypedArray: true,
-    supportTransfers: supportTransfers
-   });
-  });
-  handler.on('configure', function wphConfigure(data) {
-   setVerbosityLevel(data.verbosity);
-  });
-  handler.on('GetDocRequest', function wphSetupDoc(data) {
-   return WorkerMessageHandler.createDocumentHandler(data, port);
-  });
- },
- createDocumentHandler: function wphCreateDocumentHandler(docParams, port) {
-  var pdfManager;
-  var terminated = false;
-  var cancelXHRs = null;
-  var WorkerTasks = [];
-  var docId = docParams.docId;
-  var docBaseUrl = docParams.docBaseUrl;
-  var workerHandlerName = docParams.docId + '_worker';
-  var handler = new MessageHandler(workerHandlerName, docId, port);
-  handler.postMessageTransfers = docParams.postMessageTransfers;
-  function ensureNotTerminated() {
-   if (terminated) {
-    throw new Error('Worker was terminated');
-   }
-  }
-  function startWorkerTask(task) {
-   WorkerTasks.push(task);
-  }
-  function finishWorkerTask(task) {
-   task.finish();
-   var i = WorkerTasks.indexOf(task);
-   WorkerTasks.splice(i, 1);
-  }
-  function loadDocument(recoveryMode) {
-   var loadDocumentCapability = createPromiseCapability();
-   var parseSuccess = function parseSuccess() {
-    var numPagesPromise = pdfManager.ensureDoc('numPages');
-    var fingerprintPromise = pdfManager.ensureDoc('fingerprint');
-    var encryptedPromise = pdfManager.ensureXRef('encrypt');
-    Promise.all([
-     numPagesPromise,
-     fingerprintPromise,
-     encryptedPromise
-    ]).then(function onDocReady(results) {
-     var doc = {
-      numPages: results[0],
-      fingerprint: results[1],
-      encrypted: !!results[2]
-     };
-     loadDocumentCapability.resolve(doc);
-    }, parseFailure);
-   };
-   var parseFailure = function parseFailure(e) {
-    loadDocumentCapability.reject(e);
-   };
-   pdfManager.ensureDoc('checkHeader', []).then(function () {
-    pdfManager.ensureDoc('parseStartXRef', []).then(function () {
-     pdfManager.ensureDoc('parse', [recoveryMode]).then(parseSuccess, parseFailure);
-    }, parseFailure);
-   }, parseFailure);
-   return loadDocumentCapability.promise;
-  }
-  function getPdfManager(data, evaluatorOptions) {
-   var pdfManagerCapability = createPromiseCapability();
-   var pdfManager;
-   var source = data.source;
-   if (source.data) {
-    try {
-     pdfManager = new LocalPdfManager(docId, source.data, source.password, evaluatorOptions, docBaseUrl);
-     pdfManagerCapability.resolve(pdfManager);
-    } catch (ex) {
-     pdfManagerCapability.reject(ex);
-    }
-    return pdfManagerCapability.promise;
-   }
-   var pdfStream;
-   try {
-    if (source.chunkedViewerLoading) {
-     pdfStream = new PDFWorkerStream(source, handler);
-    } else {
-     assert(PDFNetworkStream, 'pdfjs/core/network module is not loaded');
-     pdfStream = new PDFNetworkStream(data);
-    }
-   } catch (ex) {
-    pdfManagerCapability.reject(ex);
-    return pdfManagerCapability.promise;
-   }
-   var fullRequest = pdfStream.getFullReader();
-   fullRequest.headersReady.then(function () {
-    if (!fullRequest.isStreamingSupported || !fullRequest.isRangeSupported) {
-     fullRequest.onProgress = function (evt) {
-      handler.send('DocProgress', {
-       loaded: evt.loaded,
-       total: evt.total
+  setup: function wphSetup(handler, port) {
+    var testMessageProcessed = false;
+    handler.on('test', function wphSetupTest(data) {
+      if (testMessageProcessed) {
+        return;
+      }
+      testMessageProcessed = true;
+      if (!(data instanceof Uint8Array)) {
+        handler.send('test', 'main', false);
+        return;
+      }
+      var supportTransfers = data[0] === 255;
+      handler.postMessageTransfers = supportTransfers;
+      var xhr = new XMLHttpRequest();
+      var responseExists = 'response' in xhr;
+      try {
+        xhr.responseType;
+      } catch (e) {
+        responseExists = false;
+      }
+      if (!responseExists) {
+        handler.send('test', false);
+        return;
+      }
+      handler.send('test', {
+        supportTypedArray: true,
+        supportTransfers: supportTransfers
       });
-     };
+    });
+    handler.on('configure', function wphConfigure(data) {
+      setVerbosityLevel(data.verbosity);
+    });
+    handler.on('GetDocRequest', function wphSetupDoc(data) {
+      return WorkerMessageHandler.createDocumentHandler(data, port);
+    });
+  },
+  createDocumentHandler: function wphCreateDocumentHandler(docParams, port) {
+    var pdfManager;
+    var terminated = false;
+    var cancelXHRs = null;
+    var WorkerTasks = [];
+    var docId = docParams.docId;
+    var docBaseUrl = docParams.docBaseUrl;
+    var workerHandlerName = docParams.docId + '_worker';
+    var handler = new MessageHandler(workerHandlerName, docId, port);
+    handler.postMessageTransfers = docParams.postMessageTransfers;
+    function ensureNotTerminated() {
+      if (terminated) {
+        throw new Error('Worker was terminated');
+      }
+    }
+    function startWorkerTask(task) {
+      WorkerTasks.push(task);
     }
-    if (!fullRequest.isRangeSupported) {
-     return;
+    function finishWorkerTask(task) {
+      task.finish();
+      var i = WorkerTasks.indexOf(task);
+      WorkerTasks.splice(i, 1);
     }
-    var disableAutoFetch = source.disableAutoFetch || fullRequest.isStreamingSupported;
-    pdfManager = new NetworkPdfManager(docId, pdfStream, {
-     msgHandler: handler,
-     url: source.url,
-     password: source.password,
-     length: fullRequest.contentLength,
-     disableAutoFetch: disableAutoFetch,
-     rangeChunkSize: source.rangeChunkSize
-    }, evaluatorOptions, docBaseUrl);
-    pdfManagerCapability.resolve(pdfManager);
-    cancelXHRs = null;
-   }).catch(function (reason) {
-    pdfManagerCapability.reject(reason);
-    cancelXHRs = null;
-   });
-   var cachedChunks = [], loaded = 0;
-   var flushChunks = function () {
-    var pdfFile = arraysToBytes(cachedChunks);
-    if (source.length && pdfFile.length !== source.length) {
-     warn('reported HTTP length is different from actual');
+    function loadDocument(recoveryMode) {
+      var loadDocumentCapability = createPromiseCapability();
+      var parseSuccess = function parseSuccess() {
+        var numPagesPromise = pdfManager.ensureDoc('numPages');
+        var fingerprintPromise = pdfManager.ensureDoc('fingerprint');
+        var encryptedPromise = pdfManager.ensureXRef('encrypt');
+        Promise.all([numPagesPromise, fingerprintPromise, encryptedPromise]).then(function onDocReady(results) {
+          var doc = {
+            numPages: results[0],
+            fingerprint: results[1],
+            encrypted: !!results[2]
+          };
+          loadDocumentCapability.resolve(doc);
+        }, parseFailure);
+      };
+      var parseFailure = function parseFailure(e) {
+        loadDocumentCapability.reject(e);
+      };
+      pdfManager.ensureDoc('checkHeader', []).then(function () {
+        pdfManager.ensureDoc('parseStartXRef', []).then(function () {
+          pdfManager.ensureDoc('parse', [recoveryMode]).then(parseSuccess, parseFailure);
+        }, parseFailure);
+      }, parseFailure);
+      return loadDocumentCapability.promise;
     }
-    try {
-     pdfManager = new LocalPdfManager(docId, pdfFile, source.password, evaluatorOptions, docBaseUrl);
-     pdfManagerCapability.resolve(pdfManager);
-    } catch (ex) {
-     pdfManagerCapability.reject(ex);
+    function getPdfManager(data, evaluatorOptions) {
+      var pdfManagerCapability = createPromiseCapability();
+      var pdfManager;
+      var source = data.source;
+      if (source.data) {
+        try {
+          pdfManager = new LocalPdfManager(docId, source.data, source.password, evaluatorOptions, docBaseUrl);
+          pdfManagerCapability.resolve(pdfManager);
+        } catch (ex) {
+          pdfManagerCapability.reject(ex);
+        }
+        return pdfManagerCapability.promise;
+      }
+      var pdfStream;
+      try {
+        if (source.chunkedViewerLoading) {
+          pdfStream = new PDFWorkerStream(source, handler);
+        } else {
+          assert(PDFNetworkStream, 'pdfjs/core/network module is not loaded');
+          pdfStream = new PDFNetworkStream(data);
+        }
+      } catch (ex) {
+        pdfManagerCapability.reject(ex);
+        return pdfManagerCapability.promise;
+      }
+      var fullRequest = pdfStream.getFullReader();
+      fullRequest.headersReady.then(function () {
+        if (!fullRequest.isStreamingSupported || !fullRequest.isRangeSupported) {
+          fullRequest.onProgress = function (evt) {
+            handler.send('DocProgress', {
+              loaded: evt.loaded,
+              total: evt.total
+            });
+          };
+        }
+        if (!fullRequest.isRangeSupported) {
+          return;
+        }
+        var disableAutoFetch = source.disableAutoFetch || fullRequest.isStreamingSupported;
+        pdfManager = new NetworkPdfManager(docId, pdfStream, {
+          msgHandler: handler,
+          url: source.url,
+          password: source.password,
+          length: fullRequest.contentLength,
+          disableAutoFetch: disableAutoFetch,
+          rangeChunkSize: source.rangeChunkSize
+        }, evaluatorOptions, docBaseUrl);
+        pdfManagerCapability.resolve(pdfManager);
+        cancelXHRs = null;
+      }).catch(function (reason) {
+        pdfManagerCapability.reject(reason);
+        cancelXHRs = null;
+      });
+      var cachedChunks = [],
+          loaded = 0;
+      var flushChunks = function () {
+        var pdfFile = arraysToBytes(cachedChunks);
+        if (source.length && pdfFile.length !== source.length) {
+          warn('reported HTTP length is different from actual');
+        }
+        try {
+          pdfManager = new LocalPdfManager(docId, pdfFile, source.password, evaluatorOptions, docBaseUrl);
+          pdfManagerCapability.resolve(pdfManager);
+        } catch (ex) {
+          pdfManagerCapability.reject(ex);
+        }
+        cachedChunks = [];
+      };
+      var readPromise = new Promise(function (resolve, reject) {
+        var readChunk = function (chunk) {
+          try {
+            ensureNotTerminated();
+            if (chunk.done) {
+              if (!pdfManager) {
+                flushChunks();
+              }
+              cancelXHRs = null;
+              return;
+            }
+            var data = chunk.value;
+            loaded += arrayByteLength(data);
+            if (!fullRequest.isStreamingSupported) {
+              handler.send('DocProgress', {
+                loaded: loaded,
+                total: Math.max(loaded, fullRequest.contentLength || 0)
+              });
+            }
+            if (pdfManager) {
+              pdfManager.sendProgressiveData(data);
+            } else {
+              cachedChunks.push(data);
+            }
+            fullRequest.read().then(readChunk, reject);
+          } catch (e) {
+            reject(e);
+          }
+        };
+        fullRequest.read().then(readChunk, reject);
+      });
+      readPromise.catch(function (e) {
+        pdfManagerCapability.reject(e);
+        cancelXHRs = null;
+      });
+      cancelXHRs = function () {
+        pdfStream.cancelAllRequests('abort');
+      };
+      return pdfManagerCapability.promise;
     }
-    cachedChunks = [];
-   };
-   var readPromise = new Promise(function (resolve, reject) {
-    var readChunk = function (chunk) {
-     try {
-      ensureNotTerminated();
-      if (chunk.done) {
-       if (!pdfManager) {
-        flushChunks();
-       }
-       cancelXHRs = null;
-       return;
+    function setupDoc(data) {
+      function onSuccess(doc) {
+        ensureNotTerminated();
+        handler.send('GetDoc', { pdfInfo: doc });
       }
-      var data = chunk.value;
-      loaded += arrayByteLength(data);
-      if (!fullRequest.isStreamingSupported) {
-       handler.send('DocProgress', {
-        loaded: loaded,
-        total: Math.max(loaded, fullRequest.contentLength || 0)
-       });
+      function onFailure(e) {
+        if (e instanceof PasswordException) {
+          var task = new WorkerTask('PasswordException: response ' + e.code);
+          startWorkerTask(task);
+          handler.sendWithPromise('PasswordRequest', e).then(function (data) {
+            finishWorkerTask(task);
+            pdfManager.updatePassword(data.password);
+            pdfManagerReady();
+          }).catch(function (ex) {
+            finishWorkerTask(task);
+            handler.send('PasswordException', ex);
+          }.bind(null, e));
+        } else if (e instanceof InvalidPDFException) {
+          handler.send('InvalidPDF', e);
+        } else if (e instanceof MissingPDFException) {
+          handler.send('MissingPDF', e);
+        } else if (e instanceof UnexpectedResponseException) {
+          handler.send('UnexpectedResponse', e);
+        } else {
+          handler.send('UnknownError', new UnknownErrorException(e.message, e.toString()));
+        }
       }
-      if (pdfManager) {
-       pdfManager.sendProgressiveData(data);
-      } else {
-       cachedChunks.push(data);
+      function pdfManagerReady() {
+        ensureNotTerminated();
+        loadDocument(false).then(onSuccess, function loadFailure(ex) {
+          ensureNotTerminated();
+          if (!(ex instanceof XRefParseException)) {
+            onFailure(ex);
+            return;
+          }
+          pdfManager.requestLoadedStream();
+          pdfManager.onLoadedStream().then(function () {
+            ensureNotTerminated();
+            loadDocument(true).then(onSuccess, onFailure);
+          });
+        }, onFailure);
       }
-      fullRequest.read().then(readChunk, reject);
-     } catch (e) {
-      reject(e);
-     }
-    };
-    fullRequest.read().then(readChunk, reject);
-   });
-   readPromise.catch(function (e) {
-    pdfManagerCapability.reject(e);
-    cancelXHRs = null;
-   });
-   cancelXHRs = function () {
-    pdfStream.cancelAllRequests('abort');
-   };
-   return pdfManagerCapability.promise;
-  }
-  function setupDoc(data) {
-   function onSuccess(doc) {
-    ensureNotTerminated();
-    handler.send('GetDoc', { pdfInfo: doc });
-   }
-   function onFailure(e) {
-    if (e instanceof PasswordException) {
-     var task = new WorkerTask('PasswordException: response ' + e.code);
-     startWorkerTask(task);
-     handler.sendWithPromise('PasswordRequest', e).then(function (data) {
-      finishWorkerTask(task);
-      pdfManager.updatePassword(data.password);
-      pdfManagerReady();
-     }).catch(function (ex) {
-      finishWorkerTask(task);
-      handler.send('PasswordException', ex);
-     }.bind(null, e));
-    } else if (e instanceof InvalidPDFException) {
-     handler.send('InvalidPDF', e);
-    } else if (e instanceof MissingPDFException) {
-     handler.send('MissingPDF', e);
-    } else if (e instanceof UnexpectedResponseException) {
-     handler.send('UnexpectedResponse', e);
-    } else {
-     handler.send('UnknownError', new UnknownErrorException(e.message, e.toString()));
-    }
-   }
-   function pdfManagerReady() {
-    ensureNotTerminated();
-    loadDocument(false).then(onSuccess, function loadFailure(ex) {
-     ensureNotTerminated();
-     if (!(ex instanceof XRefParseException)) {
-      onFailure(ex);
-      return;
-     }
-     pdfManager.requestLoadedStream();
-     pdfManager.onLoadedStream().then(function () {
       ensureNotTerminated();
-      loadDocument(true).then(onSuccess, onFailure);
-     });
-    }, onFailure);
-   }
-   ensureNotTerminated();
-   var evaluatorOptions = {
-    forceDataSchema: data.disableCreateObjectURL,
-    maxImageSize: data.maxImageSize === undefined ? -1 : data.maxImageSize,
-    disableFontFace: data.disableFontFace,
-    disableNativeImageDecoder: data.disableNativeImageDecoder
-   };
-   getPdfManager(data, evaluatorOptions).then(function (newPdfManager) {
-    if (terminated) {
-     newPdfManager.terminate();
-     throw new Error('Worker was terminated');
+      var evaluatorOptions = {
+        forceDataSchema: data.disableCreateObjectURL,
+        maxImageSize: data.maxImageSize === undefined ? -1 : data.maxImageSize,
+        disableFontFace: data.disableFontFace,
+        disableNativeImageDecoder: data.disableNativeImageDecoder
+      };
+      getPdfManager(data, evaluatorOptions).then(function (newPdfManager) {
+        if (terminated) {
+          newPdfManager.terminate();
+          throw new Error('Worker was terminated');
+        }
+        pdfManager = newPdfManager;
+        handler.send('PDFManagerReady', null);
+        pdfManager.onLoadedStream().then(function (stream) {
+          handler.send('DataLoaded', { length: stream.bytes.byteLength });
+        });
+      }).then(pdfManagerReady, onFailure);
     }
-    pdfManager = newPdfManager;
-    handler.send('PDFManagerReady', null);
-    pdfManager.onLoadedStream().then(function (stream) {
-     handler.send('DataLoaded', { length: stream.bytes.byteLength });
+    handler.on('GetPage', function wphSetupGetPage(data) {
+      return pdfManager.getPage(data.pageIndex).then(function (page) {
+        var rotatePromise = pdfManager.ensure(page, 'rotate');
+        var refPromise = pdfManager.ensure(page, 'ref');
+        var userUnitPromise = pdfManager.ensure(page, 'userUnit');
+        var viewPromise = pdfManager.ensure(page, 'view');
+        return Promise.all([rotatePromise, refPromise, userUnitPromise, viewPromise]).then(function (results) {
+          return {
+            rotate: results[0],
+            ref: results[1],
+            userUnit: results[2],
+            view: results[3]
+          };
+        });
+      });
     });
-   }).then(pdfManagerReady, onFailure);
-  }
-  handler.on('GetPage', function wphSetupGetPage(data) {
-   return pdfManager.getPage(data.pageIndex).then(function (page) {
-    var rotatePromise = pdfManager.ensure(page, 'rotate');
-    var refPromise = pdfManager.ensure(page, 'ref');
-    var userUnitPromise = pdfManager.ensure(page, 'userUnit');
-    var viewPromise = pdfManager.ensure(page, 'view');
-    return Promise.all([
-     rotatePromise,
-     refPromise,
-     userUnitPromise,
-     viewPromise
-    ]).then(function (results) {
-     return {
-      rotate: results[0],
-      ref: results[1],
-      userUnit: results[2],
-      view: results[3]
-     };
+    handler.on('GetPageIndex', function wphSetupGetPageIndex(data) {
+      var ref = new Ref(data.ref.num, data.ref.gen);
+      var catalog = pdfManager.pdfDocument.catalog;
+      return catalog.getPageIndex(ref);
     });
-   });
-  });
-  handler.on('GetPageIndex', function wphSetupGetPageIndex(data) {
-   var ref = new Ref(data.ref.num, data.ref.gen);
-   var catalog = pdfManager.pdfDocument.catalog;
-   return catalog.getPageIndex(ref);
-  });
-  handler.on('GetDestinations', function wphSetupGetDestinations(data) {
-   return pdfManager.ensureCatalog('destinations');
-  });
-  handler.on('GetDestination', function wphSetupGetDestination(data) {
-   return pdfManager.ensureCatalog('getDestination', [data.id]);
-  });
-  handler.on('GetPageLabels', function wphSetupGetPageLabels(data) {
-   return pdfManager.ensureCatalog('pageLabels');
-  });
-  handler.on('GetAttachments', function wphSetupGetAttachments(data) {
-   return pdfManager.ensureCatalog('attachments');
-  });
-  handler.on('GetJavaScript', function wphSetupGetJavaScript(data) {
-   return pdfManager.ensureCatalog('javaScript');
-  });
-  handler.on('GetOutline', function wphSetupGetOutline(data) {
-   return pdfManager.ensureCatalog('documentOutline');
-  });
-  handler.on('GetMetadata', function wphSetupGetMetadata(data) {
-   return Promise.all([
-    pdfManager.ensureDoc('documentInfo'),
-    pdfManager.ensureCatalog('metadata')
-   ]);
-  });
-  handler.on('GetData', function wphSetupGetData(data) {
-   pdfManager.requestLoadedStream();
-   return pdfManager.onLoadedStream().then(function (stream) {
-    return stream.bytes;
-   });
-  });
-  handler.on('GetStats', function wphSetupGetStats(data) {
-   return pdfManager.pdfDocument.xref.stats;
-  });
-  handler.on('GetAnnotations', function wphSetupGetAnnotations(data) {
-   return pdfManager.getPage(data.pageIndex).then(function (page) {
-    return pdfManager.ensure(page, 'getAnnotationsData', [data.intent]);
-   });
-  });
-  handler.on('RenderPageRequest', function wphSetupRenderPage(data) {
-   var pageIndex = data.pageIndex;
-   pdfManager.getPage(pageIndex).then(function (page) {
-    var task = new WorkerTask('RenderPageRequest: page ' + pageIndex);
-    startWorkerTask(task);
-    var pageNum = pageIndex + 1;
-    var start = Date.now();
-    page.getOperatorList(handler, task, data.intent, data.renderInteractiveForms).then(function (operatorList) {
-     finishWorkerTask(task);
-     info('page=' + pageNum + ' - getOperatorList: time=' + (Date.now() - start) + 'ms, len=' + operatorList.totalLength);
-    }, function (e) {
-     finishWorkerTask(task);
-     if (task.terminated) {
-      return;
-     }
-     handler.send('UnsupportedFeature', { featureId: UNSUPPORTED_FEATURES.unknown });
-     var minimumStackMessage = 'worker.js: while trying to getPage() and getOperatorList()';
-     var wrappedException;
-     if (typeof e === 'string') {
-      wrappedException = {
-       message: e,
-       stack: minimumStackMessage
-      };
-     } else if (typeof e === 'object') {
-      wrappedException = {
-       message: e.message || e.toString(),
-       stack: e.stack || minimumStackMessage
-      };
-     } else {
-      wrappedException = {
-       message: 'Unknown exception type: ' + typeof e,
-       stack: minimumStackMessage
-      };
-     }
-     handler.send('PageError', {
-      pageNum: pageNum,
-      error: wrappedException,
-      intent: data.intent
-     });
+    handler.on('GetDestinations', function wphSetupGetDestinations(data) {
+      return pdfManager.ensureCatalog('destinations');
+    });
+    handler.on('GetDestination', function wphSetupGetDestination(data) {
+      return pdfManager.ensureCatalog('getDestination', [data.id]);
+    });
+    handler.on('GetPageLabels', function wphSetupGetPageLabels(data) {
+      return pdfManager.ensureCatalog('pageLabels');
+    });
+    handler.on('GetAttachments', function wphSetupGetAttachments(data) {
+      return pdfManager.ensureCatalog('attachments');
+    });
+    handler.on('GetJavaScript', function wphSetupGetJavaScript(data) {
+      return pdfManager.ensureCatalog('javaScript');
+    });
+    handler.on('GetOutline', function wphSetupGetOutline(data) {
+      return pdfManager.ensureCatalog('documentOutline');
+    });
+    handler.on('GetMetadata', function wphSetupGetMetadata(data) {
+      return Promise.all([pdfManager.ensureDoc('documentInfo'), pdfManager.ensureCatalog('metadata')]);
+    });
+    handler.on('GetData', function wphSetupGetData(data) {
+      pdfManager.requestLoadedStream();
+      return pdfManager.onLoadedStream().then(function (stream) {
+        return stream.bytes;
+      });
     });
-   });
-  }, this);
-  handler.on('GetTextContent', function wphExtractText(data) {
-   var pageIndex = data.pageIndex;
-   var normalizeWhitespace = data.normalizeWhitespace;
-   var combineTextItems = data.combineTextItems;
-   return pdfManager.getPage(pageIndex).then(function (page) {
-    var task = new WorkerTask('GetTextContent: page ' + pageIndex);
-    startWorkerTask(task);
-    var pageNum = pageIndex + 1;
-    var start = Date.now();
-    return page.extractTextContent(handler, task, normalizeWhitespace, combineTextItems).then(function (textContent) {
-     finishWorkerTask(task);
-     info('text indexing: page=' + pageNum + ' - time=' + (Date.now() - start) + 'ms');
-     return textContent;
-    }, function (reason) {
-     finishWorkerTask(task);
-     if (task.terminated) {
-      return;
-     }
-     throw reason;
+    handler.on('GetStats', function wphSetupGetStats(data) {
+      return pdfManager.pdfDocument.xref.stats;
     });
-   });
-  });
-  handler.on('Cleanup', function wphCleanup(data) {
-   return pdfManager.cleanup();
-  });
-  handler.on('Terminate', function wphTerminate(data) {
-   terminated = true;
-   if (pdfManager) {
-    pdfManager.terminate();
-    pdfManager = null;
-   }
-   if (cancelXHRs) {
-    cancelXHRs();
-   }
-   var waitOn = [];
-   WorkerTasks.forEach(function (task) {
-    waitOn.push(task.finished);
-    task.terminate();
-   });
-   return Promise.all(waitOn).then(function () {
-    handler.destroy();
-    handler = null;
-   });
-  });
-  handler.on('Ready', function wphReady(data) {
-   setupDoc(docParams);
-   docParams = null;
-  });
-  return workerHandlerName;
- }
+    handler.on('GetAnnotations', function wphSetupGetAnnotations(data) {
+      return pdfManager.getPage(data.pageIndex).then(function (page) {
+        return pdfManager.ensure(page, 'getAnnotationsData', [data.intent]);
+      });
+    });
+    handler.on('RenderPageRequest', function wphSetupRenderPage(data) {
+      var pageIndex = data.pageIndex;
+      pdfManager.getPage(pageIndex).then(function (page) {
+        var task = new WorkerTask('RenderPageRequest: page ' + pageIndex);
+        startWorkerTask(task);
+        var pageNum = pageIndex + 1;
+        var start = Date.now();
+        page.getOperatorList(handler, task, data.intent, data.renderInteractiveForms).then(function (operatorList) {
+          finishWorkerTask(task);
+          info('page=' + pageNum + ' - getOperatorList: time=' + (Date.now() - start) + 'ms, len=' + operatorList.totalLength);
+        }, function (e) {
+          finishWorkerTask(task);
+          if (task.terminated) {
+            return;
+          }
+          handler.send('UnsupportedFeature', { featureId: UNSUPPORTED_FEATURES.unknown });
+          var minimumStackMessage = 'worker.js: while trying to getPage() and getOperatorList()';
+          var wrappedException;
+          if (typeof e === 'string') {
+            wrappedException = {
+              message: e,
+              stack: minimumStackMessage
+            };
+          } else if (typeof e === 'object') {
+            wrappedException = {
+              message: e.message || e.toString(),
+              stack: e.stack || minimumStackMessage
+            };
+          } else {
+            wrappedException = {
+              message: 'Unknown exception type: ' + typeof e,
+              stack: minimumStackMessage
+            };
+          }
+          handler.send('PageError', {
+            pageNum: pageNum,
+            error: wrappedException,
+            intent: data.intent
+          });
+        });
+      });
+    }, this);
+    handler.on('GetTextContent', function wphExtractText(data) {
+      var pageIndex = data.pageIndex;
+      var normalizeWhitespace = data.normalizeWhitespace;
+      var combineTextItems = data.combineTextItems;
+      return pdfManager.getPage(pageIndex).then(function (page) {
+        var task = new WorkerTask('GetTextContent: page ' + pageIndex);
+        startWorkerTask(task);
+        var pageNum = pageIndex + 1;
+        var start = Date.now();
+        return page.extractTextContent(handler, task, normalizeWhitespace, combineTextItems).then(function (textContent) {
+          finishWorkerTask(task);
+          info('text indexing: page=' + pageNum + ' - time=' + (Date.now() - start) + 'ms');
+          return textContent;
+        }, function (reason) {
+          finishWorkerTask(task);
+          if (task.terminated) {
+            return;
+          }
+          throw reason;
+        });
+      });
+    });
+    handler.on('Cleanup', function wphCleanup(data) {
+      return pdfManager.cleanup();
+    });
+    handler.on('Terminate', function wphTerminate(data) {
+      terminated = true;
+      if (pdfManager) {
+        pdfManager.terminate();
+        pdfManager = null;
+      }
+      if (cancelXHRs) {
+        cancelXHRs();
+      }
+      var waitOn = [];
+      WorkerTasks.forEach(function (task) {
+        waitOn.push(task.finished);
+        task.terminate();
+      });
+      return Promise.all(waitOn).then(function () {
+        handler.destroy();
+        handler = null;
+      });
+    });
+    handler.on('Ready', function wphReady(data) {
+      setupDoc(docParams);
+      docParams = null;
+    });
+    return workerHandlerName;
+  }
 };
 function initializeWorker() {
- var handler = new MessageHandler('worker', 'main', self);
- WorkerMessageHandler.setup(handler, self);
- handler.send('ready', null);
+  var handler = new MessageHandler('worker', 'main', self);
+  WorkerMessageHandler.setup(handler, self);
+  handler.send('ready', null);
 }
 if (typeof window === 'undefined' && !isNodeJS()) {
- initializeWorker();
+  initializeWorker();
 }
 exports.setPDFNetworkStreamClass = setPDFNetworkStreamClass;
 exports.WorkerTask = WorkerTask;

+ 542 - 549
lib/display/annotation_layer.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var displayDOMUtils = require('./dom_utils.js');
 var AnnotationBorderStyleType = sharedUtil.AnnotationBorderStyleType;
@@ -25,607 +26,599 @@ var getFilenameFromUrl = displayDOMUtils.getFilenameFromUrl;
 var warn = sharedUtil.warn;
 var CustomStyle = displayDOMUtils.CustomStyle;
 var getDefaultSetting = displayDOMUtils.getDefaultSetting;
-function AnnotationElementFactory() {
-}
+function AnnotationElementFactory() {}
 AnnotationElementFactory.prototype = {
- create: function AnnotationElementFactory_create(parameters) {
-  var subtype = parameters.data.annotationType;
-  switch (subtype) {
-  case AnnotationType.LINK:
-   return new LinkAnnotationElement(parameters);
-  case AnnotationType.TEXT:
-   return new TextAnnotationElement(parameters);
-  case AnnotationType.WIDGET:
-   var fieldType = parameters.data.fieldType;
-   switch (fieldType) {
-   case 'Tx':
-    return new TextWidgetAnnotationElement(parameters);
-   case 'Btn':
-    if (parameters.data.radioButton) {
-     return new RadioButtonWidgetAnnotationElement(parameters);
-    } else if (parameters.data.checkBox) {
-     return new CheckboxWidgetAnnotationElement(parameters);
+  create: function AnnotationElementFactory_create(parameters) {
+    var subtype = parameters.data.annotationType;
+    switch (subtype) {
+      case AnnotationType.LINK:
+        return new LinkAnnotationElement(parameters);
+      case AnnotationType.TEXT:
+        return new TextAnnotationElement(parameters);
+      case AnnotationType.WIDGET:
+        var fieldType = parameters.data.fieldType;
+        switch (fieldType) {
+          case 'Tx':
+            return new TextWidgetAnnotationElement(parameters);
+          case 'Btn':
+            if (parameters.data.radioButton) {
+              return new RadioButtonWidgetAnnotationElement(parameters);
+            } else if (parameters.data.checkBox) {
+              return new CheckboxWidgetAnnotationElement(parameters);
+            }
+            warn('Unimplemented button widget annotation: pushbutton');
+            break;
+          case 'Ch':
+            return new ChoiceWidgetAnnotationElement(parameters);
+        }
+        return new WidgetAnnotationElement(parameters);
+      case AnnotationType.POPUP:
+        return new PopupAnnotationElement(parameters);
+      case AnnotationType.HIGHLIGHT:
+        return new HighlightAnnotationElement(parameters);
+      case AnnotationType.UNDERLINE:
+        return new UnderlineAnnotationElement(parameters);
+      case AnnotationType.SQUIGGLY:
+        return new SquigglyAnnotationElement(parameters);
+      case AnnotationType.STRIKEOUT:
+        return new StrikeOutAnnotationElement(parameters);
+      case AnnotationType.FILEATTACHMENT:
+        return new FileAttachmentAnnotationElement(parameters);
+      default:
+        return new AnnotationElement(parameters);
     }
-    warn('Unimplemented button widget annotation: pushbutton');
-    break;
-   case 'Ch':
-    return new ChoiceWidgetAnnotationElement(parameters);
-   }
-   return new WidgetAnnotationElement(parameters);
-  case AnnotationType.POPUP:
-   return new PopupAnnotationElement(parameters);
-  case AnnotationType.HIGHLIGHT:
-   return new HighlightAnnotationElement(parameters);
-  case AnnotationType.UNDERLINE:
-   return new UnderlineAnnotationElement(parameters);
-  case AnnotationType.SQUIGGLY:
-   return new SquigglyAnnotationElement(parameters);
-  case AnnotationType.STRIKEOUT:
-   return new StrikeOutAnnotationElement(parameters);
-  case AnnotationType.FILEATTACHMENT:
-   return new FileAttachmentAnnotationElement(parameters);
-  default:
-   return new AnnotationElement(parameters);
   }
- }
 };
 var AnnotationElement = function AnnotationElementClosure() {
- function AnnotationElement(parameters, isRenderable) {
-  this.isRenderable = isRenderable || false;
-  this.data = parameters.data;
-  this.layer = parameters.layer;
-  this.page = parameters.page;
-  this.viewport = parameters.viewport;
-  this.linkService = parameters.linkService;
-  this.downloadManager = parameters.downloadManager;
-  this.imageResourcesPath = parameters.imageResourcesPath;
-  this.renderInteractiveForms = parameters.renderInteractiveForms;
-  if (isRenderable) {
-   this.container = this._createContainer();
-  }
- }
- AnnotationElement.prototype = {
-  _createContainer: function AnnotationElement_createContainer() {
-   var data = this.data, page = this.page, viewport = this.viewport;
-   var container = document.createElement('section');
-   var width = data.rect[2] - data.rect[0];
-   var height = data.rect[3] - data.rect[1];
-   container.setAttribute('data-annotation-id', data.id);
-   var rect = Util.normalizeRect([
-    data.rect[0],
-    page.view[3] - data.rect[1] + page.view[1],
-    data.rect[2],
-    page.view[3] - data.rect[3] + page.view[1]
-   ]);
-   CustomStyle.setProp('transform', container, 'matrix(' + viewport.transform.join(',') + ')');
-   CustomStyle.setProp('transformOrigin', container, -rect[0] + 'px ' + -rect[1] + 'px');
-   if (data.borderStyle.width > 0) {
-    container.style.borderWidth = data.borderStyle.width + 'px';
-    if (data.borderStyle.style !== AnnotationBorderStyleType.UNDERLINE) {
-     width = width - 2 * data.borderStyle.width;
-     height = height - 2 * data.borderStyle.width;
-    }
-    var horizontalRadius = data.borderStyle.horizontalCornerRadius;
-    var verticalRadius = data.borderStyle.verticalCornerRadius;
-    if (horizontalRadius > 0 || verticalRadius > 0) {
-     var radius = horizontalRadius + 'px / ' + verticalRadius + 'px';
-     CustomStyle.setProp('borderRadius', container, radius);
-    }
-    switch (data.borderStyle.style) {
-    case AnnotationBorderStyleType.SOLID:
-     container.style.borderStyle = 'solid';
-     break;
-    case AnnotationBorderStyleType.DASHED:
-     container.style.borderStyle = 'dashed';
-     break;
-    case AnnotationBorderStyleType.BEVELED:
-     warn('Unimplemented border style: beveled');
-     break;
-    case AnnotationBorderStyleType.INSET:
-     warn('Unimplemented border style: inset');
-     break;
-    case AnnotationBorderStyleType.UNDERLINE:
-     container.style.borderBottomStyle = 'solid';
-     break;
-    default:
-     break;
+  function AnnotationElement(parameters, isRenderable) {
+    this.isRenderable = isRenderable || false;
+    this.data = parameters.data;
+    this.layer = parameters.layer;
+    this.page = parameters.page;
+    this.viewport = parameters.viewport;
+    this.linkService = parameters.linkService;
+    this.downloadManager = parameters.downloadManager;
+    this.imageResourcesPath = parameters.imageResourcesPath;
+    this.renderInteractiveForms = parameters.renderInteractiveForms;
+    if (isRenderable) {
+      this.container = this._createContainer();
     }
-    if (data.color) {
-     container.style.borderColor = Util.makeCssRgb(data.color[0] | 0, data.color[1] | 0, data.color[2] | 0);
-    } else {
-     container.style.borderWidth = 0;
-    }
-   }
-   container.style.left = rect[0] + 'px';
-   container.style.top = rect[1] + 'px';
-   container.style.width = width + 'px';
-   container.style.height = height + 'px';
-   return container;
-  },
-  _createPopup: function AnnotationElement_createPopup(container, trigger, data) {
-   if (!trigger) {
-    trigger = document.createElement('div');
-    trigger.style.height = container.style.height;
-    trigger.style.width = container.style.width;
-    container.appendChild(trigger);
-   }
-   var popupElement = new PopupElement({
-    container: container,
-    trigger: trigger,
-    color: data.color,
-    title: data.title,
-    contents: data.contents,
-    hideWrapper: true
-   });
-   var popup = popupElement.render();
-   popup.style.left = container.style.width;
-   container.appendChild(popup);
-  },
-  render: function AnnotationElement_render() {
-   throw new Error('Abstract method AnnotationElement.render called');
   }
- };
- return AnnotationElement;
+  AnnotationElement.prototype = {
+    _createContainer: function AnnotationElement_createContainer() {
+      var data = this.data,
+          page = this.page,
+          viewport = this.viewport;
+      var container = document.createElement('section');
+      var width = data.rect[2] - data.rect[0];
+      var height = data.rect[3] - data.rect[1];
+      container.setAttribute('data-annotation-id', data.id);
+      var rect = Util.normalizeRect([data.rect[0], page.view[3] - data.rect[1] + page.view[1], data.rect[2], page.view[3] - data.rect[3] + page.view[1]]);
+      CustomStyle.setProp('transform', container, 'matrix(' + viewport.transform.join(',') + ')');
+      CustomStyle.setProp('transformOrigin', container, -rect[0] + 'px ' + -rect[1] + 'px');
+      if (data.borderStyle.width > 0) {
+        container.style.borderWidth = data.borderStyle.width + 'px';
+        if (data.borderStyle.style !== AnnotationBorderStyleType.UNDERLINE) {
+          width = width - 2 * data.borderStyle.width;
+          height = height - 2 * data.borderStyle.width;
+        }
+        var horizontalRadius = data.borderStyle.horizontalCornerRadius;
+        var verticalRadius = data.borderStyle.verticalCornerRadius;
+        if (horizontalRadius > 0 || verticalRadius > 0) {
+          var radius = horizontalRadius + 'px / ' + verticalRadius + 'px';
+          CustomStyle.setProp('borderRadius', container, radius);
+        }
+        switch (data.borderStyle.style) {
+          case AnnotationBorderStyleType.SOLID:
+            container.style.borderStyle = 'solid';
+            break;
+          case AnnotationBorderStyleType.DASHED:
+            container.style.borderStyle = 'dashed';
+            break;
+          case AnnotationBorderStyleType.BEVELED:
+            warn('Unimplemented border style: beveled');
+            break;
+          case AnnotationBorderStyleType.INSET:
+            warn('Unimplemented border style: inset');
+            break;
+          case AnnotationBorderStyleType.UNDERLINE:
+            container.style.borderBottomStyle = 'solid';
+            break;
+          default:
+            break;
+        }
+        if (data.color) {
+          container.style.borderColor = Util.makeCssRgb(data.color[0] | 0, data.color[1] | 0, data.color[2] | 0);
+        } else {
+          container.style.borderWidth = 0;
+        }
+      }
+      container.style.left = rect[0] + 'px';
+      container.style.top = rect[1] + 'px';
+      container.style.width = width + 'px';
+      container.style.height = height + 'px';
+      return container;
+    },
+    _createPopup: function AnnotationElement_createPopup(container, trigger, data) {
+      if (!trigger) {
+        trigger = document.createElement('div');
+        trigger.style.height = container.style.height;
+        trigger.style.width = container.style.width;
+        container.appendChild(trigger);
+      }
+      var popupElement = new PopupElement({
+        container: container,
+        trigger: trigger,
+        color: data.color,
+        title: data.title,
+        contents: data.contents,
+        hideWrapper: true
+      });
+      var popup = popupElement.render();
+      popup.style.left = container.style.width;
+      container.appendChild(popup);
+    },
+    render: function AnnotationElement_render() {
+      throw new Error('Abstract method AnnotationElement.render called');
+    }
+  };
+  return AnnotationElement;
 }();
 var LinkAnnotationElement = function LinkAnnotationElementClosure() {
- function LinkAnnotationElement(parameters) {
-  AnnotationElement.call(this, parameters, true);
- }
- Util.inherit(LinkAnnotationElement, AnnotationElement, {
-  render: function LinkAnnotationElement_render() {
-   this.container.className = 'linkAnnotation';
-   var link = document.createElement('a');
-   addLinkAttributes(link, {
-    url: this.data.url,
-    target: this.data.newWindow ? LinkTarget.BLANK : undefined
-   });
-   if (!this.data.url) {
-    if (this.data.action) {
-     this._bindNamedAction(link, this.data.action);
-    } else {
-     this._bindLink(link, this.data.dest);
-    }
-   }
-   this.container.appendChild(link);
-   return this.container;
-  },
-  _bindLink: function LinkAnnotationElement_bindLink(link, destination) {
-   var self = this;
-   link.href = this.linkService.getDestinationHash(destination);
-   link.onclick = function () {
-    if (destination) {
-     self.linkService.navigateTo(destination);
-    }
-    return false;
-   };
-   if (destination) {
-    link.className = 'internalLink';
-   }
-  },
-  _bindNamedAction: function LinkAnnotationElement_bindNamedAction(link, action) {
-   var self = this;
-   link.href = this.linkService.getAnchorUrl('');
-   link.onclick = function () {
-    self.linkService.executeNamedAction(action);
-    return false;
-   };
-   link.className = 'internalLink';
+  function LinkAnnotationElement(parameters) {
+    AnnotationElement.call(this, parameters, true);
   }
- });
- return LinkAnnotationElement;
+  Util.inherit(LinkAnnotationElement, AnnotationElement, {
+    render: function LinkAnnotationElement_render() {
+      this.container.className = 'linkAnnotation';
+      var link = document.createElement('a');
+      addLinkAttributes(link, {
+        url: this.data.url,
+        target: this.data.newWindow ? LinkTarget.BLANK : undefined
+      });
+      if (!this.data.url) {
+        if (this.data.action) {
+          this._bindNamedAction(link, this.data.action);
+        } else {
+          this._bindLink(link, this.data.dest);
+        }
+      }
+      this.container.appendChild(link);
+      return this.container;
+    },
+    _bindLink: function LinkAnnotationElement_bindLink(link, destination) {
+      var self = this;
+      link.href = this.linkService.getDestinationHash(destination);
+      link.onclick = function () {
+        if (destination) {
+          self.linkService.navigateTo(destination);
+        }
+        return false;
+      };
+      if (destination) {
+        link.className = 'internalLink';
+      }
+    },
+    _bindNamedAction: function LinkAnnotationElement_bindNamedAction(link, action) {
+      var self = this;
+      link.href = this.linkService.getAnchorUrl('');
+      link.onclick = function () {
+        self.linkService.executeNamedAction(action);
+        return false;
+      };
+      link.className = 'internalLink';
+    }
+  });
+  return LinkAnnotationElement;
 }();
 var TextAnnotationElement = function TextAnnotationElementClosure() {
- function TextAnnotationElement(parameters) {
-  var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
-  AnnotationElement.call(this, parameters, isRenderable);
- }
- Util.inherit(TextAnnotationElement, AnnotationElement, {
-  render: function TextAnnotationElement_render() {
-   this.container.className = 'textAnnotation';
-   var image = document.createElement('img');
-   image.style.height = this.container.style.height;
-   image.style.width = this.container.style.width;
-   image.src = this.imageResourcesPath + 'annotation-' + this.data.name.toLowerCase() + '.svg';
-   image.alt = '[{{type}} Annotation]';
-   image.dataset.l10nId = 'text_annotation_type';
-   image.dataset.l10nArgs = JSON.stringify({ type: this.data.name });
-   if (!this.data.hasPopup) {
-    this._createPopup(this.container, image, this.data);
-   }
-   this.container.appendChild(image);
-   return this.container;
+  function TextAnnotationElement(parameters) {
+    var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
+    AnnotationElement.call(this, parameters, isRenderable);
   }
- });
- return TextAnnotationElement;
+  Util.inherit(TextAnnotationElement, AnnotationElement, {
+    render: function TextAnnotationElement_render() {
+      this.container.className = 'textAnnotation';
+      var image = document.createElement('img');
+      image.style.height = this.container.style.height;
+      image.style.width = this.container.style.width;
+      image.src = this.imageResourcesPath + 'annotation-' + this.data.name.toLowerCase() + '.svg';
+      image.alt = '[{{type}} Annotation]';
+      image.dataset.l10nId = 'text_annotation_type';
+      image.dataset.l10nArgs = JSON.stringify({ type: this.data.name });
+      if (!this.data.hasPopup) {
+        this._createPopup(this.container, image, this.data);
+      }
+      this.container.appendChild(image);
+      return this.container;
+    }
+  });
+  return TextAnnotationElement;
 }();
 var WidgetAnnotationElement = function WidgetAnnotationElementClosure() {
- function WidgetAnnotationElement(parameters, isRenderable) {
-  AnnotationElement.call(this, parameters, isRenderable);
- }
- Util.inherit(WidgetAnnotationElement, AnnotationElement, {
-  render: function WidgetAnnotationElement_render() {
-   return this.container;
+  function WidgetAnnotationElement(parameters, isRenderable) {
+    AnnotationElement.call(this, parameters, isRenderable);
   }
- });
- return WidgetAnnotationElement;
+  Util.inherit(WidgetAnnotationElement, AnnotationElement, {
+    render: function WidgetAnnotationElement_render() {
+      return this.container;
+    }
+  });
+  return WidgetAnnotationElement;
 }();
 var TextWidgetAnnotationElement = function TextWidgetAnnotationElementClosure() {
- var TEXT_ALIGNMENT = [
-  'left',
-  'center',
-  'right'
- ];
- function TextWidgetAnnotationElement(parameters) {
-  var isRenderable = parameters.renderInteractiveForms || !parameters.data.hasAppearance && !!parameters.data.fieldValue;
-  WidgetAnnotationElement.call(this, parameters, isRenderable);
- }
- Util.inherit(TextWidgetAnnotationElement, WidgetAnnotationElement, {
-  render: function TextWidgetAnnotationElement_render() {
-   this.container.className = 'textWidgetAnnotation';
-   var element = null;
-   if (this.renderInteractiveForms) {
-    if (this.data.multiLine) {
-     element = document.createElement('textarea');
-     element.textContent = this.data.fieldValue;
-    } else {
-     element = document.createElement('input');
-     element.type = 'text';
-     element.setAttribute('value', this.data.fieldValue);
-    }
-    element.disabled = this.data.readOnly;
-    if (this.data.maxLen !== null) {
-     element.maxLength = this.data.maxLen;
-    }
-    if (this.data.comb) {
-     var fieldWidth = this.data.rect[2] - this.data.rect[0];
-     var combWidth = fieldWidth / this.data.maxLen;
-     element.classList.add('comb');
-     element.style.letterSpacing = 'calc(' + combWidth + 'px - 1ch)';
-    }
-   } else {
-    element = document.createElement('div');
-    element.textContent = this.data.fieldValue;
-    element.style.verticalAlign = 'middle';
-    element.style.display = 'table-cell';
-    var font = null;
-    if (this.data.fontRefName) {
-     font = this.page.commonObjs.getData(this.data.fontRefName);
-    }
-    this._setTextStyle(element, font);
-   }
-   if (this.data.textAlignment !== null) {
-    element.style.textAlign = TEXT_ALIGNMENT[this.data.textAlignment];
-   }
-   this.container.appendChild(element);
-   return this.container;
-  },
-  _setTextStyle: function TextWidgetAnnotationElement_setTextStyle(element, font) {
-   var style = element.style;
-   style.fontSize = this.data.fontSize + 'px';
-   style.direction = this.data.fontDirection < 0 ? 'rtl' : 'ltr';
-   if (!font) {
-    return;
-   }
-   style.fontWeight = font.black ? font.bold ? '900' : 'bold' : font.bold ? 'bold' : 'normal';
-   style.fontStyle = font.italic ? 'italic' : 'normal';
-   var fontFamily = font.loadedName ? '"' + font.loadedName + '", ' : '';
-   var fallbackName = font.fallbackName || 'Helvetica, sans-serif';
-   style.fontFamily = fontFamily + fallbackName;
+  var TEXT_ALIGNMENT = ['left', 'center', 'right'];
+  function TextWidgetAnnotationElement(parameters) {
+    var isRenderable = parameters.renderInteractiveForms || !parameters.data.hasAppearance && !!parameters.data.fieldValue;
+    WidgetAnnotationElement.call(this, parameters, isRenderable);
   }
- });
- return TextWidgetAnnotationElement;
+  Util.inherit(TextWidgetAnnotationElement, WidgetAnnotationElement, {
+    render: function TextWidgetAnnotationElement_render() {
+      this.container.className = 'textWidgetAnnotation';
+      var element = null;
+      if (this.renderInteractiveForms) {
+        if (this.data.multiLine) {
+          element = document.createElement('textarea');
+          element.textContent = this.data.fieldValue;
+        } else {
+          element = document.createElement('input');
+          element.type = 'text';
+          element.setAttribute('value', this.data.fieldValue);
+        }
+        element.disabled = this.data.readOnly;
+        if (this.data.maxLen !== null) {
+          element.maxLength = this.data.maxLen;
+        }
+        if (this.data.comb) {
+          var fieldWidth = this.data.rect[2] - this.data.rect[0];
+          var combWidth = fieldWidth / this.data.maxLen;
+          element.classList.add('comb');
+          element.style.letterSpacing = 'calc(' + combWidth + 'px - 1ch)';
+        }
+      } else {
+        element = document.createElement('div');
+        element.textContent = this.data.fieldValue;
+        element.style.verticalAlign = 'middle';
+        element.style.display = 'table-cell';
+        var font = null;
+        if (this.data.fontRefName) {
+          font = this.page.commonObjs.getData(this.data.fontRefName);
+        }
+        this._setTextStyle(element, font);
+      }
+      if (this.data.textAlignment !== null) {
+        element.style.textAlign = TEXT_ALIGNMENT[this.data.textAlignment];
+      }
+      this.container.appendChild(element);
+      return this.container;
+    },
+    _setTextStyle: function TextWidgetAnnotationElement_setTextStyle(element, font) {
+      var style = element.style;
+      style.fontSize = this.data.fontSize + 'px';
+      style.direction = this.data.fontDirection < 0 ? 'rtl' : 'ltr';
+      if (!font) {
+        return;
+      }
+      style.fontWeight = font.black ? font.bold ? '900' : 'bold' : font.bold ? 'bold' : 'normal';
+      style.fontStyle = font.italic ? 'italic' : 'normal';
+      var fontFamily = font.loadedName ? '"' + font.loadedName + '", ' : '';
+      var fallbackName = font.fallbackName || 'Helvetica, sans-serif';
+      style.fontFamily = fontFamily + fallbackName;
+    }
+  });
+  return TextWidgetAnnotationElement;
 }();
 var CheckboxWidgetAnnotationElement = function CheckboxWidgetAnnotationElementClosure() {
- function CheckboxWidgetAnnotationElement(parameters) {
-  WidgetAnnotationElement.call(this, parameters, parameters.renderInteractiveForms);
- }
- Util.inherit(CheckboxWidgetAnnotationElement, WidgetAnnotationElement, {
-  render: function CheckboxWidgetAnnotationElement_render() {
-   this.container.className = 'buttonWidgetAnnotation checkBox';
-   var element = document.createElement('input');
-   element.disabled = this.data.readOnly;
-   element.type = 'checkbox';
-   if (this.data.fieldValue && this.data.fieldValue !== 'Off') {
-    element.setAttribute('checked', true);
-   }
-   this.container.appendChild(element);
-   return this.container;
+  function CheckboxWidgetAnnotationElement(parameters) {
+    WidgetAnnotationElement.call(this, parameters, parameters.renderInteractiveForms);
   }
- });
- return CheckboxWidgetAnnotationElement;
+  Util.inherit(CheckboxWidgetAnnotationElement, WidgetAnnotationElement, {
+    render: function CheckboxWidgetAnnotationElement_render() {
+      this.container.className = 'buttonWidgetAnnotation checkBox';
+      var element = document.createElement('input');
+      element.disabled = this.data.readOnly;
+      element.type = 'checkbox';
+      if (this.data.fieldValue && this.data.fieldValue !== 'Off') {
+        element.setAttribute('checked', true);
+      }
+      this.container.appendChild(element);
+      return this.container;
+    }
+  });
+  return CheckboxWidgetAnnotationElement;
 }();
 var RadioButtonWidgetAnnotationElement = function RadioButtonWidgetAnnotationElementClosure() {
- function RadioButtonWidgetAnnotationElement(parameters) {
-  WidgetAnnotationElement.call(this, parameters, parameters.renderInteractiveForms);
- }
- Util.inherit(RadioButtonWidgetAnnotationElement, WidgetAnnotationElement, {
-  render: function RadioButtonWidgetAnnotationElement_render() {
-   this.container.className = 'buttonWidgetAnnotation radioButton';
-   var element = document.createElement('input');
-   element.disabled = this.data.readOnly;
-   element.type = 'radio';
-   element.name = this.data.fieldName;
-   if (this.data.fieldValue === this.data.buttonValue) {
-    element.setAttribute('checked', true);
-   }
-   this.container.appendChild(element);
-   return this.container;
+  function RadioButtonWidgetAnnotationElement(parameters) {
+    WidgetAnnotationElement.call(this, parameters, parameters.renderInteractiveForms);
   }
- });
- return RadioButtonWidgetAnnotationElement;
+  Util.inherit(RadioButtonWidgetAnnotationElement, WidgetAnnotationElement, {
+    render: function RadioButtonWidgetAnnotationElement_render() {
+      this.container.className = 'buttonWidgetAnnotation radioButton';
+      var element = document.createElement('input');
+      element.disabled = this.data.readOnly;
+      element.type = 'radio';
+      element.name = this.data.fieldName;
+      if (this.data.fieldValue === this.data.buttonValue) {
+        element.setAttribute('checked', true);
+      }
+      this.container.appendChild(element);
+      return this.container;
+    }
+  });
+  return RadioButtonWidgetAnnotationElement;
 }();
 var ChoiceWidgetAnnotationElement = function ChoiceWidgetAnnotationElementClosure() {
- function ChoiceWidgetAnnotationElement(parameters) {
-  WidgetAnnotationElement.call(this, parameters, parameters.renderInteractiveForms);
- }
- Util.inherit(ChoiceWidgetAnnotationElement, WidgetAnnotationElement, {
-  render: function ChoiceWidgetAnnotationElement_render() {
-   this.container.className = 'choiceWidgetAnnotation';
-   var selectElement = document.createElement('select');
-   selectElement.disabled = this.data.readOnly;
-   if (!this.data.combo) {
-    selectElement.size = this.data.options.length;
-    if (this.data.multiSelect) {
-     selectElement.multiple = true;
-    }
-   }
-   for (var i = 0, ii = this.data.options.length; i < ii; i++) {
-    var option = this.data.options[i];
-    var optionElement = document.createElement('option');
-    optionElement.textContent = option.displayValue;
-    optionElement.value = option.exportValue;
-    if (this.data.fieldValue.indexOf(option.displayValue) >= 0) {
-     optionElement.setAttribute('selected', true);
-    }
-    selectElement.appendChild(optionElement);
-   }
-   this.container.appendChild(selectElement);
-   return this.container;
+  function ChoiceWidgetAnnotationElement(parameters) {
+    WidgetAnnotationElement.call(this, parameters, parameters.renderInteractiveForms);
   }
- });
- return ChoiceWidgetAnnotationElement;
+  Util.inherit(ChoiceWidgetAnnotationElement, WidgetAnnotationElement, {
+    render: function ChoiceWidgetAnnotationElement_render() {
+      this.container.className = 'choiceWidgetAnnotation';
+      var selectElement = document.createElement('select');
+      selectElement.disabled = this.data.readOnly;
+      if (!this.data.combo) {
+        selectElement.size = this.data.options.length;
+        if (this.data.multiSelect) {
+          selectElement.multiple = true;
+        }
+      }
+      for (var i = 0, ii = this.data.options.length; i < ii; i++) {
+        var option = this.data.options[i];
+        var optionElement = document.createElement('option');
+        optionElement.textContent = option.displayValue;
+        optionElement.value = option.exportValue;
+        if (this.data.fieldValue.indexOf(option.displayValue) >= 0) {
+          optionElement.setAttribute('selected', true);
+        }
+        selectElement.appendChild(optionElement);
+      }
+      this.container.appendChild(selectElement);
+      return this.container;
+    }
+  });
+  return ChoiceWidgetAnnotationElement;
 }();
 var PopupAnnotationElement = function PopupAnnotationElementClosure() {
- function PopupAnnotationElement(parameters) {
-  var isRenderable = !!(parameters.data.title || parameters.data.contents);
-  AnnotationElement.call(this, parameters, isRenderable);
- }
- Util.inherit(PopupAnnotationElement, AnnotationElement, {
-  render: function PopupAnnotationElement_render() {
-   this.container.className = 'popupAnnotation';
-   var selector = '[data-annotation-id="' + this.data.parentId + '"]';
-   var parentElement = this.layer.querySelector(selector);
-   if (!parentElement) {
-    return this.container;
-   }
-   var popup = new PopupElement({
-    container: this.container,
-    trigger: parentElement,
-    color: this.data.color,
-    title: this.data.title,
-    contents: this.data.contents
-   });
-   var parentLeft = parseFloat(parentElement.style.left);
-   var parentWidth = parseFloat(parentElement.style.width);
-   CustomStyle.setProp('transformOrigin', this.container, -(parentLeft + parentWidth) + 'px -' + parentElement.style.top);
-   this.container.style.left = parentLeft + parentWidth + 'px';
-   this.container.appendChild(popup.render());
-   return this.container;
+  function PopupAnnotationElement(parameters) {
+    var isRenderable = !!(parameters.data.title || parameters.data.contents);
+    AnnotationElement.call(this, parameters, isRenderable);
   }
- });
- return PopupAnnotationElement;
+  Util.inherit(PopupAnnotationElement, AnnotationElement, {
+    render: function PopupAnnotationElement_render() {
+      this.container.className = 'popupAnnotation';
+      var selector = '[data-annotation-id="' + this.data.parentId + '"]';
+      var parentElement = this.layer.querySelector(selector);
+      if (!parentElement) {
+        return this.container;
+      }
+      var popup = new PopupElement({
+        container: this.container,
+        trigger: parentElement,
+        color: this.data.color,
+        title: this.data.title,
+        contents: this.data.contents
+      });
+      var parentLeft = parseFloat(parentElement.style.left);
+      var parentWidth = parseFloat(parentElement.style.width);
+      CustomStyle.setProp('transformOrigin', this.container, -(parentLeft + parentWidth) + 'px -' + parentElement.style.top);
+      this.container.style.left = parentLeft + parentWidth + 'px';
+      this.container.appendChild(popup.render());
+      return this.container;
+    }
+  });
+  return PopupAnnotationElement;
 }();
 var PopupElement = function PopupElementClosure() {
- var BACKGROUND_ENLIGHT = 0.7;
- function PopupElement(parameters) {
-  this.container = parameters.container;
-  this.trigger = parameters.trigger;
-  this.color = parameters.color;
-  this.title = parameters.title;
-  this.contents = parameters.contents;
-  this.hideWrapper = parameters.hideWrapper || false;
-  this.pinned = false;
- }
- PopupElement.prototype = {
-  render: function PopupElement_render() {
-   var wrapper = document.createElement('div');
-   wrapper.className = 'popupWrapper';
-   this.hideElement = this.hideWrapper ? wrapper : this.container;
-   this.hideElement.setAttribute('hidden', true);
-   var popup = document.createElement('div');
-   popup.className = 'popup';
-   var color = this.color;
-   if (color) {
-    var r = BACKGROUND_ENLIGHT * (255 - color[0]) + color[0];
-    var g = BACKGROUND_ENLIGHT * (255 - color[1]) + color[1];
-    var b = BACKGROUND_ENLIGHT * (255 - color[2]) + color[2];
-    popup.style.backgroundColor = Util.makeCssRgb(r | 0, g | 0, b | 0);
-   }
-   var contents = this._formatContents(this.contents);
-   var title = document.createElement('h1');
-   title.textContent = this.title;
-   this.trigger.addEventListener('click', this._toggle.bind(this));
-   this.trigger.addEventListener('mouseover', this._show.bind(this, false));
-   this.trigger.addEventListener('mouseout', this._hide.bind(this, false));
-   popup.addEventListener('click', this._hide.bind(this, true));
-   popup.appendChild(title);
-   popup.appendChild(contents);
-   wrapper.appendChild(popup);
-   return wrapper;
-  },
-  _formatContents: function PopupElement_formatContents(contents) {
-   var p = document.createElement('p');
-   var lines = contents.split(/(?:\r\n?|\n)/);
-   for (var i = 0, ii = lines.length; i < ii; ++i) {
-    var line = lines[i];
-    p.appendChild(document.createTextNode(line));
-    if (i < ii - 1) {
-     p.appendChild(document.createElement('br'));
-    }
-   }
-   return p;
-  },
-  _toggle: function PopupElement_toggle() {
-   if (this.pinned) {
-    this._hide(true);
-   } else {
-    this._show(true);
-   }
-  },
-  _show: function PopupElement_show(pin) {
-   if (pin) {
-    this.pinned = true;
-   }
-   if (this.hideElement.hasAttribute('hidden')) {
-    this.hideElement.removeAttribute('hidden');
-    this.container.style.zIndex += 1;
-   }
-  },
-  _hide: function PopupElement_hide(unpin) {
-   if (unpin) {
+  var BACKGROUND_ENLIGHT = 0.7;
+  function PopupElement(parameters) {
+    this.container = parameters.container;
+    this.trigger = parameters.trigger;
+    this.color = parameters.color;
+    this.title = parameters.title;
+    this.contents = parameters.contents;
+    this.hideWrapper = parameters.hideWrapper || false;
     this.pinned = false;
-   }
-   if (!this.hideElement.hasAttribute('hidden') && !this.pinned) {
-    this.hideElement.setAttribute('hidden', true);
-    this.container.style.zIndex -= 1;
-   }
   }
- };
- return PopupElement;
+  PopupElement.prototype = {
+    render: function PopupElement_render() {
+      var wrapper = document.createElement('div');
+      wrapper.className = 'popupWrapper';
+      this.hideElement = this.hideWrapper ? wrapper : this.container;
+      this.hideElement.setAttribute('hidden', true);
+      var popup = document.createElement('div');
+      popup.className = 'popup';
+      var color = this.color;
+      if (color) {
+        var r = BACKGROUND_ENLIGHT * (255 - color[0]) + color[0];
+        var g = BACKGROUND_ENLIGHT * (255 - color[1]) + color[1];
+        var b = BACKGROUND_ENLIGHT * (255 - color[2]) + color[2];
+        popup.style.backgroundColor = Util.makeCssRgb(r | 0, g | 0, b | 0);
+      }
+      var contents = this._formatContents(this.contents);
+      var title = document.createElement('h1');
+      title.textContent = this.title;
+      this.trigger.addEventListener('click', this._toggle.bind(this));
+      this.trigger.addEventListener('mouseover', this._show.bind(this, false));
+      this.trigger.addEventListener('mouseout', this._hide.bind(this, false));
+      popup.addEventListener('click', this._hide.bind(this, true));
+      popup.appendChild(title);
+      popup.appendChild(contents);
+      wrapper.appendChild(popup);
+      return wrapper;
+    },
+    _formatContents: function PopupElement_formatContents(contents) {
+      var p = document.createElement('p');
+      var lines = contents.split(/(?:\r\n?|\n)/);
+      for (var i = 0, ii = lines.length; i < ii; ++i) {
+        var line = lines[i];
+        p.appendChild(document.createTextNode(line));
+        if (i < ii - 1) {
+          p.appendChild(document.createElement('br'));
+        }
+      }
+      return p;
+    },
+    _toggle: function PopupElement_toggle() {
+      if (this.pinned) {
+        this._hide(true);
+      } else {
+        this._show(true);
+      }
+    },
+    _show: function PopupElement_show(pin) {
+      if (pin) {
+        this.pinned = true;
+      }
+      if (this.hideElement.hasAttribute('hidden')) {
+        this.hideElement.removeAttribute('hidden');
+        this.container.style.zIndex += 1;
+      }
+    },
+    _hide: function PopupElement_hide(unpin) {
+      if (unpin) {
+        this.pinned = false;
+      }
+      if (!this.hideElement.hasAttribute('hidden') && !this.pinned) {
+        this.hideElement.setAttribute('hidden', true);
+        this.container.style.zIndex -= 1;
+      }
+    }
+  };
+  return PopupElement;
 }();
 var HighlightAnnotationElement = function HighlightAnnotationElementClosure() {
- function HighlightAnnotationElement(parameters) {
-  var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
-  AnnotationElement.call(this, parameters, isRenderable);
- }
- Util.inherit(HighlightAnnotationElement, AnnotationElement, {
-  render: function HighlightAnnotationElement_render() {
-   this.container.className = 'highlightAnnotation';
-   if (!this.data.hasPopup) {
-    this._createPopup(this.container, null, this.data);
-   }
-   return this.container;
+  function HighlightAnnotationElement(parameters) {
+    var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
+    AnnotationElement.call(this, parameters, isRenderable);
   }
- });
- return HighlightAnnotationElement;
+  Util.inherit(HighlightAnnotationElement, AnnotationElement, {
+    render: function HighlightAnnotationElement_render() {
+      this.container.className = 'highlightAnnotation';
+      if (!this.data.hasPopup) {
+        this._createPopup(this.container, null, this.data);
+      }
+      return this.container;
+    }
+  });
+  return HighlightAnnotationElement;
 }();
 var UnderlineAnnotationElement = function UnderlineAnnotationElementClosure() {
- function UnderlineAnnotationElement(parameters) {
-  var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
-  AnnotationElement.call(this, parameters, isRenderable);
- }
- Util.inherit(UnderlineAnnotationElement, AnnotationElement, {
-  render: function UnderlineAnnotationElement_render() {
-   this.container.className = 'underlineAnnotation';
-   if (!this.data.hasPopup) {
-    this._createPopup(this.container, null, this.data);
-   }
-   return this.container;
+  function UnderlineAnnotationElement(parameters) {
+    var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
+    AnnotationElement.call(this, parameters, isRenderable);
   }
- });
- return UnderlineAnnotationElement;
+  Util.inherit(UnderlineAnnotationElement, AnnotationElement, {
+    render: function UnderlineAnnotationElement_render() {
+      this.container.className = 'underlineAnnotation';
+      if (!this.data.hasPopup) {
+        this._createPopup(this.container, null, this.data);
+      }
+      return this.container;
+    }
+  });
+  return UnderlineAnnotationElement;
 }();
 var SquigglyAnnotationElement = function SquigglyAnnotationElementClosure() {
- function SquigglyAnnotationElement(parameters) {
-  var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
-  AnnotationElement.call(this, parameters, isRenderable);
- }
- Util.inherit(SquigglyAnnotationElement, AnnotationElement, {
-  render: function SquigglyAnnotationElement_render() {
-   this.container.className = 'squigglyAnnotation';
-   if (!this.data.hasPopup) {
-    this._createPopup(this.container, null, this.data);
-   }
-   return this.container;
+  function SquigglyAnnotationElement(parameters) {
+    var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
+    AnnotationElement.call(this, parameters, isRenderable);
   }
- });
- return SquigglyAnnotationElement;
+  Util.inherit(SquigglyAnnotationElement, AnnotationElement, {
+    render: function SquigglyAnnotationElement_render() {
+      this.container.className = 'squigglyAnnotation';
+      if (!this.data.hasPopup) {
+        this._createPopup(this.container, null, this.data);
+      }
+      return this.container;
+    }
+  });
+  return SquigglyAnnotationElement;
 }();
 var StrikeOutAnnotationElement = function StrikeOutAnnotationElementClosure() {
- function StrikeOutAnnotationElement(parameters) {
-  var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
-  AnnotationElement.call(this, parameters, isRenderable);
- }
- Util.inherit(StrikeOutAnnotationElement, AnnotationElement, {
-  render: function StrikeOutAnnotationElement_render() {
-   this.container.className = 'strikeoutAnnotation';
-   if (!this.data.hasPopup) {
-    this._createPopup(this.container, null, this.data);
-   }
-   return this.container;
+  function StrikeOutAnnotationElement(parameters) {
+    var isRenderable = !!(parameters.data.hasPopup || parameters.data.title || parameters.data.contents);
+    AnnotationElement.call(this, parameters, isRenderable);
   }
- });
- return StrikeOutAnnotationElement;
+  Util.inherit(StrikeOutAnnotationElement, AnnotationElement, {
+    render: function StrikeOutAnnotationElement_render() {
+      this.container.className = 'strikeoutAnnotation';
+      if (!this.data.hasPopup) {
+        this._createPopup(this.container, null, this.data);
+      }
+      return this.container;
+    }
+  });
+  return StrikeOutAnnotationElement;
 }();
 var FileAttachmentAnnotationElement = function FileAttachmentAnnotationElementClosure() {
- function FileAttachmentAnnotationElement(parameters) {
-  AnnotationElement.call(this, parameters, true);
-  var file = this.data.file;
-  this.filename = getFilenameFromUrl(file.filename);
-  this.content = file.content;
-  this.linkService.onFileAttachmentAnnotation({
-   id: stringToPDFString(file.filename),
-   filename: file.filename,
-   content: file.content
-  });
- }
- Util.inherit(FileAttachmentAnnotationElement, AnnotationElement, {
-  render: function FileAttachmentAnnotationElement_render() {
-   this.container.className = 'fileAttachmentAnnotation';
-   var trigger = document.createElement('div');
-   trigger.style.height = this.container.style.height;
-   trigger.style.width = this.container.style.width;
-   trigger.addEventListener('dblclick', this._download.bind(this));
-   if (!this.data.hasPopup && (this.data.title || this.data.contents)) {
-    this._createPopup(this.container, trigger, this.data);
-   }
-   this.container.appendChild(trigger);
-   return this.container;
-  },
-  _download: function FileAttachmentAnnotationElement_download() {
-   if (!this.downloadManager) {
-    warn('Download cannot be started due to unavailable download manager');
-    return;
-   }
-   this.downloadManager.downloadData(this.content, this.filename, '');
+  function FileAttachmentAnnotationElement(parameters) {
+    AnnotationElement.call(this, parameters, true);
+    var file = this.data.file;
+    this.filename = getFilenameFromUrl(file.filename);
+    this.content = file.content;
+    this.linkService.onFileAttachmentAnnotation({
+      id: stringToPDFString(file.filename),
+      filename: file.filename,
+      content: file.content
+    });
   }
- });
- return FileAttachmentAnnotationElement;
+  Util.inherit(FileAttachmentAnnotationElement, AnnotationElement, {
+    render: function FileAttachmentAnnotationElement_render() {
+      this.container.className = 'fileAttachmentAnnotation';
+      var trigger = document.createElement('div');
+      trigger.style.height = this.container.style.height;
+      trigger.style.width = this.container.style.width;
+      trigger.addEventListener('dblclick', this._download.bind(this));
+      if (!this.data.hasPopup && (this.data.title || this.data.contents)) {
+        this._createPopup(this.container, trigger, this.data);
+      }
+      this.container.appendChild(trigger);
+      return this.container;
+    },
+    _download: function FileAttachmentAnnotationElement_download() {
+      if (!this.downloadManager) {
+        warn('Download cannot be started due to unavailable download manager');
+        return;
+      }
+      this.downloadManager.downloadData(this.content, this.filename, '');
+    }
+  });
+  return FileAttachmentAnnotationElement;
 }();
 var AnnotationLayer = function AnnotationLayerClosure() {
- return {
-  render: function AnnotationLayer_render(parameters) {
-   var annotationElementFactory = new AnnotationElementFactory();
-   for (var i = 0, ii = parameters.annotations.length; i < ii; i++) {
-    var data = parameters.annotations[i];
-    if (!data) {
-     continue;
-    }
-    var element = annotationElementFactory.create({
-     data: data,
-     layer: parameters.div,
-     page: parameters.page,
-     viewport: parameters.viewport,
-     linkService: parameters.linkService,
-     downloadManager: parameters.downloadManager,
-     imageResourcesPath: parameters.imageResourcesPath || getDefaultSetting('imageResourcesPath'),
-     renderInteractiveForms: parameters.renderInteractiveForms || false
-    });
-    if (element.isRenderable) {
-     parameters.div.appendChild(element.render());
+  return {
+    render: function AnnotationLayer_render(parameters) {
+      var annotationElementFactory = new AnnotationElementFactory();
+      for (var i = 0, ii = parameters.annotations.length; i < ii; i++) {
+        var data = parameters.annotations[i];
+        if (!data) {
+          continue;
+        }
+        var element = annotationElementFactory.create({
+          data: data,
+          layer: parameters.div,
+          page: parameters.page,
+          viewport: parameters.viewport,
+          linkService: parameters.linkService,
+          downloadManager: parameters.downloadManager,
+          imageResourcesPath: parameters.imageResourcesPath || getDefaultSetting('imageResourcesPath'),
+          renderInteractiveForms: parameters.renderInteractiveForms || false
+        });
+        if (element.isRenderable) {
+          parameters.div.appendChild(element.render());
+        }
+      }
+    },
+    update: function AnnotationLayer_update(parameters) {
+      for (var i = 0, ii = parameters.annotations.length; i < ii; i++) {
+        var data = parameters.annotations[i];
+        var element = parameters.div.querySelector('[data-annotation-id="' + data.id + '"]');
+        if (element) {
+          CustomStyle.setProp('transform', element, 'matrix(' + parameters.viewport.transform.join(',') + ')');
+        }
+      }
+      parameters.div.removeAttribute('hidden');
     }
-   }
-  },
-  update: function AnnotationLayer_update(parameters) {
-   for (var i = 0, ii = parameters.annotations.length; i < ii; i++) {
-    var data = parameters.annotations[i];
-    var element = parameters.div.querySelector('[data-annotation-id="' + data.id + '"]');
-    if (element) {
-     CustomStyle.setProp('transform', element, 'matrix(' + parameters.viewport.transform.join(',') + ')');
-    }
-   }
-   parameters.div.removeAttribute('hidden');
-  }
- };
+  };
 }();
 exports.AnnotationLayer = AnnotationLayer;

+ 1259 - 1261
lib/display/api.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var displayFontLoader = require('./font_loader.js');
 var displayCanvas = require('./canvas.js');
@@ -56,1326 +57,1323 @@ var pdfjsFilePath = typeof document !== 'undefined' && document.currentScript ?
 var fakeWorkerFilesLoader = null;
 var useRequireEnsure = false;
 if (typeof __pdfjsdev_webpack__ === 'undefined') {
- if (typeof window === 'undefined') {
-  isWorkerDisabled = true;
-  if (typeof require.ensure === 'undefined') {
-   require.ensure = require('node-ensure');
+  if (typeof window === 'undefined') {
+    isWorkerDisabled = true;
+    if (typeof require.ensure === 'undefined') {
+      require.ensure = require('node-ensure');
+    }
+    useRequireEnsure = true;
+  } else if (typeof require !== 'undefined' && typeof require.ensure === 'function') {
+    useRequireEnsure = true;
   }
-  useRequireEnsure = true;
- } else if (typeof require !== 'undefined' && typeof require.ensure === 'function') {
-  useRequireEnsure = true;
- }
- if (typeof requirejs !== 'undefined' && requirejs.toUrl) {
-  workerSrc = requirejs.toUrl('pdfjs-dist/build/pdf.worker.js');
- }
- var dynamicLoaderSupported = typeof requirejs !== 'undefined' && requirejs.load;
- fakeWorkerFilesLoader = useRequireEnsure ? function (callback) {
-  require.ensure([], function () {
-   var worker = require('./pdf.worker.js');
-   callback(worker.WorkerMessageHandler);
-  });
- } : dynamicLoaderSupported ? function (callback) {
-  requirejs(['pdfjs-dist/build/pdf.worker'], function (worker) {
-   callback(worker.WorkerMessageHandler);
-  });
- } : null;
+  if (typeof requirejs !== 'undefined' && requirejs.toUrl) {
+    workerSrc = requirejs.toUrl('pdfjs-dist/build/pdf.worker.js');
+  }
+  var dynamicLoaderSupported = typeof requirejs !== 'undefined' && requirejs.load;
+  fakeWorkerFilesLoader = useRequireEnsure ? function (callback) {
+    require.ensure([], function () {
+      var worker = require('./pdf.worker.js');
+      callback(worker.WorkerMessageHandler);
+    });
+  } : dynamicLoaderSupported ? function (callback) {
+    requirejs(['pdfjs-dist/build/pdf.worker'], function (worker) {
+      callback(worker.WorkerMessageHandler);
+    });
+  } : null;
 }
 function getDocument(src, pdfDataRangeTransport, passwordCallback, progressCallback) {
- var task = new PDFDocumentLoadingTask();
- if (arguments.length > 1) {
-  deprecated('getDocument is called with pdfDataRangeTransport, ' + 'passwordCallback or progressCallback argument');
- }
- if (pdfDataRangeTransport) {
-  if (!(pdfDataRangeTransport instanceof PDFDataRangeTransport)) {
-   pdfDataRangeTransport = Object.create(pdfDataRangeTransport);
-   pdfDataRangeTransport.length = src.length;
-   pdfDataRangeTransport.initialData = src.initialData;
-   if (!pdfDataRangeTransport.abort) {
-    pdfDataRangeTransport.abort = function () {
-    };
-   }
+  var task = new PDFDocumentLoadingTask();
+  if (arguments.length > 1) {
+    deprecated('getDocument is called with pdfDataRangeTransport, ' + 'passwordCallback or progressCallback argument');
   }
-  src = Object.create(src);
-  src.range = pdfDataRangeTransport;
- }
- task.onPassword = passwordCallback || null;
- task.onProgress = progressCallback || null;
- var source;
- if (typeof src === 'string') {
-  source = { url: src };
- } else if (isArrayBuffer(src)) {
-  source = { data: src };
- } else if (src instanceof PDFDataRangeTransport) {
-  source = { range: src };
- } else {
-  if (typeof src !== 'object') {
-   error('Invalid parameter in getDocument, need either Uint8Array, ' + 'string or a parameter object');
+  if (pdfDataRangeTransport) {
+    if (!(pdfDataRangeTransport instanceof PDFDataRangeTransport)) {
+      pdfDataRangeTransport = Object.create(pdfDataRangeTransport);
+      pdfDataRangeTransport.length = src.length;
+      pdfDataRangeTransport.initialData = src.initialData;
+      if (!pdfDataRangeTransport.abort) {
+        pdfDataRangeTransport.abort = function () {};
+      }
+    }
+    src = Object.create(src);
+    src.range = pdfDataRangeTransport;
   }
-  if (!src.url && !src.data && !src.range) {
-   error('Invalid parameter object: need either .data, .range or .url');
+  task.onPassword = passwordCallback || null;
+  task.onProgress = progressCallback || null;
+  var source;
+  if (typeof src === 'string') {
+    source = { url: src };
+  } else if (isArrayBuffer(src)) {
+    source = { data: src };
+  } else if (src instanceof PDFDataRangeTransport) {
+    source = { range: src };
+  } else {
+    if (typeof src !== 'object') {
+      error('Invalid parameter in getDocument, need either Uint8Array, ' + 'string or a parameter object');
+    }
+    if (!src.url && !src.data && !src.range) {
+      error('Invalid parameter object: need either .data, .range or .url');
+    }
+    source = src;
   }
-  source = src;
- }
- var params = {};
- var rangeTransport = null;
- var worker = null;
- for (var key in source) {
-  if (key === 'url' && typeof window !== 'undefined') {
-   params[key] = new URL(source[key], window.location).href;
-   continue;
-  } else if (key === 'range') {
-   rangeTransport = source[key];
-   continue;
-  } else if (key === 'worker') {
-   worker = source[key];
-   continue;
-  } else if (key === 'data' && !(source[key] instanceof Uint8Array)) {
-   var pdfBytes = source[key];
-   if (typeof pdfBytes === 'string') {
-    params[key] = stringToBytes(pdfBytes);
-   } else if (typeof pdfBytes === 'object' && pdfBytes !== null && !isNaN(pdfBytes.length)) {
-    params[key] = new Uint8Array(pdfBytes);
-   } else if (isArrayBuffer(pdfBytes)) {
-    params[key] = new Uint8Array(pdfBytes);
-   } else {
-    error('Invalid PDF binary data: either typed array, string or ' + 'array-like object is expected in the data property.');
-   }
-   continue;
+  var params = {};
+  var rangeTransport = null;
+  var worker = null;
+  for (var key in source) {
+    if (key === 'url' && typeof window !== 'undefined') {
+      params[key] = new URL(source[key], window.location).href;
+      continue;
+    } else if (key === 'range') {
+      rangeTransport = source[key];
+      continue;
+    } else if (key === 'worker') {
+      worker = source[key];
+      continue;
+    } else if (key === 'data' && !(source[key] instanceof Uint8Array)) {
+      var pdfBytes = source[key];
+      if (typeof pdfBytes === 'string') {
+        params[key] = stringToBytes(pdfBytes);
+      } else if (typeof pdfBytes === 'object' && pdfBytes !== null && !isNaN(pdfBytes.length)) {
+        params[key] = new Uint8Array(pdfBytes);
+      } else if (isArrayBuffer(pdfBytes)) {
+        params[key] = new Uint8Array(pdfBytes);
+      } else {
+        error('Invalid PDF binary data: either typed array, string or ' + 'array-like object is expected in the data property.');
+      }
+      continue;
+    }
+    params[key] = source[key];
   }
-  params[key] = source[key];
- }
- params.rangeChunkSize = params.rangeChunkSize || DEFAULT_RANGE_CHUNK_SIZE;
- params.disableNativeImageDecoder = params.disableNativeImageDecoder === true;
- var CMapReaderFactory = params.CMapReaderFactory || DOMCMapReaderFactory;
- if (!worker) {
-  var workerPort = getDefaultSetting('workerPort');
-  worker = workerPort ? new PDFWorker(null, workerPort) : new PDFWorker();
-  task._worker = worker;
- }
- var docId = task.docId;
- worker.promise.then(function () {
-  if (task.destroyed) {
-   throw new Error('Loading aborted');
+  params.rangeChunkSize = params.rangeChunkSize || DEFAULT_RANGE_CHUNK_SIZE;
+  params.disableNativeImageDecoder = params.disableNativeImageDecoder === true;
+  var CMapReaderFactory = params.CMapReaderFactory || DOMCMapReaderFactory;
+  if (!worker) {
+    var workerPort = getDefaultSetting('workerPort');
+    worker = workerPort ? new PDFWorker(null, workerPort) : new PDFWorker();
+    task._worker = worker;
   }
-  return _fetchDocument(worker, params, rangeTransport, docId).then(function (workerId) {
-   if (task.destroyed) {
-    throw new Error('Loading aborted');
-   }
-   var messageHandler = new MessageHandler(docId, workerId, worker.port);
-   var transport = new WorkerTransport(messageHandler, task, rangeTransport, CMapReaderFactory);
-   task._transport = transport;
-   messageHandler.send('Ready', null);
-  });
- }).catch(task._capability.reject);
- return task;
+  var docId = task.docId;
+  worker.promise.then(function () {
+    if (task.destroyed) {
+      throw new Error('Loading aborted');
+    }
+    return _fetchDocument(worker, params, rangeTransport, docId).then(function (workerId) {
+      if (task.destroyed) {
+        throw new Error('Loading aborted');
+      }
+      var messageHandler = new MessageHandler(docId, workerId, worker.port);
+      var transport = new WorkerTransport(messageHandler, task, rangeTransport, CMapReaderFactory);
+      task._transport = transport;
+      messageHandler.send('Ready', null);
+    });
+  }).catch(task._capability.reject);
+  return task;
 }
 function _fetchDocument(worker, source, pdfDataRangeTransport, docId) {
- if (worker.destroyed) {
-  return Promise.reject(new Error('Worker was destroyed'));
- }
- source.disableAutoFetch = getDefaultSetting('disableAutoFetch');
- source.disableStream = getDefaultSetting('disableStream');
- source.chunkedViewerLoading = !!pdfDataRangeTransport;
- if (pdfDataRangeTransport) {
-  source.length = pdfDataRangeTransport.length;
-  source.initialData = pdfDataRangeTransport.initialData;
- }
- return worker.messageHandler.sendWithPromise('GetDocRequest', {
-  docId: docId,
-  source: source,
-  disableRange: getDefaultSetting('disableRange'),
-  maxImageSize: getDefaultSetting('maxImageSize'),
-  disableFontFace: getDefaultSetting('disableFontFace'),
-  disableCreateObjectURL: getDefaultSetting('disableCreateObjectURL'),
-  postMessageTransfers: getDefaultSetting('postMessageTransfers') && !isPostMessageTransfersDisabled,
-  docBaseUrl: source.docBaseUrl,
-  disableNativeImageDecoder: source.disableNativeImageDecoder
- }).then(function (workerId) {
   if (worker.destroyed) {
-   throw new Error('Worker was destroyed');
+    return Promise.reject(new Error('Worker was destroyed'));
+  }
+  source.disableAutoFetch = getDefaultSetting('disableAutoFetch');
+  source.disableStream = getDefaultSetting('disableStream');
+  source.chunkedViewerLoading = !!pdfDataRangeTransport;
+  if (pdfDataRangeTransport) {
+    source.length = pdfDataRangeTransport.length;
+    source.initialData = pdfDataRangeTransport.initialData;
   }
-  return workerId;
- });
+  return worker.messageHandler.sendWithPromise('GetDocRequest', {
+    docId: docId,
+    source: source,
+    disableRange: getDefaultSetting('disableRange'),
+    maxImageSize: getDefaultSetting('maxImageSize'),
+    disableFontFace: getDefaultSetting('disableFontFace'),
+    disableCreateObjectURL: getDefaultSetting('disableCreateObjectURL'),
+    postMessageTransfers: getDefaultSetting('postMessageTransfers') && !isPostMessageTransfersDisabled,
+    docBaseUrl: source.docBaseUrl,
+    disableNativeImageDecoder: source.disableNativeImageDecoder
+  }).then(function (workerId) {
+    if (worker.destroyed) {
+      throw new Error('Worker was destroyed');
+    }
+    return workerId;
+  });
 }
 var PDFDocumentLoadingTask = function PDFDocumentLoadingTaskClosure() {
- var nextDocumentId = 0;
- function PDFDocumentLoadingTask() {
-  this._capability = createPromiseCapability();
-  this._transport = null;
-  this._worker = null;
-  this.docId = 'd' + nextDocumentId++;
-  this.destroyed = false;
-  this.onPassword = null;
-  this.onProgress = null;
-  this.onUnsupportedFeature = null;
- }
- PDFDocumentLoadingTask.prototype = {
-  get promise() {
-   return this._capability.promise;
-  },
-  destroy: function () {
-   this.destroyed = true;
-   var transportDestroyed = !this._transport ? Promise.resolve() : this._transport.destroy();
-   return transportDestroyed.then(function () {
+  var nextDocumentId = 0;
+  function PDFDocumentLoadingTask() {
+    this._capability = createPromiseCapability();
     this._transport = null;
-    if (this._worker) {
-     this._worker.destroy();
-     this._worker = null;
-    }
-   }.bind(this));
-  },
-  then: function PDFDocumentLoadingTask_then(onFulfilled, onRejected) {
-   return this.promise.then.apply(this.promise, arguments);
+    this._worker = null;
+    this.docId = 'd' + nextDocumentId++;
+    this.destroyed = false;
+    this.onPassword = null;
+    this.onProgress = null;
+    this.onUnsupportedFeature = null;
   }
- };
- return PDFDocumentLoadingTask;
+  PDFDocumentLoadingTask.prototype = {
+    get promise() {
+      return this._capability.promise;
+    },
+    destroy: function () {
+      this.destroyed = true;
+      var transportDestroyed = !this._transport ? Promise.resolve() : this._transport.destroy();
+      return transportDestroyed.then(function () {
+        this._transport = null;
+        if (this._worker) {
+          this._worker.destroy();
+          this._worker = null;
+        }
+      }.bind(this));
+    },
+    then: function PDFDocumentLoadingTask_then(onFulfilled, onRejected) {
+      return this.promise.then.apply(this.promise, arguments);
+    }
+  };
+  return PDFDocumentLoadingTask;
 }();
 var PDFDataRangeTransport = function pdfDataRangeTransportClosure() {
- function PDFDataRangeTransport(length, initialData) {
-  this.length = length;
-  this.initialData = initialData;
-  this._rangeListeners = [];
-  this._progressListeners = [];
-  this._progressiveReadListeners = [];
-  this._readyCapability = createPromiseCapability();
- }
- PDFDataRangeTransport.prototype = {
-  addRangeListener: function PDFDataRangeTransport_addRangeListener(listener) {
-   this._rangeListeners.push(listener);
-  },
-  addProgressListener: function PDFDataRangeTransport_addProgressListener(listener) {
-   this._progressListeners.push(listener);
-  },
-  addProgressiveReadListener: function PDFDataRangeTransport_addProgressiveReadListener(listener) {
-   this._progressiveReadListeners.push(listener);
-  },
-  onDataRange: function PDFDataRangeTransport_onDataRange(begin, chunk) {
-   var listeners = this._rangeListeners;
-   for (var i = 0, n = listeners.length; i < n; ++i) {
-    listeners[i](begin, chunk);
-   }
-  },
-  onDataProgress: function PDFDataRangeTransport_onDataProgress(loaded) {
-   this._readyCapability.promise.then(function () {
-    var listeners = this._progressListeners;
-    for (var i = 0, n = listeners.length; i < n; ++i) {
-     listeners[i](loaded);
-    }
-   }.bind(this));
-  },
-  onDataProgressiveRead: function PDFDataRangeTransport_onDataProgress(chunk) {
-   this._readyCapability.promise.then(function () {
-    var listeners = this._progressiveReadListeners;
-    for (var i = 0, n = listeners.length; i < n; ++i) {
-     listeners[i](chunk);
-    }
-   }.bind(this));
-  },
-  transportReady: function PDFDataRangeTransport_transportReady() {
-   this._readyCapability.resolve();
-  },
-  requestDataRange: function PDFDataRangeTransport_requestDataRange(begin, end) {
-   throw new Error('Abstract method PDFDataRangeTransport.requestDataRange');
-  },
-  abort: function PDFDataRangeTransport_abort() {
+  function PDFDataRangeTransport(length, initialData) {
+    this.length = length;
+    this.initialData = initialData;
+    this._rangeListeners = [];
+    this._progressListeners = [];
+    this._progressiveReadListeners = [];
+    this._readyCapability = createPromiseCapability();
   }
- };
- return PDFDataRangeTransport;
+  PDFDataRangeTransport.prototype = {
+    addRangeListener: function PDFDataRangeTransport_addRangeListener(listener) {
+      this._rangeListeners.push(listener);
+    },
+    addProgressListener: function PDFDataRangeTransport_addProgressListener(listener) {
+      this._progressListeners.push(listener);
+    },
+    addProgressiveReadListener: function PDFDataRangeTransport_addProgressiveReadListener(listener) {
+      this._progressiveReadListeners.push(listener);
+    },
+    onDataRange: function PDFDataRangeTransport_onDataRange(begin, chunk) {
+      var listeners = this._rangeListeners;
+      for (var i = 0, n = listeners.length; i < n; ++i) {
+        listeners[i](begin, chunk);
+      }
+    },
+    onDataProgress: function PDFDataRangeTransport_onDataProgress(loaded) {
+      this._readyCapability.promise.then(function () {
+        var listeners = this._progressListeners;
+        for (var i = 0, n = listeners.length; i < n; ++i) {
+          listeners[i](loaded);
+        }
+      }.bind(this));
+    },
+    onDataProgressiveRead: function PDFDataRangeTransport_onDataProgress(chunk) {
+      this._readyCapability.promise.then(function () {
+        var listeners = this._progressiveReadListeners;
+        for (var i = 0, n = listeners.length; i < n; ++i) {
+          listeners[i](chunk);
+        }
+      }.bind(this));
+    },
+    transportReady: function PDFDataRangeTransport_transportReady() {
+      this._readyCapability.resolve();
+    },
+    requestDataRange: function PDFDataRangeTransport_requestDataRange(begin, end) {
+      throw new Error('Abstract method PDFDataRangeTransport.requestDataRange');
+    },
+    abort: function PDFDataRangeTransport_abort() {}
+  };
+  return PDFDataRangeTransport;
 }();
 var PDFDocumentProxy = function PDFDocumentProxyClosure() {
- function PDFDocumentProxy(pdfInfo, transport, loadingTask) {
-  this.pdfInfo = pdfInfo;
-  this.transport = transport;
-  this.loadingTask = loadingTask;
- }
- PDFDocumentProxy.prototype = {
-  get numPages() {
-   return this.pdfInfo.numPages;
-  },
-  get fingerprint() {
-   return this.pdfInfo.fingerprint;
-  },
-  getPage: function PDFDocumentProxy_getPage(pageNumber) {
-   return this.transport.getPage(pageNumber);
-  },
-  getPageIndex: function PDFDocumentProxy_getPageIndex(ref) {
-   return this.transport.getPageIndex(ref);
-  },
-  getDestinations: function PDFDocumentProxy_getDestinations() {
-   return this.transport.getDestinations();
-  },
-  getDestination: function PDFDocumentProxy_getDestination(id) {
-   return this.transport.getDestination(id);
-  },
-  getPageLabels: function PDFDocumentProxy_getPageLabels() {
-   return this.transport.getPageLabels();
-  },
-  getAttachments: function PDFDocumentProxy_getAttachments() {
-   return this.transport.getAttachments();
-  },
-  getJavaScript: function PDFDocumentProxy_getJavaScript() {
-   return this.transport.getJavaScript();
-  },
-  getOutline: function PDFDocumentProxy_getOutline() {
-   return this.transport.getOutline();
-  },
-  getMetadata: function PDFDocumentProxy_getMetadata() {
-   return this.transport.getMetadata();
-  },
-  getData: function PDFDocumentProxy_getData() {
-   return this.transport.getData();
-  },
-  getDownloadInfo: function PDFDocumentProxy_getDownloadInfo() {
-   return this.transport.downloadInfoCapability.promise;
-  },
-  getStats: function PDFDocumentProxy_getStats() {
-   return this.transport.getStats();
-  },
-  cleanup: function PDFDocumentProxy_cleanup() {
-   this.transport.startCleanup();
-  },
-  destroy: function PDFDocumentProxy_destroy() {
-   return this.loadingTask.destroy();
+  function PDFDocumentProxy(pdfInfo, transport, loadingTask) {
+    this.pdfInfo = pdfInfo;
+    this.transport = transport;
+    this.loadingTask = loadingTask;
   }
- };
- return PDFDocumentProxy;
+  PDFDocumentProxy.prototype = {
+    get numPages() {
+      return this.pdfInfo.numPages;
+    },
+    get fingerprint() {
+      return this.pdfInfo.fingerprint;
+    },
+    getPage: function PDFDocumentProxy_getPage(pageNumber) {
+      return this.transport.getPage(pageNumber);
+    },
+    getPageIndex: function PDFDocumentProxy_getPageIndex(ref) {
+      return this.transport.getPageIndex(ref);
+    },
+    getDestinations: function PDFDocumentProxy_getDestinations() {
+      return this.transport.getDestinations();
+    },
+    getDestination: function PDFDocumentProxy_getDestination(id) {
+      return this.transport.getDestination(id);
+    },
+    getPageLabels: function PDFDocumentProxy_getPageLabels() {
+      return this.transport.getPageLabels();
+    },
+    getAttachments: function PDFDocumentProxy_getAttachments() {
+      return this.transport.getAttachments();
+    },
+    getJavaScript: function PDFDocumentProxy_getJavaScript() {
+      return this.transport.getJavaScript();
+    },
+    getOutline: function PDFDocumentProxy_getOutline() {
+      return this.transport.getOutline();
+    },
+    getMetadata: function PDFDocumentProxy_getMetadata() {
+      return this.transport.getMetadata();
+    },
+    getData: function PDFDocumentProxy_getData() {
+      return this.transport.getData();
+    },
+    getDownloadInfo: function PDFDocumentProxy_getDownloadInfo() {
+      return this.transport.downloadInfoCapability.promise;
+    },
+    getStats: function PDFDocumentProxy_getStats() {
+      return this.transport.getStats();
+    },
+    cleanup: function PDFDocumentProxy_cleanup() {
+      this.transport.startCleanup();
+    },
+    destroy: function PDFDocumentProxy_destroy() {
+      return this.loadingTask.destroy();
+    }
+  };
+  return PDFDocumentProxy;
 }();
 var PDFPageProxy = function PDFPageProxyClosure() {
- function PDFPageProxy(pageIndex, pageInfo, transport) {
-  this.pageIndex = pageIndex;
-  this.pageInfo = pageInfo;
-  this.transport = transport;
-  this.stats = new StatTimer();
-  this.stats.enabled = getDefaultSetting('enableStats');
-  this.commonObjs = transport.commonObjs;
-  this.objs = new PDFObjects();
-  this.cleanupAfterRender = false;
-  this.pendingCleanup = false;
-  this.intentStates = Object.create(null);
-  this.destroyed = false;
- }
- PDFPageProxy.prototype = {
-  get pageNumber() {
-   return this.pageIndex + 1;
-  },
-  get rotate() {
-   return this.pageInfo.rotate;
-  },
-  get ref() {
-   return this.pageInfo.ref;
-  },
-  get userUnit() {
-   return this.pageInfo.userUnit;
-  },
-  get view() {
-   return this.pageInfo.view;
-  },
-  getViewport: function PDFPageProxy_getViewport(scale, rotate) {
-   if (arguments.length < 2) {
-    rotate = this.rotate;
-   }
-   return new PageViewport(this.view, scale, rotate, 0, 0);
-  },
-  getAnnotations: function PDFPageProxy_getAnnotations(params) {
-   var intent = params && params.intent || null;
-   if (!this.annotationsPromise || this.annotationsIntent !== intent) {
-    this.annotationsPromise = this.transport.getAnnotations(this.pageIndex, intent);
-    this.annotationsIntent = intent;
-   }
-   return this.annotationsPromise;
-  },
-  render: function PDFPageProxy_render(params) {
-   var stats = this.stats;
-   stats.time('Overall');
-   this.pendingCleanup = false;
-   var renderingIntent = params.intent === 'print' ? 'print' : 'display';
-   var renderInteractiveForms = params.renderInteractiveForms === true ? true : false;
-   var canvasFactory = params.canvasFactory || new DOMCanvasFactory();
-   if (!this.intentStates[renderingIntent]) {
-    this.intentStates[renderingIntent] = Object.create(null);
-   }
-   var intentState = this.intentStates[renderingIntent];
-   if (!intentState.displayReadyCapability) {
-    intentState.receivingOperatorList = true;
-    intentState.displayReadyCapability = createPromiseCapability();
-    intentState.operatorList = {
-     fnArray: [],
-     argsArray: [],
-     lastChunk: false
-    };
-    this.stats.time('Page Request');
-    this.transport.messageHandler.send('RenderPageRequest', {
-     pageIndex: this.pageNumber - 1,
-     intent: renderingIntent,
-     renderInteractiveForms: renderInteractiveForms
-    });
-   }
-   var internalRenderTask = new InternalRenderTask(complete, params, this.objs, this.commonObjs, intentState.operatorList, this.pageNumber, canvasFactory);
-   internalRenderTask.useRequestAnimationFrame = renderingIntent !== 'print';
-   if (!intentState.renderTasks) {
-    intentState.renderTasks = [];
-   }
-   intentState.renderTasks.push(internalRenderTask);
-   var renderTask = internalRenderTask.task;
-   if (params.continueCallback) {
-    deprecated('render is used with continueCallback parameter');
-    renderTask.onContinue = params.continueCallback;
-   }
-   var self = this;
-   intentState.displayReadyCapability.promise.then(function pageDisplayReadyPromise(transparency) {
-    if (self.pendingCleanup) {
-     complete();
-     return;
-    }
-    stats.time('Rendering');
-    internalRenderTask.initializeGraphics(transparency);
-    internalRenderTask.operatorListChanged();
-   }, function pageDisplayReadPromiseError(reason) {
-    complete(reason);
-   });
-   function complete(error) {
-    var i = intentState.renderTasks.indexOf(internalRenderTask);
-    if (i >= 0) {
-     intentState.renderTasks.splice(i, 1);
-    }
-    if (self.cleanupAfterRender) {
-     self.pendingCleanup = true;
-    }
-    self._tryCleanup();
-    if (error) {
-     internalRenderTask.capability.reject(error);
-    } else {
-     internalRenderTask.capability.resolve();
-    }
-    stats.timeEnd('Rendering');
-    stats.timeEnd('Overall');
-   }
-   return renderTask;
-  },
-  getOperatorList: function PDFPageProxy_getOperatorList() {
-   function operatorListChanged() {
-    if (intentState.operatorList.lastChunk) {
-     intentState.opListReadCapability.resolve(intentState.operatorList);
-     var i = intentState.renderTasks.indexOf(opListTask);
-     if (i >= 0) {
-      intentState.renderTasks.splice(i, 1);
-     }
-    }
-   }
-   var renderingIntent = 'oplist';
-   if (!this.intentStates[renderingIntent]) {
-    this.intentStates[renderingIntent] = Object.create(null);
-   }
-   var intentState = this.intentStates[renderingIntent];
-   var opListTask;
-   if (!intentState.opListReadCapability) {
-    opListTask = {};
-    opListTask.operatorListChanged = operatorListChanged;
-    intentState.receivingOperatorList = true;
-    intentState.opListReadCapability = createPromiseCapability();
-    intentState.renderTasks = [];
-    intentState.renderTasks.push(opListTask);
-    intentState.operatorList = {
-     fnArray: [],
-     argsArray: [],
-     lastChunk: false
-    };
-    this.transport.messageHandler.send('RenderPageRequest', {
-     pageIndex: this.pageIndex,
-     intent: renderingIntent
-    });
-   }
-   return intentState.opListReadCapability.promise;
-  },
-  getTextContent: function PDFPageProxy_getTextContent(params) {
-   return this.transport.messageHandler.sendWithPromise('GetTextContent', {
-    pageIndex: this.pageNumber - 1,
-    normalizeWhitespace: params && params.normalizeWhitespace === true ? true : false,
-    combineTextItems: params && params.disableCombineTextItems === true ? false : true
-   });
-  },
-  _destroy: function PDFPageProxy_destroy() {
-   this.destroyed = true;
-   this.transport.pageCache[this.pageIndex] = null;
-   var waitOn = [];
-   Object.keys(this.intentStates).forEach(function (intent) {
-    if (intent === 'oplist') {
-     return;
-    }
-    var intentState = this.intentStates[intent];
-    intentState.renderTasks.forEach(function (renderTask) {
-     var renderCompleted = renderTask.capability.promise.catch(function () {
-     });
-     waitOn.push(renderCompleted);
-     renderTask.cancel();
-    });
-   }, this);
-   this.objs.clear();
-   this.annotationsPromise = null;
-   this.pendingCleanup = false;
-   return Promise.all(waitOn);
-  },
-  destroy: function () {
-   deprecated('page destroy method, use cleanup() instead');
-   this.cleanup();
-  },
-  cleanup: function PDFPageProxy_cleanup() {
-   this.pendingCleanup = true;
-   this._tryCleanup();
-  },
-  _tryCleanup: function PDFPageProxy_tryCleanup() {
-   if (!this.pendingCleanup || Object.keys(this.intentStates).some(function (intent) {
-     var intentState = this.intentStates[intent];
-     return intentState.renderTasks.length !== 0 || intentState.receivingOperatorList;
-    }, this)) {
-    return;
-   }
-   Object.keys(this.intentStates).forEach(function (intent) {
-    delete this.intentStates[intent];
-   }, this);
-   this.objs.clear();
-   this.annotationsPromise = null;
-   this.pendingCleanup = false;
-  },
-  _startRenderPage: function PDFPageProxy_startRenderPage(transparency, intent) {
-   var intentState = this.intentStates[intent];
-   if (intentState.displayReadyCapability) {
-    intentState.displayReadyCapability.resolve(transparency);
-   }
-  },
-  _renderPageChunk: function PDFPageProxy_renderPageChunk(operatorListChunk, intent) {
-   var intentState = this.intentStates[intent];
-   var i, ii;
-   for (i = 0, ii = operatorListChunk.length; i < ii; i++) {
-    intentState.operatorList.fnArray.push(operatorListChunk.fnArray[i]);
-    intentState.operatorList.argsArray.push(operatorListChunk.argsArray[i]);
-   }
-   intentState.operatorList.lastChunk = operatorListChunk.lastChunk;
-   for (i = 0; i < intentState.renderTasks.length; i++) {
-    intentState.renderTasks[i].operatorListChanged();
-   }
-   if (operatorListChunk.lastChunk) {
-    intentState.receivingOperatorList = false;
-    this._tryCleanup();
-   }
+  function PDFPageProxy(pageIndex, pageInfo, transport) {
+    this.pageIndex = pageIndex;
+    this.pageInfo = pageInfo;
+    this.transport = transport;
+    this.stats = new StatTimer();
+    this.stats.enabled = getDefaultSetting('enableStats');
+    this.commonObjs = transport.commonObjs;
+    this.objs = new PDFObjects();
+    this.cleanupAfterRender = false;
+    this.pendingCleanup = false;
+    this.intentStates = Object.create(null);
+    this.destroyed = false;
   }
- };
- return PDFPageProxy;
+  PDFPageProxy.prototype = {
+    get pageNumber() {
+      return this.pageIndex + 1;
+    },
+    get rotate() {
+      return this.pageInfo.rotate;
+    },
+    get ref() {
+      return this.pageInfo.ref;
+    },
+    get userUnit() {
+      return this.pageInfo.userUnit;
+    },
+    get view() {
+      return this.pageInfo.view;
+    },
+    getViewport: function PDFPageProxy_getViewport(scale, rotate) {
+      if (arguments.length < 2) {
+        rotate = this.rotate;
+      }
+      return new PageViewport(this.view, scale, rotate, 0, 0);
+    },
+    getAnnotations: function PDFPageProxy_getAnnotations(params) {
+      var intent = params && params.intent || null;
+      if (!this.annotationsPromise || this.annotationsIntent !== intent) {
+        this.annotationsPromise = this.transport.getAnnotations(this.pageIndex, intent);
+        this.annotationsIntent = intent;
+      }
+      return this.annotationsPromise;
+    },
+    render: function PDFPageProxy_render(params) {
+      var stats = this.stats;
+      stats.time('Overall');
+      this.pendingCleanup = false;
+      var renderingIntent = params.intent === 'print' ? 'print' : 'display';
+      var renderInteractiveForms = params.renderInteractiveForms === true ? true : false;
+      var canvasFactory = params.canvasFactory || new DOMCanvasFactory();
+      if (!this.intentStates[renderingIntent]) {
+        this.intentStates[renderingIntent] = Object.create(null);
+      }
+      var intentState = this.intentStates[renderingIntent];
+      if (!intentState.displayReadyCapability) {
+        intentState.receivingOperatorList = true;
+        intentState.displayReadyCapability = createPromiseCapability();
+        intentState.operatorList = {
+          fnArray: [],
+          argsArray: [],
+          lastChunk: false
+        };
+        this.stats.time('Page Request');
+        this.transport.messageHandler.send('RenderPageRequest', {
+          pageIndex: this.pageNumber - 1,
+          intent: renderingIntent,
+          renderInteractiveForms: renderInteractiveForms
+        });
+      }
+      var internalRenderTask = new InternalRenderTask(complete, params, this.objs, this.commonObjs, intentState.operatorList, this.pageNumber, canvasFactory);
+      internalRenderTask.useRequestAnimationFrame = renderingIntent !== 'print';
+      if (!intentState.renderTasks) {
+        intentState.renderTasks = [];
+      }
+      intentState.renderTasks.push(internalRenderTask);
+      var renderTask = internalRenderTask.task;
+      if (params.continueCallback) {
+        deprecated('render is used with continueCallback parameter');
+        renderTask.onContinue = params.continueCallback;
+      }
+      var self = this;
+      intentState.displayReadyCapability.promise.then(function pageDisplayReadyPromise(transparency) {
+        if (self.pendingCleanup) {
+          complete();
+          return;
+        }
+        stats.time('Rendering');
+        internalRenderTask.initializeGraphics(transparency);
+        internalRenderTask.operatorListChanged();
+      }, function pageDisplayReadPromiseError(reason) {
+        complete(reason);
+      });
+      function complete(error) {
+        var i = intentState.renderTasks.indexOf(internalRenderTask);
+        if (i >= 0) {
+          intentState.renderTasks.splice(i, 1);
+        }
+        if (self.cleanupAfterRender) {
+          self.pendingCleanup = true;
+        }
+        self._tryCleanup();
+        if (error) {
+          internalRenderTask.capability.reject(error);
+        } else {
+          internalRenderTask.capability.resolve();
+        }
+        stats.timeEnd('Rendering');
+        stats.timeEnd('Overall');
+      }
+      return renderTask;
+    },
+    getOperatorList: function PDFPageProxy_getOperatorList() {
+      function operatorListChanged() {
+        if (intentState.operatorList.lastChunk) {
+          intentState.opListReadCapability.resolve(intentState.operatorList);
+          var i = intentState.renderTasks.indexOf(opListTask);
+          if (i >= 0) {
+            intentState.renderTasks.splice(i, 1);
+          }
+        }
+      }
+      var renderingIntent = 'oplist';
+      if (!this.intentStates[renderingIntent]) {
+        this.intentStates[renderingIntent] = Object.create(null);
+      }
+      var intentState = this.intentStates[renderingIntent];
+      var opListTask;
+      if (!intentState.opListReadCapability) {
+        opListTask = {};
+        opListTask.operatorListChanged = operatorListChanged;
+        intentState.receivingOperatorList = true;
+        intentState.opListReadCapability = createPromiseCapability();
+        intentState.renderTasks = [];
+        intentState.renderTasks.push(opListTask);
+        intentState.operatorList = {
+          fnArray: [],
+          argsArray: [],
+          lastChunk: false
+        };
+        this.transport.messageHandler.send('RenderPageRequest', {
+          pageIndex: this.pageIndex,
+          intent: renderingIntent
+        });
+      }
+      return intentState.opListReadCapability.promise;
+    },
+    getTextContent: function PDFPageProxy_getTextContent(params) {
+      return this.transport.messageHandler.sendWithPromise('GetTextContent', {
+        pageIndex: this.pageNumber - 1,
+        normalizeWhitespace: params && params.normalizeWhitespace === true ? true : false,
+        combineTextItems: params && params.disableCombineTextItems === true ? false : true
+      });
+    },
+    _destroy: function PDFPageProxy_destroy() {
+      this.destroyed = true;
+      this.transport.pageCache[this.pageIndex] = null;
+      var waitOn = [];
+      Object.keys(this.intentStates).forEach(function (intent) {
+        if (intent === 'oplist') {
+          return;
+        }
+        var intentState = this.intentStates[intent];
+        intentState.renderTasks.forEach(function (renderTask) {
+          var renderCompleted = renderTask.capability.promise.catch(function () {});
+          waitOn.push(renderCompleted);
+          renderTask.cancel();
+        });
+      }, this);
+      this.objs.clear();
+      this.annotationsPromise = null;
+      this.pendingCleanup = false;
+      return Promise.all(waitOn);
+    },
+    destroy: function () {
+      deprecated('page destroy method, use cleanup() instead');
+      this.cleanup();
+    },
+    cleanup: function PDFPageProxy_cleanup() {
+      this.pendingCleanup = true;
+      this._tryCleanup();
+    },
+    _tryCleanup: function PDFPageProxy_tryCleanup() {
+      if (!this.pendingCleanup || Object.keys(this.intentStates).some(function (intent) {
+        var intentState = this.intentStates[intent];
+        return intentState.renderTasks.length !== 0 || intentState.receivingOperatorList;
+      }, this)) {
+        return;
+      }
+      Object.keys(this.intentStates).forEach(function (intent) {
+        delete this.intentStates[intent];
+      }, this);
+      this.objs.clear();
+      this.annotationsPromise = null;
+      this.pendingCleanup = false;
+    },
+    _startRenderPage: function PDFPageProxy_startRenderPage(transparency, intent) {
+      var intentState = this.intentStates[intent];
+      if (intentState.displayReadyCapability) {
+        intentState.displayReadyCapability.resolve(transparency);
+      }
+    },
+    _renderPageChunk: function PDFPageProxy_renderPageChunk(operatorListChunk, intent) {
+      var intentState = this.intentStates[intent];
+      var i, ii;
+      for (i = 0, ii = operatorListChunk.length; i < ii; i++) {
+        intentState.operatorList.fnArray.push(operatorListChunk.fnArray[i]);
+        intentState.operatorList.argsArray.push(operatorListChunk.argsArray[i]);
+      }
+      intentState.operatorList.lastChunk = operatorListChunk.lastChunk;
+      for (i = 0; i < intentState.renderTasks.length; i++) {
+        intentState.renderTasks[i].operatorListChanged();
+      }
+      if (operatorListChunk.lastChunk) {
+        intentState.receivingOperatorList = false;
+        this._tryCleanup();
+      }
+    }
+  };
+  return PDFPageProxy;
 }();
 var PDFWorker = function PDFWorkerClosure() {
- var nextFakeWorkerId = 0;
- function getWorkerSrc() {
-  if (typeof workerSrc !== 'undefined') {
-   return workerSrc;
-  }
-  if (getDefaultSetting('workerSrc')) {
-   return getDefaultSetting('workerSrc');
-  }
-  if (pdfjsFilePath) {
-   return pdfjsFilePath.replace(/\.js$/i, '.worker.js');
-  }
-  error('No PDFJS.workerSrc specified');
- }
- var fakeWorkerFilesLoadedCapability;
- function setupFakeWorkerGlobal() {
-  var WorkerMessageHandler;
-  if (fakeWorkerFilesLoadedCapability) {
-   return fakeWorkerFilesLoadedCapability.promise;
-  }
-  fakeWorkerFilesLoadedCapability = createPromiseCapability();
-  var loader = fakeWorkerFilesLoader || function (callback) {
-   Util.loadScript(getWorkerSrc(), function () {
-    callback(window.pdfjsDistBuildPdfWorker.WorkerMessageHandler);
-   });
-  };
-  loader(fakeWorkerFilesLoadedCapability.resolve);
-  return fakeWorkerFilesLoadedCapability.promise;
- }
- function FakeWorkerPort(defer) {
-  this._listeners = [];
-  this._defer = defer;
-  this._deferred = Promise.resolve(undefined);
- }
- FakeWorkerPort.prototype = {
-  postMessage: function (obj, transfers) {
-   function cloneValue(value) {
-    if (typeof value !== 'object' || value === null) {
-     return value;
+  var nextFakeWorkerId = 0;
+  function getWorkerSrc() {
+    if (typeof workerSrc !== 'undefined') {
+      return workerSrc;
     }
-    if (cloned.has(value)) {
-     return cloned.get(value);
+    if (getDefaultSetting('workerSrc')) {
+      return getDefaultSetting('workerSrc');
     }
-    var result;
-    var buffer;
-    if ((buffer = value.buffer) && isArrayBuffer(buffer)) {
-     var transferable = transfers && transfers.indexOf(buffer) >= 0;
-     if (value === buffer) {
-      result = value;
-     } else if (transferable) {
-      result = new value.constructor(buffer, value.byteOffset, value.byteLength);
-     } else {
-      result = new value.constructor(value);
-     }
-     cloned.set(value, result);
-     return result;
+    if (pdfjsFilePath) {
+      return pdfjsFilePath.replace(/\.js$/i, '.worker.js');
     }
-    result = isArray(value) ? [] : {};
-    cloned.set(value, result);
-    for (var i in value) {
-     var desc, p = value;
-     while (!(desc = Object.getOwnPropertyDescriptor(p, i))) {
-      p = Object.getPrototypeOf(p);
-     }
-     if (typeof desc.value === 'undefined' || typeof desc.value === 'function') {
-      continue;
-     }
-     result[i] = cloneValue(desc.value);
+    error('No PDFJS.workerSrc specified');
+  }
+  var fakeWorkerFilesLoadedCapability;
+  function setupFakeWorkerGlobal() {
+    var WorkerMessageHandler;
+    if (fakeWorkerFilesLoadedCapability) {
+      return fakeWorkerFilesLoadedCapability.promise;
     }
-    return result;
-   }
-   if (!this._defer) {
-    this._listeners.forEach(function (listener) {
-     listener.call(this, { data: obj });
-    }, this);
-    return;
-   }
-   var cloned = new WeakMap();
-   var e = { data: cloneValue(obj) };
-   this._deferred.then(function () {
-    this._listeners.forEach(function (listener) {
-     listener.call(this, e);
-    }, this);
-   }.bind(this));
-  },
-  addEventListener: function (name, listener) {
-   this._listeners.push(listener);
-  },
-  removeEventListener: function (name, listener) {
-   var i = this._listeners.indexOf(listener);
-   this._listeners.splice(i, 1);
-  },
-  terminate: function () {
-   this._listeners = [];
+    fakeWorkerFilesLoadedCapability = createPromiseCapability();
+    var loader = fakeWorkerFilesLoader || function (callback) {
+      Util.loadScript(getWorkerSrc(), function () {
+        callback(window.pdfjsDistBuildPdfWorker.WorkerMessageHandler);
+      });
+    };
+    loader(fakeWorkerFilesLoadedCapability.resolve);
+    return fakeWorkerFilesLoadedCapability.promise;
   }
- };
- function createCDNWrapper(url) {
-  var wrapper = 'importScripts(\'' + url + '\');';
-  return URL.createObjectURL(new Blob([wrapper]));
- }
- function PDFWorker(name, port) {
-  this.name = name;
-  this.destroyed = false;
-  this._readyCapability = createPromiseCapability();
-  this._port = null;
-  this._webWorker = null;
-  this._messageHandler = null;
-  if (port) {
-   this._initializeFromPort(port);
-   return;
+  function FakeWorkerPort(defer) {
+    this._listeners = [];
+    this._defer = defer;
+    this._deferred = Promise.resolve(undefined);
   }
-  this._initialize();
- }
- PDFWorker.prototype = {
-  get promise() {
-   return this._readyCapability.promise;
-  },
-  get port() {
-   return this._port;
-  },
-  get messageHandler() {
-   return this._messageHandler;
-  },
-  _initializeFromPort: function PDFWorker_initializeFromPort(port) {
-   this._port = port;
-   this._messageHandler = new MessageHandler('main', 'worker', port);
-   this._messageHandler.on('ready', function () {
-   });
-   this._readyCapability.resolve();
-  },
-  _initialize: function PDFWorker_initialize() {
-   if (!isWorkerDisabled && !getDefaultSetting('disableWorker') && typeof Worker !== 'undefined') {
-    var workerSrc = getWorkerSrc();
-    try {
-     if (!isSameOrigin(window.location.href, workerSrc)) {
-      workerSrc = createCDNWrapper(new URL(workerSrc, window.location).href);
-     }
-     var worker = new Worker(workerSrc);
-     var messageHandler = new MessageHandler('main', 'worker', worker);
-     var terminateEarly = function () {
-      worker.removeEventListener('error', onWorkerError);
-      messageHandler.destroy();
-      worker.terminate();
-      if (this.destroyed) {
-       this._readyCapability.reject(new Error('Worker was destroyed'));
-      } else {
-       this._setupFakeWorker();
-      }
-     }.bind(this);
-     var onWorkerError = function (event) {
-      if (!this._webWorker) {
-       terminateEarly();
+  FakeWorkerPort.prototype = {
+    postMessage: function (obj, transfers) {
+      function cloneValue(value) {
+        if (typeof value !== 'object' || value === null) {
+          return value;
+        }
+        if (cloned.has(value)) {
+          return cloned.get(value);
+        }
+        var result;
+        var buffer;
+        if ((buffer = value.buffer) && isArrayBuffer(buffer)) {
+          var transferable = transfers && transfers.indexOf(buffer) >= 0;
+          if (value === buffer) {
+            result = value;
+          } else if (transferable) {
+            result = new value.constructor(buffer, value.byteOffset, value.byteLength);
+          } else {
+            result = new value.constructor(value);
+          }
+          cloned.set(value, result);
+          return result;
+        }
+        result = isArray(value) ? [] : {};
+        cloned.set(value, result);
+        for (var i in value) {
+          var desc,
+              p = value;
+          while (!(desc = Object.getOwnPropertyDescriptor(p, i))) {
+            p = Object.getPrototypeOf(p);
+          }
+          if (typeof desc.value === 'undefined' || typeof desc.value === 'function') {
+            continue;
+          }
+          result[i] = cloneValue(desc.value);
+        }
+        return result;
       }
-     }.bind(this);
-     worker.addEventListener('error', onWorkerError);
-     messageHandler.on('test', function PDFWorker_test(data) {
-      worker.removeEventListener('error', onWorkerError);
-      if (this.destroyed) {
-       terminateEarly();
-       return;
+      if (!this._defer) {
+        this._listeners.forEach(function (listener) {
+          listener.call(this, { data: obj });
+        }, this);
+        return;
       }
-      var supportTypedArray = data && data.supportTypedArray;
-      if (supportTypedArray) {
-       this._messageHandler = messageHandler;
-       this._port = worker;
-       this._webWorker = worker;
-       if (!data.supportTransfers) {
-        isPostMessageTransfersDisabled = true;
-       }
-       this._readyCapability.resolve();
-       messageHandler.send('configure', { verbosity: getVerbosityLevel() });
-      } else {
-       this._setupFakeWorker();
-       messageHandler.destroy();
-       worker.terminate();
+      var cloned = new WeakMap();
+      var e = { data: cloneValue(obj) };
+      this._deferred.then(function () {
+        this._listeners.forEach(function (listener) {
+          listener.call(this, e);
+        }, this);
+      }.bind(this));
+    },
+    addEventListener: function (name, listener) {
+      this._listeners.push(listener);
+    },
+    removeEventListener: function (name, listener) {
+      var i = this._listeners.indexOf(listener);
+      this._listeners.splice(i, 1);
+    },
+    terminate: function () {
+      this._listeners = [];
+    }
+  };
+  function createCDNWrapper(url) {
+    var wrapper = 'importScripts(\'' + url + '\');';
+    return URL.createObjectURL(new Blob([wrapper]));
+  }
+  function PDFWorker(name, port) {
+    this.name = name;
+    this.destroyed = false;
+    this._readyCapability = createPromiseCapability();
+    this._port = null;
+    this._webWorker = null;
+    this._messageHandler = null;
+    if (port) {
+      this._initializeFromPort(port);
+      return;
+    }
+    this._initialize();
+  }
+  PDFWorker.prototype = {
+    get promise() {
+      return this._readyCapability.promise;
+    },
+    get port() {
+      return this._port;
+    },
+    get messageHandler() {
+      return this._messageHandler;
+    },
+    _initializeFromPort: function PDFWorker_initializeFromPort(port) {
+      this._port = port;
+      this._messageHandler = new MessageHandler('main', 'worker', port);
+      this._messageHandler.on('ready', function () {});
+      this._readyCapability.resolve();
+    },
+    _initialize: function PDFWorker_initialize() {
+      if (!isWorkerDisabled && !getDefaultSetting('disableWorker') && typeof Worker !== 'undefined') {
+        var workerSrc = getWorkerSrc();
+        try {
+          if (!isSameOrigin(window.location.href, workerSrc)) {
+            workerSrc = createCDNWrapper(new URL(workerSrc, window.location).href);
+          }
+          var worker = new Worker(workerSrc);
+          var messageHandler = new MessageHandler('main', 'worker', worker);
+          var terminateEarly = function () {
+            worker.removeEventListener('error', onWorkerError);
+            messageHandler.destroy();
+            worker.terminate();
+            if (this.destroyed) {
+              this._readyCapability.reject(new Error('Worker was destroyed'));
+            } else {
+              this._setupFakeWorker();
+            }
+          }.bind(this);
+          var onWorkerError = function (event) {
+            if (!this._webWorker) {
+              terminateEarly();
+            }
+          }.bind(this);
+          worker.addEventListener('error', onWorkerError);
+          messageHandler.on('test', function PDFWorker_test(data) {
+            worker.removeEventListener('error', onWorkerError);
+            if (this.destroyed) {
+              terminateEarly();
+              return;
+            }
+            var supportTypedArray = data && data.supportTypedArray;
+            if (supportTypedArray) {
+              this._messageHandler = messageHandler;
+              this._port = worker;
+              this._webWorker = worker;
+              if (!data.supportTransfers) {
+                isPostMessageTransfersDisabled = true;
+              }
+              this._readyCapability.resolve();
+              messageHandler.send('configure', { verbosity: getVerbosityLevel() });
+            } else {
+              this._setupFakeWorker();
+              messageHandler.destroy();
+              worker.terminate();
+            }
+          }.bind(this));
+          messageHandler.on('console_log', function (data) {
+            console.log.apply(console, data);
+          });
+          messageHandler.on('console_error', function (data) {
+            console.error.apply(console, data);
+          });
+          messageHandler.on('ready', function (data) {
+            worker.removeEventListener('error', onWorkerError);
+            if (this.destroyed) {
+              terminateEarly();
+              return;
+            }
+            try {
+              sendTest();
+            } catch (e) {
+              this._setupFakeWorker();
+            }
+          }.bind(this));
+          var sendTest = function () {
+            var postMessageTransfers = getDefaultSetting('postMessageTransfers') && !isPostMessageTransfersDisabled;
+            var testObj = new Uint8Array([postMessageTransfers ? 255 : 0]);
+            try {
+              messageHandler.send('test', testObj, [testObj.buffer]);
+            } catch (ex) {
+              info('Cannot use postMessage transfers');
+              testObj[0] = 0;
+              messageHandler.send('test', testObj);
+            }
+          };
+          sendTest();
+          return;
+        } catch (e) {
+          info('The worker has been disabled.');
+        }
       }
-     }.bind(this));
-     messageHandler.on('console_log', function (data) {
-      console.log.apply(console, data);
-     });
-     messageHandler.on('console_error', function (data) {
-      console.error.apply(console, data);
-     });
-     messageHandler.on('ready', function (data) {
-      worker.removeEventListener('error', onWorkerError);
-      if (this.destroyed) {
-       terminateEarly();
-       return;
+      this._setupFakeWorker();
+    },
+    _setupFakeWorker: function PDFWorker_setupFakeWorker() {
+      if (!isWorkerDisabled && !getDefaultSetting('disableWorker')) {
+        warn('Setting up fake worker.');
+        isWorkerDisabled = true;
       }
-      try {
-       sendTest();
-      } catch (e) {
-       this._setupFakeWorker();
+      setupFakeWorkerGlobal().then(function (WorkerMessageHandler) {
+        if (this.destroyed) {
+          this._readyCapability.reject(new Error('Worker was destroyed'));
+          return;
+        }
+        var isTypedArraysPresent = Uint8Array !== Float32Array;
+        var port = new FakeWorkerPort(isTypedArraysPresent);
+        this._port = port;
+        var id = 'fake' + nextFakeWorkerId++;
+        var workerHandler = new MessageHandler(id + '_worker', id, port);
+        WorkerMessageHandler.setup(workerHandler, port);
+        var messageHandler = new MessageHandler(id, id + '_worker', port);
+        this._messageHandler = messageHandler;
+        this._readyCapability.resolve();
+      }.bind(this));
+    },
+    destroy: function PDFWorker_destroy() {
+      this.destroyed = true;
+      if (this._webWorker) {
+        this._webWorker.terminate();
+        this._webWorker = null;
       }
-     }.bind(this));
-     var sendTest = function () {
-      var postMessageTransfers = getDefaultSetting('postMessageTransfers') && !isPostMessageTransfersDisabled;
-      var testObj = new Uint8Array([postMessageTransfers ? 255 : 0]);
-      try {
-       messageHandler.send('test', testObj, [testObj.buffer]);
-      } catch (ex) {
-       info('Cannot use postMessage transfers');
-       testObj[0] = 0;
-       messageHandler.send('test', testObj);
+      this._port = null;
+      if (this._messageHandler) {
+        this._messageHandler.destroy();
+        this._messageHandler = null;
       }
-     };
-     sendTest();
-     return;
-    } catch (e) {
-     info('The worker has been disabled.');
     }
-   }
-   this._setupFakeWorker();
-  },
-  _setupFakeWorker: function PDFWorker_setupFakeWorker() {
-   if (!isWorkerDisabled && !getDefaultSetting('disableWorker')) {
-    warn('Setting up fake worker.');
-    isWorkerDisabled = true;
-   }
-   setupFakeWorkerGlobal().then(function (WorkerMessageHandler) {
-    if (this.destroyed) {
-     this._readyCapability.reject(new Error('Worker was destroyed'));
-     return;
-    }
-    var isTypedArraysPresent = Uint8Array !== Float32Array;
-    var port = new FakeWorkerPort(isTypedArraysPresent);
-    this._port = port;
-    var id = 'fake' + nextFakeWorkerId++;
-    var workerHandler = new MessageHandler(id + '_worker', id, port);
-    WorkerMessageHandler.setup(workerHandler, port);
-    var messageHandler = new MessageHandler(id, id + '_worker', port);
-    this._messageHandler = messageHandler;
-    this._readyCapability.resolve();
-   }.bind(this));
-  },
-  destroy: function PDFWorker_destroy() {
-   this.destroyed = true;
-   if (this._webWorker) {
-    this._webWorker.terminate();
-    this._webWorker = null;
-   }
-   this._port = null;
-   if (this._messageHandler) {
-    this._messageHandler.destroy();
-    this._messageHandler = null;
-   }
-  }
- };
- return PDFWorker;
+  };
+  return PDFWorker;
 }();
 var WorkerTransport = function WorkerTransportClosure() {
- function WorkerTransport(messageHandler, loadingTask, pdfDataRangeTransport, CMapReaderFactory) {
-  this.messageHandler = messageHandler;
-  this.loadingTask = loadingTask;
-  this.pdfDataRangeTransport = pdfDataRangeTransport;
-  this.commonObjs = new PDFObjects();
-  this.fontLoader = new FontLoader(loadingTask.docId);
-  this.CMapReaderFactory = new CMapReaderFactory({
-   baseUrl: getDefaultSetting('cMapUrl'),
-   isCompressed: getDefaultSetting('cMapPacked')
-  });
-  this.destroyed = false;
-  this.destroyCapability = null;
-  this._passwordCapability = null;
-  this.pageCache = [];
-  this.pagePromises = [];
-  this.downloadInfoCapability = createPromiseCapability();
-  this.setupMessageHandler();
- }
- WorkerTransport.prototype = {
-  destroy: function WorkerTransport_destroy() {
-   if (this.destroyCapability) {
-    return this.destroyCapability.promise;
-   }
-   this.destroyed = true;
-   this.destroyCapability = createPromiseCapability();
-   if (this._passwordCapability) {
-    this._passwordCapability.reject(new Error('Worker was destroyed during onPassword callback'));
-   }
-   var waitOn = [];
-   this.pageCache.forEach(function (page) {
-    if (page) {
-     waitOn.push(page._destroy());
-    }
-   });
-   this.pageCache = [];
-   this.pagePromises = [];
-   var self = this;
-   var terminated = this.messageHandler.sendWithPromise('Terminate', null);
-   waitOn.push(terminated);
-   Promise.all(waitOn).then(function () {
-    self.fontLoader.clear();
-    if (self.pdfDataRangeTransport) {
-     self.pdfDataRangeTransport.abort();
-     self.pdfDataRangeTransport = null;
-    }
-    if (self.messageHandler) {
-     self.messageHandler.destroy();
-     self.messageHandler = null;
-    }
-    self.destroyCapability.resolve();
-   }, this.destroyCapability.reject);
-   return this.destroyCapability.promise;
-  },
-  setupMessageHandler: function WorkerTransport_setupMessageHandler() {
-   var messageHandler = this.messageHandler;
-   var loadingTask = this.loadingTask;
-   var pdfDataRangeTransport = this.pdfDataRangeTransport;
-   if (pdfDataRangeTransport) {
-    pdfDataRangeTransport.addRangeListener(function (begin, chunk) {
-     messageHandler.send('OnDataRange', {
-      begin: begin,
-      chunk: chunk
-     });
-    });
-    pdfDataRangeTransport.addProgressListener(function (loaded) {
-     messageHandler.send('OnDataProgress', { loaded: loaded });
+  function WorkerTransport(messageHandler, loadingTask, pdfDataRangeTransport, CMapReaderFactory) {
+    this.messageHandler = messageHandler;
+    this.loadingTask = loadingTask;
+    this.pdfDataRangeTransport = pdfDataRangeTransport;
+    this.commonObjs = new PDFObjects();
+    this.fontLoader = new FontLoader(loadingTask.docId);
+    this.CMapReaderFactory = new CMapReaderFactory({
+      baseUrl: getDefaultSetting('cMapUrl'),
+      isCompressed: getDefaultSetting('cMapPacked')
     });
-    pdfDataRangeTransport.addProgressiveReadListener(function (chunk) {
-     messageHandler.send('OnDataRange', { chunk: chunk });
-    });
-    messageHandler.on('RequestDataRange', function transportDataRange(data) {
-     pdfDataRangeTransport.requestDataRange(data.begin, data.end);
-    }, this);
-   }
-   messageHandler.on('GetDoc', function transportDoc(data) {
-    var pdfInfo = data.pdfInfo;
-    this.numPages = data.pdfInfo.numPages;
-    var loadingTask = this.loadingTask;
-    var pdfDocument = new PDFDocumentProxy(pdfInfo, this, loadingTask);
-    this.pdfDocument = pdfDocument;
-    loadingTask._capability.resolve(pdfDocument);
-   }, this);
-   messageHandler.on('PasswordRequest', function transportPasswordRequest(exception) {
-    this._passwordCapability = createPromiseCapability();
-    if (loadingTask.onPassword) {
-     var updatePassword = function (password) {
-      this._passwordCapability.resolve({ password: password });
-     }.bind(this);
-     loadingTask.onPassword(updatePassword, exception.code);
-    } else {
-     this._passwordCapability.reject(new PasswordException(exception.message, exception.code));
-    }
-    return this._passwordCapability.promise;
-   }, this);
-   messageHandler.on('PasswordException', function transportPasswordException(exception) {
-    loadingTask._capability.reject(new PasswordException(exception.message, exception.code));
-   }, this);
-   messageHandler.on('InvalidPDF', function transportInvalidPDF(exception) {
-    this.loadingTask._capability.reject(new InvalidPDFException(exception.message));
-   }, this);
-   messageHandler.on('MissingPDF', function transportMissingPDF(exception) {
-    this.loadingTask._capability.reject(new MissingPDFException(exception.message));
-   }, this);
-   messageHandler.on('UnexpectedResponse', function transportUnexpectedResponse(exception) {
-    this.loadingTask._capability.reject(new UnexpectedResponseException(exception.message, exception.status));
-   }, this);
-   messageHandler.on('UnknownError', function transportUnknownError(exception) {
-    this.loadingTask._capability.reject(new UnknownErrorException(exception.message, exception.details));
-   }, this);
-   messageHandler.on('DataLoaded', function transportPage(data) {
-    this.downloadInfoCapability.resolve(data);
-   }, this);
-   messageHandler.on('PDFManagerReady', function transportPage(data) {
-    if (this.pdfDataRangeTransport) {
-     this.pdfDataRangeTransport.transportReady();
-    }
-   }, this);
-   messageHandler.on('StartRenderPage', function transportRender(data) {
-    if (this.destroyed) {
-     return;
-    }
-    var page = this.pageCache[data.pageIndex];
-    page.stats.timeEnd('Page Request');
-    page._startRenderPage(data.transparency, data.intent);
-   }, this);
-   messageHandler.on('RenderPageChunk', function transportRender(data) {
-    if (this.destroyed) {
-     return;
-    }
-    var page = this.pageCache[data.pageIndex];
-    page._renderPageChunk(data.operatorList, data.intent);
-   }, this);
-   messageHandler.on('commonobj', function transportObj(data) {
-    if (this.destroyed) {
-     return;
-    }
-    var id = data[0];
-    var type = data[1];
-    if (this.commonObjs.hasData(id)) {
-     return;
-    }
-    switch (type) {
-    case 'Font':
-     var exportedData = data[2];
-     if ('error' in exportedData) {
-      var exportedError = exportedData.error;
-      warn('Error during font loading: ' + exportedError);
-      this.commonObjs.resolve(id, exportedError);
-      break;
-     }
-     var fontRegistry = null;
-     if (getDefaultSetting('pdfBug') && globalScope.FontInspector && globalScope['FontInspector'].enabled) {
-      fontRegistry = {
-       registerFont: function (font, url) {
-        globalScope['FontInspector'].fontAdded(font, url);
-       }
-      };
-     }
-     var font = new FontFaceObject(exportedData, {
-      isEvalSuported: getDefaultSetting('isEvalSupported'),
-      disableFontFace: getDefaultSetting('disableFontFace'),
-      fontRegistry: fontRegistry
-     });
-     this.fontLoader.bind([font], function fontReady(fontObjs) {
-      this.commonObjs.resolve(id, font);
-     }.bind(this));
-     break;
-    case 'FontPath':
-     this.commonObjs.resolve(id, data[2]);
-     break;
-    default:
-     error('Got unknown common object type ' + type);
-    }
-   }, this);
-   messageHandler.on('obj', function transportObj(data) {
-    if (this.destroyed) {
-     return;
-    }
-    var id = data[0];
-    var pageIndex = data[1];
-    var type = data[2];
-    var pageProxy = this.pageCache[pageIndex];
-    var imageData;
-    if (pageProxy.objs.hasData(id)) {
-     return;
-    }
-    switch (type) {
-    case 'JpegStream':
-     imageData = data[3];
-     loadJpegStream(id, imageData, pageProxy.objs);
-     break;
-    case 'Image':
-     imageData = data[3];
-     pageProxy.objs.resolve(id, imageData);
-     var MAX_IMAGE_SIZE_TO_STORE = 8000000;
-     if (imageData && 'data' in imageData && imageData.data.length > MAX_IMAGE_SIZE_TO_STORE) {
-      pageProxy.cleanupAfterRender = true;
-     }
-     break;
-    default:
-     error('Got unknown object type ' + type);
-    }
-   }, this);
-   messageHandler.on('DocProgress', function transportDocProgress(data) {
-    if (this.destroyed) {
-     return;
-    }
-    var loadingTask = this.loadingTask;
-    if (loadingTask.onProgress) {
-     loadingTask.onProgress({
-      loaded: data.loaded,
-      total: data.total
-     });
-    }
-   }, this);
-   messageHandler.on('PageError', function transportError(data) {
-    if (this.destroyed) {
-     return;
-    }
-    var page = this.pageCache[data.pageNum - 1];
-    var intentState = page.intentStates[data.intent];
-    if (intentState.displayReadyCapability) {
-     intentState.displayReadyCapability.reject(data.error);
-    } else {
-     error(data.error);
-    }
-    if (intentState.operatorList) {
-     intentState.operatorList.lastChunk = true;
-     for (var i = 0; i < intentState.renderTasks.length; i++) {
-      intentState.renderTasks[i].operatorListChanged();
-     }
-    }
-   }, this);
-   messageHandler.on('UnsupportedFeature', function transportUnsupportedFeature(data) {
-    if (this.destroyed) {
-     return;
-    }
-    var featureId = data.featureId;
-    var loadingTask = this.loadingTask;
-    if (loadingTask.onUnsupportedFeature) {
-     loadingTask.onUnsupportedFeature(featureId);
-    }
-    _UnsupportedManager.notify(featureId);
-   }, this);
-   messageHandler.on('JpegDecode', function (data) {
-    if (this.destroyed) {
-     return Promise.reject(new Error('Worker was destroyed'));
-    }
-    if (typeof document === 'undefined') {
-     return Promise.reject(new Error('"document" is not defined.'));
-    }
-    var imageUrl = data[0];
-    var components = data[1];
-    if (components !== 3 && components !== 1) {
-     return Promise.reject(new Error('Only 3 components or 1 component can be returned'));
-    }
-    return new Promise(function (resolve, reject) {
-     var img = new Image();
-     img.onload = function () {
-      var width = img.width;
-      var height = img.height;
-      var size = width * height;
-      var rgbaLength = size * 4;
-      var buf = new Uint8Array(size * components);
-      var tmpCanvas = document.createElement('canvas');
-      tmpCanvas.width = width;
-      tmpCanvas.height = height;
-      var tmpCtx = tmpCanvas.getContext('2d');
-      tmpCtx.drawImage(img, 0, 0);
-      var data = tmpCtx.getImageData(0, 0, width, height).data;
-      var i, j;
-      if (components === 3) {
-       for (i = 0, j = 0; i < rgbaLength; i += 4, j += 3) {
-        buf[j] = data[i];
-        buf[j + 1] = data[i + 1];
-        buf[j + 2] = data[i + 2];
-       }
-      } else if (components === 1) {
-       for (i = 0, j = 0; i < rgbaLength; i += 4, j++) {
-        buf[j] = data[i];
-       }
+    this.destroyed = false;
+    this.destroyCapability = null;
+    this._passwordCapability = null;
+    this.pageCache = [];
+    this.pagePromises = [];
+    this.downloadInfoCapability = createPromiseCapability();
+    this.setupMessageHandler();
+  }
+  WorkerTransport.prototype = {
+    destroy: function WorkerTransport_destroy() {
+      if (this.destroyCapability) {
+        return this.destroyCapability.promise;
+      }
+      this.destroyed = true;
+      this.destroyCapability = createPromiseCapability();
+      if (this._passwordCapability) {
+        this._passwordCapability.reject(new Error('Worker was destroyed during onPassword callback'));
       }
-      resolve({
-       data: buf,
-       width: width,
-       height: height
+      var waitOn = [];
+      this.pageCache.forEach(function (page) {
+        if (page) {
+          waitOn.push(page._destroy());
+        }
       });
-     };
-     img.onerror = function () {
-      reject(new Error('JpegDecode failed to load image'));
-     };
-     img.src = imageUrl;
-    });
-   }, this);
-   messageHandler.on('FetchBuiltInCMap', function (data) {
-    if (this.destroyed) {
-     return Promise.reject(new Error('Worker was destroyed'));
-    }
-    return this.CMapReaderFactory.fetch({ name: data.name });
-   }, this);
-  },
-  getData: function WorkerTransport_getData() {
-   return this.messageHandler.sendWithPromise('GetData', null);
-  },
-  getPage: function WorkerTransport_getPage(pageNumber, capability) {
-   if (!isInt(pageNumber) || pageNumber <= 0 || pageNumber > this.numPages) {
-    return Promise.reject(new Error('Invalid page request'));
-   }
-   var pageIndex = pageNumber - 1;
-   if (pageIndex in this.pagePromises) {
-    return this.pagePromises[pageIndex];
-   }
-   var promise = this.messageHandler.sendWithPromise('GetPage', { pageIndex: pageIndex }).then(function (pageInfo) {
-    if (this.destroyed) {
-     throw new Error('Transport destroyed');
-    }
-    var page = new PDFPageProxy(pageIndex, pageInfo, this);
-    this.pageCache[pageIndex] = page;
-    return page;
-   }.bind(this));
-   this.pagePromises[pageIndex] = promise;
-   return promise;
-  },
-  getPageIndex: function WorkerTransport_getPageIndexByRef(ref) {
-   return this.messageHandler.sendWithPromise('GetPageIndex', { ref: ref }).catch(function (reason) {
-    return Promise.reject(new Error(reason));
-   });
-  },
-  getAnnotations: function WorkerTransport_getAnnotations(pageIndex, intent) {
-   return this.messageHandler.sendWithPromise('GetAnnotations', {
-    pageIndex: pageIndex,
-    intent: intent
-   });
-  },
-  getDestinations: function WorkerTransport_getDestinations() {
-   return this.messageHandler.sendWithPromise('GetDestinations', null);
-  },
-  getDestination: function WorkerTransport_getDestination(id) {
-   return this.messageHandler.sendWithPromise('GetDestination', { id: id });
-  },
-  getPageLabels: function WorkerTransport_getPageLabels() {
-   return this.messageHandler.sendWithPromise('GetPageLabels', null);
-  },
-  getAttachments: function WorkerTransport_getAttachments() {
-   return this.messageHandler.sendWithPromise('GetAttachments', null);
-  },
-  getJavaScript: function WorkerTransport_getJavaScript() {
-   return this.messageHandler.sendWithPromise('GetJavaScript', null);
-  },
-  getOutline: function WorkerTransport_getOutline() {
-   return this.messageHandler.sendWithPromise('GetOutline', null);
-  },
-  getMetadata: function WorkerTransport_getMetadata() {
-   return this.messageHandler.sendWithPromise('GetMetadata', null).then(function transportMetadata(results) {
-    return {
-     info: results[0],
-     metadata: results[1] ? new Metadata(results[1]) : null
-    };
-   });
-  },
-  getStats: function WorkerTransport_getStats() {
-   return this.messageHandler.sendWithPromise('GetStats', null);
-  },
-  startCleanup: function WorkerTransport_startCleanup() {
-   this.messageHandler.sendWithPromise('Cleanup', null).then(function endCleanup() {
-    for (var i = 0, ii = this.pageCache.length; i < ii; i++) {
-     var page = this.pageCache[i];
-     if (page) {
-      page.cleanup();
-     }
+      this.pageCache = [];
+      this.pagePromises = [];
+      var self = this;
+      var terminated = this.messageHandler.sendWithPromise('Terminate', null);
+      waitOn.push(terminated);
+      Promise.all(waitOn).then(function () {
+        self.fontLoader.clear();
+        if (self.pdfDataRangeTransport) {
+          self.pdfDataRangeTransport.abort();
+          self.pdfDataRangeTransport = null;
+        }
+        if (self.messageHandler) {
+          self.messageHandler.destroy();
+          self.messageHandler = null;
+        }
+        self.destroyCapability.resolve();
+      }, this.destroyCapability.reject);
+      return this.destroyCapability.promise;
+    },
+    setupMessageHandler: function WorkerTransport_setupMessageHandler() {
+      var messageHandler = this.messageHandler;
+      var loadingTask = this.loadingTask;
+      var pdfDataRangeTransport = this.pdfDataRangeTransport;
+      if (pdfDataRangeTransport) {
+        pdfDataRangeTransport.addRangeListener(function (begin, chunk) {
+          messageHandler.send('OnDataRange', {
+            begin: begin,
+            chunk: chunk
+          });
+        });
+        pdfDataRangeTransport.addProgressListener(function (loaded) {
+          messageHandler.send('OnDataProgress', { loaded: loaded });
+        });
+        pdfDataRangeTransport.addProgressiveReadListener(function (chunk) {
+          messageHandler.send('OnDataRange', { chunk: chunk });
+        });
+        messageHandler.on('RequestDataRange', function transportDataRange(data) {
+          pdfDataRangeTransport.requestDataRange(data.begin, data.end);
+        }, this);
+      }
+      messageHandler.on('GetDoc', function transportDoc(data) {
+        var pdfInfo = data.pdfInfo;
+        this.numPages = data.pdfInfo.numPages;
+        var loadingTask = this.loadingTask;
+        var pdfDocument = new PDFDocumentProxy(pdfInfo, this, loadingTask);
+        this.pdfDocument = pdfDocument;
+        loadingTask._capability.resolve(pdfDocument);
+      }, this);
+      messageHandler.on('PasswordRequest', function transportPasswordRequest(exception) {
+        this._passwordCapability = createPromiseCapability();
+        if (loadingTask.onPassword) {
+          var updatePassword = function (password) {
+            this._passwordCapability.resolve({ password: password });
+          }.bind(this);
+          loadingTask.onPassword(updatePassword, exception.code);
+        } else {
+          this._passwordCapability.reject(new PasswordException(exception.message, exception.code));
+        }
+        return this._passwordCapability.promise;
+      }, this);
+      messageHandler.on('PasswordException', function transportPasswordException(exception) {
+        loadingTask._capability.reject(new PasswordException(exception.message, exception.code));
+      }, this);
+      messageHandler.on('InvalidPDF', function transportInvalidPDF(exception) {
+        this.loadingTask._capability.reject(new InvalidPDFException(exception.message));
+      }, this);
+      messageHandler.on('MissingPDF', function transportMissingPDF(exception) {
+        this.loadingTask._capability.reject(new MissingPDFException(exception.message));
+      }, this);
+      messageHandler.on('UnexpectedResponse', function transportUnexpectedResponse(exception) {
+        this.loadingTask._capability.reject(new UnexpectedResponseException(exception.message, exception.status));
+      }, this);
+      messageHandler.on('UnknownError', function transportUnknownError(exception) {
+        this.loadingTask._capability.reject(new UnknownErrorException(exception.message, exception.details));
+      }, this);
+      messageHandler.on('DataLoaded', function transportPage(data) {
+        this.downloadInfoCapability.resolve(data);
+      }, this);
+      messageHandler.on('PDFManagerReady', function transportPage(data) {
+        if (this.pdfDataRangeTransport) {
+          this.pdfDataRangeTransport.transportReady();
+        }
+      }, this);
+      messageHandler.on('StartRenderPage', function transportRender(data) {
+        if (this.destroyed) {
+          return;
+        }
+        var page = this.pageCache[data.pageIndex];
+        page.stats.timeEnd('Page Request');
+        page._startRenderPage(data.transparency, data.intent);
+      }, this);
+      messageHandler.on('RenderPageChunk', function transportRender(data) {
+        if (this.destroyed) {
+          return;
+        }
+        var page = this.pageCache[data.pageIndex];
+        page._renderPageChunk(data.operatorList, data.intent);
+      }, this);
+      messageHandler.on('commonobj', function transportObj(data) {
+        if (this.destroyed) {
+          return;
+        }
+        var id = data[0];
+        var type = data[1];
+        if (this.commonObjs.hasData(id)) {
+          return;
+        }
+        switch (type) {
+          case 'Font':
+            var exportedData = data[2];
+            if ('error' in exportedData) {
+              var exportedError = exportedData.error;
+              warn('Error during font loading: ' + exportedError);
+              this.commonObjs.resolve(id, exportedError);
+              break;
+            }
+            var fontRegistry = null;
+            if (getDefaultSetting('pdfBug') && globalScope.FontInspector && globalScope['FontInspector'].enabled) {
+              fontRegistry = {
+                registerFont: function (font, url) {
+                  globalScope['FontInspector'].fontAdded(font, url);
+                }
+              };
+            }
+            var font = new FontFaceObject(exportedData, {
+              isEvalSuported: getDefaultSetting('isEvalSupported'),
+              disableFontFace: getDefaultSetting('disableFontFace'),
+              fontRegistry: fontRegistry
+            });
+            this.fontLoader.bind([font], function fontReady(fontObjs) {
+              this.commonObjs.resolve(id, font);
+            }.bind(this));
+            break;
+          case 'FontPath':
+            this.commonObjs.resolve(id, data[2]);
+            break;
+          default:
+            error('Got unknown common object type ' + type);
+        }
+      }, this);
+      messageHandler.on('obj', function transportObj(data) {
+        if (this.destroyed) {
+          return;
+        }
+        var id = data[0];
+        var pageIndex = data[1];
+        var type = data[2];
+        var pageProxy = this.pageCache[pageIndex];
+        var imageData;
+        if (pageProxy.objs.hasData(id)) {
+          return;
+        }
+        switch (type) {
+          case 'JpegStream':
+            imageData = data[3];
+            loadJpegStream(id, imageData, pageProxy.objs);
+            break;
+          case 'Image':
+            imageData = data[3];
+            pageProxy.objs.resolve(id, imageData);
+            var MAX_IMAGE_SIZE_TO_STORE = 8000000;
+            if (imageData && 'data' in imageData && imageData.data.length > MAX_IMAGE_SIZE_TO_STORE) {
+              pageProxy.cleanupAfterRender = true;
+            }
+            break;
+          default:
+            error('Got unknown object type ' + type);
+        }
+      }, this);
+      messageHandler.on('DocProgress', function transportDocProgress(data) {
+        if (this.destroyed) {
+          return;
+        }
+        var loadingTask = this.loadingTask;
+        if (loadingTask.onProgress) {
+          loadingTask.onProgress({
+            loaded: data.loaded,
+            total: data.total
+          });
+        }
+      }, this);
+      messageHandler.on('PageError', function transportError(data) {
+        if (this.destroyed) {
+          return;
+        }
+        var page = this.pageCache[data.pageNum - 1];
+        var intentState = page.intentStates[data.intent];
+        if (intentState.displayReadyCapability) {
+          intentState.displayReadyCapability.reject(data.error);
+        } else {
+          error(data.error);
+        }
+        if (intentState.operatorList) {
+          intentState.operatorList.lastChunk = true;
+          for (var i = 0; i < intentState.renderTasks.length; i++) {
+            intentState.renderTasks[i].operatorListChanged();
+          }
+        }
+      }, this);
+      messageHandler.on('UnsupportedFeature', function transportUnsupportedFeature(data) {
+        if (this.destroyed) {
+          return;
+        }
+        var featureId = data.featureId;
+        var loadingTask = this.loadingTask;
+        if (loadingTask.onUnsupportedFeature) {
+          loadingTask.onUnsupportedFeature(featureId);
+        }
+        _UnsupportedManager.notify(featureId);
+      }, this);
+      messageHandler.on('JpegDecode', function (data) {
+        if (this.destroyed) {
+          return Promise.reject(new Error('Worker was destroyed'));
+        }
+        if (typeof document === 'undefined') {
+          return Promise.reject(new Error('"document" is not defined.'));
+        }
+        var imageUrl = data[0];
+        var components = data[1];
+        if (components !== 3 && components !== 1) {
+          return Promise.reject(new Error('Only 3 components or 1 component can be returned'));
+        }
+        return new Promise(function (resolve, reject) {
+          var img = new Image();
+          img.onload = function () {
+            var width = img.width;
+            var height = img.height;
+            var size = width * height;
+            var rgbaLength = size * 4;
+            var buf = new Uint8Array(size * components);
+            var tmpCanvas = document.createElement('canvas');
+            tmpCanvas.width = width;
+            tmpCanvas.height = height;
+            var tmpCtx = tmpCanvas.getContext('2d');
+            tmpCtx.drawImage(img, 0, 0);
+            var data = tmpCtx.getImageData(0, 0, width, height).data;
+            var i, j;
+            if (components === 3) {
+              for (i = 0, j = 0; i < rgbaLength; i += 4, j += 3) {
+                buf[j] = data[i];
+                buf[j + 1] = data[i + 1];
+                buf[j + 2] = data[i + 2];
+              }
+            } else if (components === 1) {
+              for (i = 0, j = 0; i < rgbaLength; i += 4, j++) {
+                buf[j] = data[i];
+              }
+            }
+            resolve({
+              data: buf,
+              width: width,
+              height: height
+            });
+          };
+          img.onerror = function () {
+            reject(new Error('JpegDecode failed to load image'));
+          };
+          img.src = imageUrl;
+        });
+      }, this);
+      messageHandler.on('FetchBuiltInCMap', function (data) {
+        if (this.destroyed) {
+          return Promise.reject(new Error('Worker was destroyed'));
+        }
+        return this.CMapReaderFactory.fetch({ name: data.name });
+      }, this);
+    },
+    getData: function WorkerTransport_getData() {
+      return this.messageHandler.sendWithPromise('GetData', null);
+    },
+    getPage: function WorkerTransport_getPage(pageNumber, capability) {
+      if (!isInt(pageNumber) || pageNumber <= 0 || pageNumber > this.numPages) {
+        return Promise.reject(new Error('Invalid page request'));
+      }
+      var pageIndex = pageNumber - 1;
+      if (pageIndex in this.pagePromises) {
+        return this.pagePromises[pageIndex];
+      }
+      var promise = this.messageHandler.sendWithPromise('GetPage', { pageIndex: pageIndex }).then(function (pageInfo) {
+        if (this.destroyed) {
+          throw new Error('Transport destroyed');
+        }
+        var page = new PDFPageProxy(pageIndex, pageInfo, this);
+        this.pageCache[pageIndex] = page;
+        return page;
+      }.bind(this));
+      this.pagePromises[pageIndex] = promise;
+      return promise;
+    },
+    getPageIndex: function WorkerTransport_getPageIndexByRef(ref) {
+      return this.messageHandler.sendWithPromise('GetPageIndex', { ref: ref }).catch(function (reason) {
+        return Promise.reject(new Error(reason));
+      });
+    },
+    getAnnotations: function WorkerTransport_getAnnotations(pageIndex, intent) {
+      return this.messageHandler.sendWithPromise('GetAnnotations', {
+        pageIndex: pageIndex,
+        intent: intent
+      });
+    },
+    getDestinations: function WorkerTransport_getDestinations() {
+      return this.messageHandler.sendWithPromise('GetDestinations', null);
+    },
+    getDestination: function WorkerTransport_getDestination(id) {
+      return this.messageHandler.sendWithPromise('GetDestination', { id: id });
+    },
+    getPageLabels: function WorkerTransport_getPageLabels() {
+      return this.messageHandler.sendWithPromise('GetPageLabels', null);
+    },
+    getAttachments: function WorkerTransport_getAttachments() {
+      return this.messageHandler.sendWithPromise('GetAttachments', null);
+    },
+    getJavaScript: function WorkerTransport_getJavaScript() {
+      return this.messageHandler.sendWithPromise('GetJavaScript', null);
+    },
+    getOutline: function WorkerTransport_getOutline() {
+      return this.messageHandler.sendWithPromise('GetOutline', null);
+    },
+    getMetadata: function WorkerTransport_getMetadata() {
+      return this.messageHandler.sendWithPromise('GetMetadata', null).then(function transportMetadata(results) {
+        return {
+          info: results[0],
+          metadata: results[1] ? new Metadata(results[1]) : null
+        };
+      });
+    },
+    getStats: function WorkerTransport_getStats() {
+      return this.messageHandler.sendWithPromise('GetStats', null);
+    },
+    startCleanup: function WorkerTransport_startCleanup() {
+      this.messageHandler.sendWithPromise('Cleanup', null).then(function endCleanup() {
+        for (var i = 0, ii = this.pageCache.length; i < ii; i++) {
+          var page = this.pageCache[i];
+          if (page) {
+            page.cleanup();
+          }
+        }
+        this.commonObjs.clear();
+        this.fontLoader.clear();
+      }.bind(this));
     }
-    this.commonObjs.clear();
-    this.fontLoader.clear();
-   }.bind(this));
-  }
- };
- return WorkerTransport;
+  };
+  return WorkerTransport;
 }();
 var PDFObjects = function PDFObjectsClosure() {
- function PDFObjects() {
-  this.objs = Object.create(null);
- }
- PDFObjects.prototype = {
-  ensureObj: function PDFObjects_ensureObj(objId) {
-   if (this.objs[objId]) {
-    return this.objs[objId];
-   }
-   var obj = {
-    capability: createPromiseCapability(),
-    data: null,
-    resolved: false
-   };
-   this.objs[objId] = obj;
-   return obj;
-  },
-  get: function PDFObjects_get(objId, callback) {
-   if (callback) {
-    this.ensureObj(objId).capability.promise.then(callback);
-    return null;
-   }
-   var obj = this.objs[objId];
-   if (!obj || !obj.resolved) {
-    error('Requesting object that isn\'t resolved yet ' + objId);
-   }
-   return obj.data;
-  },
-  resolve: function PDFObjects_resolve(objId, data) {
-   var obj = this.ensureObj(objId);
-   obj.resolved = true;
-   obj.data = data;
-   obj.capability.resolve(data);
-  },
-  isResolved: function PDFObjects_isResolved(objId) {
-   var objs = this.objs;
-   if (!objs[objId]) {
-    return false;
-   }
-   return objs[objId].resolved;
-  },
-  hasData: function PDFObjects_hasData(objId) {
-   return this.isResolved(objId);
-  },
-  getData: function PDFObjects_getData(objId) {
-   var objs = this.objs;
-   if (!objs[objId] || !objs[objId].resolved) {
-    return null;
-   }
-   return objs[objId].data;
-  },
-  clear: function PDFObjects_clear() {
-   this.objs = Object.create(null);
+  function PDFObjects() {
+    this.objs = Object.create(null);
   }
- };
- return PDFObjects;
+  PDFObjects.prototype = {
+    ensureObj: function PDFObjects_ensureObj(objId) {
+      if (this.objs[objId]) {
+        return this.objs[objId];
+      }
+      var obj = {
+        capability: createPromiseCapability(),
+        data: null,
+        resolved: false
+      };
+      this.objs[objId] = obj;
+      return obj;
+    },
+    get: function PDFObjects_get(objId, callback) {
+      if (callback) {
+        this.ensureObj(objId).capability.promise.then(callback);
+        return null;
+      }
+      var obj = this.objs[objId];
+      if (!obj || !obj.resolved) {
+        error('Requesting object that isn\'t resolved yet ' + objId);
+      }
+      return obj.data;
+    },
+    resolve: function PDFObjects_resolve(objId, data) {
+      var obj = this.ensureObj(objId);
+      obj.resolved = true;
+      obj.data = data;
+      obj.capability.resolve(data);
+    },
+    isResolved: function PDFObjects_isResolved(objId) {
+      var objs = this.objs;
+      if (!objs[objId]) {
+        return false;
+      }
+      return objs[objId].resolved;
+    },
+    hasData: function PDFObjects_hasData(objId) {
+      return this.isResolved(objId);
+    },
+    getData: function PDFObjects_getData(objId) {
+      var objs = this.objs;
+      if (!objs[objId] || !objs[objId].resolved) {
+        return null;
+      }
+      return objs[objId].data;
+    },
+    clear: function PDFObjects_clear() {
+      this.objs = Object.create(null);
+    }
+  };
+  return PDFObjects;
 }();
 var RenderTask = function RenderTaskClosure() {
- function RenderTask(internalRenderTask) {
-  this._internalRenderTask = internalRenderTask;
-  this.onContinue = null;
- }
- RenderTask.prototype = {
-  get promise() {
-   return this._internalRenderTask.capability.promise;
-  },
-  cancel: function RenderTask_cancel() {
-   this._internalRenderTask.cancel();
-  },
-  then: function RenderTask_then(onFulfilled, onRejected) {
-   return this.promise.then.apply(this.promise, arguments);
+  function RenderTask(internalRenderTask) {
+    this._internalRenderTask = internalRenderTask;
+    this.onContinue = null;
   }
- };
- return RenderTask;
+  RenderTask.prototype = {
+    get promise() {
+      return this._internalRenderTask.capability.promise;
+    },
+    cancel: function RenderTask_cancel() {
+      this._internalRenderTask.cancel();
+    },
+    then: function RenderTask_then(onFulfilled, onRejected) {
+      return this.promise.then.apply(this.promise, arguments);
+    }
+  };
+  return RenderTask;
 }();
 var InternalRenderTask = function InternalRenderTaskClosure() {
- function InternalRenderTask(callback, params, objs, commonObjs, operatorList, pageNumber, canvasFactory) {
-  this.callback = callback;
-  this.params = params;
-  this.objs = objs;
-  this.commonObjs = commonObjs;
-  this.operatorListIdx = null;
-  this.operatorList = operatorList;
-  this.pageNumber = pageNumber;
-  this.canvasFactory = canvasFactory;
-  this.running = false;
-  this.graphicsReadyCallback = null;
-  this.graphicsReady = false;
-  this.useRequestAnimationFrame = false;
-  this.cancelled = false;
-  this.capability = createPromiseCapability();
-  this.task = new RenderTask(this);
-  this._continueBound = this._continue.bind(this);
-  this._scheduleNextBound = this._scheduleNext.bind(this);
-  this._nextBound = this._next.bind(this);
- }
- InternalRenderTask.prototype = {
-  initializeGraphics: function InternalRenderTask_initializeGraphics(transparency) {
-   if (this.cancelled) {
-    return;
-   }
-   if (getDefaultSetting('pdfBug') && globalScope.StepperManager && globalScope.StepperManager.enabled) {
-    this.stepper = globalScope.StepperManager.create(this.pageNumber - 1);
-    this.stepper.init(this.operatorList);
-    this.stepper.nextBreakPoint = this.stepper.getNextBreakPoint();
-   }
-   var params = this.params;
-   this.gfx = new CanvasGraphics(params.canvasContext, this.commonObjs, this.objs, this.canvasFactory, params.imageLayer);
-   this.gfx.beginDrawing(params.transform, params.viewport, transparency);
-   this.operatorListIdx = 0;
-   this.graphicsReady = true;
-   if (this.graphicsReadyCallback) {
-    this.graphicsReadyCallback();
-   }
-  },
-  cancel: function InternalRenderTask_cancel() {
-   this.running = false;
-   this.cancelled = true;
-   this.callback('cancelled');
-  },
-  operatorListChanged: function InternalRenderTask_operatorListChanged() {
-   if (!this.graphicsReady) {
-    if (!this.graphicsReadyCallback) {
-     this.graphicsReadyCallback = this._continueBound;
-    }
-    return;
-   }
-   if (this.stepper) {
-    this.stepper.updateOperatorList(this.operatorList);
-   }
-   if (this.running) {
-    return;
-   }
-   this._continue();
-  },
-  _continue: function InternalRenderTask__continue() {
-   this.running = true;
-   if (this.cancelled) {
-    return;
-   }
-   if (this.task.onContinue) {
-    this.task.onContinue(this._scheduleNextBound);
-   } else {
-    this._scheduleNext();
-   }
-  },
-  _scheduleNext: function InternalRenderTask__scheduleNext() {
-   if (this.useRequestAnimationFrame && typeof window !== 'undefined') {
-    window.requestAnimationFrame(this._nextBound);
-   } else {
-    Promise.resolve(undefined).then(this._nextBound);
-   }
-  },
-  _next: function InternalRenderTask__next() {
-   if (this.cancelled) {
-    return;
-   }
-   this.operatorListIdx = this.gfx.executeOperatorList(this.operatorList, this.operatorListIdx, this._continueBound, this.stepper);
-   if (this.operatorListIdx === this.operatorList.argsArray.length) {
+  function InternalRenderTask(callback, params, objs, commonObjs, operatorList, pageNumber, canvasFactory) {
+    this.callback = callback;
+    this.params = params;
+    this.objs = objs;
+    this.commonObjs = commonObjs;
+    this.operatorListIdx = null;
+    this.operatorList = operatorList;
+    this.pageNumber = pageNumber;
+    this.canvasFactory = canvasFactory;
     this.running = false;
-    if (this.operatorList.lastChunk) {
-     this.gfx.endDrawing();
-     this.callback();
-    }
-   }
+    this.graphicsReadyCallback = null;
+    this.graphicsReady = false;
+    this.useRequestAnimationFrame = false;
+    this.cancelled = false;
+    this.capability = createPromiseCapability();
+    this.task = new RenderTask(this);
+    this._continueBound = this._continue.bind(this);
+    this._scheduleNextBound = this._scheduleNext.bind(this);
+    this._nextBound = this._next.bind(this);
   }
- };
- return InternalRenderTask;
+  InternalRenderTask.prototype = {
+    initializeGraphics: function InternalRenderTask_initializeGraphics(transparency) {
+      if (this.cancelled) {
+        return;
+      }
+      if (getDefaultSetting('pdfBug') && globalScope.StepperManager && globalScope.StepperManager.enabled) {
+        this.stepper = globalScope.StepperManager.create(this.pageNumber - 1);
+        this.stepper.init(this.operatorList);
+        this.stepper.nextBreakPoint = this.stepper.getNextBreakPoint();
+      }
+      var params = this.params;
+      this.gfx = new CanvasGraphics(params.canvasContext, this.commonObjs, this.objs, this.canvasFactory, params.imageLayer);
+      this.gfx.beginDrawing(params.transform, params.viewport, transparency);
+      this.operatorListIdx = 0;
+      this.graphicsReady = true;
+      if (this.graphicsReadyCallback) {
+        this.graphicsReadyCallback();
+      }
+    },
+    cancel: function InternalRenderTask_cancel() {
+      this.running = false;
+      this.cancelled = true;
+      this.callback('cancelled');
+    },
+    operatorListChanged: function InternalRenderTask_operatorListChanged() {
+      if (!this.graphicsReady) {
+        if (!this.graphicsReadyCallback) {
+          this.graphicsReadyCallback = this._continueBound;
+        }
+        return;
+      }
+      if (this.stepper) {
+        this.stepper.updateOperatorList(this.operatorList);
+      }
+      if (this.running) {
+        return;
+      }
+      this._continue();
+    },
+    _continue: function InternalRenderTask__continue() {
+      this.running = true;
+      if (this.cancelled) {
+        return;
+      }
+      if (this.task.onContinue) {
+        this.task.onContinue(this._scheduleNextBound);
+      } else {
+        this._scheduleNext();
+      }
+    },
+    _scheduleNext: function InternalRenderTask__scheduleNext() {
+      if (this.useRequestAnimationFrame && typeof window !== 'undefined') {
+        window.requestAnimationFrame(this._nextBound);
+      } else {
+        Promise.resolve(undefined).then(this._nextBound);
+      }
+    },
+    _next: function InternalRenderTask__next() {
+      if (this.cancelled) {
+        return;
+      }
+      this.operatorListIdx = this.gfx.executeOperatorList(this.operatorList, this.operatorListIdx, this._continueBound, this.stepper);
+      if (this.operatorListIdx === this.operatorList.argsArray.length) {
+        this.running = false;
+        if (this.operatorList.lastChunk) {
+          this.gfx.endDrawing();
+          this.callback();
+        }
+      }
+    }
+  };
+  return InternalRenderTask;
 }();
 var _UnsupportedManager = function UnsupportedManagerClosure() {
- var listeners = [];
- return {
-  listen: function (cb) {
-   deprecated('Global UnsupportedManager.listen is used: ' + ' use PDFDocumentLoadingTask.onUnsupportedFeature instead');
-   listeners.push(cb);
-  },
-  notify: function (featureId) {
-   for (var i = 0, ii = listeners.length; i < ii; i++) {
-    listeners[i](featureId);
-   }
-  }
- };
+  var listeners = [];
+  return {
+    listen: function (cb) {
+      deprecated('Global UnsupportedManager.listen is used: ' + ' use PDFDocumentLoadingTask.onUnsupportedFeature instead');
+      listeners.push(cb);
+    },
+    notify: function (featureId) {
+      for (var i = 0, ii = listeners.length; i < ii; i++) {
+        listeners[i](featureId);
+      }
+    }
+  };
 }();
-exports.version = '1.7.389';
-exports.build = '0423bb69';
+exports.version = '1.7.391';
+exports.build = 'cd5acf50';
 exports.getDocument = getDocument;
 exports.PDFDataRangeTransport = PDFDataRangeTransport;
 exports.PDFWorker = PDFWorker;

+ 1654 - 1788
lib/display/canvas.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var displayDOMUtils = require('./dom_utils.js');
 var displayPatternHelper = require('./pattern_helper.js');
@@ -44,1837 +45,1702 @@ var COMPILE_TYPE3_GLYPHS = true;
 var MAX_SIZE_TO_COMPILE = 1000;
 var FULL_CHUNK_HEIGHT = 16;
 var HasCanvasTypedArraysCached = {
- get value() {
-  return shadow(HasCanvasTypedArraysCached, 'value', hasCanvasTypedArrays());
- }
+  get value() {
+    return shadow(HasCanvasTypedArraysCached, 'value', hasCanvasTypedArrays());
+  }
 };
 var IsLittleEndianCached = {
- get value() {
-  return shadow(IsLittleEndianCached, 'value', isLittleEndian());
- }
+  get value() {
+    return shadow(IsLittleEndianCached, 'value', isLittleEndian());
+  }
 };
 function addContextCurrentTransform(ctx) {
- if (!ctx.mozCurrentTransform) {
-  ctx._originalSave = ctx.save;
-  ctx._originalRestore = ctx.restore;
-  ctx._originalRotate = ctx.rotate;
-  ctx._originalScale = ctx.scale;
-  ctx._originalTranslate = ctx.translate;
-  ctx._originalTransform = ctx.transform;
-  ctx._originalSetTransform = ctx.setTransform;
-  ctx._transformMatrix = ctx._transformMatrix || [
-   1,
-   0,
-   0,
-   1,
-   0,
-   0
-  ];
-  ctx._transformStack = [];
-  Object.defineProperty(ctx, 'mozCurrentTransform', {
-   get: function getCurrentTransform() {
-    return this._transformMatrix;
-   }
-  });
-  Object.defineProperty(ctx, 'mozCurrentTransformInverse', {
-   get: function getCurrentTransformInverse() {
-    var m = this._transformMatrix;
-    var a = m[0], b = m[1], c = m[2], d = m[3], e = m[4], f = m[5];
-    var ad_bc = a * d - b * c;
-    var bc_ad = b * c - a * d;
-    return [
-     d / ad_bc,
-     b / bc_ad,
-     c / bc_ad,
-     a / ad_bc,
-     (d * e - c * f) / bc_ad,
-     (b * e - a * f) / ad_bc
-    ];
-   }
-  });
-  ctx.save = function ctxSave() {
-   var old = this._transformMatrix;
-   this._transformStack.push(old);
-   this._transformMatrix = old.slice(0, 6);
-   this._originalSave();
-  };
-  ctx.restore = function ctxRestore() {
-   var prev = this._transformStack.pop();
-   if (prev) {
-    this._transformMatrix = prev;
-    this._originalRestore();
-   }
-  };
-  ctx.translate = function ctxTranslate(x, y) {
-   var m = this._transformMatrix;
-   m[4] = m[0] * x + m[2] * y + m[4];
-   m[5] = m[1] * x + m[3] * y + m[5];
-   this._originalTranslate(x, y);
-  };
-  ctx.scale = function ctxScale(x, y) {
-   var m = this._transformMatrix;
-   m[0] = m[0] * x;
-   m[1] = m[1] * x;
-   m[2] = m[2] * y;
-   m[3] = m[3] * y;
-   this._originalScale(x, y);
-  };
-  ctx.transform = function ctxTransform(a, b, c, d, e, f) {
-   var m = this._transformMatrix;
-   this._transformMatrix = [
-    m[0] * a + m[2] * b,
-    m[1] * a + m[3] * b,
-    m[0] * c + m[2] * d,
-    m[1] * c + m[3] * d,
-    m[0] * e + m[2] * f + m[4],
-    m[1] * e + m[3] * f + m[5]
-   ];
-   ctx._originalTransform(a, b, c, d, e, f);
-  };
-  ctx.setTransform = function ctxSetTransform(a, b, c, d, e, f) {
-   this._transformMatrix = [
-    a,
-    b,
-    c,
-    d,
-    e,
-    f
-   ];
-   ctx._originalSetTransform(a, b, c, d, e, f);
-  };
-  ctx.rotate = function ctxRotate(angle) {
-   var cosValue = Math.cos(angle);
-   var sinValue = Math.sin(angle);
-   var m = this._transformMatrix;
-   this._transformMatrix = [
-    m[0] * cosValue + m[2] * sinValue,
-    m[1] * cosValue + m[3] * sinValue,
-    m[0] * -sinValue + m[2] * cosValue,
-    m[1] * -sinValue + m[3] * cosValue,
-    m[4],
-    m[5]
-   ];
-   this._originalRotate(angle);
-  };
- }
+  if (!ctx.mozCurrentTransform) {
+    ctx._originalSave = ctx.save;
+    ctx._originalRestore = ctx.restore;
+    ctx._originalRotate = ctx.rotate;
+    ctx._originalScale = ctx.scale;
+    ctx._originalTranslate = ctx.translate;
+    ctx._originalTransform = ctx.transform;
+    ctx._originalSetTransform = ctx.setTransform;
+    ctx._transformMatrix = ctx._transformMatrix || [1, 0, 0, 1, 0, 0];
+    ctx._transformStack = [];
+    Object.defineProperty(ctx, 'mozCurrentTransform', {
+      get: function getCurrentTransform() {
+        return this._transformMatrix;
+      }
+    });
+    Object.defineProperty(ctx, 'mozCurrentTransformInverse', {
+      get: function getCurrentTransformInverse() {
+        var m = this._transformMatrix;
+        var a = m[0],
+            b = m[1],
+            c = m[2],
+            d = m[3],
+            e = m[4],
+            f = m[5];
+        var ad_bc = a * d - b * c;
+        var bc_ad = b * c - a * d;
+        return [d / ad_bc, b / bc_ad, c / bc_ad, a / ad_bc, (d * e - c * f) / bc_ad, (b * e - a * f) / ad_bc];
+      }
+    });
+    ctx.save = function ctxSave() {
+      var old = this._transformMatrix;
+      this._transformStack.push(old);
+      this._transformMatrix = old.slice(0, 6);
+      this._originalSave();
+    };
+    ctx.restore = function ctxRestore() {
+      var prev = this._transformStack.pop();
+      if (prev) {
+        this._transformMatrix = prev;
+        this._originalRestore();
+      }
+    };
+    ctx.translate = function ctxTranslate(x, y) {
+      var m = this._transformMatrix;
+      m[4] = m[0] * x + m[2] * y + m[4];
+      m[5] = m[1] * x + m[3] * y + m[5];
+      this._originalTranslate(x, y);
+    };
+    ctx.scale = function ctxScale(x, y) {
+      var m = this._transformMatrix;
+      m[0] = m[0] * x;
+      m[1] = m[1] * x;
+      m[2] = m[2] * y;
+      m[3] = m[3] * y;
+      this._originalScale(x, y);
+    };
+    ctx.transform = function ctxTransform(a, b, c, d, e, f) {
+      var m = this._transformMatrix;
+      this._transformMatrix = [m[0] * a + m[2] * b, m[1] * a + m[3] * b, m[0] * c + m[2] * d, m[1] * c + m[3] * d, m[0] * e + m[2] * f + m[4], m[1] * e + m[3] * f + m[5]];
+      ctx._originalTransform(a, b, c, d, e, f);
+    };
+    ctx.setTransform = function ctxSetTransform(a, b, c, d, e, f) {
+      this._transformMatrix = [a, b, c, d, e, f];
+      ctx._originalSetTransform(a, b, c, d, e, f);
+    };
+    ctx.rotate = function ctxRotate(angle) {
+      var cosValue = Math.cos(angle);
+      var sinValue = Math.sin(angle);
+      var m = this._transformMatrix;
+      this._transformMatrix = [m[0] * cosValue + m[2] * sinValue, m[1] * cosValue + m[3] * sinValue, m[0] * -sinValue + m[2] * cosValue, m[1] * -sinValue + m[3] * cosValue, m[4], m[5]];
+      this._originalRotate(angle);
+    };
+  }
 }
 var CachedCanvases = function CachedCanvasesClosure() {
- function CachedCanvases(canvasFactory) {
-  this.canvasFactory = canvasFactory;
-  this.cache = Object.create(null);
- }
- CachedCanvases.prototype = {
-  getCanvas: function CachedCanvases_getCanvas(id, width, height, trackTransform) {
-   var canvasEntry;
-   if (this.cache[id] !== undefined) {
-    canvasEntry = this.cache[id];
-    this.canvasFactory.reset(canvasEntry, width, height);
-    canvasEntry.context.setTransform(1, 0, 0, 1, 0, 0);
-   } else {
-    canvasEntry = this.canvasFactory.create(width, height);
-    this.cache[id] = canvasEntry;
-   }
-   if (trackTransform) {
-    addContextCurrentTransform(canvasEntry.context);
-   }
-   return canvasEntry;
-  },
-  clear: function () {
-   for (var id in this.cache) {
-    var canvasEntry = this.cache[id];
-    this.canvasFactory.destroy(canvasEntry);
-    delete this.cache[id];
-   }
+  function CachedCanvases(canvasFactory) {
+    this.canvasFactory = canvasFactory;
+    this.cache = Object.create(null);
   }
- };
- return CachedCanvases;
+  CachedCanvases.prototype = {
+    getCanvas: function CachedCanvases_getCanvas(id, width, height, trackTransform) {
+      var canvasEntry;
+      if (this.cache[id] !== undefined) {
+        canvasEntry = this.cache[id];
+        this.canvasFactory.reset(canvasEntry, width, height);
+        canvasEntry.context.setTransform(1, 0, 0, 1, 0, 0);
+      } else {
+        canvasEntry = this.canvasFactory.create(width, height);
+        this.cache[id] = canvasEntry;
+      }
+      if (trackTransform) {
+        addContextCurrentTransform(canvasEntry.context);
+      }
+      return canvasEntry;
+    },
+    clear: function () {
+      for (var id in this.cache) {
+        var canvasEntry = this.cache[id];
+        this.canvasFactory.destroy(canvasEntry);
+        delete this.cache[id];
+      }
+    }
+  };
+  return CachedCanvases;
 }();
 function compileType3Glyph(imgData) {
- var POINT_TO_PROCESS_LIMIT = 1000;
- var width = imgData.width, height = imgData.height;
- var i, j, j0, width1 = width + 1;
- var points = new Uint8Array(width1 * (height + 1));
- var POINT_TYPES = new Uint8Array([
-  0,
-  2,
-  4,
-  0,
-  1,
-  0,
-  5,
-  4,
-  8,
-  10,
-  0,
-  8,
-  0,
-  2,
-  1,
-  0
- ]);
- var lineSize = width + 7 & ~7, data0 = imgData.data;
- var data = new Uint8Array(lineSize * height), pos = 0, ii;
- for (i = 0, ii = data0.length; i < ii; i++) {
-  var mask = 128, elem = data0[i];
-  while (mask > 0) {
-   data[pos++] = elem & mask ? 0 : 255;
-   mask >>= 1;
-  }
- }
- var count = 0;
- pos = 0;
- if (data[pos] !== 0) {
-  points[0] = 1;
-  ++count;
- }
- for (j = 1; j < width; j++) {
-  if (data[pos] !== data[pos + 1]) {
-   points[j] = data[pos] ? 2 : 1;
-   ++count;
-  }
-  pos++;
- }
- if (data[pos] !== 0) {
-  points[j] = 2;
-  ++count;
- }
- for (i = 1; i < height; i++) {
-  pos = i * lineSize;
-  j0 = i * width1;
-  if (data[pos - lineSize] !== data[pos]) {
-   points[j0] = data[pos] ? 1 : 8;
-   ++count;
+  var POINT_TO_PROCESS_LIMIT = 1000;
+  var width = imgData.width,
+      height = imgData.height;
+  var i,
+      j,
+      j0,
+      width1 = width + 1;
+  var points = new Uint8Array(width1 * (height + 1));
+  var POINT_TYPES = new Uint8Array([0, 2, 4, 0, 1, 0, 5, 4, 8, 10, 0, 8, 0, 2, 1, 0]);
+  var lineSize = width + 7 & ~7,
+      data0 = imgData.data;
+  var data = new Uint8Array(lineSize * height),
+      pos = 0,
+      ii;
+  for (i = 0, ii = data0.length; i < ii; i++) {
+    var mask = 128,
+        elem = data0[i];
+    while (mask > 0) {
+      data[pos++] = elem & mask ? 0 : 255;
+      mask >>= 1;
+    }
   }
-  var sum = (data[pos] ? 4 : 0) + (data[pos - lineSize] ? 8 : 0);
-  for (j = 1; j < width; j++) {
-   sum = (sum >> 2) + (data[pos + 1] ? 4 : 0) + (data[pos - lineSize + 1] ? 8 : 0);
-   if (POINT_TYPES[sum]) {
-    points[j0 + j] = POINT_TYPES[sum];
+  var count = 0;
+  pos = 0;
+  if (data[pos] !== 0) {
+    points[0] = 1;
     ++count;
-   }
-   pos++;
-  }
-  if (data[pos - lineSize] !== data[pos]) {
-   points[j0 + j] = data[pos] ? 2 : 4;
-   ++count;
-  }
-  if (count > POINT_TO_PROCESS_LIMIT) {
-   return null;
-  }
- }
- pos = lineSize * (height - 1);
- j0 = i * width1;
- if (data[pos] !== 0) {
-  points[j0] = 8;
-  ++count;
- }
- for (j = 1; j < width; j++) {
-  if (data[pos] !== data[pos + 1]) {
-   points[j0 + j] = data[pos] ? 4 : 8;
-   ++count;
-  }
-  pos++;
- }
- if (data[pos] !== 0) {
-  points[j0 + j] = 4;
-  ++count;
- }
- if (count > POINT_TO_PROCESS_LIMIT) {
-  return null;
- }
- var steps = new Int32Array([
-  0,
-  width1,
-  -1,
-  0,
-  -width1,
-  0,
-  0,
-  0,
-  1
- ]);
- var outlines = [];
- for (i = 0; count && i <= height; i++) {
-  var p = i * width1;
-  var end = p + width;
-  while (p < end && !points[p]) {
-   p++;
   }
-  if (p === end) {
-   continue;
-  }
-  var coords = [
-   p % width1,
-   i
-  ];
-  var type = points[p], p0 = p, pp;
-  do {
-   var step = steps[type];
-   do {
-    p += step;
-   } while (!points[p]);
-   pp = points[p];
-   if (pp !== 5 && pp !== 10) {
-    type = pp;
-    points[p] = 0;
-   } else {
-    type = pp & 0x33 * type >> 4;
-    points[p] &= type >> 2 | type << 2;
-   }
-   coords.push(p % width1);
-   coords.push(p / width1 | 0);
-   --count;
-  } while (p0 !== p);
-  outlines.push(coords);
-  --i;
- }
- var drawOutline = function (c) {
-  c.save();
-  c.scale(1 / width, -1 / height);
-  c.translate(0, -height);
-  c.beginPath();
-  for (var i = 0, ii = outlines.length; i < ii; i++) {
-   var o = outlines[i];
-   c.moveTo(o[0], o[1]);
-   for (var j = 2, jj = o.length; j < jj; j += 2) {
-    c.lineTo(o[j], o[j + 1]);
-   }
-  }
-  c.fill();
-  c.beginPath();
-  c.restore();
- };
- return drawOutline;
-}
-var CanvasExtraState = function CanvasExtraStateClosure() {
- function CanvasExtraState(old) {
-  this.alphaIsShape = false;
-  this.fontSize = 0;
-  this.fontSizeScale = 1;
-  this.textMatrix = IDENTITY_MATRIX;
-  this.textMatrixScale = 1;
-  this.fontMatrix = FONT_IDENTITY_MATRIX;
-  this.leading = 0;
-  this.x = 0;
-  this.y = 0;
-  this.lineX = 0;
-  this.lineY = 0;
-  this.charSpacing = 0;
-  this.wordSpacing = 0;
-  this.textHScale = 1;
-  this.textRenderingMode = TextRenderingMode.FILL;
-  this.textRise = 0;
-  this.fillColor = '#000000';
-  this.strokeColor = '#000000';
-  this.patternFill = false;
-  this.fillAlpha = 1;
-  this.strokeAlpha = 1;
-  this.lineWidth = 1;
-  this.activeSMask = null;
-  this.resumeSMaskCtx = null;
-  this.old = old;
- }
- CanvasExtraState.prototype = {
-  clone: function CanvasExtraState_clone() {
-   return Object.create(this);
-  },
-  setCurrentPoint: function CanvasExtraState_setCurrentPoint(x, y) {
-   this.x = x;
-   this.y = y;
-  }
- };
- return CanvasExtraState;
-}();
-var CanvasGraphics = function CanvasGraphicsClosure() {
- var EXECUTION_TIME = 15;
- var EXECUTION_STEPS = 10;
- function CanvasGraphics(canvasCtx, commonObjs, objs, canvasFactory, imageLayer) {
-  this.ctx = canvasCtx;
-  this.current = new CanvasExtraState();
-  this.stateStack = [];
-  this.pendingClip = null;
-  this.pendingEOFill = false;
-  this.res = null;
-  this.xobjs = null;
-  this.commonObjs = commonObjs;
-  this.objs = objs;
-  this.canvasFactory = canvasFactory;
-  this.imageLayer = imageLayer;
-  this.groupStack = [];
-  this.processingType3 = null;
-  this.baseTransform = null;
-  this.baseTransformStack = [];
-  this.groupLevel = 0;
-  this.smaskStack = [];
-  this.smaskCounter = 0;
-  this.tempSMask = null;
-  this.cachedCanvases = new CachedCanvases(this.canvasFactory);
-  if (canvasCtx) {
-   addContextCurrentTransform(canvasCtx);
+  for (j = 1; j < width; j++) {
+    if (data[pos] !== data[pos + 1]) {
+      points[j] = data[pos] ? 2 : 1;
+      ++count;
+    }
+    pos++;
   }
-  this.cachedGetSinglePixelWidth = null;
- }
- function putBinaryImageData(ctx, imgData) {
-  if (typeof ImageData !== 'undefined' && imgData instanceof ImageData) {
-   ctx.putImageData(imgData, 0, 0);
-   return;
+  if (data[pos] !== 0) {
+    points[j] = 2;
+    ++count;
   }
-  var height = imgData.height, width = imgData.width;
-  var partialChunkHeight = height % FULL_CHUNK_HEIGHT;
-  var fullChunks = (height - partialChunkHeight) / FULL_CHUNK_HEIGHT;
-  var totalChunks = partialChunkHeight === 0 ? fullChunks : fullChunks + 1;
-  var chunkImgData = ctx.createImageData(width, FULL_CHUNK_HEIGHT);
-  var srcPos = 0, destPos;
-  var src = imgData.data;
-  var dest = chunkImgData.data;
-  var i, j, thisChunkHeight, elemsInThisChunk;
-  if (imgData.kind === ImageKind.GRAYSCALE_1BPP) {
-   var srcLength = src.byteLength;
-   var dest32 = HasCanvasTypedArraysCached.value ? new Uint32Array(dest.buffer) : new Uint32ArrayView(dest);
-   var dest32DataLength = dest32.length;
-   var fullSrcDiff = width + 7 >> 3;
-   var white = 0xFFFFFFFF;
-   var black = IsLittleEndianCached.value || !HasCanvasTypedArraysCached.value ? 0xFF000000 : 0x000000FF;
-   for (i = 0; i < totalChunks; i++) {
-    thisChunkHeight = i < fullChunks ? FULL_CHUNK_HEIGHT : partialChunkHeight;
-    destPos = 0;
-    for (j = 0; j < thisChunkHeight; j++) {
-     var srcDiff = srcLength - srcPos;
-     var k = 0;
-     var kEnd = srcDiff > fullSrcDiff ? width : srcDiff * 8 - 7;
-     var kEndUnrolled = kEnd & ~7;
-     var mask = 0;
-     var srcByte = 0;
-     for (; k < kEndUnrolled; k += 8) {
-      srcByte = src[srcPos++];
-      dest32[destPos++] = srcByte & 128 ? white : black;
-      dest32[destPos++] = srcByte & 64 ? white : black;
-      dest32[destPos++] = srcByte & 32 ? white : black;
-      dest32[destPos++] = srcByte & 16 ? white : black;
-      dest32[destPos++] = srcByte & 8 ? white : black;
-      dest32[destPos++] = srcByte & 4 ? white : black;
-      dest32[destPos++] = srcByte & 2 ? white : black;
-      dest32[destPos++] = srcByte & 1 ? white : black;
-     }
-     for (; k < kEnd; k++) {
-      if (mask === 0) {
-       srcByte = src[srcPos++];
-       mask = 128;
-      }
-      dest32[destPos++] = srcByte & mask ? white : black;
-      mask >>= 1;
-     }
-    }
-    while (destPos < dest32DataLength) {
-     dest32[destPos++] = 0;
+  for (i = 1; i < height; i++) {
+    pos = i * lineSize;
+    j0 = i * width1;
+    if (data[pos - lineSize] !== data[pos]) {
+      points[j0] = data[pos] ? 1 : 8;
+      ++count;
     }
-    ctx.putImageData(chunkImgData, 0, i * FULL_CHUNK_HEIGHT);
-   }
-  } else if (imgData.kind === ImageKind.RGBA_32BPP) {
-   j = 0;
-   elemsInThisChunk = width * FULL_CHUNK_HEIGHT * 4;
-   for (i = 0; i < fullChunks; i++) {
-    dest.set(src.subarray(srcPos, srcPos + elemsInThisChunk));
-    srcPos += elemsInThisChunk;
-    ctx.putImageData(chunkImgData, 0, j);
-    j += FULL_CHUNK_HEIGHT;
-   }
-   if (i < totalChunks) {
-    elemsInThisChunk = width * partialChunkHeight * 4;
-    dest.set(src.subarray(srcPos, srcPos + elemsInThisChunk));
-    ctx.putImageData(chunkImgData, 0, j);
-   }
-  } else if (imgData.kind === ImageKind.RGB_24BPP) {
-   thisChunkHeight = FULL_CHUNK_HEIGHT;
-   elemsInThisChunk = width * thisChunkHeight;
-   for (i = 0; i < totalChunks; i++) {
-    if (i >= fullChunks) {
-     thisChunkHeight = partialChunkHeight;
-     elemsInThisChunk = width * thisChunkHeight;
+    var sum = (data[pos] ? 4 : 0) + (data[pos - lineSize] ? 8 : 0);
+    for (j = 1; j < width; j++) {
+      sum = (sum >> 2) + (data[pos + 1] ? 4 : 0) + (data[pos - lineSize + 1] ? 8 : 0);
+      if (POINT_TYPES[sum]) {
+        points[j0 + j] = POINT_TYPES[sum];
+        ++count;
+      }
+      pos++;
     }
-    destPos = 0;
-    for (j = elemsInThisChunk; j--;) {
-     dest[destPos++] = src[srcPos++];
-     dest[destPos++] = src[srcPos++];
-     dest[destPos++] = src[srcPos++];
-     dest[destPos++] = 255;
+    if (data[pos - lineSize] !== data[pos]) {
+      points[j0 + j] = data[pos] ? 2 : 4;
+      ++count;
     }
-    ctx.putImageData(chunkImgData, 0, i * FULL_CHUNK_HEIGHT);
-   }
-  } else {
-   error('bad image kind: ' + imgData.kind);
-  }
- }
- function putBinaryImageMask(ctx, imgData) {
-  var height = imgData.height, width = imgData.width;
-  var partialChunkHeight = height % FULL_CHUNK_HEIGHT;
-  var fullChunks = (height - partialChunkHeight) / FULL_CHUNK_HEIGHT;
-  var totalChunks = partialChunkHeight === 0 ? fullChunks : fullChunks + 1;
-  var chunkImgData = ctx.createImageData(width, FULL_CHUNK_HEIGHT);
-  var srcPos = 0;
-  var src = imgData.data;
-  var dest = chunkImgData.data;
-  for (var i = 0; i < totalChunks; i++) {
-   var thisChunkHeight = i < fullChunks ? FULL_CHUNK_HEIGHT : partialChunkHeight;
-   var destPos = 3;
-   for (var j = 0; j < thisChunkHeight; j++) {
-    var mask = 0;
-    for (var k = 0; k < width; k++) {
-     if (!mask) {
-      var elem = src[srcPos++];
-      mask = 128;
-     }
-     dest[destPos] = elem & mask ? 0 : 255;
-     destPos += 4;
-     mask >>= 1;
+    if (count > POINT_TO_PROCESS_LIMIT) {
+      return null;
     }
-   }
-   ctx.putImageData(chunkImgData, 0, i * FULL_CHUNK_HEIGHT);
-  }
- }
- function copyCtxState(sourceCtx, destCtx) {
-  var properties = [
-   'strokeStyle',
-   'fillStyle',
-   'fillRule',
-   'globalAlpha',
-   'lineWidth',
-   'lineCap',
-   'lineJoin',
-   'miterLimit',
-   'globalCompositeOperation',
-   'font'
-  ];
-  for (var i = 0, ii = properties.length; i < ii; i++) {
-   var property = properties[i];
-   if (sourceCtx[property] !== undefined) {
-    destCtx[property] = sourceCtx[property];
-   }
-  }
-  if (sourceCtx.setLineDash !== undefined) {
-   destCtx.setLineDash(sourceCtx.getLineDash());
-   destCtx.lineDashOffset = sourceCtx.lineDashOffset;
   }
- }
- function composeSMaskBackdrop(bytes, r0, g0, b0) {
-  var length = bytes.length;
-  for (var i = 3; i < length; i += 4) {
-   var alpha = bytes[i];
-   if (alpha === 0) {
-    bytes[i - 3] = r0;
-    bytes[i - 2] = g0;
-    bytes[i - 1] = b0;
-   } else if (alpha < 255) {
-    var alpha_ = 255 - alpha;
-    bytes[i - 3] = bytes[i - 3] * alpha + r0 * alpha_ >> 8;
-    bytes[i - 2] = bytes[i - 2] * alpha + g0 * alpha_ >> 8;
-    bytes[i - 1] = bytes[i - 1] * alpha + b0 * alpha_ >> 8;
-   }
-  }
- }
- function composeSMaskAlpha(maskData, layerData, transferMap) {
-  var length = maskData.length;
-  var scale = 1 / 255;
-  for (var i = 3; i < length; i += 4) {
-   var alpha = transferMap ? transferMap[maskData[i]] : maskData[i];
-   layerData[i] = layerData[i] * alpha * scale | 0;
-  }
- }
- function composeSMaskLuminosity(maskData, layerData, transferMap) {
-  var length = maskData.length;
-  for (var i = 3; i < length; i += 4) {
-   var y = maskData[i - 3] * 77 + maskData[i - 2] * 152 + maskData[i - 1] * 28;
-   layerData[i] = transferMap ? layerData[i] * transferMap[y >> 8] >> 8 : layerData[i] * y >> 16;
+  pos = lineSize * (height - 1);
+  j0 = i * width1;
+  if (data[pos] !== 0) {
+    points[j0] = 8;
+    ++count;
   }
- }
- function genericComposeSMask(maskCtx, layerCtx, width, height, subtype, backdrop, transferMap) {
-  var hasBackdrop = !!backdrop;
-  var r0 = hasBackdrop ? backdrop[0] : 0;
-  var g0 = hasBackdrop ? backdrop[1] : 0;
-  var b0 = hasBackdrop ? backdrop[2] : 0;
-  var composeFn;
-  if (subtype === 'Luminosity') {
-   composeFn = composeSMaskLuminosity;
-  } else {
-   composeFn = composeSMaskAlpha;
+  for (j = 1; j < width; j++) {
+    if (data[pos] !== data[pos + 1]) {
+      points[j0 + j] = data[pos] ? 4 : 8;
+      ++count;
+    }
+    pos++;
   }
-  var PIXELS_TO_PROCESS = 1048576;
-  var chunkSize = Math.min(height, Math.ceil(PIXELS_TO_PROCESS / width));
-  for (var row = 0; row < height; row += chunkSize) {
-   var chunkHeight = Math.min(chunkSize, height - row);
-   var maskData = maskCtx.getImageData(0, row, width, chunkHeight);
-   var layerData = layerCtx.getImageData(0, row, width, chunkHeight);
-   if (hasBackdrop) {
-    composeSMaskBackdrop(maskData.data, r0, g0, b0);
-   }
-   composeFn(maskData.data, layerData.data, transferMap);
-   maskCtx.putImageData(layerData, 0, row);
+  if (data[pos] !== 0) {
+    points[j0 + j] = 4;
+    ++count;
   }
- }
- function composeSMask(ctx, smask, layerCtx) {
-  var mask = smask.canvas;
-  var maskCtx = smask.context;
-  ctx.setTransform(smask.scaleX, 0, 0, smask.scaleY, smask.offsetX, smask.offsetY);
-  var backdrop = smask.backdrop || null;
-  if (!smask.transferMap && WebGLUtils.isEnabled) {
-   var composed = WebGLUtils.composeSMask(layerCtx.canvas, mask, {
-    subtype: smask.subtype,
-    backdrop: backdrop
-   });
-   ctx.setTransform(1, 0, 0, 1, 0, 0);
-   ctx.drawImage(composed, smask.offsetX, smask.offsetY);
-   return;
+  if (count > POINT_TO_PROCESS_LIMIT) {
+    return null;
   }
-  genericComposeSMask(maskCtx, layerCtx, mask.width, mask.height, smask.subtype, backdrop, smask.transferMap);
-  ctx.drawImage(mask, 0, 0);
- }
- var LINE_CAP_STYLES = [
-  'butt',
-  'round',
-  'square'
- ];
- var LINE_JOIN_STYLES = [
-  'miter',
-  'round',
-  'bevel'
- ];
- var NORMAL_CLIP = {};
- var EO_CLIP = {};
- CanvasGraphics.prototype = {
-  beginDrawing: function CanvasGraphics_beginDrawing(transform, viewport, transparency) {
-   var width = this.ctx.canvas.width;
-   var height = this.ctx.canvas.height;
-   this.ctx.save();
-   this.ctx.fillStyle = 'rgb(255, 255, 255)';
-   this.ctx.fillRect(0, 0, width, height);
-   this.ctx.restore();
-   if (transparency) {
-    var transparentCanvas = this.cachedCanvases.getCanvas('transparent', width, height, true);
-    this.compositeCtx = this.ctx;
-    this.transparentCanvas = transparentCanvas.canvas;
-    this.ctx = transparentCanvas.context;
-    this.ctx.save();
-    this.ctx.transform.apply(this.ctx, this.compositeCtx.mozCurrentTransform);
-   }
-   this.ctx.save();
-   if (transform) {
-    this.ctx.transform.apply(this.ctx, transform);
-   }
-   this.ctx.transform.apply(this.ctx, viewport.transform);
-   this.baseTransform = this.ctx.mozCurrentTransform.slice();
-   if (this.imageLayer) {
-    this.imageLayer.beginLayout();
-   }
-  },
-  executeOperatorList: function CanvasGraphics_executeOperatorList(operatorList, executionStartIdx, continueCallback, stepper) {
-   var argsArray = operatorList.argsArray;
-   var fnArray = operatorList.fnArray;
-   var i = executionStartIdx || 0;
-   var argsArrayLen = argsArray.length;
-   if (argsArrayLen === i) {
-    return i;
-   }
-   var chunkOperations = argsArrayLen - i > EXECUTION_STEPS && typeof continueCallback === 'function';
-   var endTime = chunkOperations ? Date.now() + EXECUTION_TIME : 0;
-   var steps = 0;
-   var commonObjs = this.commonObjs;
-   var objs = this.objs;
-   var fnId;
-   while (true) {
-    if (stepper !== undefined && i === stepper.nextBreakPoint) {
-     stepper.breakIt(i, continueCallback);
-     return i;
+  var steps = new Int32Array([0, width1, -1, 0, -width1, 0, 0, 0, 1]);
+  var outlines = [];
+  for (i = 0; count && i <= height; i++) {
+    var p = i * width1;
+    var end = p + width;
+    while (p < end && !points[p]) {
+      p++;
     }
-    fnId = fnArray[i];
-    if (fnId !== OPS.dependency) {
-     this[fnId].apply(this, argsArray[i]);
-    } else {
-     var deps = argsArray[i];
-     for (var n = 0, nn = deps.length; n < nn; n++) {
-      var depObjId = deps[n];
-      var common = depObjId[0] === 'g' && depObjId[1] === '_';
-      var objsPool = common ? commonObjs : objs;
-      if (!objsPool.isResolved(depObjId)) {
-       objsPool.get(depObjId, continueCallback);
-       return i;
-      }
-     }
+    if (p === end) {
+      continue;
     }
-    i++;
-    if (i === argsArrayLen) {
-     return i;
-    }
-    if (chunkOperations && ++steps > EXECUTION_STEPS) {
-     if (Date.now() > endTime) {
-      continueCallback();
-      return i;
-     }
-     steps = 0;
-    }
-   }
-  },
-  endDrawing: function CanvasGraphics_endDrawing() {
-   if (this.current.activeSMask !== null) {
-    this.endSMaskGroup();
-   }
-   this.ctx.restore();
-   if (this.transparentCanvas) {
-    this.ctx = this.compositeCtx;
-    this.ctx.save();
-    this.ctx.setTransform(1, 0, 0, 1, 0, 0);
-    this.ctx.drawImage(this.transparentCanvas, 0, 0);
-    this.ctx.restore();
-    this.transparentCanvas = null;
-   }
-   this.cachedCanvases.clear();
-   WebGLUtils.clear();
-   if (this.imageLayer) {
-    this.imageLayer.endLayout();
-   }
-  },
-  setLineWidth: function CanvasGraphics_setLineWidth(width) {
-   this.current.lineWidth = width;
-   this.ctx.lineWidth = width;
-  },
-  setLineCap: function CanvasGraphics_setLineCap(style) {
-   this.ctx.lineCap = LINE_CAP_STYLES[style];
-  },
-  setLineJoin: function CanvasGraphics_setLineJoin(style) {
-   this.ctx.lineJoin = LINE_JOIN_STYLES[style];
-  },
-  setMiterLimit: function CanvasGraphics_setMiterLimit(limit) {
-   this.ctx.miterLimit = limit;
-  },
-  setDash: function CanvasGraphics_setDash(dashArray, dashPhase) {
-   var ctx = this.ctx;
-   if (ctx.setLineDash !== undefined) {
-    ctx.setLineDash(dashArray);
-    ctx.lineDashOffset = dashPhase;
-   }
-  },
-  setRenderingIntent: function CanvasGraphics_setRenderingIntent(intent) {
-  },
-  setFlatness: function CanvasGraphics_setFlatness(flatness) {
-  },
-  setGState: function CanvasGraphics_setGState(states) {
-   for (var i = 0, ii = states.length; i < ii; i++) {
-    var state = states[i];
-    var key = state[0];
-    var value = state[1];
-    switch (key) {
-    case 'LW':
-     this.setLineWidth(value);
-     break;
-    case 'LC':
-     this.setLineCap(value);
-     break;
-    case 'LJ':
-     this.setLineJoin(value);
-     break;
-    case 'ML':
-     this.setMiterLimit(value);
-     break;
-    case 'D':
-     this.setDash(value[0], value[1]);
-     break;
-    case 'RI':
-     this.setRenderingIntent(value);
-     break;
-    case 'FL':
-     this.setFlatness(value);
-     break;
-    case 'Font':
-     this.setFont(value[0], value[1]);
-     break;
-    case 'CA':
-     this.current.strokeAlpha = state[1];
-     break;
-    case 'ca':
-     this.current.fillAlpha = state[1];
-     this.ctx.globalAlpha = state[1];
-     break;
-    case 'BM':
-     if (value && value.name && value.name !== 'Normal') {
-      var mode = value.name.replace(/([A-Z])/g, function (c) {
-       return '-' + c.toLowerCase();
-      }).substring(1);
-      this.ctx.globalCompositeOperation = mode;
-      if (this.ctx.globalCompositeOperation !== mode) {
-       warn('globalCompositeOperation "' + mode + '" is not supported');
-      }
-     } else {
-      this.ctx.globalCompositeOperation = 'source-over';
-     }
-     break;
-    case 'SMask':
-     if (this.current.activeSMask) {
-      if (this.stateStack.length > 0 && this.stateStack[this.stateStack.length - 1].activeSMask === this.current.activeSMask) {
-       this.suspendSMaskGroup();
+    var coords = [p % width1, i];
+    var type = points[p],
+        p0 = p,
+        pp;
+    do {
+      var step = steps[type];
+      do {
+        p += step;
+      } while (!points[p]);
+      pp = points[p];
+      if (pp !== 5 && pp !== 10) {
+        type = pp;
+        points[p] = 0;
       } else {
-       this.endSMaskGroup();
-      }
-     }
-     this.current.activeSMask = value ? this.tempSMask : null;
-     if (this.current.activeSMask) {
-      this.beginSMaskGroup();
-     }
-     this.tempSMask = null;
-     break;
-    }
-   }
-  },
-  beginSMaskGroup: function CanvasGraphics_beginSMaskGroup() {
-   var activeSMask = this.current.activeSMask;
-   var drawnWidth = activeSMask.canvas.width;
-   var drawnHeight = activeSMask.canvas.height;
-   var cacheId = 'smaskGroupAt' + this.groupLevel;
-   var scratchCanvas = this.cachedCanvases.getCanvas(cacheId, drawnWidth, drawnHeight, true);
-   var currentCtx = this.ctx;
-   var currentTransform = currentCtx.mozCurrentTransform;
-   this.ctx.save();
-   var groupCtx = scratchCanvas.context;
-   groupCtx.scale(1 / activeSMask.scaleX, 1 / activeSMask.scaleY);
-   groupCtx.translate(-activeSMask.offsetX, -activeSMask.offsetY);
-   groupCtx.transform.apply(groupCtx, currentTransform);
-   activeSMask.startTransformInverse = groupCtx.mozCurrentTransformInverse;
-   copyCtxState(currentCtx, groupCtx);
-   this.ctx = groupCtx;
-   this.setGState([
-    [
-     'BM',
-     'Normal'
-    ],
-    [
-     'ca',
-     1
-    ],
-    [
-     'CA',
-     1
-    ]
-   ]);
-   this.groupStack.push(currentCtx);
-   this.groupLevel++;
-  },
-  suspendSMaskGroup: function CanvasGraphics_endSMaskGroup() {
-   var groupCtx = this.ctx;
-   this.groupLevel--;
-   this.ctx = this.groupStack.pop();
-   composeSMask(this.ctx, this.current.activeSMask, groupCtx);
-   this.ctx.restore();
-   this.ctx.save();
-   copyCtxState(groupCtx, this.ctx);
-   this.current.resumeSMaskCtx = groupCtx;
-   var deltaTransform = Util.transform(this.current.activeSMask.startTransformInverse, groupCtx.mozCurrentTransform);
-   this.ctx.transform.apply(this.ctx, deltaTransform);
-   groupCtx.save();
-   groupCtx.setTransform(1, 0, 0, 1, 0, 0);
-   groupCtx.clearRect(0, 0, groupCtx.canvas.width, groupCtx.canvas.height);
-   groupCtx.restore();
-  },
-  resumeSMaskGroup: function CanvasGraphics_endSMaskGroup() {
-   var groupCtx = this.current.resumeSMaskCtx;
-   var currentCtx = this.ctx;
-   this.ctx = groupCtx;
-   this.groupStack.push(currentCtx);
-   this.groupLevel++;
-  },
-  endSMaskGroup: function CanvasGraphics_endSMaskGroup() {
-   var groupCtx = this.ctx;
-   this.groupLevel--;
-   this.ctx = this.groupStack.pop();
-   composeSMask(this.ctx, this.current.activeSMask, groupCtx);
-   this.ctx.restore();
-   copyCtxState(groupCtx, this.ctx);
-   var deltaTransform = Util.transform(this.current.activeSMask.startTransformInverse, groupCtx.mozCurrentTransform);
-   this.ctx.transform.apply(this.ctx, deltaTransform);
-  },
-  save: function CanvasGraphics_save() {
-   this.ctx.save();
-   var old = this.current;
-   this.stateStack.push(old);
-   this.current = old.clone();
-   this.current.resumeSMaskCtx = null;
-  },
-  restore: function CanvasGraphics_restore() {
-   if (this.current.resumeSMaskCtx) {
-    this.resumeSMaskGroup();
-   }
-   if (this.current.activeSMask !== null && (this.stateStack.length === 0 || this.stateStack[this.stateStack.length - 1].activeSMask !== this.current.activeSMask)) {
-    this.endSMaskGroup();
-   }
-   if (this.stateStack.length !== 0) {
-    this.current = this.stateStack.pop();
-    this.ctx.restore();
-    this.pendingClip = null;
-    this.cachedGetSinglePixelWidth = null;
-   }
-  },
-  transform: function CanvasGraphics_transform(a, b, c, d, e, f) {
-   this.ctx.transform(a, b, c, d, e, f);
-   this.cachedGetSinglePixelWidth = null;
-  },
-  constructPath: function CanvasGraphics_constructPath(ops, args) {
-   var ctx = this.ctx;
-   var current = this.current;
-   var x = current.x, y = current.y;
-   for (var i = 0, j = 0, ii = ops.length; i < ii; i++) {
-    switch (ops[i] | 0) {
-    case OPS.rectangle:
-     x = args[j++];
-     y = args[j++];
-     var width = args[j++];
-     var height = args[j++];
-     if (width === 0) {
-      width = this.getSinglePixelWidth();
-     }
-     if (height === 0) {
-      height = this.getSinglePixelWidth();
-     }
-     var xw = x + width;
-     var yh = y + height;
-     this.ctx.moveTo(x, y);
-     this.ctx.lineTo(xw, y);
-     this.ctx.lineTo(xw, yh);
-     this.ctx.lineTo(x, yh);
-     this.ctx.lineTo(x, y);
-     this.ctx.closePath();
-     break;
-    case OPS.moveTo:
-     x = args[j++];
-     y = args[j++];
-     ctx.moveTo(x, y);
-     break;
-    case OPS.lineTo:
-     x = args[j++];
-     y = args[j++];
-     ctx.lineTo(x, y);
-     break;
-    case OPS.curveTo:
-     x = args[j + 4];
-     y = args[j + 5];
-     ctx.bezierCurveTo(args[j], args[j + 1], args[j + 2], args[j + 3], x, y);
-     j += 6;
-     break;
-    case OPS.curveTo2:
-     ctx.bezierCurveTo(x, y, args[j], args[j + 1], args[j + 2], args[j + 3]);
-     x = args[j + 2];
-     y = args[j + 3];
-     j += 4;
-     break;
-    case OPS.curveTo3:
-     x = args[j + 2];
-     y = args[j + 3];
-     ctx.bezierCurveTo(args[j], args[j + 1], x, y, x, y);
-     j += 4;
-     break;
-    case OPS.closePath:
-     ctx.closePath();
-     break;
+        type = pp & 0x33 * type >> 4;
+        points[p] &= type >> 2 | type << 2;
+      }
+      coords.push(p % width1);
+      coords.push(p / width1 | 0);
+      --count;
+    } while (p0 !== p);
+    outlines.push(coords);
+    --i;
+  }
+  var drawOutline = function (c) {
+    c.save();
+    c.scale(1 / width, -1 / height);
+    c.translate(0, -height);
+    c.beginPath();
+    for (var i = 0, ii = outlines.length; i < ii; i++) {
+      var o = outlines[i];
+      c.moveTo(o[0], o[1]);
+      for (var j = 2, jj = o.length; j < jj; j += 2) {
+        c.lineTo(o[j], o[j + 1]);
+      }
     }
-   }
-   current.setCurrentPoint(x, y);
-  },
-  closePath: function CanvasGraphics_closePath() {
-   this.ctx.closePath();
-  },
-  stroke: function CanvasGraphics_stroke(consumePath) {
-   consumePath = typeof consumePath !== 'undefined' ? consumePath : true;
-   var ctx = this.ctx;
-   var strokeColor = this.current.strokeColor;
-   ctx.lineWidth = Math.max(this.getSinglePixelWidth() * MIN_WIDTH_FACTOR, this.current.lineWidth);
-   ctx.globalAlpha = this.current.strokeAlpha;
-   if (strokeColor && strokeColor.hasOwnProperty('type') && strokeColor.type === 'Pattern') {
-    ctx.save();
-    ctx.strokeStyle = strokeColor.getPattern(ctx, this);
-    ctx.stroke();
-    ctx.restore();
-   } else {
-    ctx.stroke();
-   }
-   if (consumePath) {
-    this.consumePath();
-   }
-   ctx.globalAlpha = this.current.fillAlpha;
-  },
-  closeStroke: function CanvasGraphics_closeStroke() {
-   this.closePath();
-   this.stroke();
-  },
-  fill: function CanvasGraphics_fill(consumePath) {
-   consumePath = typeof consumePath !== 'undefined' ? consumePath : true;
-   var ctx = this.ctx;
-   var fillColor = this.current.fillColor;
-   var isPatternFill = this.current.patternFill;
-   var needRestore = false;
-   if (isPatternFill) {
-    ctx.save();
-    if (this.baseTransform) {
-     ctx.setTransform.apply(ctx, this.baseTransform);
+    c.fill();
+    c.beginPath();
+    c.restore();
+  };
+  return drawOutline;
+}
+var CanvasExtraState = function CanvasExtraStateClosure() {
+  function CanvasExtraState(old) {
+    this.alphaIsShape = false;
+    this.fontSize = 0;
+    this.fontSizeScale = 1;
+    this.textMatrix = IDENTITY_MATRIX;
+    this.textMatrixScale = 1;
+    this.fontMatrix = FONT_IDENTITY_MATRIX;
+    this.leading = 0;
+    this.x = 0;
+    this.y = 0;
+    this.lineX = 0;
+    this.lineY = 0;
+    this.charSpacing = 0;
+    this.wordSpacing = 0;
+    this.textHScale = 1;
+    this.textRenderingMode = TextRenderingMode.FILL;
+    this.textRise = 0;
+    this.fillColor = '#000000';
+    this.strokeColor = '#000000';
+    this.patternFill = false;
+    this.fillAlpha = 1;
+    this.strokeAlpha = 1;
+    this.lineWidth = 1;
+    this.activeSMask = null;
+    this.resumeSMaskCtx = null;
+    this.old = old;
+  }
+  CanvasExtraState.prototype = {
+    clone: function CanvasExtraState_clone() {
+      return Object.create(this);
+    },
+    setCurrentPoint: function CanvasExtraState_setCurrentPoint(x, y) {
+      this.x = x;
+      this.y = y;
     }
-    ctx.fillStyle = fillColor.getPattern(ctx, this);
-    needRestore = true;
-   }
-   if (this.pendingEOFill) {
-    ctx.fill('evenodd');
+  };
+  return CanvasExtraState;
+}();
+var CanvasGraphics = function CanvasGraphicsClosure() {
+  var EXECUTION_TIME = 15;
+  var EXECUTION_STEPS = 10;
+  function CanvasGraphics(canvasCtx, commonObjs, objs, canvasFactory, imageLayer) {
+    this.ctx = canvasCtx;
+    this.current = new CanvasExtraState();
+    this.stateStack = [];
+    this.pendingClip = null;
     this.pendingEOFill = false;
-   } else {
-    ctx.fill();
-   }
-   if (needRestore) {
-    ctx.restore();
-   }
-   if (consumePath) {
-    this.consumePath();
-   }
-  },
-  eoFill: function CanvasGraphics_eoFill() {
-   this.pendingEOFill = true;
-   this.fill();
-  },
-  fillStroke: function CanvasGraphics_fillStroke() {
-   this.fill(false);
-   this.stroke(false);
-   this.consumePath();
-  },
-  eoFillStroke: function CanvasGraphics_eoFillStroke() {
-   this.pendingEOFill = true;
-   this.fillStroke();
-  },
-  closeFillStroke: function CanvasGraphics_closeFillStroke() {
-   this.closePath();
-   this.fillStroke();
-  },
-  closeEOFillStroke: function CanvasGraphics_closeEOFillStroke() {
-   this.pendingEOFill = true;
-   this.closePath();
-   this.fillStroke();
-  },
-  endPath: function CanvasGraphics_endPath() {
-   this.consumePath();
-  },
-  clip: function CanvasGraphics_clip() {
-   this.pendingClip = NORMAL_CLIP;
-  },
-  eoClip: function CanvasGraphics_eoClip() {
-   this.pendingClip = EO_CLIP;
-  },
-  beginText: function CanvasGraphics_beginText() {
-   this.current.textMatrix = IDENTITY_MATRIX;
-   this.current.textMatrixScale = 1;
-   this.current.x = this.current.lineX = 0;
-   this.current.y = this.current.lineY = 0;
-  },
-  endText: function CanvasGraphics_endText() {
-   var paths = this.pendingTextPaths;
-   var ctx = this.ctx;
-   if (paths === undefined) {
-    ctx.beginPath();
-    return;
-   }
-   ctx.save();
-   ctx.beginPath();
-   for (var i = 0; i < paths.length; i++) {
-    var path = paths[i];
-    ctx.setTransform.apply(ctx, path.transform);
-    ctx.translate(path.x, path.y);
-    path.addToPath(ctx, path.fontSize);
-   }
-   ctx.restore();
-   ctx.clip();
-   ctx.beginPath();
-   delete this.pendingTextPaths;
-  },
-  setCharSpacing: function CanvasGraphics_setCharSpacing(spacing) {
-   this.current.charSpacing = spacing;
-  },
-  setWordSpacing: function CanvasGraphics_setWordSpacing(spacing) {
-   this.current.wordSpacing = spacing;
-  },
-  setHScale: function CanvasGraphics_setHScale(scale) {
-   this.current.textHScale = scale / 100;
-  },
-  setLeading: function CanvasGraphics_setLeading(leading) {
-   this.current.leading = -leading;
-  },
-  setFont: function CanvasGraphics_setFont(fontRefName, size) {
-   var fontObj = this.commonObjs.get(fontRefName);
-   var current = this.current;
-   if (!fontObj) {
-    error('Can\'t find font for ' + fontRefName);
-   }
-   current.fontMatrix = fontObj.fontMatrix ? fontObj.fontMatrix : FONT_IDENTITY_MATRIX;
-   if (current.fontMatrix[0] === 0 || current.fontMatrix[3] === 0) {
-    warn('Invalid font matrix for font ' + fontRefName);
-   }
-   if (size < 0) {
-    size = -size;
-    current.fontDirection = -1;
-   } else {
-    current.fontDirection = 1;
-   }
-   this.current.font = fontObj;
-   this.current.fontSize = size;
-   if (fontObj.isType3Font) {
-    return;
-   }
-   var name = fontObj.loadedName || 'sans-serif';
-   var bold = fontObj.black ? '900' : fontObj.bold ? 'bold' : 'normal';
-   var italic = fontObj.italic ? 'italic' : 'normal';
-   var typeface = '"' + name + '", ' + fontObj.fallbackName;
-   var browserFontSize = size < MIN_FONT_SIZE ? MIN_FONT_SIZE : size > MAX_FONT_SIZE ? MAX_FONT_SIZE : size;
-   this.current.fontSizeScale = size / browserFontSize;
-   var rule = italic + ' ' + bold + ' ' + browserFontSize + 'px ' + typeface;
-   this.ctx.font = rule;
-  },
-  setTextRenderingMode: function CanvasGraphics_setTextRenderingMode(mode) {
-   this.current.textRenderingMode = mode;
-  },
-  setTextRise: function CanvasGraphics_setTextRise(rise) {
-   this.current.textRise = rise;
-  },
-  moveText: function CanvasGraphics_moveText(x, y) {
-   this.current.x = this.current.lineX += x;
-   this.current.y = this.current.lineY += y;
-  },
-  setLeadingMoveText: function CanvasGraphics_setLeadingMoveText(x, y) {
-   this.setLeading(-y);
-   this.moveText(x, y);
-  },
-  setTextMatrix: function CanvasGraphics_setTextMatrix(a, b, c, d, e, f) {
-   this.current.textMatrix = [
-    a,
-    b,
-    c,
-    d,
-    e,
-    f
-   ];
-   this.current.textMatrixScale = Math.sqrt(a * a + b * b);
-   this.current.x = this.current.lineX = 0;
-   this.current.y = this.current.lineY = 0;
-  },
-  nextLine: function CanvasGraphics_nextLine() {
-   this.moveText(0, this.current.leading);
-  },
-  paintChar: function CanvasGraphics_paintChar(character, x, y) {
-   var ctx = this.ctx;
-   var current = this.current;
-   var font = current.font;
-   var textRenderingMode = current.textRenderingMode;
-   var fontSize = current.fontSize / current.fontSizeScale;
-   var fillStrokeMode = textRenderingMode & TextRenderingMode.FILL_STROKE_MASK;
-   var isAddToPathSet = !!(textRenderingMode & TextRenderingMode.ADD_TO_PATH_FLAG);
-   var addToPath;
-   if (font.disableFontFace || isAddToPathSet) {
-    addToPath = font.getPathGenerator(this.commonObjs, character);
-   }
-   if (font.disableFontFace) {
-    ctx.save();
-    ctx.translate(x, y);
-    ctx.beginPath();
-    addToPath(ctx, fontSize);
-    if (fillStrokeMode === TextRenderingMode.FILL || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
-     ctx.fill();
-    }
-    if (fillStrokeMode === TextRenderingMode.STROKE || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
-     ctx.stroke();
-    }
-    ctx.restore();
-   } else {
-    if (fillStrokeMode === TextRenderingMode.FILL || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
-     ctx.fillText(character, x, y);
-    }
-    if (fillStrokeMode === TextRenderingMode.STROKE || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
-     ctx.strokeText(character, x, y);
+    this.res = null;
+    this.xobjs = null;
+    this.commonObjs = commonObjs;
+    this.objs = objs;
+    this.canvasFactory = canvasFactory;
+    this.imageLayer = imageLayer;
+    this.groupStack = [];
+    this.processingType3 = null;
+    this.baseTransform = null;
+    this.baseTransformStack = [];
+    this.groupLevel = 0;
+    this.smaskStack = [];
+    this.smaskCounter = 0;
+    this.tempSMask = null;
+    this.cachedCanvases = new CachedCanvases(this.canvasFactory);
+    if (canvasCtx) {
+      addContextCurrentTransform(canvasCtx);
     }
-   }
-   if (isAddToPathSet) {
-    var paths = this.pendingTextPaths || (this.pendingTextPaths = []);
-    paths.push({
-     transform: ctx.mozCurrentTransform,
-     x: x,
-     y: y,
-     fontSize: fontSize,
-     addToPath: addToPath
-    });
-   }
-  },
-  get isFontSubpixelAAEnabled() {
-   var ctx = this.canvasFactory.create(10, 10).context;
-   ctx.scale(1.5, 1);
-   ctx.fillText('I', 0, 10);
-   var data = ctx.getImageData(0, 0, 10, 10).data;
-   var enabled = false;
-   for (var i = 3; i < data.length; i += 4) {
-    if (data[i] > 0 && data[i] < 255) {
-     enabled = true;
-     break;
-    }
-   }
-   return shadow(this, 'isFontSubpixelAAEnabled', enabled);
-  },
-  showText: function CanvasGraphics_showText(glyphs) {
-   var current = this.current;
-   var font = current.font;
-   if (font.isType3Font) {
-    return this.showType3Text(glyphs);
-   }
-   var fontSize = current.fontSize;
-   if (fontSize === 0) {
-    return;
-   }
-   var ctx = this.ctx;
-   var fontSizeScale = current.fontSizeScale;
-   var charSpacing = current.charSpacing;
-   var wordSpacing = current.wordSpacing;
-   var fontDirection = current.fontDirection;
-   var textHScale = current.textHScale * fontDirection;
-   var glyphsLength = glyphs.length;
-   var vertical = font.vertical;
-   var spacingDir = vertical ? 1 : -1;
-   var defaultVMetrics = font.defaultVMetrics;
-   var widthAdvanceScale = fontSize * current.fontMatrix[0];
-   var simpleFillText = current.textRenderingMode === TextRenderingMode.FILL && !font.disableFontFace;
-   ctx.save();
-   ctx.transform.apply(ctx, current.textMatrix);
-   ctx.translate(current.x, current.y + current.textRise);
-   if (current.patternFill) {
-    ctx.fillStyle = current.fillColor.getPattern(ctx, this);
-   }
-   if (fontDirection > 0) {
-    ctx.scale(textHScale, -1);
-   } else {
-    ctx.scale(textHScale, 1);
-   }
-   var lineWidth = current.lineWidth;
-   var scale = current.textMatrixScale;
-   if (scale === 0 || lineWidth === 0) {
-    var fillStrokeMode = current.textRenderingMode & TextRenderingMode.FILL_STROKE_MASK;
-    if (fillStrokeMode === TextRenderingMode.STROKE || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
-     this.cachedGetSinglePixelWidth = null;
-     lineWidth = this.getSinglePixelWidth() * MIN_WIDTH_FACTOR;
-    }
-   } else {
-    lineWidth /= scale;
-   }
-   if (fontSizeScale !== 1.0) {
-    ctx.scale(fontSizeScale, fontSizeScale);
-    lineWidth /= fontSizeScale;
-   }
-   ctx.lineWidth = lineWidth;
-   var x = 0, i;
-   for (i = 0; i < glyphsLength; ++i) {
-    var glyph = glyphs[i];
-    if (isNum(glyph)) {
-     x += spacingDir * glyph * fontSize / 1000;
-     continue;
+    this.cachedGetSinglePixelWidth = null;
+  }
+  function putBinaryImageData(ctx, imgData) {
+    if (typeof ImageData !== 'undefined' && imgData instanceof ImageData) {
+      ctx.putImageData(imgData, 0, 0);
+      return;
     }
-    var restoreNeeded = false;
-    var spacing = (glyph.isSpace ? wordSpacing : 0) + charSpacing;
-    var character = glyph.fontChar;
-    var accent = glyph.accent;
-    var scaledX, scaledY, scaledAccentX, scaledAccentY;
-    var width = glyph.width;
-    if (vertical) {
-     var vmetric, vx, vy;
-     vmetric = glyph.vmetric || defaultVMetrics;
-     vx = glyph.vmetric ? vmetric[1] : width * 0.5;
-     vx = -vx * widthAdvanceScale;
-     vy = vmetric[2] * widthAdvanceScale;
-     width = vmetric ? -vmetric[0] : width;
-     scaledX = vx / fontSizeScale;
-     scaledY = (x + vy) / fontSizeScale;
+    var height = imgData.height,
+        width = imgData.width;
+    var partialChunkHeight = height % FULL_CHUNK_HEIGHT;
+    var fullChunks = (height - partialChunkHeight) / FULL_CHUNK_HEIGHT;
+    var totalChunks = partialChunkHeight === 0 ? fullChunks : fullChunks + 1;
+    var chunkImgData = ctx.createImageData(width, FULL_CHUNK_HEIGHT);
+    var srcPos = 0,
+        destPos;
+    var src = imgData.data;
+    var dest = chunkImgData.data;
+    var i, j, thisChunkHeight, elemsInThisChunk;
+    if (imgData.kind === ImageKind.GRAYSCALE_1BPP) {
+      var srcLength = src.byteLength;
+      var dest32 = HasCanvasTypedArraysCached.value ? new Uint32Array(dest.buffer) : new Uint32ArrayView(dest);
+      var dest32DataLength = dest32.length;
+      var fullSrcDiff = width + 7 >> 3;
+      var white = 0xFFFFFFFF;
+      var black = IsLittleEndianCached.value || !HasCanvasTypedArraysCached.value ? 0xFF000000 : 0x000000FF;
+      for (i = 0; i < totalChunks; i++) {
+        thisChunkHeight = i < fullChunks ? FULL_CHUNK_HEIGHT : partialChunkHeight;
+        destPos = 0;
+        for (j = 0; j < thisChunkHeight; j++) {
+          var srcDiff = srcLength - srcPos;
+          var k = 0;
+          var kEnd = srcDiff > fullSrcDiff ? width : srcDiff * 8 - 7;
+          var kEndUnrolled = kEnd & ~7;
+          var mask = 0;
+          var srcByte = 0;
+          for (; k < kEndUnrolled; k += 8) {
+            srcByte = src[srcPos++];
+            dest32[destPos++] = srcByte & 128 ? white : black;
+            dest32[destPos++] = srcByte & 64 ? white : black;
+            dest32[destPos++] = srcByte & 32 ? white : black;
+            dest32[destPos++] = srcByte & 16 ? white : black;
+            dest32[destPos++] = srcByte & 8 ? white : black;
+            dest32[destPos++] = srcByte & 4 ? white : black;
+            dest32[destPos++] = srcByte & 2 ? white : black;
+            dest32[destPos++] = srcByte & 1 ? white : black;
+          }
+          for (; k < kEnd; k++) {
+            if (mask === 0) {
+              srcByte = src[srcPos++];
+              mask = 128;
+            }
+            dest32[destPos++] = srcByte & mask ? white : black;
+            mask >>= 1;
+          }
+        }
+        while (destPos < dest32DataLength) {
+          dest32[destPos++] = 0;
+        }
+        ctx.putImageData(chunkImgData, 0, i * FULL_CHUNK_HEIGHT);
+      }
+    } else if (imgData.kind === ImageKind.RGBA_32BPP) {
+      j = 0;
+      elemsInThisChunk = width * FULL_CHUNK_HEIGHT * 4;
+      for (i = 0; i < fullChunks; i++) {
+        dest.set(src.subarray(srcPos, srcPos + elemsInThisChunk));
+        srcPos += elemsInThisChunk;
+        ctx.putImageData(chunkImgData, 0, j);
+        j += FULL_CHUNK_HEIGHT;
+      }
+      if (i < totalChunks) {
+        elemsInThisChunk = width * partialChunkHeight * 4;
+        dest.set(src.subarray(srcPos, srcPos + elemsInThisChunk));
+        ctx.putImageData(chunkImgData, 0, j);
+      }
+    } else if (imgData.kind === ImageKind.RGB_24BPP) {
+      thisChunkHeight = FULL_CHUNK_HEIGHT;
+      elemsInThisChunk = width * thisChunkHeight;
+      for (i = 0; i < totalChunks; i++) {
+        if (i >= fullChunks) {
+          thisChunkHeight = partialChunkHeight;
+          elemsInThisChunk = width * thisChunkHeight;
+        }
+        destPos = 0;
+        for (j = elemsInThisChunk; j--;) {
+          dest[destPos++] = src[srcPos++];
+          dest[destPos++] = src[srcPos++];
+          dest[destPos++] = src[srcPos++];
+          dest[destPos++] = 255;
+        }
+        ctx.putImageData(chunkImgData, 0, i * FULL_CHUNK_HEIGHT);
+      }
     } else {
-     scaledX = x / fontSizeScale;
-     scaledY = 0;
+      error('bad image kind: ' + imgData.kind);
     }
-    if (font.remeasure && width > 0) {
-     var measuredWidth = ctx.measureText(character).width * 1000 / fontSize * fontSizeScale;
-     if (width < measuredWidth && this.isFontSubpixelAAEnabled) {
-      var characterScaleX = width / measuredWidth;
-      restoreNeeded = true;
-      ctx.save();
-      ctx.scale(characterScaleX, 1);
-      scaledX /= characterScaleX;
-     } else if (width !== measuredWidth) {
-      scaledX += (width - measuredWidth) / 2000 * fontSize / fontSizeScale;
-     }
+  }
+  function putBinaryImageMask(ctx, imgData) {
+    var height = imgData.height,
+        width = imgData.width;
+    var partialChunkHeight = height % FULL_CHUNK_HEIGHT;
+    var fullChunks = (height - partialChunkHeight) / FULL_CHUNK_HEIGHT;
+    var totalChunks = partialChunkHeight === 0 ? fullChunks : fullChunks + 1;
+    var chunkImgData = ctx.createImageData(width, FULL_CHUNK_HEIGHT);
+    var srcPos = 0;
+    var src = imgData.data;
+    var dest = chunkImgData.data;
+    for (var i = 0; i < totalChunks; i++) {
+      var thisChunkHeight = i < fullChunks ? FULL_CHUNK_HEIGHT : partialChunkHeight;
+      var destPos = 3;
+      for (var j = 0; j < thisChunkHeight; j++) {
+        var mask = 0;
+        for (var k = 0; k < width; k++) {
+          if (!mask) {
+            var elem = src[srcPos++];
+            mask = 128;
+          }
+          dest[destPos] = elem & mask ? 0 : 255;
+          destPos += 4;
+          mask >>= 1;
+        }
+      }
+      ctx.putImageData(chunkImgData, 0, i * FULL_CHUNK_HEIGHT);
     }
-    if (glyph.isInFont || font.missingFile) {
-     if (simpleFillText && !accent) {
-      ctx.fillText(character, scaledX, scaledY);
-     } else {
-      this.paintChar(character, scaledX, scaledY);
-      if (accent) {
-       scaledAccentX = scaledX + accent.offset.x / fontSizeScale;
-       scaledAccentY = scaledY - accent.offset.y / fontSizeScale;
-       this.paintChar(accent.fontChar, scaledAccentX, scaledAccentY);
-      }
-     }
+  }
+  function copyCtxState(sourceCtx, destCtx) {
+    var properties = ['strokeStyle', 'fillStyle', 'fillRule', 'globalAlpha', 'lineWidth', 'lineCap', 'lineJoin', 'miterLimit', 'globalCompositeOperation', 'font'];
+    for (var i = 0, ii = properties.length; i < ii; i++) {
+      var property = properties[i];
+      if (sourceCtx[property] !== undefined) {
+        destCtx[property] = sourceCtx[property];
+      }
     }
-    var charWidth = width * widthAdvanceScale + spacing * fontDirection;
-    x += charWidth;
-    if (restoreNeeded) {
-     ctx.restore();
+    if (sourceCtx.setLineDash !== undefined) {
+      destCtx.setLineDash(sourceCtx.getLineDash());
+      destCtx.lineDashOffset = sourceCtx.lineDashOffset;
     }
-   }
-   if (vertical) {
-    current.y -= x * textHScale;
-   } else {
-    current.x += x * textHScale;
-   }
-   ctx.restore();
-  },
-  showType3Text: function CanvasGraphics_showType3Text(glyphs) {
-   var ctx = this.ctx;
-   var current = this.current;
-   var font = current.font;
-   var fontSize = current.fontSize;
-   var fontDirection = current.fontDirection;
-   var spacingDir = font.vertical ? 1 : -1;
-   var charSpacing = current.charSpacing;
-   var wordSpacing = current.wordSpacing;
-   var textHScale = current.textHScale * fontDirection;
-   var fontMatrix = current.fontMatrix || FONT_IDENTITY_MATRIX;
-   var glyphsLength = glyphs.length;
-   var isTextInvisible = current.textRenderingMode === TextRenderingMode.INVISIBLE;
-   var i, glyph, width, spacingLength;
-   if (isTextInvisible || fontSize === 0) {
-    return;
-   }
-   this.cachedGetSinglePixelWidth = null;
-   ctx.save();
-   ctx.transform.apply(ctx, current.textMatrix);
-   ctx.translate(current.x, current.y);
-   ctx.scale(textHScale, fontDirection);
-   for (i = 0; i < glyphsLength; ++i) {
-    glyph = glyphs[i];
-    if (isNum(glyph)) {
-     spacingLength = spacingDir * glyph * fontSize / 1000;
-     this.ctx.translate(spacingLength, 0);
-     current.x += spacingLength * textHScale;
-     continue;
+  }
+  function composeSMaskBackdrop(bytes, r0, g0, b0) {
+    var length = bytes.length;
+    for (var i = 3; i < length; i += 4) {
+      var alpha = bytes[i];
+      if (alpha === 0) {
+        bytes[i - 3] = r0;
+        bytes[i - 2] = g0;
+        bytes[i - 1] = b0;
+      } else if (alpha < 255) {
+        var alpha_ = 255 - alpha;
+        bytes[i - 3] = bytes[i - 3] * alpha + r0 * alpha_ >> 8;
+        bytes[i - 2] = bytes[i - 2] * alpha + g0 * alpha_ >> 8;
+        bytes[i - 1] = bytes[i - 1] * alpha + b0 * alpha_ >> 8;
+      }
     }
-    var spacing = (glyph.isSpace ? wordSpacing : 0) + charSpacing;
-    var operatorList = font.charProcOperatorList[glyph.operatorListId];
-    if (!operatorList) {
-     warn('Type3 character \"' + glyph.operatorListId + '\" is not available');
-     continue;
+  }
+  function composeSMaskAlpha(maskData, layerData, transferMap) {
+    var length = maskData.length;
+    var scale = 1 / 255;
+    for (var i = 3; i < length; i += 4) {
+      var alpha = transferMap ? transferMap[maskData[i]] : maskData[i];
+      layerData[i] = layerData[i] * alpha * scale | 0;
     }
-    this.processingType3 = glyph;
-    this.save();
-    ctx.scale(fontSize, fontSize);
-    ctx.transform.apply(ctx, fontMatrix);
-    this.executeOperatorList(operatorList);
-    this.restore();
-    var transformed = Util.applyTransform([
-     glyph.width,
-     0
-    ], fontMatrix);
-    width = transformed[0] * fontSize + spacing;
-    ctx.translate(width, 0);
-    current.x += width * textHScale;
-   }
-   ctx.restore();
-   this.processingType3 = null;
-  },
-  setCharWidth: function CanvasGraphics_setCharWidth(xWidth, yWidth) {
-  },
-  setCharWidthAndBounds: function CanvasGraphics_setCharWidthAndBounds(xWidth, yWidth, llx, lly, urx, ury) {
-   this.ctx.rect(llx, lly, urx - llx, ury - lly);
-   this.clip();
-   this.endPath();
-  },
-  getColorN_Pattern: function CanvasGraphics_getColorN_Pattern(IR) {
-   var pattern;
-   if (IR[0] === 'TilingPattern') {
-    var color = IR[1];
-    var baseTransform = this.baseTransform || this.ctx.mozCurrentTransform.slice();
-    var self = this;
-    var canvasGraphicsFactory = {
-     createCanvasGraphics: function (ctx) {
-      return new CanvasGraphics(ctx, self.commonObjs, self.objs, self.canvasFactory);
-     }
-    };
-    pattern = new TilingPattern(IR, color, this.ctx, canvasGraphicsFactory, baseTransform);
-   } else {
-    pattern = getShadingPatternFromIR(IR);
-   }
-   return pattern;
-  },
-  setStrokeColorN: function CanvasGraphics_setStrokeColorN() {
-   this.current.strokeColor = this.getColorN_Pattern(arguments);
-  },
-  setFillColorN: function CanvasGraphics_setFillColorN() {
-   this.current.fillColor = this.getColorN_Pattern(arguments);
-   this.current.patternFill = true;
-  },
-  setStrokeRGBColor: function CanvasGraphics_setStrokeRGBColor(r, g, b) {
-   var color = Util.makeCssRgb(r, g, b);
-   this.ctx.strokeStyle = color;
-   this.current.strokeColor = color;
-  },
-  setFillRGBColor: function CanvasGraphics_setFillRGBColor(r, g, b) {
-   var color = Util.makeCssRgb(r, g, b);
-   this.ctx.fillStyle = color;
-   this.current.fillColor = color;
-   this.current.patternFill = false;
-  },
-  shadingFill: function CanvasGraphics_shadingFill(patternIR) {
-   var ctx = this.ctx;
-   this.save();
-   var pattern = getShadingPatternFromIR(patternIR);
-   ctx.fillStyle = pattern.getPattern(ctx, this, true);
-   var inv = ctx.mozCurrentTransformInverse;
-   if (inv) {
-    var canvas = ctx.canvas;
-    var width = canvas.width;
-    var height = canvas.height;
-    var bl = Util.applyTransform([
-     0,
-     0
-    ], inv);
-    var br = Util.applyTransform([
-     0,
-     height
-    ], inv);
-    var ul = Util.applyTransform([
-     width,
-     0
-    ], inv);
-    var ur = Util.applyTransform([
-     width,
-     height
-    ], inv);
-    var x0 = Math.min(bl[0], br[0], ul[0], ur[0]);
-    var y0 = Math.min(bl[1], br[1], ul[1], ur[1]);
-    var x1 = Math.max(bl[0], br[0], ul[0], ur[0]);
-    var y1 = Math.max(bl[1], br[1], ul[1], ur[1]);
-    this.ctx.fillRect(x0, y0, x1 - x0, y1 - y0);
-   } else {
-    this.ctx.fillRect(-1e10, -1e10, 2e10, 2e10);
-   }
-   this.restore();
-  },
-  beginInlineImage: function CanvasGraphics_beginInlineImage() {
-   error('Should not call beginInlineImage');
-  },
-  beginImageData: function CanvasGraphics_beginImageData() {
-   error('Should not call beginImageData');
-  },
-  paintFormXObjectBegin: function CanvasGraphics_paintFormXObjectBegin(matrix, bbox) {
-   this.save();
-   this.baseTransformStack.push(this.baseTransform);
-   if (isArray(matrix) && matrix.length === 6) {
-    this.transform.apply(this, matrix);
-   }
-   this.baseTransform = this.ctx.mozCurrentTransform;
-   if (isArray(bbox) && bbox.length === 4) {
-    var width = bbox[2] - bbox[0];
-    var height = bbox[3] - bbox[1];
-    this.ctx.rect(bbox[0], bbox[1], width, height);
-    this.clip();
-    this.endPath();
-   }
-  },
-  paintFormXObjectEnd: function CanvasGraphics_paintFormXObjectEnd() {
-   this.restore();
-   this.baseTransform = this.baseTransformStack.pop();
-  },
-  beginGroup: function CanvasGraphics_beginGroup(group) {
-   this.save();
-   var currentCtx = this.ctx;
-   if (!group.isolated) {
-    info('TODO: Support non-isolated groups.');
-   }
-   if (group.knockout) {
-    warn('Knockout groups not supported.');
-   }
-   var currentTransform = currentCtx.mozCurrentTransform;
-   if (group.matrix) {
-    currentCtx.transform.apply(currentCtx, group.matrix);
-   }
-   assert(group.bbox, 'Bounding box is required.');
-   var bounds = Util.getAxialAlignedBoundingBox(group.bbox, currentCtx.mozCurrentTransform);
-   var canvasBounds = [
-    0,
-    0,
-    currentCtx.canvas.width,
-    currentCtx.canvas.height
-   ];
-   bounds = Util.intersect(bounds, canvasBounds) || [
-    0,
-    0,
-    0,
-    0
-   ];
-   var offsetX = Math.floor(bounds[0]);
-   var offsetY = Math.floor(bounds[1]);
-   var drawnWidth = Math.max(Math.ceil(bounds[2]) - offsetX, 1);
-   var drawnHeight = Math.max(Math.ceil(bounds[3]) - offsetY, 1);
-   var scaleX = 1, scaleY = 1;
-   if (drawnWidth > MAX_GROUP_SIZE) {
-    scaleX = drawnWidth / MAX_GROUP_SIZE;
-    drawnWidth = MAX_GROUP_SIZE;
-   }
-   if (drawnHeight > MAX_GROUP_SIZE) {
-    scaleY = drawnHeight / MAX_GROUP_SIZE;
-    drawnHeight = MAX_GROUP_SIZE;
-   }
-   var cacheId = 'groupAt' + this.groupLevel;
-   if (group.smask) {
-    cacheId += '_smask_' + this.smaskCounter++ % 2;
-   }
-   var scratchCanvas = this.cachedCanvases.getCanvas(cacheId, drawnWidth, drawnHeight, true);
-   var groupCtx = scratchCanvas.context;
-   groupCtx.scale(1 / scaleX, 1 / scaleY);
-   groupCtx.translate(-offsetX, -offsetY);
-   groupCtx.transform.apply(groupCtx, currentTransform);
-   if (group.smask) {
-    this.smaskStack.push({
-     canvas: scratchCanvas.canvas,
-     context: groupCtx,
-     offsetX: offsetX,
-     offsetY: offsetY,
-     scaleX: scaleX,
-     scaleY: scaleY,
-     subtype: group.smask.subtype,
-     backdrop: group.smask.backdrop,
-     transferMap: group.smask.transferMap || null,
-     startTransformInverse: null
-    });
-   } else {
-    currentCtx.setTransform(1, 0, 0, 1, 0, 0);
-    currentCtx.translate(offsetX, offsetY);
-    currentCtx.scale(scaleX, scaleY);
-   }
-   copyCtxState(currentCtx, groupCtx);
-   this.ctx = groupCtx;
-   this.setGState([
-    [
-     'BM',
-     'Normal'
-    ],
-    [
-     'ca',
-     1
-    ],
-    [
-     'CA',
-     1
-    ]
-   ]);
-   this.groupStack.push(currentCtx);
-   this.groupLevel++;
-   this.current.activeSMask = null;
-  },
-  endGroup: function CanvasGraphics_endGroup(group) {
-   this.groupLevel--;
-   var groupCtx = this.ctx;
-   this.ctx = this.groupStack.pop();
-   if (this.ctx.imageSmoothingEnabled !== undefined) {
-    this.ctx.imageSmoothingEnabled = false;
-   } else {
-    this.ctx.mozImageSmoothingEnabled = false;
-   }
-   if (group.smask) {
-    this.tempSMask = this.smaskStack.pop();
-   } else {
-    this.ctx.drawImage(groupCtx.canvas, 0, 0);
-   }
-   this.restore();
-  },
-  beginAnnotations: function CanvasGraphics_beginAnnotations() {
-   this.save();
-   this.current = new CanvasExtraState();
-   if (this.baseTransform) {
-    this.ctx.setTransform.apply(this.ctx, this.baseTransform);
-   }
-  },
-  endAnnotations: function CanvasGraphics_endAnnotations() {
-   this.restore();
-  },
-  beginAnnotation: function CanvasGraphics_beginAnnotation(rect, transform, matrix) {
-   this.save();
-   if (isArray(rect) && rect.length === 4) {
-    var width = rect[2] - rect[0];
-    var height = rect[3] - rect[1];
-    this.ctx.rect(rect[0], rect[1], width, height);
-    this.clip();
-    this.endPath();
-   }
-   this.transform.apply(this, transform);
-   this.transform.apply(this, matrix);
-  },
-  endAnnotation: function CanvasGraphics_endAnnotation() {
-   this.restore();
-  },
-  paintJpegXObject: function CanvasGraphics_paintJpegXObject(objId, w, h) {
-   var domImage = this.objs.get(objId);
-   if (!domImage) {
-    warn('Dependent image isn\'t ready yet');
-    return;
-   }
-   this.save();
-   var ctx = this.ctx;
-   ctx.scale(1 / w, -1 / h);
-   ctx.drawImage(domImage, 0, 0, domImage.width, domImage.height, 0, -h, w, h);
-   if (this.imageLayer) {
-    var currentTransform = ctx.mozCurrentTransformInverse;
-    var position = this.getCanvasPosition(0, 0);
-    this.imageLayer.appendImage({
-     objId: objId,
-     left: position[0],
-     top: position[1],
-     width: w / currentTransform[0],
-     height: h / currentTransform[3]
-    });
-   }
-   this.restore();
-  },
-  paintImageMaskXObject: function CanvasGraphics_paintImageMaskXObject(img) {
-   var ctx = this.ctx;
-   var width = img.width, height = img.height;
-   var fillColor = this.current.fillColor;
-   var isPatternFill = this.current.patternFill;
-   var glyph = this.processingType3;
-   if (COMPILE_TYPE3_GLYPHS && glyph && glyph.compiled === undefined) {
-    if (width <= MAX_SIZE_TO_COMPILE && height <= MAX_SIZE_TO_COMPILE) {
-     glyph.compiled = compileType3Glyph({
-      data: img.data,
-      width: width,
-      height: height
-     });
-    } else {
-     glyph.compiled = null;
+  }
+  function composeSMaskLuminosity(maskData, layerData, transferMap) {
+    var length = maskData.length;
+    for (var i = 3; i < length; i += 4) {
+      var y = maskData[i - 3] * 77 + maskData[i - 2] * 152 + maskData[i - 1] * 28;
+      layerData[i] = transferMap ? layerData[i] * transferMap[y >> 8] >> 8 : layerData[i] * y >> 16;
     }
-   }
-   if (glyph && glyph.compiled) {
-    glyph.compiled(ctx);
-    return;
-   }
-   var maskCanvas = this.cachedCanvases.getCanvas('maskCanvas', width, height);
-   var maskCtx = maskCanvas.context;
-   maskCtx.save();
-   putBinaryImageMask(maskCtx, img);
-   maskCtx.globalCompositeOperation = 'source-in';
-   maskCtx.fillStyle = isPatternFill ? fillColor.getPattern(maskCtx, this) : fillColor;
-   maskCtx.fillRect(0, 0, width, height);
-   maskCtx.restore();
-   this.paintInlineImageXObject(maskCanvas.canvas);
-  },
-  paintImageMaskXObjectRepeat: function CanvasGraphics_paintImageMaskXObjectRepeat(imgData, scaleX, scaleY, positions) {
-   var width = imgData.width;
-   var height = imgData.height;
-   var fillColor = this.current.fillColor;
-   var isPatternFill = this.current.patternFill;
-   var maskCanvas = this.cachedCanvases.getCanvas('maskCanvas', width, height);
-   var maskCtx = maskCanvas.context;
-   maskCtx.save();
-   putBinaryImageMask(maskCtx, imgData);
-   maskCtx.globalCompositeOperation = 'source-in';
-   maskCtx.fillStyle = isPatternFill ? fillColor.getPattern(maskCtx, this) : fillColor;
-   maskCtx.fillRect(0, 0, width, height);
-   maskCtx.restore();
-   var ctx = this.ctx;
-   for (var i = 0, ii = positions.length; i < ii; i += 2) {
-    ctx.save();
-    ctx.transform(scaleX, 0, 0, scaleY, positions[i], positions[i + 1]);
-    ctx.scale(1, -1);
-    ctx.drawImage(maskCanvas.canvas, 0, 0, width, height, 0, -1, 1, 1);
-    ctx.restore();
-   }
-  },
-  paintImageMaskXObjectGroup: function CanvasGraphics_paintImageMaskXObjectGroup(images) {
-   var ctx = this.ctx;
-   var fillColor = this.current.fillColor;
-   var isPatternFill = this.current.patternFill;
-   for (var i = 0, ii = images.length; i < ii; i++) {
-    var image = images[i];
-    var width = image.width, height = image.height;
-    var maskCanvas = this.cachedCanvases.getCanvas('maskCanvas', width, height);
-    var maskCtx = maskCanvas.context;
-    maskCtx.save();
-    putBinaryImageMask(maskCtx, image);
-    maskCtx.globalCompositeOperation = 'source-in';
-    maskCtx.fillStyle = isPatternFill ? fillColor.getPattern(maskCtx, this) : fillColor;
-    maskCtx.fillRect(0, 0, width, height);
-    maskCtx.restore();
-    ctx.save();
-    ctx.transform.apply(ctx, image.transform);
-    ctx.scale(1, -1);
-    ctx.drawImage(maskCanvas.canvas, 0, 0, width, height, 0, -1, 1, 1);
-    ctx.restore();
-   }
-  },
-  paintImageXObject: function CanvasGraphics_paintImageXObject(objId) {
-   var imgData = this.objs.get(objId);
-   if (!imgData) {
-    warn('Dependent image isn\'t ready yet');
-    return;
-   }
-   this.paintInlineImageXObject(imgData);
-  },
-  paintImageXObjectRepeat: function CanvasGraphics_paintImageXObjectRepeat(objId, scaleX, scaleY, positions) {
-   var imgData = this.objs.get(objId);
-   if (!imgData) {
-    warn('Dependent image isn\'t ready yet');
-    return;
-   }
-   var width = imgData.width;
-   var height = imgData.height;
-   var map = [];
-   for (var i = 0, ii = positions.length; i < ii; i += 2) {
-    map.push({
-     transform: [
-      scaleX,
-      0,
-      0,
-      scaleY,
-      positions[i],
-      positions[i + 1]
-     ],
-     x: 0,
-     y: 0,
-     w: width,
-     h: height
-    });
-   }
-   this.paintInlineImageXObjectGroup(imgData, map);
-  },
-  paintInlineImageXObject: function CanvasGraphics_paintInlineImageXObject(imgData) {
-   var width = imgData.width;
-   var height = imgData.height;
-   var ctx = this.ctx;
-   this.save();
-   ctx.scale(1 / width, -1 / height);
-   var currentTransform = ctx.mozCurrentTransformInverse;
-   var a = currentTransform[0], b = currentTransform[1];
-   var widthScale = Math.max(Math.sqrt(a * a + b * b), 1);
-   var c = currentTransform[2], d = currentTransform[3];
-   var heightScale = Math.max(Math.sqrt(c * c + d * d), 1);
-   var imgToPaint, tmpCanvas;
-   if (imgData instanceof HTMLElement || !imgData.data) {
-    imgToPaint = imgData;
-   } else {
-    tmpCanvas = this.cachedCanvases.getCanvas('inlineImage', width, height);
-    var tmpCtx = tmpCanvas.context;
-    putBinaryImageData(tmpCtx, imgData);
-    imgToPaint = tmpCanvas.canvas;
-   }
-   var paintWidth = width, paintHeight = height;
-   var tmpCanvasId = 'prescale1';
-   while (widthScale > 2 && paintWidth > 1 || heightScale > 2 && paintHeight > 1) {
-    var newWidth = paintWidth, newHeight = paintHeight;
-    if (widthScale > 2 && paintWidth > 1) {
-     newWidth = Math.ceil(paintWidth / 2);
-     widthScale /= paintWidth / newWidth;
+  }
+  function genericComposeSMask(maskCtx, layerCtx, width, height, subtype, backdrop, transferMap) {
+    var hasBackdrop = !!backdrop;
+    var r0 = hasBackdrop ? backdrop[0] : 0;
+    var g0 = hasBackdrop ? backdrop[1] : 0;
+    var b0 = hasBackdrop ? backdrop[2] : 0;
+    var composeFn;
+    if (subtype === 'Luminosity') {
+      composeFn = composeSMaskLuminosity;
+    } else {
+      composeFn = composeSMaskAlpha;
     }
-    if (heightScale > 2 && paintHeight > 1) {
-     newHeight = Math.ceil(paintHeight / 2);
-     heightScale /= paintHeight / newHeight;
+    var PIXELS_TO_PROCESS = 1048576;
+    var chunkSize = Math.min(height, Math.ceil(PIXELS_TO_PROCESS / width));
+    for (var row = 0; row < height; row += chunkSize) {
+      var chunkHeight = Math.min(chunkSize, height - row);
+      var maskData = maskCtx.getImageData(0, row, width, chunkHeight);
+      var layerData = layerCtx.getImageData(0, row, width, chunkHeight);
+      if (hasBackdrop) {
+        composeSMaskBackdrop(maskData.data, r0, g0, b0);
+      }
+      composeFn(maskData.data, layerData.data, transferMap);
+      maskCtx.putImageData(layerData, 0, row);
     }
-    tmpCanvas = this.cachedCanvases.getCanvas(tmpCanvasId, newWidth, newHeight);
-    tmpCtx = tmpCanvas.context;
-    tmpCtx.clearRect(0, 0, newWidth, newHeight);
-    tmpCtx.drawImage(imgToPaint, 0, 0, paintWidth, paintHeight, 0, 0, newWidth, newHeight);
-    imgToPaint = tmpCanvas.canvas;
-    paintWidth = newWidth;
-    paintHeight = newHeight;
-    tmpCanvasId = tmpCanvasId === 'prescale1' ? 'prescale2' : 'prescale1';
-   }
-   ctx.drawImage(imgToPaint, 0, 0, paintWidth, paintHeight, 0, -height, width, height);
-   if (this.imageLayer) {
-    var position = this.getCanvasPosition(0, -height);
-    this.imageLayer.appendImage({
-     imgData: imgData,
-     left: position[0],
-     top: position[1],
-     width: width / currentTransform[0],
-     height: height / currentTransform[3]
-    });
-   }
-   this.restore();
-  },
-  paintInlineImageXObjectGroup: function CanvasGraphics_paintInlineImageXObjectGroup(imgData, map) {
-   var ctx = this.ctx;
-   var w = imgData.width;
-   var h = imgData.height;
-   var tmpCanvas = this.cachedCanvases.getCanvas('inlineImage', w, h);
-   var tmpCtx = tmpCanvas.context;
-   putBinaryImageData(tmpCtx, imgData);
-   for (var i = 0, ii = map.length; i < ii; i++) {
-    var entry = map[i];
-    ctx.save();
-    ctx.transform.apply(ctx, entry.transform);
-    ctx.scale(1, -1);
-    ctx.drawImage(tmpCanvas.canvas, entry.x, entry.y, entry.w, entry.h, 0, -1, 1, 1);
-    if (this.imageLayer) {
-     var position = this.getCanvasPosition(entry.x, entry.y);
-     this.imageLayer.appendImage({
-      imgData: imgData,
-      left: position[0],
-      top: position[1],
-      width: w,
-      height: h
-     });
+  }
+  function composeSMask(ctx, smask, layerCtx) {
+    var mask = smask.canvas;
+    var maskCtx = smask.context;
+    ctx.setTransform(smask.scaleX, 0, 0, smask.scaleY, smask.offsetX, smask.offsetY);
+    var backdrop = smask.backdrop || null;
+    if (!smask.transferMap && WebGLUtils.isEnabled) {
+      var composed = WebGLUtils.composeSMask(layerCtx.canvas, mask, {
+        subtype: smask.subtype,
+        backdrop: backdrop
+      });
+      ctx.setTransform(1, 0, 0, 1, 0, 0);
+      ctx.drawImage(composed, smask.offsetX, smask.offsetY);
+      return;
     }
-    ctx.restore();
-   }
-  },
-  paintSolidColorImageMask: function CanvasGraphics_paintSolidColorImageMask() {
-   this.ctx.fillRect(0, 0, 1, 1);
-  },
-  paintXObject: function CanvasGraphics_paintXObject() {
-   warn('Unsupported \'paintXObject\' command.');
-  },
-  markPoint: function CanvasGraphics_markPoint(tag) {
-  },
-  markPointProps: function CanvasGraphics_markPointProps(tag, properties) {
-  },
-  beginMarkedContent: function CanvasGraphics_beginMarkedContent(tag) {
-  },
-  beginMarkedContentProps: function CanvasGraphics_beginMarkedContentProps(tag, properties) {
-  },
-  endMarkedContent: function CanvasGraphics_endMarkedContent() {
-  },
-  beginCompat: function CanvasGraphics_beginCompat() {
-  },
-  endCompat: function CanvasGraphics_endCompat() {
-  },
-  consumePath: function CanvasGraphics_consumePath() {
-   var ctx = this.ctx;
-   if (this.pendingClip) {
-    if (this.pendingClip === EO_CLIP) {
-     ctx.clip('evenodd');
-    } else {
-     ctx.clip();
+    genericComposeSMask(maskCtx, layerCtx, mask.width, mask.height, smask.subtype, backdrop, smask.transferMap);
+    ctx.drawImage(mask, 0, 0);
+  }
+  var LINE_CAP_STYLES = ['butt', 'round', 'square'];
+  var LINE_JOIN_STYLES = ['miter', 'round', 'bevel'];
+  var NORMAL_CLIP = {};
+  var EO_CLIP = {};
+  CanvasGraphics.prototype = {
+    beginDrawing: function CanvasGraphics_beginDrawing(transform, viewport, transparency) {
+      var width = this.ctx.canvas.width;
+      var height = this.ctx.canvas.height;
+      this.ctx.save();
+      this.ctx.fillStyle = 'rgb(255, 255, 255)';
+      this.ctx.fillRect(0, 0, width, height);
+      this.ctx.restore();
+      if (transparency) {
+        var transparentCanvas = this.cachedCanvases.getCanvas('transparent', width, height, true);
+        this.compositeCtx = this.ctx;
+        this.transparentCanvas = transparentCanvas.canvas;
+        this.ctx = transparentCanvas.context;
+        this.ctx.save();
+        this.ctx.transform.apply(this.ctx, this.compositeCtx.mozCurrentTransform);
+      }
+      this.ctx.save();
+      if (transform) {
+        this.ctx.transform.apply(this.ctx, transform);
+      }
+      this.ctx.transform.apply(this.ctx, viewport.transform);
+      this.baseTransform = this.ctx.mozCurrentTransform.slice();
+      if (this.imageLayer) {
+        this.imageLayer.beginLayout();
+      }
+    },
+    executeOperatorList: function CanvasGraphics_executeOperatorList(operatorList, executionStartIdx, continueCallback, stepper) {
+      var argsArray = operatorList.argsArray;
+      var fnArray = operatorList.fnArray;
+      var i = executionStartIdx || 0;
+      var argsArrayLen = argsArray.length;
+      if (argsArrayLen === i) {
+        return i;
+      }
+      var chunkOperations = argsArrayLen - i > EXECUTION_STEPS && typeof continueCallback === 'function';
+      var endTime = chunkOperations ? Date.now() + EXECUTION_TIME : 0;
+      var steps = 0;
+      var commonObjs = this.commonObjs;
+      var objs = this.objs;
+      var fnId;
+      while (true) {
+        if (stepper !== undefined && i === stepper.nextBreakPoint) {
+          stepper.breakIt(i, continueCallback);
+          return i;
+        }
+        fnId = fnArray[i];
+        if (fnId !== OPS.dependency) {
+          this[fnId].apply(this, argsArray[i]);
+        } else {
+          var deps = argsArray[i];
+          for (var n = 0, nn = deps.length; n < nn; n++) {
+            var depObjId = deps[n];
+            var common = depObjId[0] === 'g' && depObjId[1] === '_';
+            var objsPool = common ? commonObjs : objs;
+            if (!objsPool.isResolved(depObjId)) {
+              objsPool.get(depObjId, continueCallback);
+              return i;
+            }
+          }
+        }
+        i++;
+        if (i === argsArrayLen) {
+          return i;
+        }
+        if (chunkOperations && ++steps > EXECUTION_STEPS) {
+          if (Date.now() > endTime) {
+            continueCallback();
+            return i;
+          }
+          steps = 0;
+        }
+      }
+    },
+    endDrawing: function CanvasGraphics_endDrawing() {
+      if (this.current.activeSMask !== null) {
+        this.endSMaskGroup();
+      }
+      this.ctx.restore();
+      if (this.transparentCanvas) {
+        this.ctx = this.compositeCtx;
+        this.ctx.save();
+        this.ctx.setTransform(1, 0, 0, 1, 0, 0);
+        this.ctx.drawImage(this.transparentCanvas, 0, 0);
+        this.ctx.restore();
+        this.transparentCanvas = null;
+      }
+      this.cachedCanvases.clear();
+      WebGLUtils.clear();
+      if (this.imageLayer) {
+        this.imageLayer.endLayout();
+      }
+    },
+    setLineWidth: function CanvasGraphics_setLineWidth(width) {
+      this.current.lineWidth = width;
+      this.ctx.lineWidth = width;
+    },
+    setLineCap: function CanvasGraphics_setLineCap(style) {
+      this.ctx.lineCap = LINE_CAP_STYLES[style];
+    },
+    setLineJoin: function CanvasGraphics_setLineJoin(style) {
+      this.ctx.lineJoin = LINE_JOIN_STYLES[style];
+    },
+    setMiterLimit: function CanvasGraphics_setMiterLimit(limit) {
+      this.ctx.miterLimit = limit;
+    },
+    setDash: function CanvasGraphics_setDash(dashArray, dashPhase) {
+      var ctx = this.ctx;
+      if (ctx.setLineDash !== undefined) {
+        ctx.setLineDash(dashArray);
+        ctx.lineDashOffset = dashPhase;
+      }
+    },
+    setRenderingIntent: function CanvasGraphics_setRenderingIntent(intent) {},
+    setFlatness: function CanvasGraphics_setFlatness(flatness) {},
+    setGState: function CanvasGraphics_setGState(states) {
+      for (var i = 0, ii = states.length; i < ii; i++) {
+        var state = states[i];
+        var key = state[0];
+        var value = state[1];
+        switch (key) {
+          case 'LW':
+            this.setLineWidth(value);
+            break;
+          case 'LC':
+            this.setLineCap(value);
+            break;
+          case 'LJ':
+            this.setLineJoin(value);
+            break;
+          case 'ML':
+            this.setMiterLimit(value);
+            break;
+          case 'D':
+            this.setDash(value[0], value[1]);
+            break;
+          case 'RI':
+            this.setRenderingIntent(value);
+            break;
+          case 'FL':
+            this.setFlatness(value);
+            break;
+          case 'Font':
+            this.setFont(value[0], value[1]);
+            break;
+          case 'CA':
+            this.current.strokeAlpha = state[1];
+            break;
+          case 'ca':
+            this.current.fillAlpha = state[1];
+            this.ctx.globalAlpha = state[1];
+            break;
+          case 'BM':
+            if (value && value.name && value.name !== 'Normal') {
+              var mode = value.name.replace(/([A-Z])/g, function (c) {
+                return '-' + c.toLowerCase();
+              }).substring(1);
+              this.ctx.globalCompositeOperation = mode;
+              if (this.ctx.globalCompositeOperation !== mode) {
+                warn('globalCompositeOperation "' + mode + '" is not supported');
+              }
+            } else {
+              this.ctx.globalCompositeOperation = 'source-over';
+            }
+            break;
+          case 'SMask':
+            if (this.current.activeSMask) {
+              if (this.stateStack.length > 0 && this.stateStack[this.stateStack.length - 1].activeSMask === this.current.activeSMask) {
+                this.suspendSMaskGroup();
+              } else {
+                this.endSMaskGroup();
+              }
+            }
+            this.current.activeSMask = value ? this.tempSMask : null;
+            if (this.current.activeSMask) {
+              this.beginSMaskGroup();
+            }
+            this.tempSMask = null;
+            break;
+        }
+      }
+    },
+    beginSMaskGroup: function CanvasGraphics_beginSMaskGroup() {
+      var activeSMask = this.current.activeSMask;
+      var drawnWidth = activeSMask.canvas.width;
+      var drawnHeight = activeSMask.canvas.height;
+      var cacheId = 'smaskGroupAt' + this.groupLevel;
+      var scratchCanvas = this.cachedCanvases.getCanvas(cacheId, drawnWidth, drawnHeight, true);
+      var currentCtx = this.ctx;
+      var currentTransform = currentCtx.mozCurrentTransform;
+      this.ctx.save();
+      var groupCtx = scratchCanvas.context;
+      groupCtx.scale(1 / activeSMask.scaleX, 1 / activeSMask.scaleY);
+      groupCtx.translate(-activeSMask.offsetX, -activeSMask.offsetY);
+      groupCtx.transform.apply(groupCtx, currentTransform);
+      activeSMask.startTransformInverse = groupCtx.mozCurrentTransformInverse;
+      copyCtxState(currentCtx, groupCtx);
+      this.ctx = groupCtx;
+      this.setGState([['BM', 'Normal'], ['ca', 1], ['CA', 1]]);
+      this.groupStack.push(currentCtx);
+      this.groupLevel++;
+    },
+    suspendSMaskGroup: function CanvasGraphics_endSMaskGroup() {
+      var groupCtx = this.ctx;
+      this.groupLevel--;
+      this.ctx = this.groupStack.pop();
+      composeSMask(this.ctx, this.current.activeSMask, groupCtx);
+      this.ctx.restore();
+      this.ctx.save();
+      copyCtxState(groupCtx, this.ctx);
+      this.current.resumeSMaskCtx = groupCtx;
+      var deltaTransform = Util.transform(this.current.activeSMask.startTransformInverse, groupCtx.mozCurrentTransform);
+      this.ctx.transform.apply(this.ctx, deltaTransform);
+      groupCtx.save();
+      groupCtx.setTransform(1, 0, 0, 1, 0, 0);
+      groupCtx.clearRect(0, 0, groupCtx.canvas.width, groupCtx.canvas.height);
+      groupCtx.restore();
+    },
+    resumeSMaskGroup: function CanvasGraphics_endSMaskGroup() {
+      var groupCtx = this.current.resumeSMaskCtx;
+      var currentCtx = this.ctx;
+      this.ctx = groupCtx;
+      this.groupStack.push(currentCtx);
+      this.groupLevel++;
+    },
+    endSMaskGroup: function CanvasGraphics_endSMaskGroup() {
+      var groupCtx = this.ctx;
+      this.groupLevel--;
+      this.ctx = this.groupStack.pop();
+      composeSMask(this.ctx, this.current.activeSMask, groupCtx);
+      this.ctx.restore();
+      copyCtxState(groupCtx, this.ctx);
+      var deltaTransform = Util.transform(this.current.activeSMask.startTransformInverse, groupCtx.mozCurrentTransform);
+      this.ctx.transform.apply(this.ctx, deltaTransform);
+    },
+    save: function CanvasGraphics_save() {
+      this.ctx.save();
+      var old = this.current;
+      this.stateStack.push(old);
+      this.current = old.clone();
+      this.current.resumeSMaskCtx = null;
+    },
+    restore: function CanvasGraphics_restore() {
+      if (this.current.resumeSMaskCtx) {
+        this.resumeSMaskGroup();
+      }
+      if (this.current.activeSMask !== null && (this.stateStack.length === 0 || this.stateStack[this.stateStack.length - 1].activeSMask !== this.current.activeSMask)) {
+        this.endSMaskGroup();
+      }
+      if (this.stateStack.length !== 0) {
+        this.current = this.stateStack.pop();
+        this.ctx.restore();
+        this.pendingClip = null;
+        this.cachedGetSinglePixelWidth = null;
+      }
+    },
+    transform: function CanvasGraphics_transform(a, b, c, d, e, f) {
+      this.ctx.transform(a, b, c, d, e, f);
+      this.cachedGetSinglePixelWidth = null;
+    },
+    constructPath: function CanvasGraphics_constructPath(ops, args) {
+      var ctx = this.ctx;
+      var current = this.current;
+      var x = current.x,
+          y = current.y;
+      for (var i = 0, j = 0, ii = ops.length; i < ii; i++) {
+        switch (ops[i] | 0) {
+          case OPS.rectangle:
+            x = args[j++];
+            y = args[j++];
+            var width = args[j++];
+            var height = args[j++];
+            if (width === 0) {
+              width = this.getSinglePixelWidth();
+            }
+            if (height === 0) {
+              height = this.getSinglePixelWidth();
+            }
+            var xw = x + width;
+            var yh = y + height;
+            this.ctx.moveTo(x, y);
+            this.ctx.lineTo(xw, y);
+            this.ctx.lineTo(xw, yh);
+            this.ctx.lineTo(x, yh);
+            this.ctx.lineTo(x, y);
+            this.ctx.closePath();
+            break;
+          case OPS.moveTo:
+            x = args[j++];
+            y = args[j++];
+            ctx.moveTo(x, y);
+            break;
+          case OPS.lineTo:
+            x = args[j++];
+            y = args[j++];
+            ctx.lineTo(x, y);
+            break;
+          case OPS.curveTo:
+            x = args[j + 4];
+            y = args[j + 5];
+            ctx.bezierCurveTo(args[j], args[j + 1], args[j + 2], args[j + 3], x, y);
+            j += 6;
+            break;
+          case OPS.curveTo2:
+            ctx.bezierCurveTo(x, y, args[j], args[j + 1], args[j + 2], args[j + 3]);
+            x = args[j + 2];
+            y = args[j + 3];
+            j += 4;
+            break;
+          case OPS.curveTo3:
+            x = args[j + 2];
+            y = args[j + 3];
+            ctx.bezierCurveTo(args[j], args[j + 1], x, y, x, y);
+            j += 4;
+            break;
+          case OPS.closePath:
+            ctx.closePath();
+            break;
+        }
+      }
+      current.setCurrentPoint(x, y);
+    },
+    closePath: function CanvasGraphics_closePath() {
+      this.ctx.closePath();
+    },
+    stroke: function CanvasGraphics_stroke(consumePath) {
+      consumePath = typeof consumePath !== 'undefined' ? consumePath : true;
+      var ctx = this.ctx;
+      var strokeColor = this.current.strokeColor;
+      ctx.lineWidth = Math.max(this.getSinglePixelWidth() * MIN_WIDTH_FACTOR, this.current.lineWidth);
+      ctx.globalAlpha = this.current.strokeAlpha;
+      if (strokeColor && strokeColor.hasOwnProperty('type') && strokeColor.type === 'Pattern') {
+        ctx.save();
+        ctx.strokeStyle = strokeColor.getPattern(ctx, this);
+        ctx.stroke();
+        ctx.restore();
+      } else {
+        ctx.stroke();
+      }
+      if (consumePath) {
+        this.consumePath();
+      }
+      ctx.globalAlpha = this.current.fillAlpha;
+    },
+    closeStroke: function CanvasGraphics_closeStroke() {
+      this.closePath();
+      this.stroke();
+    },
+    fill: function CanvasGraphics_fill(consumePath) {
+      consumePath = typeof consumePath !== 'undefined' ? consumePath : true;
+      var ctx = this.ctx;
+      var fillColor = this.current.fillColor;
+      var isPatternFill = this.current.patternFill;
+      var needRestore = false;
+      if (isPatternFill) {
+        ctx.save();
+        if (this.baseTransform) {
+          ctx.setTransform.apply(ctx, this.baseTransform);
+        }
+        ctx.fillStyle = fillColor.getPattern(ctx, this);
+        needRestore = true;
+      }
+      if (this.pendingEOFill) {
+        ctx.fill('evenodd');
+        this.pendingEOFill = false;
+      } else {
+        ctx.fill();
+      }
+      if (needRestore) {
+        ctx.restore();
+      }
+      if (consumePath) {
+        this.consumePath();
+      }
+    },
+    eoFill: function CanvasGraphics_eoFill() {
+      this.pendingEOFill = true;
+      this.fill();
+    },
+    fillStroke: function CanvasGraphics_fillStroke() {
+      this.fill(false);
+      this.stroke(false);
+      this.consumePath();
+    },
+    eoFillStroke: function CanvasGraphics_eoFillStroke() {
+      this.pendingEOFill = true;
+      this.fillStroke();
+    },
+    closeFillStroke: function CanvasGraphics_closeFillStroke() {
+      this.closePath();
+      this.fillStroke();
+    },
+    closeEOFillStroke: function CanvasGraphics_closeEOFillStroke() {
+      this.pendingEOFill = true;
+      this.closePath();
+      this.fillStroke();
+    },
+    endPath: function CanvasGraphics_endPath() {
+      this.consumePath();
+    },
+    clip: function CanvasGraphics_clip() {
+      this.pendingClip = NORMAL_CLIP;
+    },
+    eoClip: function CanvasGraphics_eoClip() {
+      this.pendingClip = EO_CLIP;
+    },
+    beginText: function CanvasGraphics_beginText() {
+      this.current.textMatrix = IDENTITY_MATRIX;
+      this.current.textMatrixScale = 1;
+      this.current.x = this.current.lineX = 0;
+      this.current.y = this.current.lineY = 0;
+    },
+    endText: function CanvasGraphics_endText() {
+      var paths = this.pendingTextPaths;
+      var ctx = this.ctx;
+      if (paths === undefined) {
+        ctx.beginPath();
+        return;
+      }
+      ctx.save();
+      ctx.beginPath();
+      for (var i = 0; i < paths.length; i++) {
+        var path = paths[i];
+        ctx.setTransform.apply(ctx, path.transform);
+        ctx.translate(path.x, path.y);
+        path.addToPath(ctx, path.fontSize);
+      }
+      ctx.restore();
+      ctx.clip();
+      ctx.beginPath();
+      delete this.pendingTextPaths;
+    },
+    setCharSpacing: function CanvasGraphics_setCharSpacing(spacing) {
+      this.current.charSpacing = spacing;
+    },
+    setWordSpacing: function CanvasGraphics_setWordSpacing(spacing) {
+      this.current.wordSpacing = spacing;
+    },
+    setHScale: function CanvasGraphics_setHScale(scale) {
+      this.current.textHScale = scale / 100;
+    },
+    setLeading: function CanvasGraphics_setLeading(leading) {
+      this.current.leading = -leading;
+    },
+    setFont: function CanvasGraphics_setFont(fontRefName, size) {
+      var fontObj = this.commonObjs.get(fontRefName);
+      var current = this.current;
+      if (!fontObj) {
+        error('Can\'t find font for ' + fontRefName);
+      }
+      current.fontMatrix = fontObj.fontMatrix ? fontObj.fontMatrix : FONT_IDENTITY_MATRIX;
+      if (current.fontMatrix[0] === 0 || current.fontMatrix[3] === 0) {
+        warn('Invalid font matrix for font ' + fontRefName);
+      }
+      if (size < 0) {
+        size = -size;
+        current.fontDirection = -1;
+      } else {
+        current.fontDirection = 1;
+      }
+      this.current.font = fontObj;
+      this.current.fontSize = size;
+      if (fontObj.isType3Font) {
+        return;
+      }
+      var name = fontObj.loadedName || 'sans-serif';
+      var bold = fontObj.black ? '900' : fontObj.bold ? 'bold' : 'normal';
+      var italic = fontObj.italic ? 'italic' : 'normal';
+      var typeface = '"' + name + '", ' + fontObj.fallbackName;
+      var browserFontSize = size < MIN_FONT_SIZE ? MIN_FONT_SIZE : size > MAX_FONT_SIZE ? MAX_FONT_SIZE : size;
+      this.current.fontSizeScale = size / browserFontSize;
+      var rule = italic + ' ' + bold + ' ' + browserFontSize + 'px ' + typeface;
+      this.ctx.font = rule;
+    },
+    setTextRenderingMode: function CanvasGraphics_setTextRenderingMode(mode) {
+      this.current.textRenderingMode = mode;
+    },
+    setTextRise: function CanvasGraphics_setTextRise(rise) {
+      this.current.textRise = rise;
+    },
+    moveText: function CanvasGraphics_moveText(x, y) {
+      this.current.x = this.current.lineX += x;
+      this.current.y = this.current.lineY += y;
+    },
+    setLeadingMoveText: function CanvasGraphics_setLeadingMoveText(x, y) {
+      this.setLeading(-y);
+      this.moveText(x, y);
+    },
+    setTextMatrix: function CanvasGraphics_setTextMatrix(a, b, c, d, e, f) {
+      this.current.textMatrix = [a, b, c, d, e, f];
+      this.current.textMatrixScale = Math.sqrt(a * a + b * b);
+      this.current.x = this.current.lineX = 0;
+      this.current.y = this.current.lineY = 0;
+    },
+    nextLine: function CanvasGraphics_nextLine() {
+      this.moveText(0, this.current.leading);
+    },
+    paintChar: function CanvasGraphics_paintChar(character, x, y) {
+      var ctx = this.ctx;
+      var current = this.current;
+      var font = current.font;
+      var textRenderingMode = current.textRenderingMode;
+      var fontSize = current.fontSize / current.fontSizeScale;
+      var fillStrokeMode = textRenderingMode & TextRenderingMode.FILL_STROKE_MASK;
+      var isAddToPathSet = !!(textRenderingMode & TextRenderingMode.ADD_TO_PATH_FLAG);
+      var addToPath;
+      if (font.disableFontFace || isAddToPathSet) {
+        addToPath = font.getPathGenerator(this.commonObjs, character);
+      }
+      if (font.disableFontFace) {
+        ctx.save();
+        ctx.translate(x, y);
+        ctx.beginPath();
+        addToPath(ctx, fontSize);
+        if (fillStrokeMode === TextRenderingMode.FILL || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
+          ctx.fill();
+        }
+        if (fillStrokeMode === TextRenderingMode.STROKE || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
+          ctx.stroke();
+        }
+        ctx.restore();
+      } else {
+        if (fillStrokeMode === TextRenderingMode.FILL || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
+          ctx.fillText(character, x, y);
+        }
+        if (fillStrokeMode === TextRenderingMode.STROKE || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
+          ctx.strokeText(character, x, y);
+        }
+      }
+      if (isAddToPathSet) {
+        var paths = this.pendingTextPaths || (this.pendingTextPaths = []);
+        paths.push({
+          transform: ctx.mozCurrentTransform,
+          x: x,
+          y: y,
+          fontSize: fontSize,
+          addToPath: addToPath
+        });
+      }
+    },
+    get isFontSubpixelAAEnabled() {
+      var ctx = this.canvasFactory.create(10, 10).context;
+      ctx.scale(1.5, 1);
+      ctx.fillText('I', 0, 10);
+      var data = ctx.getImageData(0, 0, 10, 10).data;
+      var enabled = false;
+      for (var i = 3; i < data.length; i += 4) {
+        if (data[i] > 0 && data[i] < 255) {
+          enabled = true;
+          break;
+        }
+      }
+      return shadow(this, 'isFontSubpixelAAEnabled', enabled);
+    },
+    showText: function CanvasGraphics_showText(glyphs) {
+      var current = this.current;
+      var font = current.font;
+      if (font.isType3Font) {
+        return this.showType3Text(glyphs);
+      }
+      var fontSize = current.fontSize;
+      if (fontSize === 0) {
+        return;
+      }
+      var ctx = this.ctx;
+      var fontSizeScale = current.fontSizeScale;
+      var charSpacing = current.charSpacing;
+      var wordSpacing = current.wordSpacing;
+      var fontDirection = current.fontDirection;
+      var textHScale = current.textHScale * fontDirection;
+      var glyphsLength = glyphs.length;
+      var vertical = font.vertical;
+      var spacingDir = vertical ? 1 : -1;
+      var defaultVMetrics = font.defaultVMetrics;
+      var widthAdvanceScale = fontSize * current.fontMatrix[0];
+      var simpleFillText = current.textRenderingMode === TextRenderingMode.FILL && !font.disableFontFace;
+      ctx.save();
+      ctx.transform.apply(ctx, current.textMatrix);
+      ctx.translate(current.x, current.y + current.textRise);
+      if (current.patternFill) {
+        ctx.fillStyle = current.fillColor.getPattern(ctx, this);
+      }
+      if (fontDirection > 0) {
+        ctx.scale(textHScale, -1);
+      } else {
+        ctx.scale(textHScale, 1);
+      }
+      var lineWidth = current.lineWidth;
+      var scale = current.textMatrixScale;
+      if (scale === 0 || lineWidth === 0) {
+        var fillStrokeMode = current.textRenderingMode & TextRenderingMode.FILL_STROKE_MASK;
+        if (fillStrokeMode === TextRenderingMode.STROKE || fillStrokeMode === TextRenderingMode.FILL_STROKE) {
+          this.cachedGetSinglePixelWidth = null;
+          lineWidth = this.getSinglePixelWidth() * MIN_WIDTH_FACTOR;
+        }
+      } else {
+        lineWidth /= scale;
+      }
+      if (fontSizeScale !== 1.0) {
+        ctx.scale(fontSizeScale, fontSizeScale);
+        lineWidth /= fontSizeScale;
+      }
+      ctx.lineWidth = lineWidth;
+      var x = 0,
+          i;
+      for (i = 0; i < glyphsLength; ++i) {
+        var glyph = glyphs[i];
+        if (isNum(glyph)) {
+          x += spacingDir * glyph * fontSize / 1000;
+          continue;
+        }
+        var restoreNeeded = false;
+        var spacing = (glyph.isSpace ? wordSpacing : 0) + charSpacing;
+        var character = glyph.fontChar;
+        var accent = glyph.accent;
+        var scaledX, scaledY, scaledAccentX, scaledAccentY;
+        var width = glyph.width;
+        if (vertical) {
+          var vmetric, vx, vy;
+          vmetric = glyph.vmetric || defaultVMetrics;
+          vx = glyph.vmetric ? vmetric[1] : width * 0.5;
+          vx = -vx * widthAdvanceScale;
+          vy = vmetric[2] * widthAdvanceScale;
+          width = vmetric ? -vmetric[0] : width;
+          scaledX = vx / fontSizeScale;
+          scaledY = (x + vy) / fontSizeScale;
+        } else {
+          scaledX = x / fontSizeScale;
+          scaledY = 0;
+        }
+        if (font.remeasure && width > 0) {
+          var measuredWidth = ctx.measureText(character).width * 1000 / fontSize * fontSizeScale;
+          if (width < measuredWidth && this.isFontSubpixelAAEnabled) {
+            var characterScaleX = width / measuredWidth;
+            restoreNeeded = true;
+            ctx.save();
+            ctx.scale(characterScaleX, 1);
+            scaledX /= characterScaleX;
+          } else if (width !== measuredWidth) {
+            scaledX += (width - measuredWidth) / 2000 * fontSize / fontSizeScale;
+          }
+        }
+        if (glyph.isInFont || font.missingFile) {
+          if (simpleFillText && !accent) {
+            ctx.fillText(character, scaledX, scaledY);
+          } else {
+            this.paintChar(character, scaledX, scaledY);
+            if (accent) {
+              scaledAccentX = scaledX + accent.offset.x / fontSizeScale;
+              scaledAccentY = scaledY - accent.offset.y / fontSizeScale;
+              this.paintChar(accent.fontChar, scaledAccentX, scaledAccentY);
+            }
+          }
+        }
+        var charWidth = width * widthAdvanceScale + spacing * fontDirection;
+        x += charWidth;
+        if (restoreNeeded) {
+          ctx.restore();
+        }
+      }
+      if (vertical) {
+        current.y -= x * textHScale;
+      } else {
+        current.x += x * textHScale;
+      }
+      ctx.restore();
+    },
+    showType3Text: function CanvasGraphics_showType3Text(glyphs) {
+      var ctx = this.ctx;
+      var current = this.current;
+      var font = current.font;
+      var fontSize = current.fontSize;
+      var fontDirection = current.fontDirection;
+      var spacingDir = font.vertical ? 1 : -1;
+      var charSpacing = current.charSpacing;
+      var wordSpacing = current.wordSpacing;
+      var textHScale = current.textHScale * fontDirection;
+      var fontMatrix = current.fontMatrix || FONT_IDENTITY_MATRIX;
+      var glyphsLength = glyphs.length;
+      var isTextInvisible = current.textRenderingMode === TextRenderingMode.INVISIBLE;
+      var i, glyph, width, spacingLength;
+      if (isTextInvisible || fontSize === 0) {
+        return;
+      }
+      this.cachedGetSinglePixelWidth = null;
+      ctx.save();
+      ctx.transform.apply(ctx, current.textMatrix);
+      ctx.translate(current.x, current.y);
+      ctx.scale(textHScale, fontDirection);
+      for (i = 0; i < glyphsLength; ++i) {
+        glyph = glyphs[i];
+        if (isNum(glyph)) {
+          spacingLength = spacingDir * glyph * fontSize / 1000;
+          this.ctx.translate(spacingLength, 0);
+          current.x += spacingLength * textHScale;
+          continue;
+        }
+        var spacing = (glyph.isSpace ? wordSpacing : 0) + charSpacing;
+        var operatorList = font.charProcOperatorList[glyph.operatorListId];
+        if (!operatorList) {
+          warn('Type3 character \"' + glyph.operatorListId + '\" is not available');
+          continue;
+        }
+        this.processingType3 = glyph;
+        this.save();
+        ctx.scale(fontSize, fontSize);
+        ctx.transform.apply(ctx, fontMatrix);
+        this.executeOperatorList(operatorList);
+        this.restore();
+        var transformed = Util.applyTransform([glyph.width, 0], fontMatrix);
+        width = transformed[0] * fontSize + spacing;
+        ctx.translate(width, 0);
+        current.x += width * textHScale;
+      }
+      ctx.restore();
+      this.processingType3 = null;
+    },
+    setCharWidth: function CanvasGraphics_setCharWidth(xWidth, yWidth) {},
+    setCharWidthAndBounds: function CanvasGraphics_setCharWidthAndBounds(xWidth, yWidth, llx, lly, urx, ury) {
+      this.ctx.rect(llx, lly, urx - llx, ury - lly);
+      this.clip();
+      this.endPath();
+    },
+    getColorN_Pattern: function CanvasGraphics_getColorN_Pattern(IR) {
+      var pattern;
+      if (IR[0] === 'TilingPattern') {
+        var color = IR[1];
+        var baseTransform = this.baseTransform || this.ctx.mozCurrentTransform.slice();
+        var self = this;
+        var canvasGraphicsFactory = {
+          createCanvasGraphics: function (ctx) {
+            return new CanvasGraphics(ctx, self.commonObjs, self.objs, self.canvasFactory);
+          }
+        };
+        pattern = new TilingPattern(IR, color, this.ctx, canvasGraphicsFactory, baseTransform);
+      } else {
+        pattern = getShadingPatternFromIR(IR);
+      }
+      return pattern;
+    },
+    setStrokeColorN: function CanvasGraphics_setStrokeColorN() {
+      this.current.strokeColor = this.getColorN_Pattern(arguments);
+    },
+    setFillColorN: function CanvasGraphics_setFillColorN() {
+      this.current.fillColor = this.getColorN_Pattern(arguments);
+      this.current.patternFill = true;
+    },
+    setStrokeRGBColor: function CanvasGraphics_setStrokeRGBColor(r, g, b) {
+      var color = Util.makeCssRgb(r, g, b);
+      this.ctx.strokeStyle = color;
+      this.current.strokeColor = color;
+    },
+    setFillRGBColor: function CanvasGraphics_setFillRGBColor(r, g, b) {
+      var color = Util.makeCssRgb(r, g, b);
+      this.ctx.fillStyle = color;
+      this.current.fillColor = color;
+      this.current.patternFill = false;
+    },
+    shadingFill: function CanvasGraphics_shadingFill(patternIR) {
+      var ctx = this.ctx;
+      this.save();
+      var pattern = getShadingPatternFromIR(patternIR);
+      ctx.fillStyle = pattern.getPattern(ctx, this, true);
+      var inv = ctx.mozCurrentTransformInverse;
+      if (inv) {
+        var canvas = ctx.canvas;
+        var width = canvas.width;
+        var height = canvas.height;
+        var bl = Util.applyTransform([0, 0], inv);
+        var br = Util.applyTransform([0, height], inv);
+        var ul = Util.applyTransform([width, 0], inv);
+        var ur = Util.applyTransform([width, height], inv);
+        var x0 = Math.min(bl[0], br[0], ul[0], ur[0]);
+        var y0 = Math.min(bl[1], br[1], ul[1], ur[1]);
+        var x1 = Math.max(bl[0], br[0], ul[0], ur[0]);
+        var y1 = Math.max(bl[1], br[1], ul[1], ur[1]);
+        this.ctx.fillRect(x0, y0, x1 - x0, y1 - y0);
+      } else {
+        this.ctx.fillRect(-1e10, -1e10, 2e10, 2e10);
+      }
+      this.restore();
+    },
+    beginInlineImage: function CanvasGraphics_beginInlineImage() {
+      error('Should not call beginInlineImage');
+    },
+    beginImageData: function CanvasGraphics_beginImageData() {
+      error('Should not call beginImageData');
+    },
+    paintFormXObjectBegin: function CanvasGraphics_paintFormXObjectBegin(matrix, bbox) {
+      this.save();
+      this.baseTransformStack.push(this.baseTransform);
+      if (isArray(matrix) && matrix.length === 6) {
+        this.transform.apply(this, matrix);
+      }
+      this.baseTransform = this.ctx.mozCurrentTransform;
+      if (isArray(bbox) && bbox.length === 4) {
+        var width = bbox[2] - bbox[0];
+        var height = bbox[3] - bbox[1];
+        this.ctx.rect(bbox[0], bbox[1], width, height);
+        this.clip();
+        this.endPath();
+      }
+    },
+    paintFormXObjectEnd: function CanvasGraphics_paintFormXObjectEnd() {
+      this.restore();
+      this.baseTransform = this.baseTransformStack.pop();
+    },
+    beginGroup: function CanvasGraphics_beginGroup(group) {
+      this.save();
+      var currentCtx = this.ctx;
+      if (!group.isolated) {
+        info('TODO: Support non-isolated groups.');
+      }
+      if (group.knockout) {
+        warn('Knockout groups not supported.');
+      }
+      var currentTransform = currentCtx.mozCurrentTransform;
+      if (group.matrix) {
+        currentCtx.transform.apply(currentCtx, group.matrix);
+      }
+      assert(group.bbox, 'Bounding box is required.');
+      var bounds = Util.getAxialAlignedBoundingBox(group.bbox, currentCtx.mozCurrentTransform);
+      var canvasBounds = [0, 0, currentCtx.canvas.width, currentCtx.canvas.height];
+      bounds = Util.intersect(bounds, canvasBounds) || [0, 0, 0, 0];
+      var offsetX = Math.floor(bounds[0]);
+      var offsetY = Math.floor(bounds[1]);
+      var drawnWidth = Math.max(Math.ceil(bounds[2]) - offsetX, 1);
+      var drawnHeight = Math.max(Math.ceil(bounds[3]) - offsetY, 1);
+      var scaleX = 1,
+          scaleY = 1;
+      if (drawnWidth > MAX_GROUP_SIZE) {
+        scaleX = drawnWidth / MAX_GROUP_SIZE;
+        drawnWidth = MAX_GROUP_SIZE;
+      }
+      if (drawnHeight > MAX_GROUP_SIZE) {
+        scaleY = drawnHeight / MAX_GROUP_SIZE;
+        drawnHeight = MAX_GROUP_SIZE;
+      }
+      var cacheId = 'groupAt' + this.groupLevel;
+      if (group.smask) {
+        cacheId += '_smask_' + this.smaskCounter++ % 2;
+      }
+      var scratchCanvas = this.cachedCanvases.getCanvas(cacheId, drawnWidth, drawnHeight, true);
+      var groupCtx = scratchCanvas.context;
+      groupCtx.scale(1 / scaleX, 1 / scaleY);
+      groupCtx.translate(-offsetX, -offsetY);
+      groupCtx.transform.apply(groupCtx, currentTransform);
+      if (group.smask) {
+        this.smaskStack.push({
+          canvas: scratchCanvas.canvas,
+          context: groupCtx,
+          offsetX: offsetX,
+          offsetY: offsetY,
+          scaleX: scaleX,
+          scaleY: scaleY,
+          subtype: group.smask.subtype,
+          backdrop: group.smask.backdrop,
+          transferMap: group.smask.transferMap || null,
+          startTransformInverse: null
+        });
+      } else {
+        currentCtx.setTransform(1, 0, 0, 1, 0, 0);
+        currentCtx.translate(offsetX, offsetY);
+        currentCtx.scale(scaleX, scaleY);
+      }
+      copyCtxState(currentCtx, groupCtx);
+      this.ctx = groupCtx;
+      this.setGState([['BM', 'Normal'], ['ca', 1], ['CA', 1]]);
+      this.groupStack.push(currentCtx);
+      this.groupLevel++;
+      this.current.activeSMask = null;
+    },
+    endGroup: function CanvasGraphics_endGroup(group) {
+      this.groupLevel--;
+      var groupCtx = this.ctx;
+      this.ctx = this.groupStack.pop();
+      if (this.ctx.imageSmoothingEnabled !== undefined) {
+        this.ctx.imageSmoothingEnabled = false;
+      } else {
+        this.ctx.mozImageSmoothingEnabled = false;
+      }
+      if (group.smask) {
+        this.tempSMask = this.smaskStack.pop();
+      } else {
+        this.ctx.drawImage(groupCtx.canvas, 0, 0);
+      }
+      this.restore();
+    },
+    beginAnnotations: function CanvasGraphics_beginAnnotations() {
+      this.save();
+      this.current = new CanvasExtraState();
+      if (this.baseTransform) {
+        this.ctx.setTransform.apply(this.ctx, this.baseTransform);
+      }
+    },
+    endAnnotations: function CanvasGraphics_endAnnotations() {
+      this.restore();
+    },
+    beginAnnotation: function CanvasGraphics_beginAnnotation(rect, transform, matrix) {
+      this.save();
+      if (isArray(rect) && rect.length === 4) {
+        var width = rect[2] - rect[0];
+        var height = rect[3] - rect[1];
+        this.ctx.rect(rect[0], rect[1], width, height);
+        this.clip();
+        this.endPath();
+      }
+      this.transform.apply(this, transform);
+      this.transform.apply(this, matrix);
+    },
+    endAnnotation: function CanvasGraphics_endAnnotation() {
+      this.restore();
+    },
+    paintJpegXObject: function CanvasGraphics_paintJpegXObject(objId, w, h) {
+      var domImage = this.objs.get(objId);
+      if (!domImage) {
+        warn('Dependent image isn\'t ready yet');
+        return;
+      }
+      this.save();
+      var ctx = this.ctx;
+      ctx.scale(1 / w, -1 / h);
+      ctx.drawImage(domImage, 0, 0, domImage.width, domImage.height, 0, -h, w, h);
+      if (this.imageLayer) {
+        var currentTransform = ctx.mozCurrentTransformInverse;
+        var position = this.getCanvasPosition(0, 0);
+        this.imageLayer.appendImage({
+          objId: objId,
+          left: position[0],
+          top: position[1],
+          width: w / currentTransform[0],
+          height: h / currentTransform[3]
+        });
+      }
+      this.restore();
+    },
+    paintImageMaskXObject: function CanvasGraphics_paintImageMaskXObject(img) {
+      var ctx = this.ctx;
+      var width = img.width,
+          height = img.height;
+      var fillColor = this.current.fillColor;
+      var isPatternFill = this.current.patternFill;
+      var glyph = this.processingType3;
+      if (COMPILE_TYPE3_GLYPHS && glyph && glyph.compiled === undefined) {
+        if (width <= MAX_SIZE_TO_COMPILE && height <= MAX_SIZE_TO_COMPILE) {
+          glyph.compiled = compileType3Glyph({
+            data: img.data,
+            width: width,
+            height: height
+          });
+        } else {
+          glyph.compiled = null;
+        }
+      }
+      if (glyph && glyph.compiled) {
+        glyph.compiled(ctx);
+        return;
+      }
+      var maskCanvas = this.cachedCanvases.getCanvas('maskCanvas', width, height);
+      var maskCtx = maskCanvas.context;
+      maskCtx.save();
+      putBinaryImageMask(maskCtx, img);
+      maskCtx.globalCompositeOperation = 'source-in';
+      maskCtx.fillStyle = isPatternFill ? fillColor.getPattern(maskCtx, this) : fillColor;
+      maskCtx.fillRect(0, 0, width, height);
+      maskCtx.restore();
+      this.paintInlineImageXObject(maskCanvas.canvas);
+    },
+    paintImageMaskXObjectRepeat: function CanvasGraphics_paintImageMaskXObjectRepeat(imgData, scaleX, scaleY, positions) {
+      var width = imgData.width;
+      var height = imgData.height;
+      var fillColor = this.current.fillColor;
+      var isPatternFill = this.current.patternFill;
+      var maskCanvas = this.cachedCanvases.getCanvas('maskCanvas', width, height);
+      var maskCtx = maskCanvas.context;
+      maskCtx.save();
+      putBinaryImageMask(maskCtx, imgData);
+      maskCtx.globalCompositeOperation = 'source-in';
+      maskCtx.fillStyle = isPatternFill ? fillColor.getPattern(maskCtx, this) : fillColor;
+      maskCtx.fillRect(0, 0, width, height);
+      maskCtx.restore();
+      var ctx = this.ctx;
+      for (var i = 0, ii = positions.length; i < ii; i += 2) {
+        ctx.save();
+        ctx.transform(scaleX, 0, 0, scaleY, positions[i], positions[i + 1]);
+        ctx.scale(1, -1);
+        ctx.drawImage(maskCanvas.canvas, 0, 0, width, height, 0, -1, 1, 1);
+        ctx.restore();
+      }
+    },
+    paintImageMaskXObjectGroup: function CanvasGraphics_paintImageMaskXObjectGroup(images) {
+      var ctx = this.ctx;
+      var fillColor = this.current.fillColor;
+      var isPatternFill = this.current.patternFill;
+      for (var i = 0, ii = images.length; i < ii; i++) {
+        var image = images[i];
+        var width = image.width,
+            height = image.height;
+        var maskCanvas = this.cachedCanvases.getCanvas('maskCanvas', width, height);
+        var maskCtx = maskCanvas.context;
+        maskCtx.save();
+        putBinaryImageMask(maskCtx, image);
+        maskCtx.globalCompositeOperation = 'source-in';
+        maskCtx.fillStyle = isPatternFill ? fillColor.getPattern(maskCtx, this) : fillColor;
+        maskCtx.fillRect(0, 0, width, height);
+        maskCtx.restore();
+        ctx.save();
+        ctx.transform.apply(ctx, image.transform);
+        ctx.scale(1, -1);
+        ctx.drawImage(maskCanvas.canvas, 0, 0, width, height, 0, -1, 1, 1);
+        ctx.restore();
+      }
+    },
+    paintImageXObject: function CanvasGraphics_paintImageXObject(objId) {
+      var imgData = this.objs.get(objId);
+      if (!imgData) {
+        warn('Dependent image isn\'t ready yet');
+        return;
+      }
+      this.paintInlineImageXObject(imgData);
+    },
+    paintImageXObjectRepeat: function CanvasGraphics_paintImageXObjectRepeat(objId, scaleX, scaleY, positions) {
+      var imgData = this.objs.get(objId);
+      if (!imgData) {
+        warn('Dependent image isn\'t ready yet');
+        return;
+      }
+      var width = imgData.width;
+      var height = imgData.height;
+      var map = [];
+      for (var i = 0, ii = positions.length; i < ii; i += 2) {
+        map.push({
+          transform: [scaleX, 0, 0, scaleY, positions[i], positions[i + 1]],
+          x: 0,
+          y: 0,
+          w: width,
+          h: height
+        });
+      }
+      this.paintInlineImageXObjectGroup(imgData, map);
+    },
+    paintInlineImageXObject: function CanvasGraphics_paintInlineImageXObject(imgData) {
+      var width = imgData.width;
+      var height = imgData.height;
+      var ctx = this.ctx;
+      this.save();
+      ctx.scale(1 / width, -1 / height);
+      var currentTransform = ctx.mozCurrentTransformInverse;
+      var a = currentTransform[0],
+          b = currentTransform[1];
+      var widthScale = Math.max(Math.sqrt(a * a + b * b), 1);
+      var c = currentTransform[2],
+          d = currentTransform[3];
+      var heightScale = Math.max(Math.sqrt(c * c + d * d), 1);
+      var imgToPaint, tmpCanvas;
+      if (imgData instanceof HTMLElement || !imgData.data) {
+        imgToPaint = imgData;
+      } else {
+        tmpCanvas = this.cachedCanvases.getCanvas('inlineImage', width, height);
+        var tmpCtx = tmpCanvas.context;
+        putBinaryImageData(tmpCtx, imgData);
+        imgToPaint = tmpCanvas.canvas;
+      }
+      var paintWidth = width,
+          paintHeight = height;
+      var tmpCanvasId = 'prescale1';
+      while (widthScale > 2 && paintWidth > 1 || heightScale > 2 && paintHeight > 1) {
+        var newWidth = paintWidth,
+            newHeight = paintHeight;
+        if (widthScale > 2 && paintWidth > 1) {
+          newWidth = Math.ceil(paintWidth / 2);
+          widthScale /= paintWidth / newWidth;
+        }
+        if (heightScale > 2 && paintHeight > 1) {
+          newHeight = Math.ceil(paintHeight / 2);
+          heightScale /= paintHeight / newHeight;
+        }
+        tmpCanvas = this.cachedCanvases.getCanvas(tmpCanvasId, newWidth, newHeight);
+        tmpCtx = tmpCanvas.context;
+        tmpCtx.clearRect(0, 0, newWidth, newHeight);
+        tmpCtx.drawImage(imgToPaint, 0, 0, paintWidth, paintHeight, 0, 0, newWidth, newHeight);
+        imgToPaint = tmpCanvas.canvas;
+        paintWidth = newWidth;
+        paintHeight = newHeight;
+        tmpCanvasId = tmpCanvasId === 'prescale1' ? 'prescale2' : 'prescale1';
+      }
+      ctx.drawImage(imgToPaint, 0, 0, paintWidth, paintHeight, 0, -height, width, height);
+      if (this.imageLayer) {
+        var position = this.getCanvasPosition(0, -height);
+        this.imageLayer.appendImage({
+          imgData: imgData,
+          left: position[0],
+          top: position[1],
+          width: width / currentTransform[0],
+          height: height / currentTransform[3]
+        });
+      }
+      this.restore();
+    },
+    paintInlineImageXObjectGroup: function CanvasGraphics_paintInlineImageXObjectGroup(imgData, map) {
+      var ctx = this.ctx;
+      var w = imgData.width;
+      var h = imgData.height;
+      var tmpCanvas = this.cachedCanvases.getCanvas('inlineImage', w, h);
+      var tmpCtx = tmpCanvas.context;
+      putBinaryImageData(tmpCtx, imgData);
+      for (var i = 0, ii = map.length; i < ii; i++) {
+        var entry = map[i];
+        ctx.save();
+        ctx.transform.apply(ctx, entry.transform);
+        ctx.scale(1, -1);
+        ctx.drawImage(tmpCanvas.canvas, entry.x, entry.y, entry.w, entry.h, 0, -1, 1, 1);
+        if (this.imageLayer) {
+          var position = this.getCanvasPosition(entry.x, entry.y);
+          this.imageLayer.appendImage({
+            imgData: imgData,
+            left: position[0],
+            top: position[1],
+            width: w,
+            height: h
+          });
+        }
+        ctx.restore();
+      }
+    },
+    paintSolidColorImageMask: function CanvasGraphics_paintSolidColorImageMask() {
+      this.ctx.fillRect(0, 0, 1, 1);
+    },
+    paintXObject: function CanvasGraphics_paintXObject() {
+      warn('Unsupported \'paintXObject\' command.');
+    },
+    markPoint: function CanvasGraphics_markPoint(tag) {},
+    markPointProps: function CanvasGraphics_markPointProps(tag, properties) {},
+    beginMarkedContent: function CanvasGraphics_beginMarkedContent(tag) {},
+    beginMarkedContentProps: function CanvasGraphics_beginMarkedContentProps(tag, properties) {},
+    endMarkedContent: function CanvasGraphics_endMarkedContent() {},
+    beginCompat: function CanvasGraphics_beginCompat() {},
+    endCompat: function CanvasGraphics_endCompat() {},
+    consumePath: function CanvasGraphics_consumePath() {
+      var ctx = this.ctx;
+      if (this.pendingClip) {
+        if (this.pendingClip === EO_CLIP) {
+          ctx.clip('evenodd');
+        } else {
+          ctx.clip();
+        }
+        this.pendingClip = null;
+      }
+      ctx.beginPath();
+    },
+    getSinglePixelWidth: function CanvasGraphics_getSinglePixelWidth(scale) {
+      if (this.cachedGetSinglePixelWidth === null) {
+        this.ctx.save();
+        var inverse = this.ctx.mozCurrentTransformInverse;
+        this.ctx.restore();
+        this.cachedGetSinglePixelWidth = Math.sqrt(Math.max(inverse[0] * inverse[0] + inverse[1] * inverse[1], inverse[2] * inverse[2] + inverse[3] * inverse[3]));
+      }
+      return this.cachedGetSinglePixelWidth;
+    },
+    getCanvasPosition: function CanvasGraphics_getCanvasPosition(x, y) {
+      var transform = this.ctx.mozCurrentTransform;
+      return [transform[0] * x + transform[2] * y + transform[4], transform[1] * x + transform[3] * y + transform[5]];
     }
-    this.pendingClip = null;
-   }
-   ctx.beginPath();
-  },
-  getSinglePixelWidth: function CanvasGraphics_getSinglePixelWidth(scale) {
-   if (this.cachedGetSinglePixelWidth === null) {
-    this.ctx.save();
-    var inverse = this.ctx.mozCurrentTransformInverse;
-    this.ctx.restore();
-    this.cachedGetSinglePixelWidth = Math.sqrt(Math.max(inverse[0] * inverse[0] + inverse[1] * inverse[1], inverse[2] * inverse[2] + inverse[3] * inverse[3]));
-   }
-   return this.cachedGetSinglePixelWidth;
-  },
-  getCanvasPosition: function CanvasGraphics_getCanvasPosition(x, y) {
-   var transform = this.ctx.mozCurrentTransform;
-   return [
-    transform[0] * x + transform[2] * y + transform[4],
-    transform[1] * x + transform[3] * y + transform[5]
-   ];
+  };
+  for (var op in OPS) {
+    CanvasGraphics.prototype[OPS[op]] = CanvasGraphics.prototype[op];
   }
- };
- for (var op in OPS) {
-  CanvasGraphics.prototype[OPS[op]] = CanvasGraphics.prototype[op];
- }
- return CanvasGraphics;
+  return CanvasGraphics;
 }();
 exports.CanvasGraphics = CanvasGraphics;

+ 191 - 201
lib/display/dom_utils.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var assert = sharedUtil.assert;
 var removeNullCharacters = sharedUtil.removeNullCharacters;
@@ -22,230 +23,219 @@ var createValidAbsoluteUrl = sharedUtil.createValidAbsoluteUrl;
 var stringToBytes = sharedUtil.stringToBytes;
 var CMapCompressionType = sharedUtil.CMapCompressionType;
 var DEFAULT_LINK_REL = 'noopener noreferrer nofollow';
-function DOMCanvasFactory() {
-}
+function DOMCanvasFactory() {}
 DOMCanvasFactory.prototype = {
- create: function DOMCanvasFactory_create(width, height) {
-  assert(width > 0 && height > 0, 'invalid canvas size');
-  var canvas = document.createElement('canvas');
-  var context = canvas.getContext('2d');
-  canvas.width = width;
-  canvas.height = height;
-  return {
-   canvas: canvas,
-   context: context
-  };
- },
- reset: function DOMCanvasFactory_reset(canvasAndContextPair, width, height) {
-  assert(canvasAndContextPair.canvas, 'canvas is not specified');
-  assert(width > 0 && height > 0, 'invalid canvas size');
-  canvasAndContextPair.canvas.width = width;
-  canvasAndContextPair.canvas.height = height;
- },
- destroy: function DOMCanvasFactory_destroy(canvasAndContextPair) {
-  assert(canvasAndContextPair.canvas, 'canvas is not specified');
-  canvasAndContextPair.canvas.width = 0;
-  canvasAndContextPair.canvas.height = 0;
-  canvasAndContextPair.canvas = null;
-  canvasAndContextPair.context = null;
- }
+  create: function DOMCanvasFactory_create(width, height) {
+    assert(width > 0 && height > 0, 'invalid canvas size');
+    var canvas = document.createElement('canvas');
+    var context = canvas.getContext('2d');
+    canvas.width = width;
+    canvas.height = height;
+    return {
+      canvas: canvas,
+      context: context
+    };
+  },
+  reset: function DOMCanvasFactory_reset(canvasAndContextPair, width, height) {
+    assert(canvasAndContextPair.canvas, 'canvas is not specified');
+    assert(width > 0 && height > 0, 'invalid canvas size');
+    canvasAndContextPair.canvas.width = width;
+    canvasAndContextPair.canvas.height = height;
+  },
+  destroy: function DOMCanvasFactory_destroy(canvasAndContextPair) {
+    assert(canvasAndContextPair.canvas, 'canvas is not specified');
+    canvasAndContextPair.canvas.width = 0;
+    canvasAndContextPair.canvas.height = 0;
+    canvasAndContextPair.canvas = null;
+    canvasAndContextPair.context = null;
+  }
 };
 var DOMCMapReaderFactory = function DOMCMapReaderFactoryClosure() {
- function DOMCMapReaderFactory(params) {
-  this.baseUrl = params.baseUrl || null;
-  this.isCompressed = params.isCompressed || false;
- }
- DOMCMapReaderFactory.prototype = {
-  fetch: function (params) {
-   var name = params.name;
-   if (!name) {
-    return Promise.reject(new Error('CMap name must be specified.'));
-   }
-   return new Promise(function (resolve, reject) {
-    var url = this.baseUrl + name + (this.isCompressed ? '.bcmap' : '');
-    var request = new XMLHttpRequest();
-    request.open('GET', url, true);
-    if (this.isCompressed) {
-     request.responseType = 'arraybuffer';
-    }
-    request.onreadystatechange = function () {
-     if (request.readyState === XMLHttpRequest.DONE && (request.status === 200 || request.status === 0)) {
-      var data;
-      if (this.isCompressed && request.response) {
-       data = new Uint8Array(request.response);
-      } else if (!this.isCompressed && request.responseText) {
-       data = stringToBytes(request.responseText);
-      }
-      if (data) {
-       resolve({
-        cMapData: data,
-        compressionType: this.isCompressed ? CMapCompressionType.BINARY : CMapCompressionType.NONE
-       });
-       return;
-      }
-      reject(new Error('Unable to load ' + (this.isCompressed ? 'binary ' : '') + 'CMap at: ' + url));
-     }
-    }.bind(this);
-    request.send(null);
-   }.bind(this));
+  function DOMCMapReaderFactory(params) {
+    this.baseUrl = params.baseUrl || null;
+    this.isCompressed = params.isCompressed || false;
   }
- };
- return DOMCMapReaderFactory;
+  DOMCMapReaderFactory.prototype = {
+    fetch: function (params) {
+      var name = params.name;
+      if (!name) {
+        return Promise.reject(new Error('CMap name must be specified.'));
+      }
+      return new Promise(function (resolve, reject) {
+        var url = this.baseUrl + name + (this.isCompressed ? '.bcmap' : '');
+        var request = new XMLHttpRequest();
+        request.open('GET', url, true);
+        if (this.isCompressed) {
+          request.responseType = 'arraybuffer';
+        }
+        request.onreadystatechange = function () {
+          if (request.readyState === XMLHttpRequest.DONE && (request.status === 200 || request.status === 0)) {
+            var data;
+            if (this.isCompressed && request.response) {
+              data = new Uint8Array(request.response);
+            } else if (!this.isCompressed && request.responseText) {
+              data = stringToBytes(request.responseText);
+            }
+            if (data) {
+              resolve({
+                cMapData: data,
+                compressionType: this.isCompressed ? CMapCompressionType.BINARY : CMapCompressionType.NONE
+              });
+              return;
+            }
+            reject(new Error('Unable to load ' + (this.isCompressed ? 'binary ' : '') + 'CMap at: ' + url));
+          }
+        }.bind(this);
+        request.send(null);
+      }.bind(this));
+    }
+  };
+  return DOMCMapReaderFactory;
 }();
 var CustomStyle = function CustomStyleClosure() {
- var prefixes = [
-  'ms',
-  'Moz',
-  'Webkit',
-  'O'
- ];
- var _cache = Object.create(null);
- function CustomStyle() {
- }
- CustomStyle.getProp = function get(propName, element) {
-  if (arguments.length === 1 && typeof _cache[propName] === 'string') {
-   return _cache[propName];
-  }
-  element = element || document.documentElement;
-  var style = element.style, prefixed, uPropName;
-  if (typeof style[propName] === 'string') {
-   return _cache[propName] = propName;
-  }
-  uPropName = propName.charAt(0).toUpperCase() + propName.slice(1);
-  for (var i = 0, l = prefixes.length; i < l; i++) {
-   prefixed = prefixes[i] + uPropName;
-   if (typeof style[prefixed] === 'string') {
-    return _cache[propName] = prefixed;
-   }
-  }
-  return _cache[propName] = 'undefined';
- };
- CustomStyle.setProp = function set(propName, element, str) {
-  var prop = this.getProp(propName);
-  if (prop !== 'undefined') {
-   element.style[prop] = str;
-  }
- };
- return CustomStyle;
+  var prefixes = ['ms', 'Moz', 'Webkit', 'O'];
+  var _cache = Object.create(null);
+  function CustomStyle() {}
+  CustomStyle.getProp = function get(propName, element) {
+    if (arguments.length === 1 && typeof _cache[propName] === 'string') {
+      return _cache[propName];
+    }
+    element = element || document.documentElement;
+    var style = element.style,
+        prefixed,
+        uPropName;
+    if (typeof style[propName] === 'string') {
+      return _cache[propName] = propName;
+    }
+    uPropName = propName.charAt(0).toUpperCase() + propName.slice(1);
+    for (var i = 0, l = prefixes.length; i < l; i++) {
+      prefixed = prefixes[i] + uPropName;
+      if (typeof style[prefixed] === 'string') {
+        return _cache[propName] = prefixed;
+      }
+    }
+    return _cache[propName] = 'undefined';
+  };
+  CustomStyle.setProp = function set(propName, element, str) {
+    var prop = this.getProp(propName);
+    if (prop !== 'undefined') {
+      element.style[prop] = str;
+    }
+  };
+  return CustomStyle;
 }();
 var hasCanvasTypedArrays;
 hasCanvasTypedArrays = function hasCanvasTypedArrays() {
- var canvas = document.createElement('canvas');
- canvas.width = canvas.height = 1;
- var ctx = canvas.getContext('2d');
- var imageData = ctx.createImageData(1, 1);
- return typeof imageData.data.buffer !== 'undefined';
+  var canvas = document.createElement('canvas');
+  canvas.width = canvas.height = 1;
+  var ctx = canvas.getContext('2d');
+  var imageData = ctx.createImageData(1, 1);
+  return typeof imageData.data.buffer !== 'undefined';
 };
 var LinkTarget = {
- NONE: 0,
- SELF: 1,
- BLANK: 2,
- PARENT: 3,
- TOP: 4
+  NONE: 0,
+  SELF: 1,
+  BLANK: 2,
+  PARENT: 3,
+  TOP: 4
 };
-var LinkTargetStringMap = [
- '',
- '_self',
- '_blank',
- '_parent',
- '_top'
-];
+var LinkTargetStringMap = ['', '_self', '_blank', '_parent', '_top'];
 function addLinkAttributes(link, params) {
- var url = params && params.url;
- link.href = link.title = url ? removeNullCharacters(url) : '';
- if (url) {
-  var target = params.target;
-  if (typeof target === 'undefined') {
-   target = getDefaultSetting('externalLinkTarget');
-  }
-  link.target = LinkTargetStringMap[target];
-  var rel = params.rel;
-  if (typeof rel === 'undefined') {
-   rel = getDefaultSetting('externalLinkRel');
+  var url = params && params.url;
+  link.href = link.title = url ? removeNullCharacters(url) : '';
+  if (url) {
+    var target = params.target;
+    if (typeof target === 'undefined') {
+      target = getDefaultSetting('externalLinkTarget');
+    }
+    link.target = LinkTargetStringMap[target];
+    var rel = params.rel;
+    if (typeof rel === 'undefined') {
+      rel = getDefaultSetting('externalLinkRel');
+    }
+    link.rel = rel;
   }
-  link.rel = rel;
- }
 }
 function getFilenameFromUrl(url) {
- var anchor = url.indexOf('#');
- var query = url.indexOf('?');
- var end = Math.min(anchor > 0 ? anchor : url.length, query > 0 ? query : url.length);
- return url.substring(url.lastIndexOf('/', end) + 1, end);
+  var anchor = url.indexOf('#');
+  var query = url.indexOf('?');
+  var end = Math.min(anchor > 0 ? anchor : url.length, query > 0 ? query : url.length);
+  return url.substring(url.lastIndexOf('/', end) + 1, end);
 }
 function getDefaultSetting(id) {
- var globalSettings = sharedUtil.globalScope.PDFJS;
- switch (id) {
- case 'pdfBug':
-  return globalSettings ? globalSettings.pdfBug : false;
- case 'disableAutoFetch':
-  return globalSettings ? globalSettings.disableAutoFetch : false;
- case 'disableStream':
-  return globalSettings ? globalSettings.disableStream : false;
- case 'disableRange':
-  return globalSettings ? globalSettings.disableRange : false;
- case 'disableFontFace':
-  return globalSettings ? globalSettings.disableFontFace : false;
- case 'disableCreateObjectURL':
-  return globalSettings ? globalSettings.disableCreateObjectURL : false;
- case 'disableWebGL':
-  return globalSettings ? globalSettings.disableWebGL : true;
- case 'cMapUrl':
-  return globalSettings ? globalSettings.cMapUrl : null;
- case 'cMapPacked':
-  return globalSettings ? globalSettings.cMapPacked : false;
- case 'postMessageTransfers':
-  return globalSettings ? globalSettings.postMessageTransfers : true;
- case 'workerPort':
-  return globalSettings ? globalSettings.workerPort : null;
- case 'workerSrc':
-  return globalSettings ? globalSettings.workerSrc : null;
- case 'disableWorker':
-  return globalSettings ? globalSettings.disableWorker : false;
- case 'maxImageSize':
-  return globalSettings ? globalSettings.maxImageSize : -1;
- case 'imageResourcesPath':
-  return globalSettings ? globalSettings.imageResourcesPath : '';
- case 'isEvalSupported':
-  return globalSettings ? globalSettings.isEvalSupported : true;
- case 'externalLinkTarget':
-  if (!globalSettings) {
-   return LinkTarget.NONE;
-  }
-  switch (globalSettings.externalLinkTarget) {
-  case LinkTarget.NONE:
-  case LinkTarget.SELF:
-  case LinkTarget.BLANK:
-  case LinkTarget.PARENT:
-  case LinkTarget.TOP:
-   return globalSettings.externalLinkTarget;
+  var globalSettings = sharedUtil.globalScope.PDFJS;
+  switch (id) {
+    case 'pdfBug':
+      return globalSettings ? globalSettings.pdfBug : false;
+    case 'disableAutoFetch':
+      return globalSettings ? globalSettings.disableAutoFetch : false;
+    case 'disableStream':
+      return globalSettings ? globalSettings.disableStream : false;
+    case 'disableRange':
+      return globalSettings ? globalSettings.disableRange : false;
+    case 'disableFontFace':
+      return globalSettings ? globalSettings.disableFontFace : false;
+    case 'disableCreateObjectURL':
+      return globalSettings ? globalSettings.disableCreateObjectURL : false;
+    case 'disableWebGL':
+      return globalSettings ? globalSettings.disableWebGL : true;
+    case 'cMapUrl':
+      return globalSettings ? globalSettings.cMapUrl : null;
+    case 'cMapPacked':
+      return globalSettings ? globalSettings.cMapPacked : false;
+    case 'postMessageTransfers':
+      return globalSettings ? globalSettings.postMessageTransfers : true;
+    case 'workerPort':
+      return globalSettings ? globalSettings.workerPort : null;
+    case 'workerSrc':
+      return globalSettings ? globalSettings.workerSrc : null;
+    case 'disableWorker':
+      return globalSettings ? globalSettings.disableWorker : false;
+    case 'maxImageSize':
+      return globalSettings ? globalSettings.maxImageSize : -1;
+    case 'imageResourcesPath':
+      return globalSettings ? globalSettings.imageResourcesPath : '';
+    case 'isEvalSupported':
+      return globalSettings ? globalSettings.isEvalSupported : true;
+    case 'externalLinkTarget':
+      if (!globalSettings) {
+        return LinkTarget.NONE;
+      }
+      switch (globalSettings.externalLinkTarget) {
+        case LinkTarget.NONE:
+        case LinkTarget.SELF:
+        case LinkTarget.BLANK:
+        case LinkTarget.PARENT:
+        case LinkTarget.TOP:
+          return globalSettings.externalLinkTarget;
+      }
+      warn('PDFJS.externalLinkTarget is invalid: ' + globalSettings.externalLinkTarget);
+      globalSettings.externalLinkTarget = LinkTarget.NONE;
+      return LinkTarget.NONE;
+    case 'externalLinkRel':
+      return globalSettings ? globalSettings.externalLinkRel : DEFAULT_LINK_REL;
+    case 'enableStats':
+      return !!(globalSettings && globalSettings.enableStats);
+    default:
+      throw new Error('Unknown default setting: ' + id);
   }
-  warn('PDFJS.externalLinkTarget is invalid: ' + globalSettings.externalLinkTarget);
-  globalSettings.externalLinkTarget = LinkTarget.NONE;
-  return LinkTarget.NONE;
- case 'externalLinkRel':
-  return globalSettings ? globalSettings.externalLinkRel : DEFAULT_LINK_REL;
- case 'enableStats':
-  return !!(globalSettings && globalSettings.enableStats);
- default:
-  throw new Error('Unknown default setting: ' + id);
- }
 }
 function isExternalLinkTargetSet() {
- var externalLinkTarget = getDefaultSetting('externalLinkTarget');
- switch (externalLinkTarget) {
- case LinkTarget.NONE:
-  return false;
- case LinkTarget.SELF:
- case LinkTarget.BLANK:
- case LinkTarget.PARENT:
- case LinkTarget.TOP:
-  return true;
- }
+  var externalLinkTarget = getDefaultSetting('externalLinkTarget');
+  switch (externalLinkTarget) {
+    case LinkTarget.NONE:
+      return false;
+    case LinkTarget.SELF:
+    case LinkTarget.BLANK:
+    case LinkTarget.PARENT:
+    case LinkTarget.TOP:
+      return true;
+  }
 }
 function isValidUrl(url, allowRelative) {
- deprecated('isValidUrl(), please use createValidAbsoluteUrl() instead.');
- var baseUrl = allowRelative ? 'http://example.com' : null;
- return createValidAbsoluteUrl(url, baseUrl) !== null;
+  deprecated('isValidUrl(), please use createValidAbsoluteUrl() instead.');
+  var baseUrl = allowRelative ? 'http://example.com' : null;
+  return createValidAbsoluteUrl(url, baseUrl) !== null;
 }
 exports.CustomStyle = CustomStyle;
 exports.addLinkAttributes = addLinkAttributes;

+ 236 - 237
lib/display/font_loader.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var assert = sharedUtil.assert;
 var bytesToString = sharedUtil.bytesToString;
@@ -20,278 +21,276 @@ var string32 = sharedUtil.string32;
 var shadow = sharedUtil.shadow;
 var warn = sharedUtil.warn;
 function FontLoader(docId) {
- this.docId = docId;
- this.styleElement = null;
- this.nativeFontFaces = [];
- this.loadTestFontId = 0;
- this.loadingContext = {
-  requests: [],
-  nextRequestId: 0
- };
+  this.docId = docId;
+  this.styleElement = null;
+  this.nativeFontFaces = [];
+  this.loadTestFontId = 0;
+  this.loadingContext = {
+    requests: [],
+    nextRequestId: 0
+  };
 }
 FontLoader.prototype = {
- insertRule: function fontLoaderInsertRule(rule) {
-  var styleElement = this.styleElement;
-  if (!styleElement) {
-   styleElement = this.styleElement = document.createElement('style');
-   styleElement.id = 'PDFJS_FONT_STYLE_TAG_' + this.docId;
-   document.documentElement.getElementsByTagName('head')[0].appendChild(styleElement);
-  }
-  var styleSheet = styleElement.sheet;
-  styleSheet.insertRule(rule, styleSheet.cssRules.length);
- },
- clear: function fontLoaderClear() {
-  if (this.styleElement) {
-   this.styleElement.remove();
-   this.styleElement = null;
+  insertRule: function fontLoaderInsertRule(rule) {
+    var styleElement = this.styleElement;
+    if (!styleElement) {
+      styleElement = this.styleElement = document.createElement('style');
+      styleElement.id = 'PDFJS_FONT_STYLE_TAG_' + this.docId;
+      document.documentElement.getElementsByTagName('head')[0].appendChild(styleElement);
+    }
+    var styleSheet = styleElement.sheet;
+    styleSheet.insertRule(rule, styleSheet.cssRules.length);
+  },
+  clear: function fontLoaderClear() {
+    if (this.styleElement) {
+      this.styleElement.remove();
+      this.styleElement = null;
+    }
+    this.nativeFontFaces.forEach(function (nativeFontFace) {
+      document.fonts.delete(nativeFontFace);
+    });
+    this.nativeFontFaces.length = 0;
   }
-  this.nativeFontFaces.forEach(function (nativeFontFace) {
-   document.fonts.delete(nativeFontFace);
-  });
-  this.nativeFontFaces.length = 0;
- }
 };
 var getLoadTestFont = function () {
- return atob('T1RUTwALAIAAAwAwQ0ZGIDHtZg4AAAOYAAAAgUZGVE1lkzZwAAAEHAAAABxHREVGABQAFQ' + 'AABDgAAAAeT1MvMlYNYwkAAAEgAAAAYGNtYXABDQLUAAACNAAAAUJoZWFk/xVFDQAAALwA' + 'AAA2aGhlYQdkA+oAAAD0AAAAJGhtdHgD6AAAAAAEWAAAAAZtYXhwAAJQAAAAARgAAAAGbm' + 'FtZVjmdH4AAAGAAAAAsXBvc3T/hgAzAAADeAAAACAAAQAAAAEAALZRFsRfDzz1AAsD6AAA' + 'AADOBOTLAAAAAM4KHDwAAAAAA+gDIQAAAAgAAgAAAAAAAAABAAADIQAAAFoD6AAAAAAD6A' + 'ABAAAAAAAAAAAAAAAAAAAAAQAAUAAAAgAAAAQD6AH0AAUAAAKKArwAAACMAooCvAAAAeAA' + 'MQECAAACAAYJAAAAAAAAAAAAAQAAAAAAAAAAAAAAAFBmRWQAwAAuAC4DIP84AFoDIQAAAA' + 'AAAQAAAAAAAAAAACAAIAABAAAADgCuAAEAAAAAAAAAAQAAAAEAAAAAAAEAAQAAAAEAAAAA' + 'AAIAAQAAAAEAAAAAAAMAAQAAAAEAAAAAAAQAAQAAAAEAAAAAAAUAAQAAAAEAAAAAAAYAAQ' + 'AAAAMAAQQJAAAAAgABAAMAAQQJAAEAAgABAAMAAQQJAAIAAgABAAMAAQQJAAMAAgABAAMA' + 'AQQJAAQAAgABAAMAAQQJAAUAAgABAAMAAQQJAAYAAgABWABYAAAAAAAAAwAAAAMAAAAcAA' + 'EAAAAAADwAAwABAAAAHAAEACAAAAAEAAQAAQAAAC7//wAAAC7////TAAEAAAAAAAABBgAA' + 'AQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAA' + 'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' + 'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' + 'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' + 'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAMAAA' + 'AAAAD/gwAyAAAAAQAAAAAAAAAAAAAAAAAAAAABAAQEAAEBAQJYAAEBASH4DwD4GwHEAvgc' + 'A/gXBIwMAYuL+nz5tQXkD5j3CBLnEQACAQEBIVhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWF' + 'hYWFhYWFhYAAABAQAADwACAQEEE/t3Dov6fAH6fAT+fPp8+nwHDosMCvm1Cvm1DAz6fBQA' + 'AAAAAAABAAAAAMmJbzEAAAAAzgTjFQAAAADOBOQpAAEAAAAAAAAADAAUAAQAAAABAAAAAg' + 'ABAAAAAAAAAAAD6AAAAAAAAA==');
+  return atob('T1RUTwALAIAAAwAwQ0ZGIDHtZg4AAAOYAAAAgUZGVE1lkzZwAAAEHAAAABxHREVGABQAFQ' + 'AABDgAAAAeT1MvMlYNYwkAAAEgAAAAYGNtYXABDQLUAAACNAAAAUJoZWFk/xVFDQAAALwA' + 'AAA2aGhlYQdkA+oAAAD0AAAAJGhtdHgD6AAAAAAEWAAAAAZtYXhwAAJQAAAAARgAAAAGbm' + 'FtZVjmdH4AAAGAAAAAsXBvc3T/hgAzAAADeAAAACAAAQAAAAEAALZRFsRfDzz1AAsD6AAA' + 'AADOBOTLAAAAAM4KHDwAAAAAA+gDIQAAAAgAAgAAAAAAAAABAAADIQAAAFoD6AAAAAAD6A' + 'ABAAAAAAAAAAAAAAAAAAAAAQAAUAAAAgAAAAQD6AH0AAUAAAKKArwAAACMAooCvAAAAeAA' + 'MQECAAACAAYJAAAAAAAAAAAAAQAAAAAAAAAAAAAAAFBmRWQAwAAuAC4DIP84AFoDIQAAAA' + 'AAAQAAAAAAAAAAACAAIAABAAAADgCuAAEAAAAAAAAAAQAAAAEAAAAAAAEAAQAAAAEAAAAA' + 'AAIAAQAAAAEAAAAAAAMAAQAAAAEAAAAAAAQAAQAAAAEAAAAAAAUAAQAAAAEAAAAAAAYAAQ' + 'AAAAMAAQQJAAAAAgABAAMAAQQJAAEAAgABAAMAAQQJAAIAAgABAAMAAQQJAAMAAgABAAMA' + 'AQQJAAQAAgABAAMAAQQJAAUAAgABAAMAAQQJAAYAAgABWABYAAAAAAAAAwAAAAMAAAAcAA' + 'EAAAAAADwAAwABAAAAHAAEACAAAAAEAAQAAQAAAC7//wAAAC7////TAAEAAAAAAAABBgAA' + 'AQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAA' + 'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' + 'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' + 'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA' + 'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAMAAA' + 'AAAAD/gwAyAAAAAQAAAAAAAAAAAAAAAAAAAAABAAQEAAEBAQJYAAEBASH4DwD4GwHEAvgc' + 'A/gXBIwMAYuL+nz5tQXkD5j3CBLnEQACAQEBIVhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWF' + 'hYWFhYWFhYAAABAQAADwACAQEEE/t3Dov6fAH6fAT+fPp8+nwHDosMCvm1Cvm1DAz6fBQA' + 'AAAAAAABAAAAAMmJbzEAAAAAzgTjFQAAAADOBOQpAAEAAAAAAAAADAAUAAQAAAABAAAAAg' + 'ABAAAAAAAAAAAD6AAAAAAAAA==');
 };
 Object.defineProperty(FontLoader.prototype, 'loadTestFont', {
- get: function () {
-  return shadow(this, 'loadTestFont', getLoadTestFont());
- },
- configurable: true
+  get: function () {
+    return shadow(this, 'loadTestFont', getLoadTestFont());
+  },
+  configurable: true
 });
 FontLoader.prototype.addNativeFontFace = function fontLoader_addNativeFontFace(nativeFontFace) {
- this.nativeFontFaces.push(nativeFontFace);
- document.fonts.add(nativeFontFace);
+  this.nativeFontFaces.push(nativeFontFace);
+  document.fonts.add(nativeFontFace);
 };
 FontLoader.prototype.bind = function fontLoaderBind(fonts, callback) {
- var rules = [];
- var fontsToLoad = [];
- var fontLoadPromises = [];
- var getNativeFontPromise = function (nativeFontFace) {
-  return nativeFontFace.loaded.catch(function (e) {
-   warn('Failed to load font "' + nativeFontFace.family + '": ' + e);
-  });
- };
- var isFontLoadingAPISupported = FontLoader.isFontLoadingAPISupported && !FontLoader.isSyncFontLoadingSupported;
- for (var i = 0, ii = fonts.length; i < ii; i++) {
-  var font = fonts[i];
-  if (font.attached || font.loading === false) {
-   continue;
+  var rules = [];
+  var fontsToLoad = [];
+  var fontLoadPromises = [];
+  var getNativeFontPromise = function (nativeFontFace) {
+    return nativeFontFace.loaded.catch(function (e) {
+      warn('Failed to load font "' + nativeFontFace.family + '": ' + e);
+    });
+  };
+  var isFontLoadingAPISupported = FontLoader.isFontLoadingAPISupported && !FontLoader.isSyncFontLoadingSupported;
+  for (var i = 0, ii = fonts.length; i < ii; i++) {
+    var font = fonts[i];
+    if (font.attached || font.loading === false) {
+      continue;
+    }
+    font.attached = true;
+    if (isFontLoadingAPISupported) {
+      var nativeFontFace = font.createNativeFontFace();
+      if (nativeFontFace) {
+        this.addNativeFontFace(nativeFontFace);
+        fontLoadPromises.push(getNativeFontPromise(nativeFontFace));
+      }
+    } else {
+      var rule = font.createFontFaceRule();
+      if (rule) {
+        this.insertRule(rule);
+        rules.push(rule);
+        fontsToLoad.push(font);
+      }
+    }
   }
-  font.attached = true;
+  var request = this.queueLoadingCallback(callback);
   if (isFontLoadingAPISupported) {
-   var nativeFontFace = font.createNativeFontFace();
-   if (nativeFontFace) {
-    this.addNativeFontFace(nativeFontFace);
-    fontLoadPromises.push(getNativeFontPromise(nativeFontFace));
-   }
+    Promise.all(fontLoadPromises).then(function () {
+      request.complete();
+    });
+  } else if (rules.length > 0 && !FontLoader.isSyncFontLoadingSupported) {
+    this.prepareFontLoadEvent(rules, fontsToLoad, request);
   } else {
-   var rule = font.createFontFaceRule();
-   if (rule) {
-    this.insertRule(rule);
-    rules.push(rule);
-    fontsToLoad.push(font);
-   }
+    request.complete();
   }
- }
- var request = this.queueLoadingCallback(callback);
- if (isFontLoadingAPISupported) {
-  Promise.all(fontLoadPromises).then(function () {
-   request.complete();
-  });
- } else if (rules.length > 0 && !FontLoader.isSyncFontLoadingSupported) {
-  this.prepareFontLoadEvent(rules, fontsToLoad, request);
- } else {
-  request.complete();
- }
 };
 FontLoader.prototype.queueLoadingCallback = function FontLoader_queueLoadingCallback(callback) {
- function LoadLoader_completeRequest() {
-  assert(!request.end, 'completeRequest() cannot be called twice');
-  request.end = Date.now();
-  while (context.requests.length > 0 && context.requests[0].end) {
-   var otherRequest = context.requests.shift();
-   setTimeout(otherRequest.callback, 0);
+  function LoadLoader_completeRequest() {
+    assert(!request.end, 'completeRequest() cannot be called twice');
+    request.end = Date.now();
+    while (context.requests.length > 0 && context.requests[0].end) {
+      var otherRequest = context.requests.shift();
+      setTimeout(otherRequest.callback, 0);
+    }
   }
- }
- var context = this.loadingContext;
- var requestId = 'pdfjs-font-loading-' + context.nextRequestId++;
- var request = {
-  id: requestId,
-  complete: LoadLoader_completeRequest,
-  callback: callback,
-  started: Date.now()
- };
- context.requests.push(request);
- return request;
+  var context = this.loadingContext;
+  var requestId = 'pdfjs-font-loading-' + context.nextRequestId++;
+  var request = {
+    id: requestId,
+    complete: LoadLoader_completeRequest,
+    callback: callback,
+    started: Date.now()
+  };
+  context.requests.push(request);
+  return request;
 };
 FontLoader.prototype.prepareFontLoadEvent = function fontLoaderPrepareFontLoadEvent(rules, fonts, request) {
- function int32(data, offset) {
-  return data.charCodeAt(offset) << 24 | data.charCodeAt(offset + 1) << 16 | data.charCodeAt(offset + 2) << 8 | data.charCodeAt(offset + 3) & 0xff;
- }
- function spliceString(s, offset, remove, insert) {
-  var chunk1 = s.substr(0, offset);
-  var chunk2 = s.substr(offset + remove);
-  return chunk1 + insert + chunk2;
- }
- var i, ii;
- var canvas = document.createElement('canvas');
- canvas.width = 1;
- canvas.height = 1;
- var ctx = canvas.getContext('2d');
- var called = 0;
- function isFontReady(name, callback) {
-  called++;
-  if (called > 30) {
-   warn('Load test font never loaded.');
-   callback();
-   return;
+  function int32(data, offset) {
+    return data.charCodeAt(offset) << 24 | data.charCodeAt(offset + 1) << 16 | data.charCodeAt(offset + 2) << 8 | data.charCodeAt(offset + 3) & 0xff;
+  }
+  function spliceString(s, offset, remove, insert) {
+    var chunk1 = s.substr(0, offset);
+    var chunk2 = s.substr(offset + remove);
+    return chunk1 + insert + chunk2;
+  }
+  var i, ii;
+  var canvas = document.createElement('canvas');
+  canvas.width = 1;
+  canvas.height = 1;
+  var ctx = canvas.getContext('2d');
+  var called = 0;
+  function isFontReady(name, callback) {
+    called++;
+    if (called > 30) {
+      warn('Load test font never loaded.');
+      callback();
+      return;
+    }
+    ctx.font = '30px ' + name;
+    ctx.fillText('.', 0, 20);
+    var imageData = ctx.getImageData(0, 0, 1, 1);
+    if (imageData.data[3] > 0) {
+      callback();
+      return;
+    }
+    setTimeout(isFontReady.bind(null, name, callback));
   }
-  ctx.font = '30px ' + name;
-  ctx.fillText('.', 0, 20);
-  var imageData = ctx.getImageData(0, 0, 1, 1);
-  if (imageData.data[3] > 0) {
-   callback();
-   return;
+  var loadTestFontId = 'lt' + Date.now() + this.loadTestFontId++;
+  var data = this.loadTestFont;
+  var COMMENT_OFFSET = 976;
+  data = spliceString(data, COMMENT_OFFSET, loadTestFontId.length, loadTestFontId);
+  var CFF_CHECKSUM_OFFSET = 16;
+  var XXXX_VALUE = 0x58585858;
+  var checksum = int32(data, CFF_CHECKSUM_OFFSET);
+  for (i = 0, ii = loadTestFontId.length - 3; i < ii; i += 4) {
+    checksum = checksum - XXXX_VALUE + int32(loadTestFontId, i) | 0;
   }
-  setTimeout(isFontReady.bind(null, name, callback));
- }
- var loadTestFontId = 'lt' + Date.now() + this.loadTestFontId++;
- var data = this.loadTestFont;
- var COMMENT_OFFSET = 976;
- data = spliceString(data, COMMENT_OFFSET, loadTestFontId.length, loadTestFontId);
- var CFF_CHECKSUM_OFFSET = 16;
- var XXXX_VALUE = 0x58585858;
- var checksum = int32(data, CFF_CHECKSUM_OFFSET);
- for (i = 0, ii = loadTestFontId.length - 3; i < ii; i += 4) {
-  checksum = checksum - XXXX_VALUE + int32(loadTestFontId, i) | 0;
- }
- if (i < loadTestFontId.length) {
-  checksum = checksum - XXXX_VALUE + int32(loadTestFontId + 'XXX', i) | 0;
- }
- data = spliceString(data, CFF_CHECKSUM_OFFSET, 4, string32(checksum));
- var url = 'url(data:font/opentype;base64,' + btoa(data) + ');';
- var rule = '@font-face { font-family:"' + loadTestFontId + '";src:' + url + '}';
- this.insertRule(rule);
- var names = [];
- for (i = 0, ii = fonts.length; i < ii; i++) {
-  names.push(fonts[i].loadedName);
- }
- names.push(loadTestFontId);
- var div = document.createElement('div');
- div.setAttribute('style', 'visibility: hidden;' + 'width: 10px; height: 10px;' + 'position: absolute; top: 0px; left: 0px;');
- for (i = 0, ii = names.length; i < ii; ++i) {
-  var span = document.createElement('span');
-  span.textContent = 'Hi';
-  span.style.fontFamily = names[i];
-  div.appendChild(span);
- }
- document.body.appendChild(div);
- isFontReady(loadTestFontId, function () {
-  document.body.removeChild(div);
-  request.complete();
- });
+  if (i < loadTestFontId.length) {
+    checksum = checksum - XXXX_VALUE + int32(loadTestFontId + 'XXX', i) | 0;
+  }
+  data = spliceString(data, CFF_CHECKSUM_OFFSET, 4, string32(checksum));
+  var url = 'url(data:font/opentype;base64,' + btoa(data) + ');';
+  var rule = '@font-face { font-family:"' + loadTestFontId + '";src:' + url + '}';
+  this.insertRule(rule);
+  var names = [];
+  for (i = 0, ii = fonts.length; i < ii; i++) {
+    names.push(fonts[i].loadedName);
+  }
+  names.push(loadTestFontId);
+  var div = document.createElement('div');
+  div.setAttribute('style', 'visibility: hidden;' + 'width: 10px; height: 10px;' + 'position: absolute; top: 0px; left: 0px;');
+  for (i = 0, ii = names.length; i < ii; ++i) {
+    var span = document.createElement('span');
+    span.textContent = 'Hi';
+    span.style.fontFamily = names[i];
+    div.appendChild(span);
+  }
+  document.body.appendChild(div);
+  isFontReady(loadTestFontId, function () {
+    document.body.removeChild(div);
+    request.complete();
+  });
 };
 FontLoader.isFontLoadingAPISupported = typeof document !== 'undefined' && !!document.fonts;
 var isSyncFontLoadingSupported = function isSyncFontLoadingSupported() {
- if (typeof navigator === 'undefined') {
-  return true;
- }
- var supported = false;
- var m = /Mozilla\/5.0.*?rv:(\d+).*? Gecko/.exec(navigator.userAgent);
- if (m && m[1] >= 14) {
-  supported = true;
- }
- return supported;
+  if (typeof navigator === 'undefined') {
+    return true;
+  }
+  var supported = false;
+  var m = /Mozilla\/5.0.*?rv:(\d+).*? Gecko/.exec(navigator.userAgent);
+  if (m && m[1] >= 14) {
+    supported = true;
+  }
+  return supported;
 };
 Object.defineProperty(FontLoader, 'isSyncFontLoadingSupported', {
- get: function () {
-  return shadow(FontLoader, 'isSyncFontLoadingSupported', isSyncFontLoadingSupported());
- },
- enumerable: true,
- configurable: true
+  get: function () {
+    return shadow(FontLoader, 'isSyncFontLoadingSupported', isSyncFontLoadingSupported());
+  },
+  enumerable: true,
+  configurable: true
 });
 var IsEvalSupportedCached = {
- get value() {
-  return shadow(this, 'value', sharedUtil.isEvalSupported());
- }
+  get value() {
+    return shadow(this, 'value', sharedUtil.isEvalSupported());
+  }
 };
 var FontFaceObject = function FontFaceObjectClosure() {
- function FontFaceObject(translatedData, options) {
-  this.compiledGlyphs = Object.create(null);
-  for (var i in translatedData) {
-   this[i] = translatedData[i];
+  function FontFaceObject(translatedData, options) {
+    this.compiledGlyphs = Object.create(null);
+    for (var i in translatedData) {
+      this[i] = translatedData[i];
+    }
+    this.options = options;
   }
-  this.options = options;
- }
- FontFaceObject.prototype = {
-  createNativeFontFace: function FontFaceObject_createNativeFontFace() {
-   if (!this.data) {
-    return null;
-   }
-   if (this.options.disableFontFace) {
-    this.disableFontFace = true;
-    return null;
-   }
-   var nativeFontFace = new FontFace(this.loadedName, this.data, {});
-   if (this.options.fontRegistry) {
-    this.options.fontRegistry.registerFont(this);
-   }
-   return nativeFontFace;
-  },
-  createFontFaceRule: function FontFaceObject_createFontFaceRule() {
-   if (!this.data) {
-    return null;
-   }
-   if (this.options.disableFontFace) {
-    this.disableFontFace = true;
-    return null;
-   }
-   var data = bytesToString(new Uint8Array(this.data));
-   var fontName = this.loadedName;
-   var url = 'url(data:' + this.mimetype + ';base64,' + btoa(data) + ');';
-   var rule = '@font-face { font-family:"' + fontName + '";src:' + url + '}';
-   if (this.options.fontRegistry) {
-    this.options.fontRegistry.registerFont(this, url);
-   }
-   return rule;
-  },
-  getPathGenerator: function FontFaceObject_getPathGenerator(objs, character) {
-   if (!(character in this.compiledGlyphs)) {
-    var cmds = objs.get(this.loadedName + '_path_' + character);
-    var current, i, len;
-    if (this.options.isEvalSupported && IsEvalSupportedCached.value) {
-     var args, js = '';
-     for (i = 0, len = cmds.length; i < len; i++) {
-      current = cmds[i];
-      if (current.args !== undefined) {
-       args = current.args.join(',');
-      } else {
-       args = '';
+  FontFaceObject.prototype = {
+    createNativeFontFace: function FontFaceObject_createNativeFontFace() {
+      if (!this.data) {
+        return null;
       }
-      js += 'c.' + current.cmd + '(' + args + ');\n';
-     }
-     this.compiledGlyphs[character] = new Function('c', 'size', js);
-    } else {
-     this.compiledGlyphs[character] = function (c, size) {
-      for (i = 0, len = cmds.length; i < len; i++) {
-       current = cmds[i];
-       if (current.cmd === 'scale') {
-        current.args = [
-         size,
-         -size
-        ];
-       }
-       c[current.cmd].apply(c, current.args);
+      if (this.options.disableFontFace) {
+        this.disableFontFace = true;
+        return null;
+      }
+      var nativeFontFace = new FontFace(this.loadedName, this.data, {});
+      if (this.options.fontRegistry) {
+        this.options.fontRegistry.registerFont(this);
+      }
+      return nativeFontFace;
+    },
+    createFontFaceRule: function FontFaceObject_createFontFaceRule() {
+      if (!this.data) {
+        return null;
       }
-     };
+      if (this.options.disableFontFace) {
+        this.disableFontFace = true;
+        return null;
+      }
+      var data = bytesToString(new Uint8Array(this.data));
+      var fontName = this.loadedName;
+      var url = 'url(data:' + this.mimetype + ';base64,' + btoa(data) + ');';
+      var rule = '@font-face { font-family:"' + fontName + '";src:' + url + '}';
+      if (this.options.fontRegistry) {
+        this.options.fontRegistry.registerFont(this, url);
+      }
+      return rule;
+    },
+    getPathGenerator: function FontFaceObject_getPathGenerator(objs, character) {
+      if (!(character in this.compiledGlyphs)) {
+        var cmds = objs.get(this.loadedName + '_path_' + character);
+        var current, i, len;
+        if (this.options.isEvalSupported && IsEvalSupportedCached.value) {
+          var args,
+              js = '';
+          for (i = 0, len = cmds.length; i < len; i++) {
+            current = cmds[i];
+            if (current.args !== undefined) {
+              args = current.args.join(',');
+            } else {
+              args = '';
+            }
+            js += 'c.' + current.cmd + '(' + args + ');\n';
+          }
+          this.compiledGlyphs[character] = new Function('c', 'size', js);
+        } else {
+          this.compiledGlyphs[character] = function (c, size) {
+            for (i = 0, len = cmds.length; i < len; i++) {
+              current = cmds[i];
+              if (current.cmd === 'scale') {
+                current.args = [size, -size];
+              }
+              c[current.cmd].apply(c, current.args);
+            }
+          };
+        }
+      }
+      return this.compiledGlyphs[character];
     }
-   }
-   return this.compiledGlyphs[character];
-  }
- };
- return FontFaceObject;
+  };
+  return FontFaceObject;
 }();
 exports.FontFaceObject = FontFaceObject;
 exports.FontLoader = FontLoader;

+ 40 - 39
lib/display/global.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var displayDOMUtils = require('./dom_utils.js');
 var displayAPI = require('./api.js');
@@ -27,25 +28,25 @@ var LinkTarget = displayDOMUtils.LinkTarget;
 var DEFAULT_LINK_REL = displayDOMUtils.DEFAULT_LINK_REL;
 var isWorker = typeof window === 'undefined';
 if (!globalScope.PDFJS) {
- globalScope.PDFJS = {};
+  globalScope.PDFJS = {};
 }
 var PDFJS = globalScope.PDFJS;
-PDFJS.version = '1.7.389';
-PDFJS.build = '0423bb69';
+PDFJS.version = '1.7.391';
+PDFJS.build = 'cd5acf50';
 PDFJS.pdfBug = false;
 if (PDFJS.verbosity !== undefined) {
- sharedUtil.setVerbosityLevel(PDFJS.verbosity);
+  sharedUtil.setVerbosityLevel(PDFJS.verbosity);
 }
 delete PDFJS.verbosity;
 Object.defineProperty(PDFJS, 'verbosity', {
- get: function () {
-  return sharedUtil.getVerbosityLevel();
- },
- set: function (level) {
-  sharedUtil.setVerbosityLevel(level);
- },
- enumerable: true,
- configurable: true
+  get: function () {
+    return sharedUtil.getVerbosityLevel();
+  },
+  set: function (level) {
+    sharedUtil.setVerbosityLevel(level);
+  },
+  enumerable: true,
+  configurable: true
 });
 PDFJS.VERBOSITY_LEVELS = sharedUtil.VERBOSITY_LEVELS;
 PDFJS.OPS = sharedUtil.OPS;
@@ -54,14 +55,14 @@ PDFJS.isValidUrl = displayDOMUtils.isValidUrl;
 PDFJS.shadow = sharedUtil.shadow;
 PDFJS.createBlob = sharedUtil.createBlob;
 PDFJS.createObjectURL = function PDFJS_createObjectURL(data, contentType) {
- return sharedUtil.createObjectURL(data, contentType, PDFJS.disableCreateObjectURL);
+  return sharedUtil.createObjectURL(data, contentType, PDFJS.disableCreateObjectURL);
 };
 Object.defineProperty(PDFJS, 'isLittleEndian', {
- configurable: true,
- get: function PDFJS_isLittleEndian() {
-  var value = sharedUtil.isLittleEndian();
-  return sharedUtil.shadow(PDFJS, 'isLittleEndian', value);
- }
+  configurable: true,
+  get: function PDFJS_isLittleEndian() {
+    var value = sharedUtil.isLittleEndian();
+    return sharedUtil.shadow(PDFJS, 'isLittleEndian', value);
+  }
 });
 PDFJS.removeNullCharacters = sharedUtil.removeNullCharacters;
 PDFJS.PasswordResponses = sharedUtil.PasswordResponses;
@@ -94,34 +95,34 @@ PDFJS.isEvalSupported = PDFJS.isEvalSupported === undefined ? true : PDFJS.isEva
 var savedOpenExternalLinksInNewWindow = PDFJS.openExternalLinksInNewWindow;
 delete PDFJS.openExternalLinksInNewWindow;
 Object.defineProperty(PDFJS, 'openExternalLinksInNewWindow', {
- get: function () {
-  return PDFJS.externalLinkTarget === LinkTarget.BLANK;
- },
- set: function (value) {
-  if (value) {
-   deprecated('PDFJS.openExternalLinksInNewWindow, please use ' + '"PDFJS.externalLinkTarget = PDFJS.LinkTarget.BLANK" instead.');
-  }
-  if (PDFJS.externalLinkTarget !== LinkTarget.NONE) {
-   warn('PDFJS.externalLinkTarget is already initialized');
-   return;
-  }
-  PDFJS.externalLinkTarget = value ? LinkTarget.BLANK : LinkTarget.NONE;
- },
- enumerable: true,
- configurable: true
+  get: function () {
+    return PDFJS.externalLinkTarget === LinkTarget.BLANK;
+  },
+  set: function (value) {
+    if (value) {
+      deprecated('PDFJS.openExternalLinksInNewWindow, please use ' + '"PDFJS.externalLinkTarget = PDFJS.LinkTarget.BLANK" instead.');
+    }
+    if (PDFJS.externalLinkTarget !== LinkTarget.NONE) {
+      warn('PDFJS.externalLinkTarget is already initialized');
+      return;
+    }
+    PDFJS.externalLinkTarget = value ? LinkTarget.BLANK : LinkTarget.NONE;
+  },
+  enumerable: true,
+  configurable: true
 });
 if (savedOpenExternalLinksInNewWindow) {
- PDFJS.openExternalLinksInNewWindow = savedOpenExternalLinksInNewWindow;
+  PDFJS.openExternalLinksInNewWindow = savedOpenExternalLinksInNewWindow;
 }
 PDFJS.getDocument = displayAPI.getDocument;
 PDFJS.PDFDataRangeTransport = displayAPI.PDFDataRangeTransport;
 PDFJS.PDFWorker = displayAPI.PDFWorker;
 Object.defineProperty(PDFJS, 'hasCanvasTypedArrays', {
- configurable: true,
- get: function PDFJS_hasCanvasTypedArrays() {
-  var value = displayDOMUtils.hasCanvasTypedArrays();
-  return sharedUtil.shadow(PDFJS, 'hasCanvasTypedArrays', value);
- }
+  configurable: true,
+  get: function PDFJS_hasCanvasTypedArrays() {
+    var value = displayDOMUtils.hasCanvasTypedArrays();
+    return sharedUtil.shadow(PDFJS, 'hasCanvasTypedArrays', value);
+  }
 });
 PDFJS.CustomStyle = displayDOMUtils.CustomStyle;
 PDFJS.LinkTarget = LinkTarget;

+ 60 - 52
lib/display/metadata.js

@@ -13,67 +13,75 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var error = sharedUtil.error;
 function fixMetadata(meta) {
- return meta.replace(/>\\376\\377([^<]+)/g, function (all, codes) {
-  var bytes = codes.replace(/\\([0-3])([0-7])([0-7])/g, function (code, d1, d2, d3) {
-   return String.fromCharCode(d1 * 64 + d2 * 8 + d3 * 1);
+  return meta.replace(/>\\376\\377([^<]+)/g, function (all, codes) {
+    var bytes = codes.replace(/\\([0-3])([0-7])([0-7])/g, function (code, d1, d2, d3) {
+      return String.fromCharCode(d1 * 64 + d2 * 8 + d3 * 1);
+    });
+    var chars = '';
+    for (var i = 0; i < bytes.length; i += 2) {
+      var code = bytes.charCodeAt(i) * 256 + bytes.charCodeAt(i + 1);
+      chars += code >= 32 && code < 127 && code !== 60 && code !== 62 && code !== 38 ? String.fromCharCode(code) : '&#x' + (0x10000 + code).toString(16).substring(1) + ';';
+    }
+    return '>' + chars;
   });
-  var chars = '';
-  for (var i = 0; i < bytes.length; i += 2) {
-   var code = bytes.charCodeAt(i) * 256 + bytes.charCodeAt(i + 1);
-   chars += code >= 32 && code < 127 && code !== 60 && code !== 62 && code !== 38 ? String.fromCharCode(code) : '&#x' + (0x10000 + code).toString(16).substring(1) + ';';
-  }
-  return '>' + chars;
- });
 }
 function Metadata(meta) {
- if (typeof meta === 'string') {
-  meta = fixMetadata(meta);
-  var parser = new DOMParser();
-  meta = parser.parseFromString(meta, 'application/xml');
- } else if (!(meta instanceof Document)) {
-  error('Metadata: Invalid metadata object');
- }
- this.metaDocument = meta;
- this.metadata = Object.create(null);
- this.parse();
+  if (typeof meta === 'string') {
+    meta = fixMetadata(meta);
+    var parser = new DOMParser();
+    meta = parser.parseFromString(meta, 'application/xml');
+  } else if (!(meta instanceof Document)) {
+    error('Metadata: Invalid metadata object');
+  }
+  this.metaDocument = meta;
+  this.metadata = Object.create(null);
+  this.parse();
 }
 Metadata.prototype = {
- parse: function Metadata_parse() {
-  var doc = this.metaDocument;
-  var rdf = doc.documentElement;
-  if (rdf.nodeName.toLowerCase() !== 'rdf:rdf') {
-   rdf = rdf.firstChild;
-   while (rdf && rdf.nodeName.toLowerCase() !== 'rdf:rdf') {
-    rdf = rdf.nextSibling;
-   }
-  }
-  var nodeName = rdf ? rdf.nodeName.toLowerCase() : null;
-  if (!rdf || nodeName !== 'rdf:rdf' || !rdf.hasChildNodes()) {
-   return;
-  }
-  var children = rdf.childNodes, desc, entry, name, i, ii, length, iLength;
-  for (i = 0, length = children.length; i < length; i++) {
-   desc = children[i];
-   if (desc.nodeName.toLowerCase() !== 'rdf:description') {
-    continue;
-   }
-   for (ii = 0, iLength = desc.childNodes.length; ii < iLength; ii++) {
-    if (desc.childNodes[ii].nodeName.toLowerCase() !== '#text') {
-     entry = desc.childNodes[ii];
-     name = entry.nodeName.toLowerCase();
-     this.metadata[name] = entry.textContent.trim();
+  parse: function Metadata_parse() {
+    var doc = this.metaDocument;
+    var rdf = doc.documentElement;
+    if (rdf.nodeName.toLowerCase() !== 'rdf:rdf') {
+      rdf = rdf.firstChild;
+      while (rdf && rdf.nodeName.toLowerCase() !== 'rdf:rdf') {
+        rdf = rdf.nextSibling;
+      }
+    }
+    var nodeName = rdf ? rdf.nodeName.toLowerCase() : null;
+    if (!rdf || nodeName !== 'rdf:rdf' || !rdf.hasChildNodes()) {
+      return;
+    }
+    var children = rdf.childNodes,
+        desc,
+        entry,
+        name,
+        i,
+        ii,
+        length,
+        iLength;
+    for (i = 0, length = children.length; i < length; i++) {
+      desc = children[i];
+      if (desc.nodeName.toLowerCase() !== 'rdf:description') {
+        continue;
+      }
+      for (ii = 0, iLength = desc.childNodes.length; ii < iLength; ii++) {
+        if (desc.childNodes[ii].nodeName.toLowerCase() !== '#text') {
+          entry = desc.childNodes[ii];
+          name = entry.nodeName.toLowerCase();
+          this.metadata[name] = entry.textContent.trim();
+        }
+      }
     }
-   }
+  },
+  get: function Metadata_get(name) {
+    return this.metadata[name] || null;
+  },
+  has: function Metadata_has(name) {
+    return typeof this.metadata[name] !== 'undefined';
   }
- },
- get: function Metadata_get(name) {
-  return this.metadata[name] || null;
- },
- has: function Metadata_has(name) {
-  return typeof this.metadata[name] !== 'undefined';
- }
 };
 exports.Metadata = Metadata;

+ 336 - 359
lib/display/pattern_helper.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var displayWebGL = require('./webgl.js');
 var Util = sharedUtil.Util;
@@ -22,381 +23,357 @@ var error = sharedUtil.error;
 var WebGLUtils = displayWebGL.WebGLUtils;
 var ShadingIRs = {};
 ShadingIRs.RadialAxial = {
- fromIR: function RadialAxial_fromIR(raw) {
-  var type = raw[1];
-  var colorStops = raw[2];
-  var p0 = raw[3];
-  var p1 = raw[4];
-  var r0 = raw[5];
-  var r1 = raw[6];
-  return {
-   type: 'Pattern',
-   getPattern: function RadialAxial_getPattern(ctx) {
-    var grad;
-    if (type === 'axial') {
-     grad = ctx.createLinearGradient(p0[0], p0[1], p1[0], p1[1]);
-    } else if (type === 'radial') {
-     grad = ctx.createRadialGradient(p0[0], p0[1], r0, p1[0], p1[1], r1);
-    }
-    for (var i = 0, ii = colorStops.length; i < ii; ++i) {
-     var c = colorStops[i];
-     grad.addColorStop(c[0], c[1]);
-    }
-    return grad;
-   }
-  };
- }
+  fromIR: function RadialAxial_fromIR(raw) {
+    var type = raw[1];
+    var colorStops = raw[2];
+    var p0 = raw[3];
+    var p1 = raw[4];
+    var r0 = raw[5];
+    var r1 = raw[6];
+    return {
+      type: 'Pattern',
+      getPattern: function RadialAxial_getPattern(ctx) {
+        var grad;
+        if (type === 'axial') {
+          grad = ctx.createLinearGradient(p0[0], p0[1], p1[0], p1[1]);
+        } else if (type === 'radial') {
+          grad = ctx.createRadialGradient(p0[0], p0[1], r0, p1[0], p1[1], r1);
+        }
+        for (var i = 0, ii = colorStops.length; i < ii; ++i) {
+          var c = colorStops[i];
+          grad.addColorStop(c[0], c[1]);
+        }
+        return grad;
+      }
+    };
+  }
 };
 var createMeshCanvas = function createMeshCanvasClosure() {
- function drawTriangle(data, context, p1, p2, p3, c1, c2, c3) {
-  var coords = context.coords, colors = context.colors;
-  var bytes = data.data, rowSize = data.width * 4;
-  var tmp;
-  if (coords[p1 + 1] > coords[p2 + 1]) {
-   tmp = p1;
-   p1 = p2;
-   p2 = tmp;
-   tmp = c1;
-   c1 = c2;
-   c2 = tmp;
-  }
-  if (coords[p2 + 1] > coords[p3 + 1]) {
-   tmp = p2;
-   p2 = p3;
-   p3 = tmp;
-   tmp = c2;
-   c2 = c3;
-   c3 = tmp;
-  }
-  if (coords[p1 + 1] > coords[p2 + 1]) {
-   tmp = p1;
-   p1 = p2;
-   p2 = tmp;
-   tmp = c1;
-   c1 = c2;
-   c2 = tmp;
-  }
-  var x1 = (coords[p1] + context.offsetX) * context.scaleX;
-  var y1 = (coords[p1 + 1] + context.offsetY) * context.scaleY;
-  var x2 = (coords[p2] + context.offsetX) * context.scaleX;
-  var y2 = (coords[p2 + 1] + context.offsetY) * context.scaleY;
-  var x3 = (coords[p3] + context.offsetX) * context.scaleX;
-  var y3 = (coords[p3 + 1] + context.offsetY) * context.scaleY;
-  if (y1 >= y3) {
-   return;
-  }
-  var c1r = colors[c1], c1g = colors[c1 + 1], c1b = colors[c1 + 2];
-  var c2r = colors[c2], c2g = colors[c2 + 1], c2b = colors[c2 + 2];
-  var c3r = colors[c3], c3g = colors[c3 + 1], c3b = colors[c3 + 2];
-  var minY = Math.round(y1), maxY = Math.round(y3);
-  var xa, car, cag, cab;
-  var xb, cbr, cbg, cbb;
-  var k;
-  for (var y = minY; y <= maxY; y++) {
-   if (y < y2) {
-    k = y < y1 ? 0 : y1 === y2 ? 1 : (y1 - y) / (y1 - y2);
-    xa = x1 - (x1 - x2) * k;
-    car = c1r - (c1r - c2r) * k;
-    cag = c1g - (c1g - c2g) * k;
-    cab = c1b - (c1b - c2b) * k;
-   } else {
-    k = y > y3 ? 1 : y2 === y3 ? 0 : (y2 - y) / (y2 - y3);
-    xa = x2 - (x2 - x3) * k;
-    car = c2r - (c2r - c3r) * k;
-    cag = c2g - (c2g - c3g) * k;
-    cab = c2b - (c2b - c3b) * k;
-   }
-   k = y < y1 ? 0 : y > y3 ? 1 : (y1 - y) / (y1 - y3);
-   xb = x1 - (x1 - x3) * k;
-   cbr = c1r - (c1r - c3r) * k;
-   cbg = c1g - (c1g - c3g) * k;
-   cbb = c1b - (c1b - c3b) * k;
-   var x1_ = Math.round(Math.min(xa, xb));
-   var x2_ = Math.round(Math.max(xa, xb));
-   var j = rowSize * y + x1_ * 4;
-   for (var x = x1_; x <= x2_; x++) {
-    k = (xa - x) / (xa - xb);
-    k = k < 0 ? 0 : k > 1 ? 1 : k;
-    bytes[j++] = car - (car - cbr) * k | 0;
-    bytes[j++] = cag - (cag - cbg) * k | 0;
-    bytes[j++] = cab - (cab - cbb) * k | 0;
-    bytes[j++] = 255;
-   }
+  function drawTriangle(data, context, p1, p2, p3, c1, c2, c3) {
+    var coords = context.coords,
+        colors = context.colors;
+    var bytes = data.data,
+        rowSize = data.width * 4;
+    var tmp;
+    if (coords[p1 + 1] > coords[p2 + 1]) {
+      tmp = p1;
+      p1 = p2;
+      p2 = tmp;
+      tmp = c1;
+      c1 = c2;
+      c2 = tmp;
+    }
+    if (coords[p2 + 1] > coords[p3 + 1]) {
+      tmp = p2;
+      p2 = p3;
+      p3 = tmp;
+      tmp = c2;
+      c2 = c3;
+      c3 = tmp;
+    }
+    if (coords[p1 + 1] > coords[p2 + 1]) {
+      tmp = p1;
+      p1 = p2;
+      p2 = tmp;
+      tmp = c1;
+      c1 = c2;
+      c2 = tmp;
+    }
+    var x1 = (coords[p1] + context.offsetX) * context.scaleX;
+    var y1 = (coords[p1 + 1] + context.offsetY) * context.scaleY;
+    var x2 = (coords[p2] + context.offsetX) * context.scaleX;
+    var y2 = (coords[p2 + 1] + context.offsetY) * context.scaleY;
+    var x3 = (coords[p3] + context.offsetX) * context.scaleX;
+    var y3 = (coords[p3 + 1] + context.offsetY) * context.scaleY;
+    if (y1 >= y3) {
+      return;
+    }
+    var c1r = colors[c1],
+        c1g = colors[c1 + 1],
+        c1b = colors[c1 + 2];
+    var c2r = colors[c2],
+        c2g = colors[c2 + 1],
+        c2b = colors[c2 + 2];
+    var c3r = colors[c3],
+        c3g = colors[c3 + 1],
+        c3b = colors[c3 + 2];
+    var minY = Math.round(y1),
+        maxY = Math.round(y3);
+    var xa, car, cag, cab;
+    var xb, cbr, cbg, cbb;
+    var k;
+    for (var y = minY; y <= maxY; y++) {
+      if (y < y2) {
+        k = y < y1 ? 0 : y1 === y2 ? 1 : (y1 - y) / (y1 - y2);
+        xa = x1 - (x1 - x2) * k;
+        car = c1r - (c1r - c2r) * k;
+        cag = c1g - (c1g - c2g) * k;
+        cab = c1b - (c1b - c2b) * k;
+      } else {
+        k = y > y3 ? 1 : y2 === y3 ? 0 : (y2 - y) / (y2 - y3);
+        xa = x2 - (x2 - x3) * k;
+        car = c2r - (c2r - c3r) * k;
+        cag = c2g - (c2g - c3g) * k;
+        cab = c2b - (c2b - c3b) * k;
+      }
+      k = y < y1 ? 0 : y > y3 ? 1 : (y1 - y) / (y1 - y3);
+      xb = x1 - (x1 - x3) * k;
+      cbr = c1r - (c1r - c3r) * k;
+      cbg = c1g - (c1g - c3g) * k;
+      cbb = c1b - (c1b - c3b) * k;
+      var x1_ = Math.round(Math.min(xa, xb));
+      var x2_ = Math.round(Math.max(xa, xb));
+      var j = rowSize * y + x1_ * 4;
+      for (var x = x1_; x <= x2_; x++) {
+        k = (xa - x) / (xa - xb);
+        k = k < 0 ? 0 : k > 1 ? 1 : k;
+        bytes[j++] = car - (car - cbr) * k | 0;
+        bytes[j++] = cag - (cag - cbg) * k | 0;
+        bytes[j++] = cab - (cab - cbb) * k | 0;
+        bytes[j++] = 255;
+      }
+    }
   }
- }
- function drawFigure(data, figure, context) {
-  var ps = figure.coords;
-  var cs = figure.colors;
-  var i, ii;
-  switch (figure.type) {
-  case 'lattice':
-   var verticesPerRow = figure.verticesPerRow;
-   var rows = Math.floor(ps.length / verticesPerRow) - 1;
-   var cols = verticesPerRow - 1;
-   for (i = 0; i < rows; i++) {
-    var q = i * verticesPerRow;
-    for (var j = 0; j < cols; j++, q++) {
-     drawTriangle(data, context, ps[q], ps[q + 1], ps[q + verticesPerRow], cs[q], cs[q + 1], cs[q + verticesPerRow]);
-     drawTriangle(data, context, ps[q + verticesPerRow + 1], ps[q + 1], ps[q + verticesPerRow], cs[q + verticesPerRow + 1], cs[q + 1], cs[q + verticesPerRow]);
+  function drawFigure(data, figure, context) {
+    var ps = figure.coords;
+    var cs = figure.colors;
+    var i, ii;
+    switch (figure.type) {
+      case 'lattice':
+        var verticesPerRow = figure.verticesPerRow;
+        var rows = Math.floor(ps.length / verticesPerRow) - 1;
+        var cols = verticesPerRow - 1;
+        for (i = 0; i < rows; i++) {
+          var q = i * verticesPerRow;
+          for (var j = 0; j < cols; j++, q++) {
+            drawTriangle(data, context, ps[q], ps[q + 1], ps[q + verticesPerRow], cs[q], cs[q + 1], cs[q + verticesPerRow]);
+            drawTriangle(data, context, ps[q + verticesPerRow + 1], ps[q + 1], ps[q + verticesPerRow], cs[q + verticesPerRow + 1], cs[q + 1], cs[q + verticesPerRow]);
+          }
+        }
+        break;
+      case 'triangles':
+        for (i = 0, ii = ps.length; i < ii; i += 3) {
+          drawTriangle(data, context, ps[i], ps[i + 1], ps[i + 2], cs[i], cs[i + 1], cs[i + 2]);
+        }
+        break;
+      default:
+        error('illigal figure');
+        break;
     }
-   }
-   break;
-  case 'triangles':
-   for (i = 0, ii = ps.length; i < ii; i += 3) {
-    drawTriangle(data, context, ps[i], ps[i + 1], ps[i + 2], cs[i], cs[i + 1], cs[i + 2]);
-   }
-   break;
-  default:
-   error('illigal figure');
-   break;
   }
- }
- function createMeshCanvas(bounds, combinesScale, coords, colors, figures, backgroundColor, cachedCanvases) {
-  var EXPECTED_SCALE = 1.1;
-  var MAX_PATTERN_SIZE = 3000;
-  var BORDER_SIZE = 2;
-  var offsetX = Math.floor(bounds[0]);
-  var offsetY = Math.floor(bounds[1]);
-  var boundsWidth = Math.ceil(bounds[2]) - offsetX;
-  var boundsHeight = Math.ceil(bounds[3]) - offsetY;
-  var width = Math.min(Math.ceil(Math.abs(boundsWidth * combinesScale[0] * EXPECTED_SCALE)), MAX_PATTERN_SIZE);
-  var height = Math.min(Math.ceil(Math.abs(boundsHeight * combinesScale[1] * EXPECTED_SCALE)), MAX_PATTERN_SIZE);
-  var scaleX = boundsWidth / width;
-  var scaleY = boundsHeight / height;
-  var context = {
-   coords: coords,
-   colors: colors,
-   offsetX: -offsetX,
-   offsetY: -offsetY,
-   scaleX: 1 / scaleX,
-   scaleY: 1 / scaleY
-  };
-  var paddedWidth = width + BORDER_SIZE * 2;
-  var paddedHeight = height + BORDER_SIZE * 2;
-  var canvas, tmpCanvas, i, ii;
-  if (WebGLUtils.isEnabled) {
-   canvas = WebGLUtils.drawFigures(width, height, backgroundColor, figures, context);
-   tmpCanvas = cachedCanvases.getCanvas('mesh', paddedWidth, paddedHeight, false);
-   tmpCanvas.context.drawImage(canvas, BORDER_SIZE, BORDER_SIZE);
-   canvas = tmpCanvas.canvas;
-  } else {
-   tmpCanvas = cachedCanvases.getCanvas('mesh', paddedWidth, paddedHeight, false);
-   var tmpCtx = tmpCanvas.context;
-   var data = tmpCtx.createImageData(width, height);
-   if (backgroundColor) {
-    var bytes = data.data;
-    for (i = 0, ii = bytes.length; i < ii; i += 4) {
-     bytes[i] = backgroundColor[0];
-     bytes[i + 1] = backgroundColor[1];
-     bytes[i + 2] = backgroundColor[2];
-     bytes[i + 3] = 255;
+  function createMeshCanvas(bounds, combinesScale, coords, colors, figures, backgroundColor, cachedCanvases) {
+    var EXPECTED_SCALE = 1.1;
+    var MAX_PATTERN_SIZE = 3000;
+    var BORDER_SIZE = 2;
+    var offsetX = Math.floor(bounds[0]);
+    var offsetY = Math.floor(bounds[1]);
+    var boundsWidth = Math.ceil(bounds[2]) - offsetX;
+    var boundsHeight = Math.ceil(bounds[3]) - offsetY;
+    var width = Math.min(Math.ceil(Math.abs(boundsWidth * combinesScale[0] * EXPECTED_SCALE)), MAX_PATTERN_SIZE);
+    var height = Math.min(Math.ceil(Math.abs(boundsHeight * combinesScale[1] * EXPECTED_SCALE)), MAX_PATTERN_SIZE);
+    var scaleX = boundsWidth / width;
+    var scaleY = boundsHeight / height;
+    var context = {
+      coords: coords,
+      colors: colors,
+      offsetX: -offsetX,
+      offsetY: -offsetY,
+      scaleX: 1 / scaleX,
+      scaleY: 1 / scaleY
+    };
+    var paddedWidth = width + BORDER_SIZE * 2;
+    var paddedHeight = height + BORDER_SIZE * 2;
+    var canvas, tmpCanvas, i, ii;
+    if (WebGLUtils.isEnabled) {
+      canvas = WebGLUtils.drawFigures(width, height, backgroundColor, figures, context);
+      tmpCanvas = cachedCanvases.getCanvas('mesh', paddedWidth, paddedHeight, false);
+      tmpCanvas.context.drawImage(canvas, BORDER_SIZE, BORDER_SIZE);
+      canvas = tmpCanvas.canvas;
+    } else {
+      tmpCanvas = cachedCanvases.getCanvas('mesh', paddedWidth, paddedHeight, false);
+      var tmpCtx = tmpCanvas.context;
+      var data = tmpCtx.createImageData(width, height);
+      if (backgroundColor) {
+        var bytes = data.data;
+        for (i = 0, ii = bytes.length; i < ii; i += 4) {
+          bytes[i] = backgroundColor[0];
+          bytes[i + 1] = backgroundColor[1];
+          bytes[i + 2] = backgroundColor[2];
+          bytes[i + 3] = 255;
+        }
+      }
+      for (i = 0; i < figures.length; i++) {
+        drawFigure(data, figures[i], context);
+      }
+      tmpCtx.putImageData(data, BORDER_SIZE, BORDER_SIZE);
+      canvas = tmpCanvas.canvas;
     }
-   }
-   for (i = 0; i < figures.length; i++) {
-    drawFigure(data, figures[i], context);
-   }
-   tmpCtx.putImageData(data, BORDER_SIZE, BORDER_SIZE);
-   canvas = tmpCanvas.canvas;
+    return {
+      canvas: canvas,
+      offsetX: offsetX - BORDER_SIZE * scaleX,
+      offsetY: offsetY - BORDER_SIZE * scaleY,
+      scaleX: scaleX,
+      scaleY: scaleY
+    };
   }
-  return {
-   canvas: canvas,
-   offsetX: offsetX - BORDER_SIZE * scaleX,
-   offsetY: offsetY - BORDER_SIZE * scaleY,
-   scaleX: scaleX,
-   scaleY: scaleY
-  };
- }
- return createMeshCanvas;
+  return createMeshCanvas;
 }();
 ShadingIRs.Mesh = {
- fromIR: function Mesh_fromIR(raw) {
-  var coords = raw[2];
-  var colors = raw[3];
-  var figures = raw[4];
-  var bounds = raw[5];
-  var matrix = raw[6];
-  var background = raw[8];
-  return {
-   type: 'Pattern',
-   getPattern: function Mesh_getPattern(ctx, owner, shadingFill) {
-    var scale;
-    if (shadingFill) {
-     scale = Util.singularValueDecompose2dScale(ctx.mozCurrentTransform);
-    } else {
-     scale = Util.singularValueDecompose2dScale(owner.baseTransform);
-     if (matrix) {
-      var matrixScale = Util.singularValueDecompose2dScale(matrix);
-      scale = [
-       scale[0] * matrixScale[0],
-       scale[1] * matrixScale[1]
-      ];
-     }
-    }
-    var temporaryPatternCanvas = createMeshCanvas(bounds, scale, coords, colors, figures, shadingFill ? null : background, owner.cachedCanvases);
-    if (!shadingFill) {
-     ctx.setTransform.apply(ctx, owner.baseTransform);
-     if (matrix) {
-      ctx.transform.apply(ctx, matrix);
-     }
-    }
-    ctx.translate(temporaryPatternCanvas.offsetX, temporaryPatternCanvas.offsetY);
-    ctx.scale(temporaryPatternCanvas.scaleX, temporaryPatternCanvas.scaleY);
-    return ctx.createPattern(temporaryPatternCanvas.canvas, 'no-repeat');
-   }
-  };
- }
+  fromIR: function Mesh_fromIR(raw) {
+    var coords = raw[2];
+    var colors = raw[3];
+    var figures = raw[4];
+    var bounds = raw[5];
+    var matrix = raw[6];
+    var background = raw[8];
+    return {
+      type: 'Pattern',
+      getPattern: function Mesh_getPattern(ctx, owner, shadingFill) {
+        var scale;
+        if (shadingFill) {
+          scale = Util.singularValueDecompose2dScale(ctx.mozCurrentTransform);
+        } else {
+          scale = Util.singularValueDecompose2dScale(owner.baseTransform);
+          if (matrix) {
+            var matrixScale = Util.singularValueDecompose2dScale(matrix);
+            scale = [scale[0] * matrixScale[0], scale[1] * matrixScale[1]];
+          }
+        }
+        var temporaryPatternCanvas = createMeshCanvas(bounds, scale, coords, colors, figures, shadingFill ? null : background, owner.cachedCanvases);
+        if (!shadingFill) {
+          ctx.setTransform.apply(ctx, owner.baseTransform);
+          if (matrix) {
+            ctx.transform.apply(ctx, matrix);
+          }
+        }
+        ctx.translate(temporaryPatternCanvas.offsetX, temporaryPatternCanvas.offsetY);
+        ctx.scale(temporaryPatternCanvas.scaleX, temporaryPatternCanvas.scaleY);
+        return ctx.createPattern(temporaryPatternCanvas.canvas, 'no-repeat');
+      }
+    };
+  }
 };
 ShadingIRs.Dummy = {
- fromIR: function Dummy_fromIR() {
-  return {
-   type: 'Pattern',
-   getPattern: function Dummy_fromIR_getPattern() {
-    return 'hotpink';
-   }
-  };
- }
+  fromIR: function Dummy_fromIR() {
+    return {
+      type: 'Pattern',
+      getPattern: function Dummy_fromIR_getPattern() {
+        return 'hotpink';
+      }
+    };
+  }
 };
 function getShadingPatternFromIR(raw) {
- var shadingIR = ShadingIRs[raw[0]];
- if (!shadingIR) {
-  error('Unknown IR type: ' + raw[0]);
- }
- return shadingIR.fromIR(raw);
+  var shadingIR = ShadingIRs[raw[0]];
+  if (!shadingIR) {
+    error('Unknown IR type: ' + raw[0]);
+  }
+  return shadingIR.fromIR(raw);
 }
 var TilingPattern = function TilingPatternClosure() {
- var PaintType = {
-  COLORED: 1,
-  UNCOLORED: 2
- };
- var MAX_PATTERN_SIZE = 3000;
- function TilingPattern(IR, color, ctx, canvasGraphicsFactory, baseTransform) {
-  this.operatorList = IR[2];
-  this.matrix = IR[3] || [
-   1,
-   0,
-   0,
-   1,
-   0,
-   0
-  ];
-  this.bbox = Util.normalizeRect(IR[4]);
-  this.xstep = IR[5];
-  this.ystep = IR[6];
-  this.paintType = IR[7];
-  this.tilingType = IR[8];
-  this.color = color;
-  this.canvasGraphicsFactory = canvasGraphicsFactory;
-  this.baseTransform = baseTransform;
-  this.type = 'Pattern';
-  this.ctx = ctx;
- }
- TilingPattern.prototype = {
-  createPatternCanvas: function TilinPattern_createPatternCanvas(owner) {
-   var operatorList = this.operatorList;
-   var bbox = this.bbox;
-   var xstep = this.xstep;
-   var ystep = this.ystep;
-   var paintType = this.paintType;
-   var tilingType = this.tilingType;
-   var color = this.color;
-   var canvasGraphicsFactory = this.canvasGraphicsFactory;
-   info('TilingType: ' + tilingType);
-   var x0 = bbox[0], y0 = bbox[1], x1 = bbox[2], y1 = bbox[3];
-   var topLeft = [
-    x0,
-    y0
-   ];
-   var botRight = [
-    x0 + xstep,
-    y0 + ystep
-   ];
-   var width = botRight[0] - topLeft[0];
-   var height = botRight[1] - topLeft[1];
-   var matrixScale = Util.singularValueDecompose2dScale(this.matrix);
-   var curMatrixScale = Util.singularValueDecompose2dScale(this.baseTransform);
-   var combinedScale = [
-    matrixScale[0] * curMatrixScale[0],
-    matrixScale[1] * curMatrixScale[1]
-   ];
-   width = Math.min(Math.ceil(Math.abs(width * combinedScale[0])), MAX_PATTERN_SIZE);
-   height = Math.min(Math.ceil(Math.abs(height * combinedScale[1])), MAX_PATTERN_SIZE);
-   var tmpCanvas = owner.cachedCanvases.getCanvas('pattern', width, height, true);
-   var tmpCtx = tmpCanvas.context;
-   var graphics = canvasGraphicsFactory.createCanvasGraphics(tmpCtx);
-   graphics.groupLevel = owner.groupLevel;
-   this.setFillAndStrokeStyleToContext(tmpCtx, paintType, color);
-   this.setScale(width, height, xstep, ystep);
-   this.transformToScale(graphics);
-   var tmpTranslate = [
-    1,
-    0,
-    0,
-    1,
-    -topLeft[0],
-    -topLeft[1]
-   ];
-   graphics.transform.apply(graphics, tmpTranslate);
-   this.clipBbox(graphics, bbox, x0, y0, x1, y1);
-   graphics.executeOperatorList(operatorList);
-   return tmpCanvas.canvas;
-  },
-  setScale: function TilingPattern_setScale(width, height, xstep, ystep) {
-   this.scale = [
-    width / xstep,
-    height / ystep
-   ];
-  },
-  transformToScale: function TilingPattern_transformToScale(graphics) {
-   var scale = this.scale;
-   var tmpScale = [
-    scale[0],
-    0,
-    0,
-    scale[1],
-    0,
-    0
-   ];
-   graphics.transform.apply(graphics, tmpScale);
-  },
-  scaleToContext: function TilingPattern_scaleToContext() {
-   var scale = this.scale;
-   this.ctx.scale(1 / scale[0], 1 / scale[1]);
-  },
-  clipBbox: function clipBbox(graphics, bbox, x0, y0, x1, y1) {
-   if (isArray(bbox) && bbox.length === 4) {
-    var bboxWidth = x1 - x0;
-    var bboxHeight = y1 - y0;
-    graphics.ctx.rect(x0, y0, bboxWidth, bboxHeight);
-    graphics.clip();
-    graphics.endPath();
-   }
-  },
-  setFillAndStrokeStyleToContext: function setFillAndStrokeStyleToContext(context, paintType, color) {
-   switch (paintType) {
-   case PaintType.COLORED:
-    var ctx = this.ctx;
-    context.fillStyle = ctx.fillStyle;
-    context.strokeStyle = ctx.strokeStyle;
-    break;
-   case PaintType.UNCOLORED:
-    var cssColor = Util.makeCssRgb(color[0], color[1], color[2]);
-    context.fillStyle = cssColor;
-    context.strokeStyle = cssColor;
-    break;
-   default:
-    error('Unsupported paint type: ' + paintType);
-   }
-  },
-  getPattern: function TilingPattern_getPattern(ctx, owner) {
-   var temporaryPatternCanvas = this.createPatternCanvas(owner);
-   ctx = this.ctx;
-   ctx.setTransform.apply(ctx, this.baseTransform);
-   ctx.transform.apply(ctx, this.matrix);
-   this.scaleToContext();
-   return ctx.createPattern(temporaryPatternCanvas, 'repeat');
+  var PaintType = {
+    COLORED: 1,
+    UNCOLORED: 2
+  };
+  var MAX_PATTERN_SIZE = 3000;
+  function TilingPattern(IR, color, ctx, canvasGraphicsFactory, baseTransform) {
+    this.operatorList = IR[2];
+    this.matrix = IR[3] || [1, 0, 0, 1, 0, 0];
+    this.bbox = Util.normalizeRect(IR[4]);
+    this.xstep = IR[5];
+    this.ystep = IR[6];
+    this.paintType = IR[7];
+    this.tilingType = IR[8];
+    this.color = color;
+    this.canvasGraphicsFactory = canvasGraphicsFactory;
+    this.baseTransform = baseTransform;
+    this.type = 'Pattern';
+    this.ctx = ctx;
   }
- };
- return TilingPattern;
+  TilingPattern.prototype = {
+    createPatternCanvas: function TilinPattern_createPatternCanvas(owner) {
+      var operatorList = this.operatorList;
+      var bbox = this.bbox;
+      var xstep = this.xstep;
+      var ystep = this.ystep;
+      var paintType = this.paintType;
+      var tilingType = this.tilingType;
+      var color = this.color;
+      var canvasGraphicsFactory = this.canvasGraphicsFactory;
+      info('TilingType: ' + tilingType);
+      var x0 = bbox[0],
+          y0 = bbox[1],
+          x1 = bbox[2],
+          y1 = bbox[3];
+      var topLeft = [x0, y0];
+      var botRight = [x0 + xstep, y0 + ystep];
+      var width = botRight[0] - topLeft[0];
+      var height = botRight[1] - topLeft[1];
+      var matrixScale = Util.singularValueDecompose2dScale(this.matrix);
+      var curMatrixScale = Util.singularValueDecompose2dScale(this.baseTransform);
+      var combinedScale = [matrixScale[0] * curMatrixScale[0], matrixScale[1] * curMatrixScale[1]];
+      width = Math.min(Math.ceil(Math.abs(width * combinedScale[0])), MAX_PATTERN_SIZE);
+      height = Math.min(Math.ceil(Math.abs(height * combinedScale[1])), MAX_PATTERN_SIZE);
+      var tmpCanvas = owner.cachedCanvases.getCanvas('pattern', width, height, true);
+      var tmpCtx = tmpCanvas.context;
+      var graphics = canvasGraphicsFactory.createCanvasGraphics(tmpCtx);
+      graphics.groupLevel = owner.groupLevel;
+      this.setFillAndStrokeStyleToContext(tmpCtx, paintType, color);
+      this.setScale(width, height, xstep, ystep);
+      this.transformToScale(graphics);
+      var tmpTranslate = [1, 0, 0, 1, -topLeft[0], -topLeft[1]];
+      graphics.transform.apply(graphics, tmpTranslate);
+      this.clipBbox(graphics, bbox, x0, y0, x1, y1);
+      graphics.executeOperatorList(operatorList);
+      return tmpCanvas.canvas;
+    },
+    setScale: function TilingPattern_setScale(width, height, xstep, ystep) {
+      this.scale = [width / xstep, height / ystep];
+    },
+    transformToScale: function TilingPattern_transformToScale(graphics) {
+      var scale = this.scale;
+      var tmpScale = [scale[0], 0, 0, scale[1], 0, 0];
+      graphics.transform.apply(graphics, tmpScale);
+    },
+    scaleToContext: function TilingPattern_scaleToContext() {
+      var scale = this.scale;
+      this.ctx.scale(1 / scale[0], 1 / scale[1]);
+    },
+    clipBbox: function clipBbox(graphics, bbox, x0, y0, x1, y1) {
+      if (isArray(bbox) && bbox.length === 4) {
+        var bboxWidth = x1 - x0;
+        var bboxHeight = y1 - y0;
+        graphics.ctx.rect(x0, y0, bboxWidth, bboxHeight);
+        graphics.clip();
+        graphics.endPath();
+      }
+    },
+    setFillAndStrokeStyleToContext: function setFillAndStrokeStyleToContext(context, paintType, color) {
+      switch (paintType) {
+        case PaintType.COLORED:
+          var ctx = this.ctx;
+          context.fillStyle = ctx.fillStyle;
+          context.strokeStyle = ctx.strokeStyle;
+          break;
+        case PaintType.UNCOLORED:
+          var cssColor = Util.makeCssRgb(color[0], color[1], color[2]);
+          context.fillStyle = cssColor;
+          context.strokeStyle = cssColor;
+          break;
+        default:
+          error('Unsupported paint type: ' + paintType);
+      }
+    },
+    getPattern: function TilingPattern_getPattern(ctx, owner) {
+      var temporaryPatternCanvas = this.createPatternCanvas(owner);
+      ctx = this.ctx;
+      ctx.setTransform.apply(ctx, this.baseTransform);
+      ctx.transform.apply(ctx, this.matrix);
+      this.scaleToContext();
+      return ctx.createPattern(temporaryPatternCanvas, 'repeat');
+    }
+  };
+  return TilingPattern;
 }();
 exports.getShadingPatternFromIR = getShadingPatternFromIR;
 exports.TilingPattern = TilingPattern;

+ 922 - 966
lib/display/svg.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var FONT_IDENTITY_MATRIX = sharedUtil.FONT_IDENTITY_MATRIX;
 var IDENTITY_MATRIX = sharedUtil.IDENTITY_MATRIX;
@@ -24,992 +25,947 @@ var isArray = sharedUtil.isArray;
 var warn = sharedUtil.warn;
 var createObjectURL = sharedUtil.createObjectURL;
 var SVG_DEFAULTS = {
- fontStyle: 'normal',
- fontWeight: 'normal',
- fillColor: '#000000'
+  fontStyle: 'normal',
+  fontWeight: 'normal',
+  fillColor: '#000000'
 };
 var convertImgDataToPng = function convertImgDataToPngClosure() {
- var PNG_HEADER = new Uint8Array([
-  0x89,
-  0x50,
-  0x4e,
-  0x47,
-  0x0d,
-  0x0a,
-  0x1a,
-  0x0a
- ]);
- var CHUNK_WRAPPER_SIZE = 12;
- var crcTable = new Int32Array(256);
- for (var i = 0; i < 256; i++) {
-  var c = i;
-  for (var h = 0; h < 8; h++) {
-   if (c & 1) {
-    c = 0xedB88320 ^ c >> 1 & 0x7fffffff;
-   } else {
-    c = c >> 1 & 0x7fffffff;
-   }
-  }
-  crcTable[i] = c;
- }
- function crc32(data, start, end) {
-  var crc = -1;
-  for (var i = start; i < end; i++) {
-   var a = (crc ^ data[i]) & 0xff;
-   var b = crcTable[a];
-   crc = crc >>> 8 ^ b;
-  }
-  return crc ^ -1;
- }
- function writePngChunk(type, body, data, offset) {
-  var p = offset;
-  var len = body.length;
-  data[p] = len >> 24 & 0xff;
-  data[p + 1] = len >> 16 & 0xff;
-  data[p + 2] = len >> 8 & 0xff;
-  data[p + 3] = len & 0xff;
-  p += 4;
-  data[p] = type.charCodeAt(0) & 0xff;
-  data[p + 1] = type.charCodeAt(1) & 0xff;
-  data[p + 2] = type.charCodeAt(2) & 0xff;
-  data[p + 3] = type.charCodeAt(3) & 0xff;
-  p += 4;
-  data.set(body, p);
-  p += body.length;
-  var crc = crc32(data, offset + 4, p);
-  data[p] = crc >> 24 & 0xff;
-  data[p + 1] = crc >> 16 & 0xff;
-  data[p + 2] = crc >> 8 & 0xff;
-  data[p + 3] = crc & 0xff;
- }
- function adler32(data, start, end) {
-  var a = 1;
-  var b = 0;
-  for (var i = start; i < end; ++i) {
-   a = (a + (data[i] & 0xff)) % 65521;
-   b = (b + a) % 65521;
+  var PNG_HEADER = new Uint8Array([0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a]);
+  var CHUNK_WRAPPER_SIZE = 12;
+  var crcTable = new Int32Array(256);
+  for (var i = 0; i < 256; i++) {
+    var c = i;
+    for (var h = 0; h < 8; h++) {
+      if (c & 1) {
+        c = 0xedB88320 ^ c >> 1 & 0x7fffffff;
+      } else {
+        c = c >> 1 & 0x7fffffff;
+      }
+    }
+    crcTable[i] = c;
   }
-  return b << 16 | a;
- }
- function encode(imgData, kind, forceDataSchema) {
-  var width = imgData.width;
-  var height = imgData.height;
-  var bitDepth, colorType, lineSize;
-  var bytes = imgData.data;
-  switch (kind) {
-  case ImageKind.GRAYSCALE_1BPP:
-   colorType = 0;
-   bitDepth = 1;
-   lineSize = width + 7 >> 3;
-   break;
-  case ImageKind.RGB_24BPP:
-   colorType = 2;
-   bitDepth = 8;
-   lineSize = width * 3;
-   break;
-  case ImageKind.RGBA_32BPP:
-   colorType = 6;
-   bitDepth = 8;
-   lineSize = width * 4;
-   break;
-  default:
-   throw new Error('invalid format');
+  function crc32(data, start, end) {
+    var crc = -1;
+    for (var i = start; i < end; i++) {
+      var a = (crc ^ data[i]) & 0xff;
+      var b = crcTable[a];
+      crc = crc >>> 8 ^ b;
+    }
+    return crc ^ -1;
   }
-  var literals = new Uint8Array((1 + lineSize) * height);
-  var offsetLiterals = 0, offsetBytes = 0;
-  var y, i;
-  for (y = 0; y < height; ++y) {
-   literals[offsetLiterals++] = 0;
-   literals.set(bytes.subarray(offsetBytes, offsetBytes + lineSize), offsetLiterals);
-   offsetBytes += lineSize;
-   offsetLiterals += lineSize;
+  function writePngChunk(type, body, data, offset) {
+    var p = offset;
+    var len = body.length;
+    data[p] = len >> 24 & 0xff;
+    data[p + 1] = len >> 16 & 0xff;
+    data[p + 2] = len >> 8 & 0xff;
+    data[p + 3] = len & 0xff;
+    p += 4;
+    data[p] = type.charCodeAt(0) & 0xff;
+    data[p + 1] = type.charCodeAt(1) & 0xff;
+    data[p + 2] = type.charCodeAt(2) & 0xff;
+    data[p + 3] = type.charCodeAt(3) & 0xff;
+    p += 4;
+    data.set(body, p);
+    p += body.length;
+    var crc = crc32(data, offset + 4, p);
+    data[p] = crc >> 24 & 0xff;
+    data[p + 1] = crc >> 16 & 0xff;
+    data[p + 2] = crc >> 8 & 0xff;
+    data[p + 3] = crc & 0xff;
   }
-  if (kind === ImageKind.GRAYSCALE_1BPP) {
-   offsetLiterals = 0;
-   for (y = 0; y < height; y++) {
-    offsetLiterals++;
-    for (i = 0; i < lineSize; i++) {
-     literals[offsetLiterals++] ^= 0xFF;
+  function adler32(data, start, end) {
+    var a = 1;
+    var b = 0;
+    for (var i = start; i < end; ++i) {
+      a = (a + (data[i] & 0xff)) % 65521;
+      b = (b + a) % 65521;
     }
-   }
+    return b << 16 | a;
   }
-  var ihdr = new Uint8Array([
-   width >> 24 & 0xff,
-   width >> 16 & 0xff,
-   width >> 8 & 0xff,
-   width & 0xff,
-   height >> 24 & 0xff,
-   height >> 16 & 0xff,
-   height >> 8 & 0xff,
-   height & 0xff,
-   bitDepth,
-   colorType,
-   0x00,
-   0x00,
-   0x00
-  ]);
-  var len = literals.length;
-  var maxBlockLength = 0xFFFF;
-  var deflateBlocks = Math.ceil(len / maxBlockLength);
-  var idat = new Uint8Array(2 + len + deflateBlocks * 5 + 4);
-  var pi = 0;
-  idat[pi++] = 0x78;
-  idat[pi++] = 0x9c;
-  var pos = 0;
-  while (len > maxBlockLength) {
-   idat[pi++] = 0x00;
-   idat[pi++] = 0xff;
-   idat[pi++] = 0xff;
-   idat[pi++] = 0x00;
-   idat[pi++] = 0x00;
-   idat.set(literals.subarray(pos, pos + maxBlockLength), pi);
-   pi += maxBlockLength;
-   pos += maxBlockLength;
-   len -= maxBlockLength;
+  function encode(imgData, kind, forceDataSchema) {
+    var width = imgData.width;
+    var height = imgData.height;
+    var bitDepth, colorType, lineSize;
+    var bytes = imgData.data;
+    switch (kind) {
+      case ImageKind.GRAYSCALE_1BPP:
+        colorType = 0;
+        bitDepth = 1;
+        lineSize = width + 7 >> 3;
+        break;
+      case ImageKind.RGB_24BPP:
+        colorType = 2;
+        bitDepth = 8;
+        lineSize = width * 3;
+        break;
+      case ImageKind.RGBA_32BPP:
+        colorType = 6;
+        bitDepth = 8;
+        lineSize = width * 4;
+        break;
+      default:
+        throw new Error('invalid format');
+    }
+    var literals = new Uint8Array((1 + lineSize) * height);
+    var offsetLiterals = 0,
+        offsetBytes = 0;
+    var y, i;
+    for (y = 0; y < height; ++y) {
+      literals[offsetLiterals++] = 0;
+      literals.set(bytes.subarray(offsetBytes, offsetBytes + lineSize), offsetLiterals);
+      offsetBytes += lineSize;
+      offsetLiterals += lineSize;
+    }
+    if (kind === ImageKind.GRAYSCALE_1BPP) {
+      offsetLiterals = 0;
+      for (y = 0; y < height; y++) {
+        offsetLiterals++;
+        for (i = 0; i < lineSize; i++) {
+          literals[offsetLiterals++] ^= 0xFF;
+        }
+      }
+    }
+    var ihdr = new Uint8Array([width >> 24 & 0xff, width >> 16 & 0xff, width >> 8 & 0xff, width & 0xff, height >> 24 & 0xff, height >> 16 & 0xff, height >> 8 & 0xff, height & 0xff, bitDepth, colorType, 0x00, 0x00, 0x00]);
+    var len = literals.length;
+    var maxBlockLength = 0xFFFF;
+    var deflateBlocks = Math.ceil(len / maxBlockLength);
+    var idat = new Uint8Array(2 + len + deflateBlocks * 5 + 4);
+    var pi = 0;
+    idat[pi++] = 0x78;
+    idat[pi++] = 0x9c;
+    var pos = 0;
+    while (len > maxBlockLength) {
+      idat[pi++] = 0x00;
+      idat[pi++] = 0xff;
+      idat[pi++] = 0xff;
+      idat[pi++] = 0x00;
+      idat[pi++] = 0x00;
+      idat.set(literals.subarray(pos, pos + maxBlockLength), pi);
+      pi += maxBlockLength;
+      pos += maxBlockLength;
+      len -= maxBlockLength;
+    }
+    idat[pi++] = 0x01;
+    idat[pi++] = len & 0xff;
+    idat[pi++] = len >> 8 & 0xff;
+    idat[pi++] = ~len & 0xffff & 0xff;
+    idat[pi++] = (~len & 0xffff) >> 8 & 0xff;
+    idat.set(literals.subarray(pos), pi);
+    pi += literals.length - pos;
+    var adler = adler32(literals, 0, literals.length);
+    idat[pi++] = adler >> 24 & 0xff;
+    idat[pi++] = adler >> 16 & 0xff;
+    idat[pi++] = adler >> 8 & 0xff;
+    idat[pi++] = adler & 0xff;
+    var pngLength = PNG_HEADER.length + CHUNK_WRAPPER_SIZE * 3 + ihdr.length + idat.length;
+    var data = new Uint8Array(pngLength);
+    var offset = 0;
+    data.set(PNG_HEADER, offset);
+    offset += PNG_HEADER.length;
+    writePngChunk('IHDR', ihdr, data, offset);
+    offset += CHUNK_WRAPPER_SIZE + ihdr.length;
+    writePngChunk('IDATA', idat, data, offset);
+    offset += CHUNK_WRAPPER_SIZE + idat.length;
+    writePngChunk('IEND', new Uint8Array(0), data, offset);
+    return createObjectURL(data, 'image/png', forceDataSchema);
   }
-  idat[pi++] = 0x01;
-  idat[pi++] = len & 0xff;
-  idat[pi++] = len >> 8 & 0xff;
-  idat[pi++] = ~len & 0xffff & 0xff;
-  idat[pi++] = (~len & 0xffff) >> 8 & 0xff;
-  idat.set(literals.subarray(pos), pi);
-  pi += literals.length - pos;
-  var adler = adler32(literals, 0, literals.length);
-  idat[pi++] = adler >> 24 & 0xff;
-  idat[pi++] = adler >> 16 & 0xff;
-  idat[pi++] = adler >> 8 & 0xff;
-  idat[pi++] = adler & 0xff;
-  var pngLength = PNG_HEADER.length + CHUNK_WRAPPER_SIZE * 3 + ihdr.length + idat.length;
-  var data = new Uint8Array(pngLength);
-  var offset = 0;
-  data.set(PNG_HEADER, offset);
-  offset += PNG_HEADER.length;
-  writePngChunk('IHDR', ihdr, data, offset);
-  offset += CHUNK_WRAPPER_SIZE + ihdr.length;
-  writePngChunk('IDATA', idat, data, offset);
-  offset += CHUNK_WRAPPER_SIZE + idat.length;
-  writePngChunk('IEND', new Uint8Array(0), data, offset);
-  return createObjectURL(data, 'image/png', forceDataSchema);
- }
- return function convertImgDataToPng(imgData, forceDataSchema) {
-  var kind = imgData.kind === undefined ? ImageKind.GRAYSCALE_1BPP : imgData.kind;
-  return encode(imgData, kind, forceDataSchema);
- };
+  return function convertImgDataToPng(imgData, forceDataSchema) {
+    var kind = imgData.kind === undefined ? ImageKind.GRAYSCALE_1BPP : imgData.kind;
+    return encode(imgData, kind, forceDataSchema);
+  };
 }();
 var SVGExtraState = function SVGExtraStateClosure() {
- function SVGExtraState() {
-  this.fontSizeScale = 1;
-  this.fontWeight = SVG_DEFAULTS.fontWeight;
-  this.fontSize = 0;
-  this.textMatrix = IDENTITY_MATRIX;
-  this.fontMatrix = FONT_IDENTITY_MATRIX;
-  this.leading = 0;
-  this.x = 0;
-  this.y = 0;
-  this.lineX = 0;
-  this.lineY = 0;
-  this.charSpacing = 0;
-  this.wordSpacing = 0;
-  this.textHScale = 1;
-  this.textRise = 0;
-  this.fillColor = SVG_DEFAULTS.fillColor;
-  this.strokeColor = '#000000';
-  this.fillAlpha = 1;
-  this.strokeAlpha = 1;
-  this.lineWidth = 1;
-  this.lineJoin = '';
-  this.lineCap = '';
-  this.miterLimit = 0;
-  this.dashArray = [];
-  this.dashPhase = 0;
-  this.dependencies = [];
-  this.activeClipUrl = null;
-  this.clipGroup = null;
-  this.maskId = '';
- }
- SVGExtraState.prototype = {
-  clone: function SVGExtraState_clone() {
-   return Object.create(this);
-  },
-  setCurrentPoint: function SVGExtraState_setCurrentPoint(x, y) {
-   this.x = x;
-   this.y = y;
+  function SVGExtraState() {
+    this.fontSizeScale = 1;
+    this.fontWeight = SVG_DEFAULTS.fontWeight;
+    this.fontSize = 0;
+    this.textMatrix = IDENTITY_MATRIX;
+    this.fontMatrix = FONT_IDENTITY_MATRIX;
+    this.leading = 0;
+    this.x = 0;
+    this.y = 0;
+    this.lineX = 0;
+    this.lineY = 0;
+    this.charSpacing = 0;
+    this.wordSpacing = 0;
+    this.textHScale = 1;
+    this.textRise = 0;
+    this.fillColor = SVG_DEFAULTS.fillColor;
+    this.strokeColor = '#000000';
+    this.fillAlpha = 1;
+    this.strokeAlpha = 1;
+    this.lineWidth = 1;
+    this.lineJoin = '';
+    this.lineCap = '';
+    this.miterLimit = 0;
+    this.dashArray = [];
+    this.dashPhase = 0;
+    this.dependencies = [];
+    this.activeClipUrl = null;
+    this.clipGroup = null;
+    this.maskId = '';
   }
- };
- return SVGExtraState;
+  SVGExtraState.prototype = {
+    clone: function SVGExtraState_clone() {
+      return Object.create(this);
+    },
+    setCurrentPoint: function SVGExtraState_setCurrentPoint(x, y) {
+      this.x = x;
+      this.y = y;
+    }
+  };
+  return SVGExtraState;
 }();
 var SVGGraphics = function SVGGraphicsClosure() {
- function opListToTree(opList) {
-  var opTree = [];
-  var tmp = [];
-  var opListLen = opList.length;
-  for (var x = 0; x < opListLen; x++) {
-   if (opList[x].fn === 'save') {
-    opTree.push({
-     'fnId': 92,
-     'fn': 'group',
-     'items': []
-    });
-    tmp.push(opTree);
-    opTree = opTree[opTree.length - 1].items;
-    continue;
-   }
-   if (opList[x].fn === 'restore') {
-    opTree = tmp.pop();
-   } else {
-    opTree.push(opList[x]);
-   }
-  }
-  return opTree;
- }
- function pf(value) {
-  if (value === (value | 0)) {
-   return value.toString();
-  }
-  var s = value.toFixed(10);
-  var i = s.length - 1;
-  if (s[i] !== '0') {
-   return s;
-  }
-  do {
-   i--;
-  } while (s[i] === '0');
-  return s.substr(0, s[i] === '.' ? i : i + 1);
- }
- function pm(m) {
-  if (m[4] === 0 && m[5] === 0) {
-   if (m[1] === 0 && m[2] === 0) {
-    if (m[0] === 1 && m[3] === 1) {
-     return '';
-    }
-    return 'scale(' + pf(m[0]) + ' ' + pf(m[3]) + ')';
-   }
-   if (m[0] === m[3] && m[1] === -m[2]) {
-    var a = Math.acos(m[0]) * 180 / Math.PI;
-    return 'rotate(' + pf(a) + ')';
-   }
-  } else {
-   if (m[0] === 1 && m[1] === 0 && m[2] === 0 && m[3] === 1) {
-    return 'translate(' + pf(m[4]) + ' ' + pf(m[5]) + ')';
-   }
-  }
-  return 'matrix(' + pf(m[0]) + ' ' + pf(m[1]) + ' ' + pf(m[2]) + ' ' + pf(m[3]) + ' ' + pf(m[4]) + ' ' + pf(m[5]) + ')';
- }
- function SVGGraphics(commonObjs, objs, forceDataSchema) {
-  this.current = new SVGExtraState();
-  this.transformMatrix = IDENTITY_MATRIX;
-  this.transformStack = [];
-  this.extraStack = [];
-  this.commonObjs = commonObjs;
-  this.objs = objs;
-  this.pendingEOFill = false;
-  this.embedFonts = false;
-  this.embeddedFonts = Object.create(null);
-  this.cssStyle = null;
-  this.forceDataSchema = !!forceDataSchema;
- }
- var NS = 'http://www.w3.org/2000/svg';
- var XML_NS = 'http://www.w3.org/XML/1998/namespace';
- var XLINK_NS = 'http://www.w3.org/1999/xlink';
- var LINE_CAP_STYLES = [
-  'butt',
-  'round',
-  'square'
- ];
- var LINE_JOIN_STYLES = [
-  'miter',
-  'round',
-  'bevel'
- ];
- var clipCount = 0;
- var maskCount = 0;
- SVGGraphics.prototype = {
-  save: function SVGGraphics_save() {
-   this.transformStack.push(this.transformMatrix);
-   var old = this.current;
-   this.extraStack.push(old);
-   this.current = old.clone();
-  },
-  restore: function SVGGraphics_restore() {
-   this.transformMatrix = this.transformStack.pop();
-   this.current = this.extraStack.pop();
-   this.tgrp = null;
-  },
-  group: function SVGGraphics_group(items) {
-   this.save();
-   this.executeOpTree(items);
-   this.restore();
-  },
-  loadDependencies: function SVGGraphics_loadDependencies(operatorList) {
-   var fnArray = operatorList.fnArray;
-   var fnArrayLen = fnArray.length;
-   var argsArray = operatorList.argsArray;
-   var self = this;
-   for (var i = 0; i < fnArrayLen; i++) {
-    if (OPS.dependency === fnArray[i]) {
-     var deps = argsArray[i];
-     for (var n = 0, nn = deps.length; n < nn; n++) {
-      var obj = deps[n];
-      var common = obj.substring(0, 2) === 'g_';
-      var promise;
-      if (common) {
-       promise = new Promise(function (resolve) {
-        self.commonObjs.get(obj, resolve);
-       });
+  function opListToTree(opList) {
+    var opTree = [];
+    var tmp = [];
+    var opListLen = opList.length;
+    for (var x = 0; x < opListLen; x++) {
+      if (opList[x].fn === 'save') {
+        opTree.push({
+          'fnId': 92,
+          'fn': 'group',
+          'items': []
+        });
+        tmp.push(opTree);
+        opTree = opTree[opTree.length - 1].items;
+        continue;
+      }
+      if (opList[x].fn === 'restore') {
+        opTree = tmp.pop();
       } else {
-       promise = new Promise(function (resolve) {
-        self.objs.get(obj, resolve);
-       });
+        opTree.push(opList[x]);
       }
-      this.current.dependencies.push(promise);
-     }
-    }
-   }
-   return Promise.all(this.current.dependencies);
-  },
-  transform: function SVGGraphics_transform(a, b, c, d, e, f) {
-   var transformMatrix = [
-    a,
-    b,
-    c,
-    d,
-    e,
-    f
-   ];
-   this.transformMatrix = Util.transform(this.transformMatrix, transformMatrix);
-   this.tgrp = null;
-  },
-  getSVG: function SVGGraphics_getSVG(operatorList, viewport) {
-   this.viewport = viewport;
-   var svgElement = this._initialize(viewport);
-   return this.loadDependencies(operatorList).then(function () {
-    this.transformMatrix = IDENTITY_MATRIX;
-    var opTree = this.convertOpList(operatorList);
-    this.executeOpTree(opTree);
-    return svgElement;
-   }.bind(this));
-  },
-  convertOpList: function SVGGraphics_convertOpList(operatorList) {
-   var argsArray = operatorList.argsArray;
-   var fnArray = operatorList.fnArray;
-   var fnArrayLen = fnArray.length;
-   var REVOPS = [];
-   var opList = [];
-   for (var op in OPS) {
-    REVOPS[OPS[op]] = op;
-   }
-   for (var x = 0; x < fnArrayLen; x++) {
-    var fnId = fnArray[x];
-    opList.push({
-     'fnId': fnId,
-     'fn': REVOPS[fnId],
-     'args': argsArray[x]
-    });
-   }
-   return opListToTree(opList);
-  },
-  executeOpTree: function SVGGraphics_executeOpTree(opTree) {
-   var opTreeLen = opTree.length;
-   for (var x = 0; x < opTreeLen; x++) {
-    var fn = opTree[x].fn;
-    var fnId = opTree[x].fnId;
-    var args = opTree[x].args;
-    switch (fnId | 0) {
-    case OPS.beginText:
-     this.beginText();
-     break;
-    case OPS.setLeading:
-     this.setLeading(args);
-     break;
-    case OPS.setLeadingMoveText:
-     this.setLeadingMoveText(args[0], args[1]);
-     break;
-    case OPS.setFont:
-     this.setFont(args);
-     break;
-    case OPS.showText:
-     this.showText(args[0]);
-     break;
-    case OPS.showSpacedText:
-     this.showText(args[0]);
-     break;
-    case OPS.endText:
-     this.endText();
-     break;
-    case OPS.moveText:
-     this.moveText(args[0], args[1]);
-     break;
-    case OPS.setCharSpacing:
-     this.setCharSpacing(args[0]);
-     break;
-    case OPS.setWordSpacing:
-     this.setWordSpacing(args[0]);
-     break;
-    case OPS.setHScale:
-     this.setHScale(args[0]);
-     break;
-    case OPS.setTextMatrix:
-     this.setTextMatrix(args[0], args[1], args[2], args[3], args[4], args[5]);
-     break;
-    case OPS.setLineWidth:
-     this.setLineWidth(args[0]);
-     break;
-    case OPS.setLineJoin:
-     this.setLineJoin(args[0]);
-     break;
-    case OPS.setLineCap:
-     this.setLineCap(args[0]);
-     break;
-    case OPS.setMiterLimit:
-     this.setMiterLimit(args[0]);
-     break;
-    case OPS.setFillRGBColor:
-     this.setFillRGBColor(args[0], args[1], args[2]);
-     break;
-    case OPS.setStrokeRGBColor:
-     this.setStrokeRGBColor(args[0], args[1], args[2]);
-     break;
-    case OPS.setDash:
-     this.setDash(args[0], args[1]);
-     break;
-    case OPS.setGState:
-     this.setGState(args[0]);
-     break;
-    case OPS.fill:
-     this.fill();
-     break;
-    case OPS.eoFill:
-     this.eoFill();
-     break;
-    case OPS.stroke:
-     this.stroke();
-     break;
-    case OPS.fillStroke:
-     this.fillStroke();
-     break;
-    case OPS.eoFillStroke:
-     this.eoFillStroke();
-     break;
-    case OPS.clip:
-     this.clip('nonzero');
-     break;
-    case OPS.eoClip:
-     this.clip('evenodd');
-     break;
-    case OPS.paintSolidColorImageMask:
-     this.paintSolidColorImageMask();
-     break;
-    case OPS.paintJpegXObject:
-     this.paintJpegXObject(args[0], args[1], args[2]);
-     break;
-    case OPS.paintImageXObject:
-     this.paintImageXObject(args[0]);
-     break;
-    case OPS.paintInlineImageXObject:
-     this.paintInlineImageXObject(args[0]);
-     break;
-    case OPS.paintImageMaskXObject:
-     this.paintImageMaskXObject(args[0]);
-     break;
-    case OPS.paintFormXObjectBegin:
-     this.paintFormXObjectBegin(args[0], args[1]);
-     break;
-    case OPS.paintFormXObjectEnd:
-     this.paintFormXObjectEnd();
-     break;
-    case OPS.closePath:
-     this.closePath();
-     break;
-    case OPS.closeStroke:
-     this.closeStroke();
-     break;
-    case OPS.closeFillStroke:
-     this.closeFillStroke();
-     break;
-    case OPS.nextLine:
-     this.nextLine();
-     break;
-    case OPS.transform:
-     this.transform(args[0], args[1], args[2], args[3], args[4], args[5]);
-     break;
-    case OPS.constructPath:
-     this.constructPath(args[0], args[1]);
-     break;
-    case OPS.endPath:
-     this.endPath();
-     break;
-    case 92:
-     this.group(opTree[x].items);
-     break;
-    default:
-     warn('Unimplemented operator ' + fn);
-     break;
-    }
-   }
-  },
-  setWordSpacing: function SVGGraphics_setWordSpacing(wordSpacing) {
-   this.current.wordSpacing = wordSpacing;
-  },
-  setCharSpacing: function SVGGraphics_setCharSpacing(charSpacing) {
-   this.current.charSpacing = charSpacing;
-  },
-  nextLine: function SVGGraphics_nextLine() {
-   this.moveText(0, this.current.leading);
-  },
-  setTextMatrix: function SVGGraphics_setTextMatrix(a, b, c, d, e, f) {
-   var current = this.current;
-   this.current.textMatrix = this.current.lineMatrix = [
-    a,
-    b,
-    c,
-    d,
-    e,
-    f
-   ];
-   this.current.x = this.current.lineX = 0;
-   this.current.y = this.current.lineY = 0;
-   current.xcoords = [];
-   current.tspan = document.createElementNS(NS, 'svg:tspan');
-   current.tspan.setAttributeNS(null, 'font-family', current.fontFamily);
-   current.tspan.setAttributeNS(null, 'font-size', pf(current.fontSize) + 'px');
-   current.tspan.setAttributeNS(null, 'y', pf(-current.y));
-   current.txtElement = document.createElementNS(NS, 'svg:text');
-   current.txtElement.appendChild(current.tspan);
-  },
-  beginText: function SVGGraphics_beginText() {
-   this.current.x = this.current.lineX = 0;
-   this.current.y = this.current.lineY = 0;
-   this.current.textMatrix = IDENTITY_MATRIX;
-   this.current.lineMatrix = IDENTITY_MATRIX;
-   this.current.tspan = document.createElementNS(NS, 'svg:tspan');
-   this.current.txtElement = document.createElementNS(NS, 'svg:text');
-   this.current.txtgrp = document.createElementNS(NS, 'svg:g');
-   this.current.xcoords = [];
-  },
-  moveText: function SVGGraphics_moveText(x, y) {
-   var current = this.current;
-   this.current.x = this.current.lineX += x;
-   this.current.y = this.current.lineY += y;
-   current.xcoords = [];
-   current.tspan = document.createElementNS(NS, 'svg:tspan');
-   current.tspan.setAttributeNS(null, 'font-family', current.fontFamily);
-   current.tspan.setAttributeNS(null, 'font-size', pf(current.fontSize) + 'px');
-   current.tspan.setAttributeNS(null, 'y', pf(-current.y));
-  },
-  showText: function SVGGraphics_showText(glyphs) {
-   var current = this.current;
-   var font = current.font;
-   var fontSize = current.fontSize;
-   if (fontSize === 0) {
-    return;
-   }
-   var charSpacing = current.charSpacing;
-   var wordSpacing = current.wordSpacing;
-   var fontDirection = current.fontDirection;
-   var textHScale = current.textHScale * fontDirection;
-   var glyphsLength = glyphs.length;
-   var vertical = font.vertical;
-   var widthAdvanceScale = fontSize * current.fontMatrix[0];
-   var x = 0, i;
-   for (i = 0; i < glyphsLength; ++i) {
-    var glyph = glyphs[i];
-    if (glyph === null) {
-     x += fontDirection * wordSpacing;
-     continue;
-    } else if (isNum(glyph)) {
-     x += -glyph * fontSize * 0.001;
-     continue;
     }
-    current.xcoords.push(current.x + x * textHScale);
-    var width = glyph.width;
-    var character = glyph.fontChar;
-    var charWidth = width * widthAdvanceScale + charSpacing * fontDirection;
-    x += charWidth;
-    current.tspan.textContent += character;
-   }
-   if (vertical) {
-    current.y -= x * textHScale;
-   } else {
-    current.x += x * textHScale;
-   }
-   current.tspan.setAttributeNS(null, 'x', current.xcoords.map(pf).join(' '));
-   current.tspan.setAttributeNS(null, 'y', pf(-current.y));
-   current.tspan.setAttributeNS(null, 'font-family', current.fontFamily);
-   current.tspan.setAttributeNS(null, 'font-size', pf(current.fontSize) + 'px');
-   if (current.fontStyle !== SVG_DEFAULTS.fontStyle) {
-    current.tspan.setAttributeNS(null, 'font-style', current.fontStyle);
-   }
-   if (current.fontWeight !== SVG_DEFAULTS.fontWeight) {
-    current.tspan.setAttributeNS(null, 'font-weight', current.fontWeight);
-   }
-   if (current.fillColor !== SVG_DEFAULTS.fillColor) {
-    current.tspan.setAttributeNS(null, 'fill', current.fillColor);
-   }
-   current.txtElement.setAttributeNS(null, 'transform', pm(current.textMatrix) + ' scale(1, -1)');
-   current.txtElement.setAttributeNS(XML_NS, 'xml:space', 'preserve');
-   current.txtElement.appendChild(current.tspan);
-   current.txtgrp.appendChild(current.txtElement);
-   this._ensureTransformGroup().appendChild(current.txtElement);
-  },
-  setLeadingMoveText: function SVGGraphics_setLeadingMoveText(x, y) {
-   this.setLeading(-y);
-   this.moveText(x, y);
-  },
-  addFontStyle: function SVGGraphics_addFontStyle(fontObj) {
-   if (!this.cssStyle) {
-    this.cssStyle = document.createElementNS(NS, 'svg:style');
-    this.cssStyle.setAttributeNS(null, 'type', 'text/css');
-    this.defs.appendChild(this.cssStyle);
-   }
-   var url = createObjectURL(fontObj.data, fontObj.mimetype, this.forceDataSchema);
-   this.cssStyle.textContent += '@font-face { font-family: "' + fontObj.loadedName + '";' + ' src: url(' + url + '); }\n';
-  },
-  setFont: function SVGGraphics_setFont(details) {
-   var current = this.current;
-   var fontObj = this.commonObjs.get(details[0]);
-   var size = details[1];
-   this.current.font = fontObj;
-   if (this.embedFonts && fontObj.data && !this.embeddedFonts[fontObj.loadedName]) {
-    this.addFontStyle(fontObj);
-    this.embeddedFonts[fontObj.loadedName] = fontObj;
-   }
-   current.fontMatrix = fontObj.fontMatrix ? fontObj.fontMatrix : FONT_IDENTITY_MATRIX;
-   var bold = fontObj.black ? fontObj.bold ? 'bolder' : 'bold' : fontObj.bold ? 'bold' : 'normal';
-   var italic = fontObj.italic ? 'italic' : 'normal';
-   if (size < 0) {
-    size = -size;
-    current.fontDirection = -1;
-   } else {
-    current.fontDirection = 1;
-   }
-   current.fontSize = size;
-   current.fontFamily = fontObj.loadedName;
-   current.fontWeight = bold;
-   current.fontStyle = italic;
-   current.tspan = document.createElementNS(NS, 'svg:tspan');
-   current.tspan.setAttributeNS(null, 'y', pf(-current.y));
-   current.xcoords = [];
-  },
-  endText: function SVGGraphics_endText() {
-  },
-  setLineWidth: function SVGGraphics_setLineWidth(width) {
-   this.current.lineWidth = width;
-  },
-  setLineCap: function SVGGraphics_setLineCap(style) {
-   this.current.lineCap = LINE_CAP_STYLES[style];
-  },
-  setLineJoin: function SVGGraphics_setLineJoin(style) {
-   this.current.lineJoin = LINE_JOIN_STYLES[style];
-  },
-  setMiterLimit: function SVGGraphics_setMiterLimit(limit) {
-   this.current.miterLimit = limit;
-  },
-  setStrokeRGBColor: function SVGGraphics_setStrokeRGBColor(r, g, b) {
-   var color = Util.makeCssRgb(r, g, b);
-   this.current.strokeColor = color;
-  },
-  setFillRGBColor: function SVGGraphics_setFillRGBColor(r, g, b) {
-   var color = Util.makeCssRgb(r, g, b);
-   this.current.fillColor = color;
-   this.current.tspan = document.createElementNS(NS, 'svg:tspan');
-   this.current.xcoords = [];
-  },
-  setDash: function SVGGraphics_setDash(dashArray, dashPhase) {
-   this.current.dashArray = dashArray;
-   this.current.dashPhase = dashPhase;
-  },
-  constructPath: function SVGGraphics_constructPath(ops, args) {
-   var current = this.current;
-   var x = current.x, y = current.y;
-   current.path = document.createElementNS(NS, 'svg:path');
-   var d = [];
-   var opLength = ops.length;
-   for (var i = 0, j = 0; i < opLength; i++) {
-    switch (ops[i] | 0) {
-    case OPS.rectangle:
-     x = args[j++];
-     y = args[j++];
-     var width = args[j++];
-     var height = args[j++];
-     var xw = x + width;
-     var yh = y + height;
-     d.push('M', pf(x), pf(y), 'L', pf(xw), pf(y), 'L', pf(xw), pf(yh), 'L', pf(x), pf(yh), 'Z');
-     break;
-    case OPS.moveTo:
-     x = args[j++];
-     y = args[j++];
-     d.push('M', pf(x), pf(y));
-     break;
-    case OPS.lineTo:
-     x = args[j++];
-     y = args[j++];
-     d.push('L', pf(x), pf(y));
-     break;
-    case OPS.curveTo:
-     x = args[j + 4];
-     y = args[j + 5];
-     d.push('C', pf(args[j]), pf(args[j + 1]), pf(args[j + 2]), pf(args[j + 3]), pf(x), pf(y));
-     j += 6;
-     break;
-    case OPS.curveTo2:
-     x = args[j + 2];
-     y = args[j + 3];
-     d.push('C', pf(x), pf(y), pf(args[j]), pf(args[j + 1]), pf(args[j + 2]), pf(args[j + 3]));
-     j += 4;
-     break;
-    case OPS.curveTo3:
-     x = args[j + 2];
-     y = args[j + 3];
-     d.push('C', pf(args[j]), pf(args[j + 1]), pf(x), pf(y), pf(x), pf(y));
-     j += 4;
-     break;
-    case OPS.closePath:
-     d.push('Z');
-     break;
+    return opTree;
+  }
+  function pf(value) {
+    if (value === (value | 0)) {
+      return value.toString();
     }
-   }
-   current.path.setAttributeNS(null, 'd', d.join(' '));
-   current.path.setAttributeNS(null, 'stroke-miterlimit', pf(current.miterLimit));
-   current.path.setAttributeNS(null, 'stroke-linecap', current.lineCap);
-   current.path.setAttributeNS(null, 'stroke-linejoin', current.lineJoin);
-   current.path.setAttributeNS(null, 'stroke-width', pf(current.lineWidth) + 'px');
-   current.path.setAttributeNS(null, 'stroke-dasharray', current.dashArray.map(pf).join(' '));
-   current.path.setAttributeNS(null, 'stroke-dashoffset', pf(current.dashPhase) + 'px');
-   current.path.setAttributeNS(null, 'fill', 'none');
-   this._ensureTransformGroup().appendChild(current.path);
-   current.element = current.path;
-   current.setCurrentPoint(x, y);
-  },
-  endPath: function SVGGraphics_endPath() {
-  },
-  clip: function SVGGraphics_clip(type) {
-   var current = this.current;
-   var clipId = 'clippath' + clipCount;
-   clipCount++;
-   var clipPath = document.createElementNS(NS, 'svg:clipPath');
-   clipPath.setAttributeNS(null, 'id', clipId);
-   clipPath.setAttributeNS(null, 'transform', pm(this.transformMatrix));
-   var clipElement = current.element.cloneNode();
-   if (type === 'evenodd') {
-    clipElement.setAttributeNS(null, 'clip-rule', 'evenodd');
-   } else {
-    clipElement.setAttributeNS(null, 'clip-rule', 'nonzero');
-   }
-   clipPath.appendChild(clipElement);
-   this.defs.appendChild(clipPath);
-   if (current.activeClipUrl) {
-    current.clipGroup = null;
-    this.extraStack.forEach(function (prev) {
-     prev.clipGroup = null;
-    });
-   }
-   current.activeClipUrl = 'url(#' + clipId + ')';
-   this.tgrp = null;
-  },
-  closePath: function SVGGraphics_closePath() {
-   var current = this.current;
-   var d = current.path.getAttributeNS(null, 'd');
-   d += 'Z';
-   current.path.setAttributeNS(null, 'd', d);
-  },
-  setLeading: function SVGGraphics_setLeading(leading) {
-   this.current.leading = -leading;
-  },
-  setTextRise: function SVGGraphics_setTextRise(textRise) {
-   this.current.textRise = textRise;
-  },
-  setHScale: function SVGGraphics_setHScale(scale) {
-   this.current.textHScale = scale / 100;
-  },
-  setGState: function SVGGraphics_setGState(states) {
-   for (var i = 0, ii = states.length; i < ii; i++) {
-    var state = states[i];
-    var key = state[0];
-    var value = state[1];
-    switch (key) {
-    case 'LW':
-     this.setLineWidth(value);
-     break;
-    case 'LC':
-     this.setLineCap(value);
-     break;
-    case 'LJ':
-     this.setLineJoin(value);
-     break;
-    case 'ML':
-     this.setMiterLimit(value);
-     break;
-    case 'D':
-     this.setDash(value[0], value[1]);
-     break;
-    case 'Font':
-     this.setFont(value);
-     break;
-    default:
-     warn('Unimplemented graphic state ' + key);
-     break;
+    var s = value.toFixed(10);
+    var i = s.length - 1;
+    if (s[i] !== '0') {
+      return s;
     }
-   }
-  },
-  fill: function SVGGraphics_fill() {
-   var current = this.current;
-   current.element.setAttributeNS(null, 'fill', current.fillColor);
-  },
-  stroke: function SVGGraphics_stroke() {
-   var current = this.current;
-   current.element.setAttributeNS(null, 'stroke', current.strokeColor);
-   current.element.setAttributeNS(null, 'fill', 'none');
-  },
-  eoFill: function SVGGraphics_eoFill() {
-   var current = this.current;
-   current.element.setAttributeNS(null, 'fill', current.fillColor);
-   current.element.setAttributeNS(null, 'fill-rule', 'evenodd');
-  },
-  fillStroke: function SVGGraphics_fillStroke() {
-   this.stroke();
-   this.fill();
-  },
-  eoFillStroke: function SVGGraphics_eoFillStroke() {
-   this.current.element.setAttributeNS(null, 'fill-rule', 'evenodd');
-   this.fillStroke();
-  },
-  closeStroke: function SVGGraphics_closeStroke() {
-   this.closePath();
-   this.stroke();
-  },
-  closeFillStroke: function SVGGraphics_closeFillStroke() {
-   this.closePath();
-   this.fillStroke();
-  },
-  paintSolidColorImageMask: function SVGGraphics_paintSolidColorImageMask() {
-   var current = this.current;
-   var rect = document.createElementNS(NS, 'svg:rect');
-   rect.setAttributeNS(null, 'x', '0');
-   rect.setAttributeNS(null, 'y', '0');
-   rect.setAttributeNS(null, 'width', '1px');
-   rect.setAttributeNS(null, 'height', '1px');
-   rect.setAttributeNS(null, 'fill', current.fillColor);
-   this._ensureTransformGroup().appendChild(rect);
-  },
-  paintJpegXObject: function SVGGraphics_paintJpegXObject(objId, w, h) {
-   var imgObj = this.objs.get(objId);
-   var imgEl = document.createElementNS(NS, 'svg:image');
-   imgEl.setAttributeNS(XLINK_NS, 'xlink:href', imgObj.src);
-   imgEl.setAttributeNS(null, 'width', imgObj.width + 'px');
-   imgEl.setAttributeNS(null, 'height', imgObj.height + 'px');
-   imgEl.setAttributeNS(null, 'x', '0');
-   imgEl.setAttributeNS(null, 'y', pf(-h));
-   imgEl.setAttributeNS(null, 'transform', 'scale(' + pf(1 / w) + ' ' + pf(-1 / h) + ')');
-   this._ensureTransformGroup().appendChild(imgEl);
-  },
-  paintImageXObject: function SVGGraphics_paintImageXObject(objId) {
-   var imgData = this.objs.get(objId);
-   if (!imgData) {
-    warn('Dependent image isn\'t ready yet');
-    return;
-   }
-   this.paintInlineImageXObject(imgData);
-  },
-  paintInlineImageXObject: function SVGGraphics_paintInlineImageXObject(imgData, mask) {
-   var width = imgData.width;
-   var height = imgData.height;
-   var imgSrc = convertImgDataToPng(imgData, this.forceDataSchema);
-   var cliprect = document.createElementNS(NS, 'svg:rect');
-   cliprect.setAttributeNS(null, 'x', '0');
-   cliprect.setAttributeNS(null, 'y', '0');
-   cliprect.setAttributeNS(null, 'width', pf(width));
-   cliprect.setAttributeNS(null, 'height', pf(height));
-   this.current.element = cliprect;
-   this.clip('nonzero');
-   var imgEl = document.createElementNS(NS, 'svg:image');
-   imgEl.setAttributeNS(XLINK_NS, 'xlink:href', imgSrc);
-   imgEl.setAttributeNS(null, 'x', '0');
-   imgEl.setAttributeNS(null, 'y', pf(-height));
-   imgEl.setAttributeNS(null, 'width', pf(width) + 'px');
-   imgEl.setAttributeNS(null, 'height', pf(height) + 'px');
-   imgEl.setAttributeNS(null, 'transform', 'scale(' + pf(1 / width) + ' ' + pf(-1 / height) + ')');
-   if (mask) {
-    mask.appendChild(imgEl);
-   } else {
-    this._ensureTransformGroup().appendChild(imgEl);
-   }
-  },
-  paintImageMaskXObject: function SVGGraphics_paintImageMaskXObject(imgData) {
-   var current = this.current;
-   var width = imgData.width;
-   var height = imgData.height;
-   var fillColor = current.fillColor;
-   current.maskId = 'mask' + maskCount++;
-   var mask = document.createElementNS(NS, 'svg:mask');
-   mask.setAttributeNS(null, 'id', current.maskId);
-   var rect = document.createElementNS(NS, 'svg:rect');
-   rect.setAttributeNS(null, 'x', '0');
-   rect.setAttributeNS(null, 'y', '0');
-   rect.setAttributeNS(null, 'width', pf(width));
-   rect.setAttributeNS(null, 'height', pf(height));
-   rect.setAttributeNS(null, 'fill', fillColor);
-   rect.setAttributeNS(null, 'mask', 'url(#' + current.maskId + ')');
-   this.defs.appendChild(mask);
-   this._ensureTransformGroup().appendChild(rect);
-   this.paintInlineImageXObject(imgData, mask);
-  },
-  paintFormXObjectBegin: function SVGGraphics_paintFormXObjectBegin(matrix, bbox) {
-   if (isArray(matrix) && matrix.length === 6) {
-    this.transform(matrix[0], matrix[1], matrix[2], matrix[3], matrix[4], matrix[5]);
-   }
-   if (isArray(bbox) && bbox.length === 4) {
-    var width = bbox[2] - bbox[0];
-    var height = bbox[3] - bbox[1];
-    var cliprect = document.createElementNS(NS, 'svg:rect');
-    cliprect.setAttributeNS(null, 'x', bbox[0]);
-    cliprect.setAttributeNS(null, 'y', bbox[1]);
-    cliprect.setAttributeNS(null, 'width', pf(width));
-    cliprect.setAttributeNS(null, 'height', pf(height));
-    this.current.element = cliprect;
-    this.clip('nonzero');
-    this.endPath();
-   }
-  },
-  paintFormXObjectEnd: function SVGGraphics_paintFormXObjectEnd() {
-  },
-  _initialize: function SVGGraphics_initialize(viewport) {
-   var svg = document.createElementNS(NS, 'svg:svg');
-   svg.setAttributeNS(null, 'version', '1.1');
-   svg.setAttributeNS(null, 'width', viewport.width + 'px');
-   svg.setAttributeNS(null, 'height', viewport.height + 'px');
-   svg.setAttributeNS(null, 'preserveAspectRatio', 'none');
-   svg.setAttributeNS(null, 'viewBox', '0 0 ' + viewport.width + ' ' + viewport.height);
-   var definitions = document.createElementNS(NS, 'svg:defs');
-   svg.appendChild(definitions);
-   this.defs = definitions;
-   var rootGroup = document.createElementNS(NS, 'svg:g');
-   rootGroup.setAttributeNS(null, 'transform', pm(viewport.transform));
-   svg.appendChild(rootGroup);
-   this.svg = rootGroup;
-   return svg;
-  },
-  _ensureClipGroup: function SVGGraphics_ensureClipGroup() {
-   if (!this.current.clipGroup) {
-    var clipGroup = document.createElementNS(NS, 'svg:g');
-    clipGroup.setAttributeNS(null, 'clip-path', this.current.activeClipUrl);
-    this.svg.appendChild(clipGroup);
-    this.current.clipGroup = clipGroup;
-   }
-   return this.current.clipGroup;
-  },
-  _ensureTransformGroup: function SVGGraphics_ensureTransformGroup() {
-   if (!this.tgrp) {
-    this.tgrp = document.createElementNS(NS, 'svg:g');
-    this.tgrp.setAttributeNS(null, 'transform', pm(this.transformMatrix));
-    if (this.current.activeClipUrl) {
-     this._ensureClipGroup().appendChild(this.tgrp);
+    do {
+      i--;
+    } while (s[i] === '0');
+    return s.substr(0, s[i] === '.' ? i : i + 1);
+  }
+  function pm(m) {
+    if (m[4] === 0 && m[5] === 0) {
+      if (m[1] === 0 && m[2] === 0) {
+        if (m[0] === 1 && m[3] === 1) {
+          return '';
+        }
+        return 'scale(' + pf(m[0]) + ' ' + pf(m[3]) + ')';
+      }
+      if (m[0] === m[3] && m[1] === -m[2]) {
+        var a = Math.acos(m[0]) * 180 / Math.PI;
+        return 'rotate(' + pf(a) + ')';
+      }
     } else {
-     this.svg.appendChild(this.tgrp);
+      if (m[0] === 1 && m[1] === 0 && m[2] === 0 && m[3] === 1) {
+        return 'translate(' + pf(m[4]) + ' ' + pf(m[5]) + ')';
+      }
     }
-   }
-   return this.tgrp;
+    return 'matrix(' + pf(m[0]) + ' ' + pf(m[1]) + ' ' + pf(m[2]) + ' ' + pf(m[3]) + ' ' + pf(m[4]) + ' ' + pf(m[5]) + ')';
+  }
+  function SVGGraphics(commonObjs, objs, forceDataSchema) {
+    this.current = new SVGExtraState();
+    this.transformMatrix = IDENTITY_MATRIX;
+    this.transformStack = [];
+    this.extraStack = [];
+    this.commonObjs = commonObjs;
+    this.objs = objs;
+    this.pendingEOFill = false;
+    this.embedFonts = false;
+    this.embeddedFonts = Object.create(null);
+    this.cssStyle = null;
+    this.forceDataSchema = !!forceDataSchema;
   }
- };
- return SVGGraphics;
+  var NS = 'http://www.w3.org/2000/svg';
+  var XML_NS = 'http://www.w3.org/XML/1998/namespace';
+  var XLINK_NS = 'http://www.w3.org/1999/xlink';
+  var LINE_CAP_STYLES = ['butt', 'round', 'square'];
+  var LINE_JOIN_STYLES = ['miter', 'round', 'bevel'];
+  var clipCount = 0;
+  var maskCount = 0;
+  SVGGraphics.prototype = {
+    save: function SVGGraphics_save() {
+      this.transformStack.push(this.transformMatrix);
+      var old = this.current;
+      this.extraStack.push(old);
+      this.current = old.clone();
+    },
+    restore: function SVGGraphics_restore() {
+      this.transformMatrix = this.transformStack.pop();
+      this.current = this.extraStack.pop();
+      this.tgrp = null;
+    },
+    group: function SVGGraphics_group(items) {
+      this.save();
+      this.executeOpTree(items);
+      this.restore();
+    },
+    loadDependencies: function SVGGraphics_loadDependencies(operatorList) {
+      var fnArray = operatorList.fnArray;
+      var fnArrayLen = fnArray.length;
+      var argsArray = operatorList.argsArray;
+      var self = this;
+      for (var i = 0; i < fnArrayLen; i++) {
+        if (OPS.dependency === fnArray[i]) {
+          var deps = argsArray[i];
+          for (var n = 0, nn = deps.length; n < nn; n++) {
+            var obj = deps[n];
+            var common = obj.substring(0, 2) === 'g_';
+            var promise;
+            if (common) {
+              promise = new Promise(function (resolve) {
+                self.commonObjs.get(obj, resolve);
+              });
+            } else {
+              promise = new Promise(function (resolve) {
+                self.objs.get(obj, resolve);
+              });
+            }
+            this.current.dependencies.push(promise);
+          }
+        }
+      }
+      return Promise.all(this.current.dependencies);
+    },
+    transform: function SVGGraphics_transform(a, b, c, d, e, f) {
+      var transformMatrix = [a, b, c, d, e, f];
+      this.transformMatrix = Util.transform(this.transformMatrix, transformMatrix);
+      this.tgrp = null;
+    },
+    getSVG: function SVGGraphics_getSVG(operatorList, viewport) {
+      this.viewport = viewport;
+      var svgElement = this._initialize(viewport);
+      return this.loadDependencies(operatorList).then(function () {
+        this.transformMatrix = IDENTITY_MATRIX;
+        var opTree = this.convertOpList(operatorList);
+        this.executeOpTree(opTree);
+        return svgElement;
+      }.bind(this));
+    },
+    convertOpList: function SVGGraphics_convertOpList(operatorList) {
+      var argsArray = operatorList.argsArray;
+      var fnArray = operatorList.fnArray;
+      var fnArrayLen = fnArray.length;
+      var REVOPS = [];
+      var opList = [];
+      for (var op in OPS) {
+        REVOPS[OPS[op]] = op;
+      }
+      for (var x = 0; x < fnArrayLen; x++) {
+        var fnId = fnArray[x];
+        opList.push({
+          'fnId': fnId,
+          'fn': REVOPS[fnId],
+          'args': argsArray[x]
+        });
+      }
+      return opListToTree(opList);
+    },
+    executeOpTree: function SVGGraphics_executeOpTree(opTree) {
+      var opTreeLen = opTree.length;
+      for (var x = 0; x < opTreeLen; x++) {
+        var fn = opTree[x].fn;
+        var fnId = opTree[x].fnId;
+        var args = opTree[x].args;
+        switch (fnId | 0) {
+          case OPS.beginText:
+            this.beginText();
+            break;
+          case OPS.setLeading:
+            this.setLeading(args);
+            break;
+          case OPS.setLeadingMoveText:
+            this.setLeadingMoveText(args[0], args[1]);
+            break;
+          case OPS.setFont:
+            this.setFont(args);
+            break;
+          case OPS.showText:
+            this.showText(args[0]);
+            break;
+          case OPS.showSpacedText:
+            this.showText(args[0]);
+            break;
+          case OPS.endText:
+            this.endText();
+            break;
+          case OPS.moveText:
+            this.moveText(args[0], args[1]);
+            break;
+          case OPS.setCharSpacing:
+            this.setCharSpacing(args[0]);
+            break;
+          case OPS.setWordSpacing:
+            this.setWordSpacing(args[0]);
+            break;
+          case OPS.setHScale:
+            this.setHScale(args[0]);
+            break;
+          case OPS.setTextMatrix:
+            this.setTextMatrix(args[0], args[1], args[2], args[3], args[4], args[5]);
+            break;
+          case OPS.setLineWidth:
+            this.setLineWidth(args[0]);
+            break;
+          case OPS.setLineJoin:
+            this.setLineJoin(args[0]);
+            break;
+          case OPS.setLineCap:
+            this.setLineCap(args[0]);
+            break;
+          case OPS.setMiterLimit:
+            this.setMiterLimit(args[0]);
+            break;
+          case OPS.setFillRGBColor:
+            this.setFillRGBColor(args[0], args[1], args[2]);
+            break;
+          case OPS.setStrokeRGBColor:
+            this.setStrokeRGBColor(args[0], args[1], args[2]);
+            break;
+          case OPS.setDash:
+            this.setDash(args[0], args[1]);
+            break;
+          case OPS.setGState:
+            this.setGState(args[0]);
+            break;
+          case OPS.fill:
+            this.fill();
+            break;
+          case OPS.eoFill:
+            this.eoFill();
+            break;
+          case OPS.stroke:
+            this.stroke();
+            break;
+          case OPS.fillStroke:
+            this.fillStroke();
+            break;
+          case OPS.eoFillStroke:
+            this.eoFillStroke();
+            break;
+          case OPS.clip:
+            this.clip('nonzero');
+            break;
+          case OPS.eoClip:
+            this.clip('evenodd');
+            break;
+          case OPS.paintSolidColorImageMask:
+            this.paintSolidColorImageMask();
+            break;
+          case OPS.paintJpegXObject:
+            this.paintJpegXObject(args[0], args[1], args[2]);
+            break;
+          case OPS.paintImageXObject:
+            this.paintImageXObject(args[0]);
+            break;
+          case OPS.paintInlineImageXObject:
+            this.paintInlineImageXObject(args[0]);
+            break;
+          case OPS.paintImageMaskXObject:
+            this.paintImageMaskXObject(args[0]);
+            break;
+          case OPS.paintFormXObjectBegin:
+            this.paintFormXObjectBegin(args[0], args[1]);
+            break;
+          case OPS.paintFormXObjectEnd:
+            this.paintFormXObjectEnd();
+            break;
+          case OPS.closePath:
+            this.closePath();
+            break;
+          case OPS.closeStroke:
+            this.closeStroke();
+            break;
+          case OPS.closeFillStroke:
+            this.closeFillStroke();
+            break;
+          case OPS.nextLine:
+            this.nextLine();
+            break;
+          case OPS.transform:
+            this.transform(args[0], args[1], args[2], args[3], args[4], args[5]);
+            break;
+          case OPS.constructPath:
+            this.constructPath(args[0], args[1]);
+            break;
+          case OPS.endPath:
+            this.endPath();
+            break;
+          case 92:
+            this.group(opTree[x].items);
+            break;
+          default:
+            warn('Unimplemented operator ' + fn);
+            break;
+        }
+      }
+    },
+    setWordSpacing: function SVGGraphics_setWordSpacing(wordSpacing) {
+      this.current.wordSpacing = wordSpacing;
+    },
+    setCharSpacing: function SVGGraphics_setCharSpacing(charSpacing) {
+      this.current.charSpacing = charSpacing;
+    },
+    nextLine: function SVGGraphics_nextLine() {
+      this.moveText(0, this.current.leading);
+    },
+    setTextMatrix: function SVGGraphics_setTextMatrix(a, b, c, d, e, f) {
+      var current = this.current;
+      this.current.textMatrix = this.current.lineMatrix = [a, b, c, d, e, f];
+      this.current.x = this.current.lineX = 0;
+      this.current.y = this.current.lineY = 0;
+      current.xcoords = [];
+      current.tspan = document.createElementNS(NS, 'svg:tspan');
+      current.tspan.setAttributeNS(null, 'font-family', current.fontFamily);
+      current.tspan.setAttributeNS(null, 'font-size', pf(current.fontSize) + 'px');
+      current.tspan.setAttributeNS(null, 'y', pf(-current.y));
+      current.txtElement = document.createElementNS(NS, 'svg:text');
+      current.txtElement.appendChild(current.tspan);
+    },
+    beginText: function SVGGraphics_beginText() {
+      this.current.x = this.current.lineX = 0;
+      this.current.y = this.current.lineY = 0;
+      this.current.textMatrix = IDENTITY_MATRIX;
+      this.current.lineMatrix = IDENTITY_MATRIX;
+      this.current.tspan = document.createElementNS(NS, 'svg:tspan');
+      this.current.txtElement = document.createElementNS(NS, 'svg:text');
+      this.current.txtgrp = document.createElementNS(NS, 'svg:g');
+      this.current.xcoords = [];
+    },
+    moveText: function SVGGraphics_moveText(x, y) {
+      var current = this.current;
+      this.current.x = this.current.lineX += x;
+      this.current.y = this.current.lineY += y;
+      current.xcoords = [];
+      current.tspan = document.createElementNS(NS, 'svg:tspan');
+      current.tspan.setAttributeNS(null, 'font-family', current.fontFamily);
+      current.tspan.setAttributeNS(null, 'font-size', pf(current.fontSize) + 'px');
+      current.tspan.setAttributeNS(null, 'y', pf(-current.y));
+    },
+    showText: function SVGGraphics_showText(glyphs) {
+      var current = this.current;
+      var font = current.font;
+      var fontSize = current.fontSize;
+      if (fontSize === 0) {
+        return;
+      }
+      var charSpacing = current.charSpacing;
+      var wordSpacing = current.wordSpacing;
+      var fontDirection = current.fontDirection;
+      var textHScale = current.textHScale * fontDirection;
+      var glyphsLength = glyphs.length;
+      var vertical = font.vertical;
+      var widthAdvanceScale = fontSize * current.fontMatrix[0];
+      var x = 0,
+          i;
+      for (i = 0; i < glyphsLength; ++i) {
+        var glyph = glyphs[i];
+        if (glyph === null) {
+          x += fontDirection * wordSpacing;
+          continue;
+        } else if (isNum(glyph)) {
+          x += -glyph * fontSize * 0.001;
+          continue;
+        }
+        current.xcoords.push(current.x + x * textHScale);
+        var width = glyph.width;
+        var character = glyph.fontChar;
+        var charWidth = width * widthAdvanceScale + charSpacing * fontDirection;
+        x += charWidth;
+        current.tspan.textContent += character;
+      }
+      if (vertical) {
+        current.y -= x * textHScale;
+      } else {
+        current.x += x * textHScale;
+      }
+      current.tspan.setAttributeNS(null, 'x', current.xcoords.map(pf).join(' '));
+      current.tspan.setAttributeNS(null, 'y', pf(-current.y));
+      current.tspan.setAttributeNS(null, 'font-family', current.fontFamily);
+      current.tspan.setAttributeNS(null, 'font-size', pf(current.fontSize) + 'px');
+      if (current.fontStyle !== SVG_DEFAULTS.fontStyle) {
+        current.tspan.setAttributeNS(null, 'font-style', current.fontStyle);
+      }
+      if (current.fontWeight !== SVG_DEFAULTS.fontWeight) {
+        current.tspan.setAttributeNS(null, 'font-weight', current.fontWeight);
+      }
+      if (current.fillColor !== SVG_DEFAULTS.fillColor) {
+        current.tspan.setAttributeNS(null, 'fill', current.fillColor);
+      }
+      current.txtElement.setAttributeNS(null, 'transform', pm(current.textMatrix) + ' scale(1, -1)');
+      current.txtElement.setAttributeNS(XML_NS, 'xml:space', 'preserve');
+      current.txtElement.appendChild(current.tspan);
+      current.txtgrp.appendChild(current.txtElement);
+      this._ensureTransformGroup().appendChild(current.txtElement);
+    },
+    setLeadingMoveText: function SVGGraphics_setLeadingMoveText(x, y) {
+      this.setLeading(-y);
+      this.moveText(x, y);
+    },
+    addFontStyle: function SVGGraphics_addFontStyle(fontObj) {
+      if (!this.cssStyle) {
+        this.cssStyle = document.createElementNS(NS, 'svg:style');
+        this.cssStyle.setAttributeNS(null, 'type', 'text/css');
+        this.defs.appendChild(this.cssStyle);
+      }
+      var url = createObjectURL(fontObj.data, fontObj.mimetype, this.forceDataSchema);
+      this.cssStyle.textContent += '@font-face { font-family: "' + fontObj.loadedName + '";' + ' src: url(' + url + '); }\n';
+    },
+    setFont: function SVGGraphics_setFont(details) {
+      var current = this.current;
+      var fontObj = this.commonObjs.get(details[0]);
+      var size = details[1];
+      this.current.font = fontObj;
+      if (this.embedFonts && fontObj.data && !this.embeddedFonts[fontObj.loadedName]) {
+        this.addFontStyle(fontObj);
+        this.embeddedFonts[fontObj.loadedName] = fontObj;
+      }
+      current.fontMatrix = fontObj.fontMatrix ? fontObj.fontMatrix : FONT_IDENTITY_MATRIX;
+      var bold = fontObj.black ? fontObj.bold ? 'bolder' : 'bold' : fontObj.bold ? 'bold' : 'normal';
+      var italic = fontObj.italic ? 'italic' : 'normal';
+      if (size < 0) {
+        size = -size;
+        current.fontDirection = -1;
+      } else {
+        current.fontDirection = 1;
+      }
+      current.fontSize = size;
+      current.fontFamily = fontObj.loadedName;
+      current.fontWeight = bold;
+      current.fontStyle = italic;
+      current.tspan = document.createElementNS(NS, 'svg:tspan');
+      current.tspan.setAttributeNS(null, 'y', pf(-current.y));
+      current.xcoords = [];
+    },
+    endText: function SVGGraphics_endText() {},
+    setLineWidth: function SVGGraphics_setLineWidth(width) {
+      this.current.lineWidth = width;
+    },
+    setLineCap: function SVGGraphics_setLineCap(style) {
+      this.current.lineCap = LINE_CAP_STYLES[style];
+    },
+    setLineJoin: function SVGGraphics_setLineJoin(style) {
+      this.current.lineJoin = LINE_JOIN_STYLES[style];
+    },
+    setMiterLimit: function SVGGraphics_setMiterLimit(limit) {
+      this.current.miterLimit = limit;
+    },
+    setStrokeRGBColor: function SVGGraphics_setStrokeRGBColor(r, g, b) {
+      var color = Util.makeCssRgb(r, g, b);
+      this.current.strokeColor = color;
+    },
+    setFillRGBColor: function SVGGraphics_setFillRGBColor(r, g, b) {
+      var color = Util.makeCssRgb(r, g, b);
+      this.current.fillColor = color;
+      this.current.tspan = document.createElementNS(NS, 'svg:tspan');
+      this.current.xcoords = [];
+    },
+    setDash: function SVGGraphics_setDash(dashArray, dashPhase) {
+      this.current.dashArray = dashArray;
+      this.current.dashPhase = dashPhase;
+    },
+    constructPath: function SVGGraphics_constructPath(ops, args) {
+      var current = this.current;
+      var x = current.x,
+          y = current.y;
+      current.path = document.createElementNS(NS, 'svg:path');
+      var d = [];
+      var opLength = ops.length;
+      for (var i = 0, j = 0; i < opLength; i++) {
+        switch (ops[i] | 0) {
+          case OPS.rectangle:
+            x = args[j++];
+            y = args[j++];
+            var width = args[j++];
+            var height = args[j++];
+            var xw = x + width;
+            var yh = y + height;
+            d.push('M', pf(x), pf(y), 'L', pf(xw), pf(y), 'L', pf(xw), pf(yh), 'L', pf(x), pf(yh), 'Z');
+            break;
+          case OPS.moveTo:
+            x = args[j++];
+            y = args[j++];
+            d.push('M', pf(x), pf(y));
+            break;
+          case OPS.lineTo:
+            x = args[j++];
+            y = args[j++];
+            d.push('L', pf(x), pf(y));
+            break;
+          case OPS.curveTo:
+            x = args[j + 4];
+            y = args[j + 5];
+            d.push('C', pf(args[j]), pf(args[j + 1]), pf(args[j + 2]), pf(args[j + 3]), pf(x), pf(y));
+            j += 6;
+            break;
+          case OPS.curveTo2:
+            x = args[j + 2];
+            y = args[j + 3];
+            d.push('C', pf(x), pf(y), pf(args[j]), pf(args[j + 1]), pf(args[j + 2]), pf(args[j + 3]));
+            j += 4;
+            break;
+          case OPS.curveTo3:
+            x = args[j + 2];
+            y = args[j + 3];
+            d.push('C', pf(args[j]), pf(args[j + 1]), pf(x), pf(y), pf(x), pf(y));
+            j += 4;
+            break;
+          case OPS.closePath:
+            d.push('Z');
+            break;
+        }
+      }
+      current.path.setAttributeNS(null, 'd', d.join(' '));
+      current.path.setAttributeNS(null, 'stroke-miterlimit', pf(current.miterLimit));
+      current.path.setAttributeNS(null, 'stroke-linecap', current.lineCap);
+      current.path.setAttributeNS(null, 'stroke-linejoin', current.lineJoin);
+      current.path.setAttributeNS(null, 'stroke-width', pf(current.lineWidth) + 'px');
+      current.path.setAttributeNS(null, 'stroke-dasharray', current.dashArray.map(pf).join(' '));
+      current.path.setAttributeNS(null, 'stroke-dashoffset', pf(current.dashPhase) + 'px');
+      current.path.setAttributeNS(null, 'fill', 'none');
+      this._ensureTransformGroup().appendChild(current.path);
+      current.element = current.path;
+      current.setCurrentPoint(x, y);
+    },
+    endPath: function SVGGraphics_endPath() {},
+    clip: function SVGGraphics_clip(type) {
+      var current = this.current;
+      var clipId = 'clippath' + clipCount;
+      clipCount++;
+      var clipPath = document.createElementNS(NS, 'svg:clipPath');
+      clipPath.setAttributeNS(null, 'id', clipId);
+      clipPath.setAttributeNS(null, 'transform', pm(this.transformMatrix));
+      var clipElement = current.element.cloneNode();
+      if (type === 'evenodd') {
+        clipElement.setAttributeNS(null, 'clip-rule', 'evenodd');
+      } else {
+        clipElement.setAttributeNS(null, 'clip-rule', 'nonzero');
+      }
+      clipPath.appendChild(clipElement);
+      this.defs.appendChild(clipPath);
+      if (current.activeClipUrl) {
+        current.clipGroup = null;
+        this.extraStack.forEach(function (prev) {
+          prev.clipGroup = null;
+        });
+      }
+      current.activeClipUrl = 'url(#' + clipId + ')';
+      this.tgrp = null;
+    },
+    closePath: function SVGGraphics_closePath() {
+      var current = this.current;
+      var d = current.path.getAttributeNS(null, 'd');
+      d += 'Z';
+      current.path.setAttributeNS(null, 'd', d);
+    },
+    setLeading: function SVGGraphics_setLeading(leading) {
+      this.current.leading = -leading;
+    },
+    setTextRise: function SVGGraphics_setTextRise(textRise) {
+      this.current.textRise = textRise;
+    },
+    setHScale: function SVGGraphics_setHScale(scale) {
+      this.current.textHScale = scale / 100;
+    },
+    setGState: function SVGGraphics_setGState(states) {
+      for (var i = 0, ii = states.length; i < ii; i++) {
+        var state = states[i];
+        var key = state[0];
+        var value = state[1];
+        switch (key) {
+          case 'LW':
+            this.setLineWidth(value);
+            break;
+          case 'LC':
+            this.setLineCap(value);
+            break;
+          case 'LJ':
+            this.setLineJoin(value);
+            break;
+          case 'ML':
+            this.setMiterLimit(value);
+            break;
+          case 'D':
+            this.setDash(value[0], value[1]);
+            break;
+          case 'Font':
+            this.setFont(value);
+            break;
+          default:
+            warn('Unimplemented graphic state ' + key);
+            break;
+        }
+      }
+    },
+    fill: function SVGGraphics_fill() {
+      var current = this.current;
+      current.element.setAttributeNS(null, 'fill', current.fillColor);
+    },
+    stroke: function SVGGraphics_stroke() {
+      var current = this.current;
+      current.element.setAttributeNS(null, 'stroke', current.strokeColor);
+      current.element.setAttributeNS(null, 'fill', 'none');
+    },
+    eoFill: function SVGGraphics_eoFill() {
+      var current = this.current;
+      current.element.setAttributeNS(null, 'fill', current.fillColor);
+      current.element.setAttributeNS(null, 'fill-rule', 'evenodd');
+    },
+    fillStroke: function SVGGraphics_fillStroke() {
+      this.stroke();
+      this.fill();
+    },
+    eoFillStroke: function SVGGraphics_eoFillStroke() {
+      this.current.element.setAttributeNS(null, 'fill-rule', 'evenodd');
+      this.fillStroke();
+    },
+    closeStroke: function SVGGraphics_closeStroke() {
+      this.closePath();
+      this.stroke();
+    },
+    closeFillStroke: function SVGGraphics_closeFillStroke() {
+      this.closePath();
+      this.fillStroke();
+    },
+    paintSolidColorImageMask: function SVGGraphics_paintSolidColorImageMask() {
+      var current = this.current;
+      var rect = document.createElementNS(NS, 'svg:rect');
+      rect.setAttributeNS(null, 'x', '0');
+      rect.setAttributeNS(null, 'y', '0');
+      rect.setAttributeNS(null, 'width', '1px');
+      rect.setAttributeNS(null, 'height', '1px');
+      rect.setAttributeNS(null, 'fill', current.fillColor);
+      this._ensureTransformGroup().appendChild(rect);
+    },
+    paintJpegXObject: function SVGGraphics_paintJpegXObject(objId, w, h) {
+      var imgObj = this.objs.get(objId);
+      var imgEl = document.createElementNS(NS, 'svg:image');
+      imgEl.setAttributeNS(XLINK_NS, 'xlink:href', imgObj.src);
+      imgEl.setAttributeNS(null, 'width', imgObj.width + 'px');
+      imgEl.setAttributeNS(null, 'height', imgObj.height + 'px');
+      imgEl.setAttributeNS(null, 'x', '0');
+      imgEl.setAttributeNS(null, 'y', pf(-h));
+      imgEl.setAttributeNS(null, 'transform', 'scale(' + pf(1 / w) + ' ' + pf(-1 / h) + ')');
+      this._ensureTransformGroup().appendChild(imgEl);
+    },
+    paintImageXObject: function SVGGraphics_paintImageXObject(objId) {
+      var imgData = this.objs.get(objId);
+      if (!imgData) {
+        warn('Dependent image isn\'t ready yet');
+        return;
+      }
+      this.paintInlineImageXObject(imgData);
+    },
+    paintInlineImageXObject: function SVGGraphics_paintInlineImageXObject(imgData, mask) {
+      var width = imgData.width;
+      var height = imgData.height;
+      var imgSrc = convertImgDataToPng(imgData, this.forceDataSchema);
+      var cliprect = document.createElementNS(NS, 'svg:rect');
+      cliprect.setAttributeNS(null, 'x', '0');
+      cliprect.setAttributeNS(null, 'y', '0');
+      cliprect.setAttributeNS(null, 'width', pf(width));
+      cliprect.setAttributeNS(null, 'height', pf(height));
+      this.current.element = cliprect;
+      this.clip('nonzero');
+      var imgEl = document.createElementNS(NS, 'svg:image');
+      imgEl.setAttributeNS(XLINK_NS, 'xlink:href', imgSrc);
+      imgEl.setAttributeNS(null, 'x', '0');
+      imgEl.setAttributeNS(null, 'y', pf(-height));
+      imgEl.setAttributeNS(null, 'width', pf(width) + 'px');
+      imgEl.setAttributeNS(null, 'height', pf(height) + 'px');
+      imgEl.setAttributeNS(null, 'transform', 'scale(' + pf(1 / width) + ' ' + pf(-1 / height) + ')');
+      if (mask) {
+        mask.appendChild(imgEl);
+      } else {
+        this._ensureTransformGroup().appendChild(imgEl);
+      }
+    },
+    paintImageMaskXObject: function SVGGraphics_paintImageMaskXObject(imgData) {
+      var current = this.current;
+      var width = imgData.width;
+      var height = imgData.height;
+      var fillColor = current.fillColor;
+      current.maskId = 'mask' + maskCount++;
+      var mask = document.createElementNS(NS, 'svg:mask');
+      mask.setAttributeNS(null, 'id', current.maskId);
+      var rect = document.createElementNS(NS, 'svg:rect');
+      rect.setAttributeNS(null, 'x', '0');
+      rect.setAttributeNS(null, 'y', '0');
+      rect.setAttributeNS(null, 'width', pf(width));
+      rect.setAttributeNS(null, 'height', pf(height));
+      rect.setAttributeNS(null, 'fill', fillColor);
+      rect.setAttributeNS(null, 'mask', 'url(#' + current.maskId + ')');
+      this.defs.appendChild(mask);
+      this._ensureTransformGroup().appendChild(rect);
+      this.paintInlineImageXObject(imgData, mask);
+    },
+    paintFormXObjectBegin: function SVGGraphics_paintFormXObjectBegin(matrix, bbox) {
+      if (isArray(matrix) && matrix.length === 6) {
+        this.transform(matrix[0], matrix[1], matrix[2], matrix[3], matrix[4], matrix[5]);
+      }
+      if (isArray(bbox) && bbox.length === 4) {
+        var width = bbox[2] - bbox[0];
+        var height = bbox[3] - bbox[1];
+        var cliprect = document.createElementNS(NS, 'svg:rect');
+        cliprect.setAttributeNS(null, 'x', bbox[0]);
+        cliprect.setAttributeNS(null, 'y', bbox[1]);
+        cliprect.setAttributeNS(null, 'width', pf(width));
+        cliprect.setAttributeNS(null, 'height', pf(height));
+        this.current.element = cliprect;
+        this.clip('nonzero');
+        this.endPath();
+      }
+    },
+    paintFormXObjectEnd: function SVGGraphics_paintFormXObjectEnd() {},
+    _initialize: function SVGGraphics_initialize(viewport) {
+      var svg = document.createElementNS(NS, 'svg:svg');
+      svg.setAttributeNS(null, 'version', '1.1');
+      svg.setAttributeNS(null, 'width', viewport.width + 'px');
+      svg.setAttributeNS(null, 'height', viewport.height + 'px');
+      svg.setAttributeNS(null, 'preserveAspectRatio', 'none');
+      svg.setAttributeNS(null, 'viewBox', '0 0 ' + viewport.width + ' ' + viewport.height);
+      var definitions = document.createElementNS(NS, 'svg:defs');
+      svg.appendChild(definitions);
+      this.defs = definitions;
+      var rootGroup = document.createElementNS(NS, 'svg:g');
+      rootGroup.setAttributeNS(null, 'transform', pm(viewport.transform));
+      svg.appendChild(rootGroup);
+      this.svg = rootGroup;
+      return svg;
+    },
+    _ensureClipGroup: function SVGGraphics_ensureClipGroup() {
+      if (!this.current.clipGroup) {
+        var clipGroup = document.createElementNS(NS, 'svg:g');
+        clipGroup.setAttributeNS(null, 'clip-path', this.current.activeClipUrl);
+        this.svg.appendChild(clipGroup);
+        this.current.clipGroup = clipGroup;
+      }
+      return this.current.clipGroup;
+    },
+    _ensureTransformGroup: function SVGGraphics_ensureTransformGroup() {
+      if (!this.tgrp) {
+        this.tgrp = document.createElementNS(NS, 'svg:g');
+        this.tgrp.setAttributeNS(null, 'transform', pm(this.transformMatrix));
+        if (this.current.activeClipUrl) {
+          this._ensureClipGroup().appendChild(this.tgrp);
+        } else {
+          this.svg.appendChild(this.tgrp);
+        }
+      }
+      return this.tgrp;
+    }
+  };
+  return SVGGraphics;
 }();
 exports.SVGGraphics = SVGGraphics;

+ 454 - 491
lib/display/text_layer.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var displayDOMUtils = require('./dom_utils.js');
 var Util = sharedUtil.Util;
@@ -20,511 +21,473 @@ var createPromiseCapability = sharedUtil.createPromiseCapability;
 var CustomStyle = displayDOMUtils.CustomStyle;
 var getDefaultSetting = displayDOMUtils.getDefaultSetting;
 var renderTextLayer = function renderTextLayerClosure() {
- var MAX_TEXT_DIVS_TO_RENDER = 100000;
- var NonWhitespaceRegexp = /\S/;
- function isAllWhitespace(str) {
-  return !NonWhitespaceRegexp.test(str);
- }
- var styleBuf = [
-  'left: ',
-  0,
-  'px; top: ',
-  0,
-  'px; font-size: ',
-  0,
-  'px; font-family: ',
-  '',
-  ';'
- ];
- function appendText(task, geom, styles) {
-  var textDiv = document.createElement('div');
-  var textDivProperties = {
-   style: null,
-   angle: 0,
-   canvasWidth: 0,
-   isWhitespace: false,
-   originalTransform: null,
-   paddingBottom: 0,
-   paddingLeft: 0,
-   paddingRight: 0,
-   paddingTop: 0,
-   scale: 1
-  };
-  task._textDivs.push(textDiv);
-  if (isAllWhitespace(geom.str)) {
-   textDivProperties.isWhitespace = true;
-   task._textDivProperties.set(textDiv, textDivProperties);
-   return;
-  }
-  var tx = Util.transform(task._viewport.transform, geom.transform);
-  var angle = Math.atan2(tx[1], tx[0]);
-  var style = styles[geom.fontName];
-  if (style.vertical) {
-   angle += Math.PI / 2;
-  }
-  var fontHeight = Math.sqrt(tx[2] * tx[2] + tx[3] * tx[3]);
-  var fontAscent = fontHeight;
-  if (style.ascent) {
-   fontAscent = style.ascent * fontAscent;
-  } else if (style.descent) {
-   fontAscent = (1 + style.descent) * fontAscent;
-  }
-  var left;
-  var top;
-  if (angle === 0) {
-   left = tx[4];
-   top = tx[5] - fontAscent;
-  } else {
-   left = tx[4] + fontAscent * Math.sin(angle);
-   top = tx[5] - fontAscent * Math.cos(angle);
-  }
-  styleBuf[1] = left;
-  styleBuf[3] = top;
-  styleBuf[5] = fontHeight;
-  styleBuf[7] = style.fontFamily;
-  textDivProperties.style = styleBuf.join('');
-  textDiv.setAttribute('style', textDivProperties.style);
-  textDiv.textContent = geom.str;
-  if (getDefaultSetting('pdfBug')) {
-   textDiv.dataset.fontName = geom.fontName;
-  }
-  if (angle !== 0) {
-   textDivProperties.angle = angle * (180 / Math.PI);
-  }
-  if (geom.str.length > 1) {
-   if (style.vertical) {
-    textDivProperties.canvasWidth = geom.height * task._viewport.scale;
-   } else {
-    textDivProperties.canvasWidth = geom.width * task._viewport.scale;
-   }
-  }
-  task._textDivProperties.set(textDiv, textDivProperties);
-  if (task._enhanceTextSelection) {
-   var angleCos = 1, angleSin = 0;
-   if (angle !== 0) {
-    angleCos = Math.cos(angle);
-    angleSin = Math.sin(angle);
-   }
-   var divWidth = (style.vertical ? geom.height : geom.width) * task._viewport.scale;
-   var divHeight = fontHeight;
-   var m, b;
-   if (angle !== 0) {
-    m = [
-     angleCos,
-     angleSin,
-     -angleSin,
-     angleCos,
-     left,
-     top
-    ];
-    b = Util.getAxialAlignedBoundingBox([
-     0,
-     0,
-     divWidth,
-     divHeight
-    ], m);
-   } else {
-    b = [
-     left,
-     top,
-     left + divWidth,
-     top + divHeight
-    ];
-   }
-   task._bounds.push({
-    left: b[0],
-    top: b[1],
-    right: b[2],
-    bottom: b[3],
-    div: textDiv,
-    size: [
-     divWidth,
-     divHeight
-    ],
-    m: m
-   });
+  var MAX_TEXT_DIVS_TO_RENDER = 100000;
+  var NonWhitespaceRegexp = /\S/;
+  function isAllWhitespace(str) {
+    return !NonWhitespaceRegexp.test(str);
   }
- }
- function render(task) {
-  if (task._canceled) {
-   return;
-  }
-  var textLayerFrag = task._container;
-  var textDivs = task._textDivs;
-  var capability = task._capability;
-  var textDivsLength = textDivs.length;
-  if (textDivsLength > MAX_TEXT_DIVS_TO_RENDER) {
-   task._renderingDone = true;
-   capability.resolve();
-   return;
-  }
-  var canvas = document.createElement('canvas');
-  canvas.mozOpaque = true;
-  var ctx = canvas.getContext('2d', { alpha: false });
-  var lastFontSize;
-  var lastFontFamily;
-  for (var i = 0; i < textDivsLength; i++) {
-   var textDiv = textDivs[i];
-   var textDivProperties = task._textDivProperties.get(textDiv);
-   if (textDivProperties.isWhitespace) {
-    continue;
-   }
-   var fontSize = textDiv.style.fontSize;
-   var fontFamily = textDiv.style.fontFamily;
-   if (fontSize !== lastFontSize || fontFamily !== lastFontFamily) {
-    ctx.font = fontSize + ' ' + fontFamily;
-    lastFontSize = fontSize;
-    lastFontFamily = fontFamily;
-   }
-   var width = ctx.measureText(textDiv.textContent).width;
-   textLayerFrag.appendChild(textDiv);
-   var transform = '';
-   if (textDivProperties.canvasWidth !== 0 && width > 0) {
-    textDivProperties.scale = textDivProperties.canvasWidth / width;
-    transform = 'scaleX(' + textDivProperties.scale + ')';
-   }
-   if (textDivProperties.angle !== 0) {
-    transform = 'rotate(' + textDivProperties.angle + 'deg) ' + transform;
-   }
-   if (transform !== '') {
-    textDivProperties.originalTransform = transform;
-    CustomStyle.setProp('transform', textDiv, transform);
-   }
-   task._textDivProperties.set(textDiv, textDivProperties);
-  }
-  task._renderingDone = true;
-  capability.resolve();
- }
- function expand(task) {
-  var bounds = task._bounds;
-  var viewport = task._viewport;
-  var expanded = expandBounds(viewport.width, viewport.height, bounds);
-  for (var i = 0; i < expanded.length; i++) {
-   var div = bounds[i].div;
-   var divProperties = task._textDivProperties.get(div);
-   if (divProperties.angle === 0) {
-    divProperties.paddingLeft = bounds[i].left - expanded[i].left;
-    divProperties.paddingTop = bounds[i].top - expanded[i].top;
-    divProperties.paddingRight = expanded[i].right - bounds[i].right;
-    divProperties.paddingBottom = expanded[i].bottom - bounds[i].bottom;
-    task._textDivProperties.set(div, divProperties);
-    continue;
-   }
-   var e = expanded[i], b = bounds[i];
-   var m = b.m, c = m[0], s = m[1];
-   var points = [
-    [
-     0,
-     0
-    ],
-    [
-     0,
-     b.size[1]
-    ],
-    [
-     b.size[0],
-     0
-    ],
-    b.size
-   ];
-   var ts = new Float64Array(64);
-   points.forEach(function (p, i) {
-    var t = Util.applyTransform(p, m);
-    ts[i + 0] = c && (e.left - t[0]) / c;
-    ts[i + 4] = s && (e.top - t[1]) / s;
-    ts[i + 8] = c && (e.right - t[0]) / c;
-    ts[i + 12] = s && (e.bottom - t[1]) / s;
-    ts[i + 16] = s && (e.left - t[0]) / -s;
-    ts[i + 20] = c && (e.top - t[1]) / c;
-    ts[i + 24] = s && (e.right - t[0]) / -s;
-    ts[i + 28] = c && (e.bottom - t[1]) / c;
-    ts[i + 32] = c && (e.left - t[0]) / -c;
-    ts[i + 36] = s && (e.top - t[1]) / -s;
-    ts[i + 40] = c && (e.right - t[0]) / -c;
-    ts[i + 44] = s && (e.bottom - t[1]) / -s;
-    ts[i + 48] = s && (e.left - t[0]) / s;
-    ts[i + 52] = c && (e.top - t[1]) / -c;
-    ts[i + 56] = s && (e.right - t[0]) / s;
-    ts[i + 60] = c && (e.bottom - t[1]) / -c;
-   });
-   var findPositiveMin = function (ts, offset, count) {
-    var result = 0;
-    for (var i = 0; i < count; i++) {
-     var t = ts[offset++];
-     if (t > 0) {
-      result = result ? Math.min(t, result) : t;
-     }
+  var styleBuf = ['left: ', 0, 'px; top: ', 0, 'px; font-size: ', 0, 'px; font-family: ', '', ';'];
+  function appendText(task, geom, styles) {
+    var textDiv = document.createElement('div');
+    var textDivProperties = {
+      style: null,
+      angle: 0,
+      canvasWidth: 0,
+      isWhitespace: false,
+      originalTransform: null,
+      paddingBottom: 0,
+      paddingLeft: 0,
+      paddingRight: 0,
+      paddingTop: 0,
+      scale: 1
+    };
+    task._textDivs.push(textDiv);
+    if (isAllWhitespace(geom.str)) {
+      textDivProperties.isWhitespace = true;
+      task._textDivProperties.set(textDiv, textDivProperties);
+      return;
     }
-    return result;
-   };
-   var boxScale = 1 + Math.min(Math.abs(c), Math.abs(s));
-   divProperties.paddingLeft = findPositiveMin(ts, 32, 16) / boxScale;
-   divProperties.paddingTop = findPositiveMin(ts, 48, 16) / boxScale;
-   divProperties.paddingRight = findPositiveMin(ts, 0, 16) / boxScale;
-   divProperties.paddingBottom = findPositiveMin(ts, 16, 16) / boxScale;
-   task._textDivProperties.set(div, divProperties);
-  }
- }
- function expandBounds(width, height, boxes) {
-  var bounds = boxes.map(function (box, i) {
-   return {
-    x1: box.left,
-    y1: box.top,
-    x2: box.right,
-    y2: box.bottom,
-    index: i,
-    x1New: undefined,
-    x2New: undefined
-   };
-  });
-  expandBoundsLTR(width, bounds);
-  var expanded = new Array(boxes.length);
-  bounds.forEach(function (b) {
-   var i = b.index;
-   expanded[i] = {
-    left: b.x1New,
-    top: 0,
-    right: b.x2New,
-    bottom: 0
-   };
-  });
-  boxes.map(function (box, i) {
-   var e = expanded[i], b = bounds[i];
-   b.x1 = box.top;
-   b.y1 = width - e.right;
-   b.x2 = box.bottom;
-   b.y2 = width - e.left;
-   b.index = i;
-   b.x1New = undefined;
-   b.x2New = undefined;
-  });
-  expandBoundsLTR(height, bounds);
-  bounds.forEach(function (b) {
-   var i = b.index;
-   expanded[i].top = b.x1New;
-   expanded[i].bottom = b.x2New;
-  });
-  return expanded;
- }
- function expandBoundsLTR(width, bounds) {
-  bounds.sort(function (a, b) {
-   return a.x1 - b.x1 || a.index - b.index;
-  });
-  var fakeBoundary = {
-   x1: -Infinity,
-   y1: -Infinity,
-   x2: 0,
-   y2: Infinity,
-   index: -1,
-   x1New: 0,
-   x2New: 0
-  };
-  var horizon = [{
-    start: -Infinity,
-    end: Infinity,
-    boundary: fakeBoundary
-   }];
-  bounds.forEach(function (boundary) {
-   var i = 0;
-   while (i < horizon.length && horizon[i].end <= boundary.y1) {
-    i++;
-   }
-   var j = horizon.length - 1;
-   while (j >= 0 && horizon[j].start >= boundary.y2) {
-    j--;
-   }
-   var horizonPart, affectedBoundary;
-   var q, k, maxXNew = -Infinity;
-   for (q = i; q <= j; q++) {
-    horizonPart = horizon[q];
-    affectedBoundary = horizonPart.boundary;
-    var xNew;
-    if (affectedBoundary.x2 > boundary.x1) {
-     xNew = affectedBoundary.index > boundary.index ? affectedBoundary.x1New : boundary.x1;
-    } else if (affectedBoundary.x2New === undefined) {
-     xNew = (affectedBoundary.x2 + boundary.x1) / 2;
+    var tx = Util.transform(task._viewport.transform, geom.transform);
+    var angle = Math.atan2(tx[1], tx[0]);
+    var style = styles[geom.fontName];
+    if (style.vertical) {
+      angle += Math.PI / 2;
+    }
+    var fontHeight = Math.sqrt(tx[2] * tx[2] + tx[3] * tx[3]);
+    var fontAscent = fontHeight;
+    if (style.ascent) {
+      fontAscent = style.ascent * fontAscent;
+    } else if (style.descent) {
+      fontAscent = (1 + style.descent) * fontAscent;
+    }
+    var left;
+    var top;
+    if (angle === 0) {
+      left = tx[4];
+      top = tx[5] - fontAscent;
     } else {
-     xNew = affectedBoundary.x2New;
+      left = tx[4] + fontAscent * Math.sin(angle);
+      top = tx[5] - fontAscent * Math.cos(angle);
     }
-    if (xNew > maxXNew) {
-     maxXNew = xNew;
+    styleBuf[1] = left;
+    styleBuf[3] = top;
+    styleBuf[5] = fontHeight;
+    styleBuf[7] = style.fontFamily;
+    textDivProperties.style = styleBuf.join('');
+    textDiv.setAttribute('style', textDivProperties.style);
+    textDiv.textContent = geom.str;
+    if (getDefaultSetting('pdfBug')) {
+      textDiv.dataset.fontName = geom.fontName;
     }
-   }
-   boundary.x1New = maxXNew;
-   for (q = i; q <= j; q++) {
-    horizonPart = horizon[q];
-    affectedBoundary = horizonPart.boundary;
-    if (affectedBoundary.x2New === undefined) {
-     if (affectedBoundary.x2 > boundary.x1) {
-      if (affectedBoundary.index > boundary.index) {
-       affectedBoundary.x2New = affectedBoundary.x2;
-      }
-     } else {
-      affectedBoundary.x2New = maxXNew;
-     }
-    } else if (affectedBoundary.x2New > maxXNew) {
-     affectedBoundary.x2New = Math.max(maxXNew, affectedBoundary.x2);
+    if (angle !== 0) {
+      textDivProperties.angle = angle * (180 / Math.PI);
     }
-   }
-   var changedHorizon = [], lastBoundary = null;
-   for (q = i; q <= j; q++) {
-    horizonPart = horizon[q];
-    affectedBoundary = horizonPart.boundary;
-    var useBoundary = affectedBoundary.x2 > boundary.x2 ? affectedBoundary : boundary;
-    if (lastBoundary === useBoundary) {
-     changedHorizon[changedHorizon.length - 1].end = horizonPart.end;
-    } else {
-     changedHorizon.push({
-      start: horizonPart.start,
-      end: horizonPart.end,
-      boundary: useBoundary
-     });
-     lastBoundary = useBoundary;
+    if (geom.str.length > 1) {
+      if (style.vertical) {
+        textDivProperties.canvasWidth = geom.height * task._viewport.scale;
+      } else {
+        textDivProperties.canvasWidth = geom.width * task._viewport.scale;
+      }
     }
-   }
-   if (horizon[i].start < boundary.y1) {
-    changedHorizon[0].start = boundary.y1;
-    changedHorizon.unshift({
-     start: horizon[i].start,
-     end: boundary.y1,
-     boundary: horizon[i].boundary
-    });
-   }
-   if (boundary.y2 < horizon[j].end) {
-    changedHorizon[changedHorizon.length - 1].end = boundary.y2;
-    changedHorizon.push({
-     start: boundary.y2,
-     end: horizon[j].end,
-     boundary: horizon[j].boundary
-    });
-   }
-   for (q = i; q <= j; q++) {
-    horizonPart = horizon[q];
-    affectedBoundary = horizonPart.boundary;
-    if (affectedBoundary.x2New !== undefined) {
-     continue;
+    task._textDivProperties.set(textDiv, textDivProperties);
+    if (task._enhanceTextSelection) {
+      var angleCos = 1,
+          angleSin = 0;
+      if (angle !== 0) {
+        angleCos = Math.cos(angle);
+        angleSin = Math.sin(angle);
+      }
+      var divWidth = (style.vertical ? geom.height : geom.width) * task._viewport.scale;
+      var divHeight = fontHeight;
+      var m, b;
+      if (angle !== 0) {
+        m = [angleCos, angleSin, -angleSin, angleCos, left, top];
+        b = Util.getAxialAlignedBoundingBox([0, 0, divWidth, divHeight], m);
+      } else {
+        b = [left, top, left + divWidth, top + divHeight];
+      }
+      task._bounds.push({
+        left: b[0],
+        top: b[1],
+        right: b[2],
+        bottom: b[3],
+        div: textDiv,
+        size: [divWidth, divHeight],
+        m: m
+      });
     }
-    var used = false;
-    for (k = i - 1; !used && k >= 0 && horizon[k].start >= affectedBoundary.y1; k--) {
-     used = horizon[k].boundary === affectedBoundary;
+  }
+  function render(task) {
+    if (task._canceled) {
+      return;
     }
-    for (k = j + 1; !used && k < horizon.length && horizon[k].end <= affectedBoundary.y2; k++) {
-     used = horizon[k].boundary === affectedBoundary;
+    var textLayerFrag = task._container;
+    var textDivs = task._textDivs;
+    var capability = task._capability;
+    var textDivsLength = textDivs.length;
+    if (textDivsLength > MAX_TEXT_DIVS_TO_RENDER) {
+      task._renderingDone = true;
+      capability.resolve();
+      return;
     }
-    for (k = 0; !used && k < changedHorizon.length; k++) {
-     used = changedHorizon[k].boundary === affectedBoundary;
+    var canvas = document.createElement('canvas');
+    canvas.mozOpaque = true;
+    var ctx = canvas.getContext('2d', { alpha: false });
+    var lastFontSize;
+    var lastFontFamily;
+    for (var i = 0; i < textDivsLength; i++) {
+      var textDiv = textDivs[i];
+      var textDivProperties = task._textDivProperties.get(textDiv);
+      if (textDivProperties.isWhitespace) {
+        continue;
+      }
+      var fontSize = textDiv.style.fontSize;
+      var fontFamily = textDiv.style.fontFamily;
+      if (fontSize !== lastFontSize || fontFamily !== lastFontFamily) {
+        ctx.font = fontSize + ' ' + fontFamily;
+        lastFontSize = fontSize;
+        lastFontFamily = fontFamily;
+      }
+      var width = ctx.measureText(textDiv.textContent).width;
+      textLayerFrag.appendChild(textDiv);
+      var transform = '';
+      if (textDivProperties.canvasWidth !== 0 && width > 0) {
+        textDivProperties.scale = textDivProperties.canvasWidth / width;
+        transform = 'scaleX(' + textDivProperties.scale + ')';
+      }
+      if (textDivProperties.angle !== 0) {
+        transform = 'rotate(' + textDivProperties.angle + 'deg) ' + transform;
+      }
+      if (transform !== '') {
+        textDivProperties.originalTransform = transform;
+        CustomStyle.setProp('transform', textDiv, transform);
+      }
+      task._textDivProperties.set(textDiv, textDivProperties);
     }
-    if (!used) {
-     affectedBoundary.x2New = maxXNew;
+    task._renderingDone = true;
+    capability.resolve();
+  }
+  function expand(task) {
+    var bounds = task._bounds;
+    var viewport = task._viewport;
+    var expanded = expandBounds(viewport.width, viewport.height, bounds);
+    for (var i = 0; i < expanded.length; i++) {
+      var div = bounds[i].div;
+      var divProperties = task._textDivProperties.get(div);
+      if (divProperties.angle === 0) {
+        divProperties.paddingLeft = bounds[i].left - expanded[i].left;
+        divProperties.paddingTop = bounds[i].top - expanded[i].top;
+        divProperties.paddingRight = expanded[i].right - bounds[i].right;
+        divProperties.paddingBottom = expanded[i].bottom - bounds[i].bottom;
+        task._textDivProperties.set(div, divProperties);
+        continue;
+      }
+      var e = expanded[i],
+          b = bounds[i];
+      var m = b.m,
+          c = m[0],
+          s = m[1];
+      var points = [[0, 0], [0, b.size[1]], [b.size[0], 0], b.size];
+      var ts = new Float64Array(64);
+      points.forEach(function (p, i) {
+        var t = Util.applyTransform(p, m);
+        ts[i + 0] = c && (e.left - t[0]) / c;
+        ts[i + 4] = s && (e.top - t[1]) / s;
+        ts[i + 8] = c && (e.right - t[0]) / c;
+        ts[i + 12] = s && (e.bottom - t[1]) / s;
+        ts[i + 16] = s && (e.left - t[0]) / -s;
+        ts[i + 20] = c && (e.top - t[1]) / c;
+        ts[i + 24] = s && (e.right - t[0]) / -s;
+        ts[i + 28] = c && (e.bottom - t[1]) / c;
+        ts[i + 32] = c && (e.left - t[0]) / -c;
+        ts[i + 36] = s && (e.top - t[1]) / -s;
+        ts[i + 40] = c && (e.right - t[0]) / -c;
+        ts[i + 44] = s && (e.bottom - t[1]) / -s;
+        ts[i + 48] = s && (e.left - t[0]) / s;
+        ts[i + 52] = c && (e.top - t[1]) / -c;
+        ts[i + 56] = s && (e.right - t[0]) / s;
+        ts[i + 60] = c && (e.bottom - t[1]) / -c;
+      });
+      var findPositiveMin = function (ts, offset, count) {
+        var result = 0;
+        for (var i = 0; i < count; i++) {
+          var t = ts[offset++];
+          if (t > 0) {
+            result = result ? Math.min(t, result) : t;
+          }
+        }
+        return result;
+      };
+      var boxScale = 1 + Math.min(Math.abs(c), Math.abs(s));
+      divProperties.paddingLeft = findPositiveMin(ts, 32, 16) / boxScale;
+      divProperties.paddingTop = findPositiveMin(ts, 48, 16) / boxScale;
+      divProperties.paddingRight = findPositiveMin(ts, 0, 16) / boxScale;
+      divProperties.paddingBottom = findPositiveMin(ts, 16, 16) / boxScale;
+      task._textDivProperties.set(div, divProperties);
     }
-   }
-   Array.prototype.splice.apply(horizon, [
-    i,
-    j - i + 1
-   ].concat(changedHorizon));
-  });
-  horizon.forEach(function (horizonPart) {
-   var affectedBoundary = horizonPart.boundary;
-   if (affectedBoundary.x2New === undefined) {
-    affectedBoundary.x2New = Math.max(width, affectedBoundary.x2);
-   }
-  });
- }
- function TextLayerRenderTask(textContent, container, viewport, textDivs, enhanceTextSelection) {
-  this._textContent = textContent;
-  this._container = container;
-  this._viewport = viewport;
-  this._textDivs = textDivs || [];
-  this._textDivProperties = new WeakMap();
-  this._renderingDone = false;
-  this._canceled = false;
-  this._capability = createPromiseCapability();
-  this._renderTimer = null;
-  this._bounds = [];
-  this._enhanceTextSelection = !!enhanceTextSelection;
- }
- TextLayerRenderTask.prototype = {
-  get promise() {
-   return this._capability.promise;
-  },
-  cancel: function TextLayer_cancel() {
-   this._canceled = true;
-   if (this._renderTimer !== null) {
-    clearTimeout(this._renderTimer);
+  }
+  function expandBounds(width, height, boxes) {
+    var bounds = boxes.map(function (box, i) {
+      return {
+        x1: box.left,
+        y1: box.top,
+        x2: box.right,
+        y2: box.bottom,
+        index: i,
+        x1New: undefined,
+        x2New: undefined
+      };
+    });
+    expandBoundsLTR(width, bounds);
+    var expanded = new Array(boxes.length);
+    bounds.forEach(function (b) {
+      var i = b.index;
+      expanded[i] = {
+        left: b.x1New,
+        top: 0,
+        right: b.x2New,
+        bottom: 0
+      };
+    });
+    boxes.map(function (box, i) {
+      var e = expanded[i],
+          b = bounds[i];
+      b.x1 = box.top;
+      b.y1 = width - e.right;
+      b.x2 = box.bottom;
+      b.y2 = width - e.left;
+      b.index = i;
+      b.x1New = undefined;
+      b.x2New = undefined;
+    });
+    expandBoundsLTR(height, bounds);
+    bounds.forEach(function (b) {
+      var i = b.index;
+      expanded[i].top = b.x1New;
+      expanded[i].bottom = b.x2New;
+    });
+    return expanded;
+  }
+  function expandBoundsLTR(width, bounds) {
+    bounds.sort(function (a, b) {
+      return a.x1 - b.x1 || a.index - b.index;
+    });
+    var fakeBoundary = {
+      x1: -Infinity,
+      y1: -Infinity,
+      x2: 0,
+      y2: Infinity,
+      index: -1,
+      x1New: 0,
+      x2New: 0
+    };
+    var horizon = [{
+      start: -Infinity,
+      end: Infinity,
+      boundary: fakeBoundary
+    }];
+    bounds.forEach(function (boundary) {
+      var i = 0;
+      while (i < horizon.length && horizon[i].end <= boundary.y1) {
+        i++;
+      }
+      var j = horizon.length - 1;
+      while (j >= 0 && horizon[j].start >= boundary.y2) {
+        j--;
+      }
+      var horizonPart, affectedBoundary;
+      var q,
+          k,
+          maxXNew = -Infinity;
+      for (q = i; q <= j; q++) {
+        horizonPart = horizon[q];
+        affectedBoundary = horizonPart.boundary;
+        var xNew;
+        if (affectedBoundary.x2 > boundary.x1) {
+          xNew = affectedBoundary.index > boundary.index ? affectedBoundary.x1New : boundary.x1;
+        } else if (affectedBoundary.x2New === undefined) {
+          xNew = (affectedBoundary.x2 + boundary.x1) / 2;
+        } else {
+          xNew = affectedBoundary.x2New;
+        }
+        if (xNew > maxXNew) {
+          maxXNew = xNew;
+        }
+      }
+      boundary.x1New = maxXNew;
+      for (q = i; q <= j; q++) {
+        horizonPart = horizon[q];
+        affectedBoundary = horizonPart.boundary;
+        if (affectedBoundary.x2New === undefined) {
+          if (affectedBoundary.x2 > boundary.x1) {
+            if (affectedBoundary.index > boundary.index) {
+              affectedBoundary.x2New = affectedBoundary.x2;
+            }
+          } else {
+            affectedBoundary.x2New = maxXNew;
+          }
+        } else if (affectedBoundary.x2New > maxXNew) {
+          affectedBoundary.x2New = Math.max(maxXNew, affectedBoundary.x2);
+        }
+      }
+      var changedHorizon = [],
+          lastBoundary = null;
+      for (q = i; q <= j; q++) {
+        horizonPart = horizon[q];
+        affectedBoundary = horizonPart.boundary;
+        var useBoundary = affectedBoundary.x2 > boundary.x2 ? affectedBoundary : boundary;
+        if (lastBoundary === useBoundary) {
+          changedHorizon[changedHorizon.length - 1].end = horizonPart.end;
+        } else {
+          changedHorizon.push({
+            start: horizonPart.start,
+            end: horizonPart.end,
+            boundary: useBoundary
+          });
+          lastBoundary = useBoundary;
+        }
+      }
+      if (horizon[i].start < boundary.y1) {
+        changedHorizon[0].start = boundary.y1;
+        changedHorizon.unshift({
+          start: horizon[i].start,
+          end: boundary.y1,
+          boundary: horizon[i].boundary
+        });
+      }
+      if (boundary.y2 < horizon[j].end) {
+        changedHorizon[changedHorizon.length - 1].end = boundary.y2;
+        changedHorizon.push({
+          start: boundary.y2,
+          end: horizon[j].end,
+          boundary: horizon[j].boundary
+        });
+      }
+      for (q = i; q <= j; q++) {
+        horizonPart = horizon[q];
+        affectedBoundary = horizonPart.boundary;
+        if (affectedBoundary.x2New !== undefined) {
+          continue;
+        }
+        var used = false;
+        for (k = i - 1; !used && k >= 0 && horizon[k].start >= affectedBoundary.y1; k--) {
+          used = horizon[k].boundary === affectedBoundary;
+        }
+        for (k = j + 1; !used && k < horizon.length && horizon[k].end <= affectedBoundary.y2; k++) {
+          used = horizon[k].boundary === affectedBoundary;
+        }
+        for (k = 0; !used && k < changedHorizon.length; k++) {
+          used = changedHorizon[k].boundary === affectedBoundary;
+        }
+        if (!used) {
+          affectedBoundary.x2New = maxXNew;
+        }
+      }
+      Array.prototype.splice.apply(horizon, [i, j - i + 1].concat(changedHorizon));
+    });
+    horizon.forEach(function (horizonPart) {
+      var affectedBoundary = horizonPart.boundary;
+      if (affectedBoundary.x2New === undefined) {
+        affectedBoundary.x2New = Math.max(width, affectedBoundary.x2);
+      }
+    });
+  }
+  function TextLayerRenderTask(textContent, container, viewport, textDivs, enhanceTextSelection) {
+    this._textContent = textContent;
+    this._container = container;
+    this._viewport = viewport;
+    this._textDivs = textDivs || [];
+    this._textDivProperties = new WeakMap();
+    this._renderingDone = false;
+    this._canceled = false;
+    this._capability = createPromiseCapability();
     this._renderTimer = null;
-   }
-   this._capability.reject('canceled');
-  },
-  _render: function TextLayer_render(timeout) {
-   var textItems = this._textContent.items;
-   var textStyles = this._textContent.styles;
-   for (var i = 0, len = textItems.length; i < len; i++) {
-    appendText(this, textItems[i], textStyles);
-   }
-   if (!timeout) {
-    render(this);
-   } else {
-    var self = this;
-    this._renderTimer = setTimeout(function () {
-     render(self);
-     self._renderTimer = null;
-    }, timeout);
-   }
-  },
-  expandTextDivs: function TextLayer_expandTextDivs(expandDivs) {
-   if (!this._enhanceTextSelection || !this._renderingDone) {
-    return;
-   }
-   if (this._bounds !== null) {
-    expand(this);
-    this._bounds = null;
-   }
-   for (var i = 0, ii = this._textDivs.length; i < ii; i++) {
-    var div = this._textDivs[i];
-    var divProperties = this._textDivProperties.get(div);
-    if (divProperties.isWhitespace) {
-     continue;
-    }
-    if (expandDivs) {
-     var transform = '', padding = '';
-     if (divProperties.scale !== 1) {
-      transform = 'scaleX(' + divProperties.scale + ')';
-     }
-     if (divProperties.angle !== 0) {
-      transform = 'rotate(' + divProperties.angle + 'deg) ' + transform;
-     }
-     if (divProperties.paddingLeft !== 0) {
-      padding += ' padding-left: ' + divProperties.paddingLeft / divProperties.scale + 'px;';
-      transform += ' translateX(' + -divProperties.paddingLeft / divProperties.scale + 'px)';
-     }
-     if (divProperties.paddingTop !== 0) {
-      padding += ' padding-top: ' + divProperties.paddingTop + 'px;';
-      transform += ' translateY(' + -divProperties.paddingTop + 'px)';
-     }
-     if (divProperties.paddingRight !== 0) {
-      padding += ' padding-right: ' + divProperties.paddingRight / divProperties.scale + 'px;';
-     }
-     if (divProperties.paddingBottom !== 0) {
-      padding += ' padding-bottom: ' + divProperties.paddingBottom + 'px;';
-     }
-     if (padding !== '') {
-      div.setAttribute('style', divProperties.style + padding);
-     }
-     if (transform !== '') {
-      CustomStyle.setProp('transform', div, transform);
-     }
-    } else {
-     div.style.padding = 0;
-     CustomStyle.setProp('transform', div, divProperties.originalTransform || '');
+    this._bounds = [];
+    this._enhanceTextSelection = !!enhanceTextSelection;
+  }
+  TextLayerRenderTask.prototype = {
+    get promise() {
+      return this._capability.promise;
+    },
+    cancel: function TextLayer_cancel() {
+      this._canceled = true;
+      if (this._renderTimer !== null) {
+        clearTimeout(this._renderTimer);
+        this._renderTimer = null;
+      }
+      this._capability.reject('canceled');
+    },
+    _render: function TextLayer_render(timeout) {
+      var textItems = this._textContent.items;
+      var textStyles = this._textContent.styles;
+      for (var i = 0, len = textItems.length; i < len; i++) {
+        appendText(this, textItems[i], textStyles);
+      }
+      if (!timeout) {
+        render(this);
+      } else {
+        var self = this;
+        this._renderTimer = setTimeout(function () {
+          render(self);
+          self._renderTimer = null;
+        }, timeout);
+      }
+    },
+    expandTextDivs: function TextLayer_expandTextDivs(expandDivs) {
+      if (!this._enhanceTextSelection || !this._renderingDone) {
+        return;
+      }
+      if (this._bounds !== null) {
+        expand(this);
+        this._bounds = null;
+      }
+      for (var i = 0, ii = this._textDivs.length; i < ii; i++) {
+        var div = this._textDivs[i];
+        var divProperties = this._textDivProperties.get(div);
+        if (divProperties.isWhitespace) {
+          continue;
+        }
+        if (expandDivs) {
+          var transform = '',
+              padding = '';
+          if (divProperties.scale !== 1) {
+            transform = 'scaleX(' + divProperties.scale + ')';
+          }
+          if (divProperties.angle !== 0) {
+            transform = 'rotate(' + divProperties.angle + 'deg) ' + transform;
+          }
+          if (divProperties.paddingLeft !== 0) {
+            padding += ' padding-left: ' + divProperties.paddingLeft / divProperties.scale + 'px;';
+            transform += ' translateX(' + -divProperties.paddingLeft / divProperties.scale + 'px)';
+          }
+          if (divProperties.paddingTop !== 0) {
+            padding += ' padding-top: ' + divProperties.paddingTop + 'px;';
+            transform += ' translateY(' + -divProperties.paddingTop + 'px)';
+          }
+          if (divProperties.paddingRight !== 0) {
+            padding += ' padding-right: ' + divProperties.paddingRight / divProperties.scale + 'px;';
+          }
+          if (divProperties.paddingBottom !== 0) {
+            padding += ' padding-bottom: ' + divProperties.paddingBottom + 'px;';
+          }
+          if (padding !== '') {
+            div.setAttribute('style', divProperties.style + padding);
+          }
+          if (transform !== '') {
+            CustomStyle.setProp('transform', div, transform);
+          }
+        } else {
+          div.style.padding = 0;
+          CustomStyle.setProp('transform', div, divProperties.originalTransform || '');
+        }
+      }
     }
-   }
+  };
+  function renderTextLayer(renderParameters) {
+    var task = new TextLayerRenderTask(renderParameters.textContent, renderParameters.container, renderParameters.viewport, renderParameters.textDivs, renderParameters.enhanceTextSelection);
+    task._render(renderParameters.timeout);
+    return task;
   }
- };
- function renderTextLayer(renderParameters) {
-  var task = new TextLayerRenderTask(renderParameters.textContent, renderParameters.container, renderParameters.viewport, renderParameters.textDivs, renderParameters.enhanceTextSelection);
-  task._render(renderParameters.timeout);
-  return task;
- }
- return renderTextLayer;
+  return renderTextLayer;
 }();
 exports.renderTextLayer = renderTextLayer;

+ 274 - 297
lib/display/webgl.js

@@ -13,61 +13,62 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../shared/util.js');
 var displayDOMUtils = require('./dom_utils.js');
 var shadow = sharedUtil.shadow;
 var getDefaultSetting = displayDOMUtils.getDefaultSetting;
 var WebGLUtils = function WebGLUtilsClosure() {
- function loadShader(gl, code, shaderType) {
-  var shader = gl.createShader(shaderType);
-  gl.shaderSource(shader, code);
-  gl.compileShader(shader);
-  var compiled = gl.getShaderParameter(shader, gl.COMPILE_STATUS);
-  if (!compiled) {
-   var errorMsg = gl.getShaderInfoLog(shader);
-   throw new Error('Error during shader compilation: ' + errorMsg);
+  function loadShader(gl, code, shaderType) {
+    var shader = gl.createShader(shaderType);
+    gl.shaderSource(shader, code);
+    gl.compileShader(shader);
+    var compiled = gl.getShaderParameter(shader, gl.COMPILE_STATUS);
+    if (!compiled) {
+      var errorMsg = gl.getShaderInfoLog(shader);
+      throw new Error('Error during shader compilation: ' + errorMsg);
+    }
+    return shader;
   }
-  return shader;
- }
- function createVertexShader(gl, code) {
-  return loadShader(gl, code, gl.VERTEX_SHADER);
- }
- function createFragmentShader(gl, code) {
-  return loadShader(gl, code, gl.FRAGMENT_SHADER);
- }
- function createProgram(gl, shaders) {
-  var program = gl.createProgram();
-  for (var i = 0, ii = shaders.length; i < ii; ++i) {
-   gl.attachShader(program, shaders[i]);
+  function createVertexShader(gl, code) {
+    return loadShader(gl, code, gl.VERTEX_SHADER);
   }
-  gl.linkProgram(program);
-  var linked = gl.getProgramParameter(program, gl.LINK_STATUS);
-  if (!linked) {
-   var errorMsg = gl.getProgramInfoLog(program);
-   throw new Error('Error during program linking: ' + errorMsg);
+  function createFragmentShader(gl, code) {
+    return loadShader(gl, code, gl.FRAGMENT_SHADER);
   }
-  return program;
- }
- function createTexture(gl, image, textureId) {
-  gl.activeTexture(textureId);
-  var texture = gl.createTexture();
-  gl.bindTexture(gl.TEXTURE_2D, texture);
-  gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
-  gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
-  gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
-  gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);
-  gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, image);
-  return texture;
- }
- var currentGL, currentCanvas;
- function generateGL() {
-  if (currentGL) {
-   return;
+  function createProgram(gl, shaders) {
+    var program = gl.createProgram();
+    for (var i = 0, ii = shaders.length; i < ii; ++i) {
+      gl.attachShader(program, shaders[i]);
+    }
+    gl.linkProgram(program);
+    var linked = gl.getProgramParameter(program, gl.LINK_STATUS);
+    if (!linked) {
+      var errorMsg = gl.getProgramInfoLog(program);
+      throw new Error('Error during program linking: ' + errorMsg);
+    }
+    return program;
   }
-  currentCanvas = document.createElement('canvas');
-  currentGL = currentCanvas.getContext('webgl', { premultipliedalpha: false });
- }
- var smaskVertexShaderCode = '\
+  function createTexture(gl, image, textureId) {
+    gl.activeTexture(textureId);
+    var texture = gl.createTexture();
+    gl.bindTexture(gl.TEXTURE_2D, texture);
+    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
+    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
+    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
+    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);
+    gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, image);
+    return texture;
+  }
+  var currentGL, currentCanvas;
+  function generateGL() {
+    if (currentGL) {
+      return;
+    }
+    currentCanvas = document.createElement('canvas');
+    currentGL = currentCanvas.getContext('webgl', { premultipliedalpha: false });
+  }
+  var smaskVertexShaderCode = '\
   attribute vec2 a_position;                                    \
   attribute vec2 a_texCoord;                                    \
                                                                 \
@@ -81,7 +82,7 @@ var WebGLUtils = function WebGLUtilsClosure() {
                                                                 \
     v_texCoord = a_texCoord;                                    \
   }                                                             ';
- var smaskFragmentShaderCode = '\
+  var smaskFragmentShaderCode = '\
   precision mediump float;                                      \
                                                                 \
   uniform vec4 u_backdrop;                                      \
@@ -109,101 +110,75 @@ var WebGLUtils = function WebGLUtilsClosure() {
     imageColor.rgb *= imageColor.a;                             \
     gl_FragColor = imageColor;                                  \
   }                                                             ';
- var smaskCache = null;
- function initSmaskGL() {
-  var canvas, gl;
-  generateGL();
-  canvas = currentCanvas;
-  currentCanvas = null;
-  gl = currentGL;
-  currentGL = null;
-  var vertexShader = createVertexShader(gl, smaskVertexShaderCode);
-  var fragmentShader = createFragmentShader(gl, smaskFragmentShaderCode);
-  var program = createProgram(gl, [
-   vertexShader,
-   fragmentShader
-  ]);
-  gl.useProgram(program);
-  var cache = {};
-  cache.gl = gl;
-  cache.canvas = canvas;
-  cache.resolutionLocation = gl.getUniformLocation(program, 'u_resolution');
-  cache.positionLocation = gl.getAttribLocation(program, 'a_position');
-  cache.backdropLocation = gl.getUniformLocation(program, 'u_backdrop');
-  cache.subtypeLocation = gl.getUniformLocation(program, 'u_subtype');
-  var texCoordLocation = gl.getAttribLocation(program, 'a_texCoord');
-  var texLayerLocation = gl.getUniformLocation(program, 'u_image');
-  var texMaskLocation = gl.getUniformLocation(program, 'u_mask');
-  var texCoordBuffer = gl.createBuffer();
-  gl.bindBuffer(gl.ARRAY_BUFFER, texCoordBuffer);
-  gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([
-   0.0,
-   0.0,
-   1.0,
-   0.0,
-   0.0,
-   1.0,
-   0.0,
-   1.0,
-   1.0,
-   0.0,
-   1.0,
-   1.0
-  ]), gl.STATIC_DRAW);
-  gl.enableVertexAttribArray(texCoordLocation);
-  gl.vertexAttribPointer(texCoordLocation, 2, gl.FLOAT, false, 0, 0);
-  gl.uniform1i(texLayerLocation, 0);
-  gl.uniform1i(texMaskLocation, 1);
-  smaskCache = cache;
- }
- function composeSMask(layer, mask, properties) {
-  var width = layer.width, height = layer.height;
-  if (!smaskCache) {
-   initSmaskGL();
+  var smaskCache = null;
+  function initSmaskGL() {
+    var canvas, gl;
+    generateGL();
+    canvas = currentCanvas;
+    currentCanvas = null;
+    gl = currentGL;
+    currentGL = null;
+    var vertexShader = createVertexShader(gl, smaskVertexShaderCode);
+    var fragmentShader = createFragmentShader(gl, smaskFragmentShaderCode);
+    var program = createProgram(gl, [vertexShader, fragmentShader]);
+    gl.useProgram(program);
+    var cache = {};
+    cache.gl = gl;
+    cache.canvas = canvas;
+    cache.resolutionLocation = gl.getUniformLocation(program, 'u_resolution');
+    cache.positionLocation = gl.getAttribLocation(program, 'a_position');
+    cache.backdropLocation = gl.getUniformLocation(program, 'u_backdrop');
+    cache.subtypeLocation = gl.getUniformLocation(program, 'u_subtype');
+    var texCoordLocation = gl.getAttribLocation(program, 'a_texCoord');
+    var texLayerLocation = gl.getUniformLocation(program, 'u_image');
+    var texMaskLocation = gl.getUniformLocation(program, 'u_mask');
+    var texCoordBuffer = gl.createBuffer();
+    gl.bindBuffer(gl.ARRAY_BUFFER, texCoordBuffer);
+    gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([0.0, 0.0, 1.0, 0.0, 0.0, 1.0, 0.0, 1.0, 1.0, 0.0, 1.0, 1.0]), gl.STATIC_DRAW);
+    gl.enableVertexAttribArray(texCoordLocation);
+    gl.vertexAttribPointer(texCoordLocation, 2, gl.FLOAT, false, 0, 0);
+    gl.uniform1i(texLayerLocation, 0);
+    gl.uniform1i(texMaskLocation, 1);
+    smaskCache = cache;
   }
-  var cache = smaskCache, canvas = cache.canvas, gl = cache.gl;
-  canvas.width = width;
-  canvas.height = height;
-  gl.viewport(0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight);
-  gl.uniform2f(cache.resolutionLocation, width, height);
-  if (properties.backdrop) {
-   gl.uniform4f(cache.resolutionLocation, properties.backdrop[0], properties.backdrop[1], properties.backdrop[2], 1);
-  } else {
-   gl.uniform4f(cache.resolutionLocation, 0, 0, 0, 0);
+  function composeSMask(layer, mask, properties) {
+    var width = layer.width,
+        height = layer.height;
+    if (!smaskCache) {
+      initSmaskGL();
+    }
+    var cache = smaskCache,
+        canvas = cache.canvas,
+        gl = cache.gl;
+    canvas.width = width;
+    canvas.height = height;
+    gl.viewport(0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight);
+    gl.uniform2f(cache.resolutionLocation, width, height);
+    if (properties.backdrop) {
+      gl.uniform4f(cache.resolutionLocation, properties.backdrop[0], properties.backdrop[1], properties.backdrop[2], 1);
+    } else {
+      gl.uniform4f(cache.resolutionLocation, 0, 0, 0, 0);
+    }
+    gl.uniform1i(cache.subtypeLocation, properties.subtype === 'Luminosity' ? 1 : 0);
+    var texture = createTexture(gl, layer, gl.TEXTURE0);
+    var maskTexture = createTexture(gl, mask, gl.TEXTURE1);
+    var buffer = gl.createBuffer();
+    gl.bindBuffer(gl.ARRAY_BUFFER, buffer);
+    gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([0, 0, width, 0, 0, height, 0, height, width, 0, width, height]), gl.STATIC_DRAW);
+    gl.enableVertexAttribArray(cache.positionLocation);
+    gl.vertexAttribPointer(cache.positionLocation, 2, gl.FLOAT, false, 0, 0);
+    gl.clearColor(0, 0, 0, 0);
+    gl.enable(gl.BLEND);
+    gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);
+    gl.clear(gl.COLOR_BUFFER_BIT);
+    gl.drawArrays(gl.TRIANGLES, 0, 6);
+    gl.flush();
+    gl.deleteTexture(texture);
+    gl.deleteTexture(maskTexture);
+    gl.deleteBuffer(buffer);
+    return canvas;
   }
-  gl.uniform1i(cache.subtypeLocation, properties.subtype === 'Luminosity' ? 1 : 0);
-  var texture = createTexture(gl, layer, gl.TEXTURE0);
-  var maskTexture = createTexture(gl, mask, gl.TEXTURE1);
-  var buffer = gl.createBuffer();
-  gl.bindBuffer(gl.ARRAY_BUFFER, buffer);
-  gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([
-   0,
-   0,
-   width,
-   0,
-   0,
-   height,
-   0,
-   height,
-   width,
-   0,
-   width,
-   height
-  ]), gl.STATIC_DRAW);
-  gl.enableVertexAttribArray(cache.positionLocation);
-  gl.vertexAttribPointer(cache.positionLocation, 2, gl.FLOAT, false, 0, 0);
-  gl.clearColor(0, 0, 0, 0);
-  gl.enable(gl.BLEND);
-  gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);
-  gl.clear(gl.COLOR_BUFFER_BIT);
-  gl.drawArrays(gl.TRIANGLES, 0, 6);
-  gl.flush();
-  gl.deleteTexture(texture);
-  gl.deleteTexture(maskTexture);
-  gl.deleteBuffer(buffer);
-  return canvas;
- }
- var figuresVertexShaderCode = '\
+  var figuresVertexShaderCode = '\
   attribute vec2 a_position;                                    \
   attribute vec3 a_color;                                       \
                                                                 \
@@ -220,7 +195,7 @@ var WebGLUtils = function WebGLUtilsClosure() {
                                                                 \
     v_color = vec4(a_color / 255.0, 1.0);                       \
   }                                                             ';
- var figuresFragmentShaderCode = '\
+  var figuresFragmentShaderCode = '\
   precision mediump float;                                      \
                                                                 \
   varying vec4 v_color;                                         \
@@ -228,166 +203,168 @@ var WebGLUtils = function WebGLUtilsClosure() {
   void main() {                                                 \
     gl_FragColor = v_color;                                     \
   }                                                             ';
- var figuresCache = null;
- function initFiguresGL() {
-  var canvas, gl;
-  generateGL();
-  canvas = currentCanvas;
-  currentCanvas = null;
-  gl = currentGL;
-  currentGL = null;
-  var vertexShader = createVertexShader(gl, figuresVertexShaderCode);
-  var fragmentShader = createFragmentShader(gl, figuresFragmentShaderCode);
-  var program = createProgram(gl, [
-   vertexShader,
-   fragmentShader
-  ]);
-  gl.useProgram(program);
-  var cache = {};
-  cache.gl = gl;
-  cache.canvas = canvas;
-  cache.resolutionLocation = gl.getUniformLocation(program, 'u_resolution');
-  cache.scaleLocation = gl.getUniformLocation(program, 'u_scale');
-  cache.offsetLocation = gl.getUniformLocation(program, 'u_offset');
-  cache.positionLocation = gl.getAttribLocation(program, 'a_position');
-  cache.colorLocation = gl.getAttribLocation(program, 'a_color');
-  figuresCache = cache;
- }
- function drawFigures(width, height, backgroundColor, figures, context) {
-  if (!figuresCache) {
-   initFiguresGL();
-  }
-  var cache = figuresCache, canvas = cache.canvas, gl = cache.gl;
-  canvas.width = width;
-  canvas.height = height;
-  gl.viewport(0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight);
-  gl.uniform2f(cache.resolutionLocation, width, height);
-  var count = 0;
-  var i, ii, rows;
-  for (i = 0, ii = figures.length; i < ii; i++) {
-   switch (figures[i].type) {
-   case 'lattice':
-    rows = figures[i].coords.length / figures[i].verticesPerRow | 0;
-    count += (rows - 1) * (figures[i].verticesPerRow - 1) * 6;
-    break;
-   case 'triangles':
-    count += figures[i].coords.length;
-    break;
-   }
+  var figuresCache = null;
+  function initFiguresGL() {
+    var canvas, gl;
+    generateGL();
+    canvas = currentCanvas;
+    currentCanvas = null;
+    gl = currentGL;
+    currentGL = null;
+    var vertexShader = createVertexShader(gl, figuresVertexShaderCode);
+    var fragmentShader = createFragmentShader(gl, figuresFragmentShaderCode);
+    var program = createProgram(gl, [vertexShader, fragmentShader]);
+    gl.useProgram(program);
+    var cache = {};
+    cache.gl = gl;
+    cache.canvas = canvas;
+    cache.resolutionLocation = gl.getUniformLocation(program, 'u_resolution');
+    cache.scaleLocation = gl.getUniformLocation(program, 'u_scale');
+    cache.offsetLocation = gl.getUniformLocation(program, 'u_offset');
+    cache.positionLocation = gl.getAttribLocation(program, 'a_position');
+    cache.colorLocation = gl.getAttribLocation(program, 'a_color');
+    figuresCache = cache;
   }
-  var coords = new Float32Array(count * 2);
-  var colors = new Uint8Array(count * 3);
-  var coordsMap = context.coords, colorsMap = context.colors;
-  var pIndex = 0, cIndex = 0;
-  for (i = 0, ii = figures.length; i < ii; i++) {
-   var figure = figures[i], ps = figure.coords, cs = figure.colors;
-   switch (figure.type) {
-   case 'lattice':
-    var cols = figure.verticesPerRow;
-    rows = ps.length / cols | 0;
-    for (var row = 1; row < rows; row++) {
-     var offset = row * cols + 1;
-     for (var col = 1; col < cols; col++, offset++) {
-      coords[pIndex] = coordsMap[ps[offset - cols - 1]];
-      coords[pIndex + 1] = coordsMap[ps[offset - cols - 1] + 1];
-      coords[pIndex + 2] = coordsMap[ps[offset - cols]];
-      coords[pIndex + 3] = coordsMap[ps[offset - cols] + 1];
-      coords[pIndex + 4] = coordsMap[ps[offset - 1]];
-      coords[pIndex + 5] = coordsMap[ps[offset - 1] + 1];
-      colors[cIndex] = colorsMap[cs[offset - cols - 1]];
-      colors[cIndex + 1] = colorsMap[cs[offset - cols - 1] + 1];
-      colors[cIndex + 2] = colorsMap[cs[offset - cols - 1] + 2];
-      colors[cIndex + 3] = colorsMap[cs[offset - cols]];
-      colors[cIndex + 4] = colorsMap[cs[offset - cols] + 1];
-      colors[cIndex + 5] = colorsMap[cs[offset - cols] + 2];
-      colors[cIndex + 6] = colorsMap[cs[offset - 1]];
-      colors[cIndex + 7] = colorsMap[cs[offset - 1] + 1];
-      colors[cIndex + 8] = colorsMap[cs[offset - 1] + 2];
-      coords[pIndex + 6] = coords[pIndex + 2];
-      coords[pIndex + 7] = coords[pIndex + 3];
-      coords[pIndex + 8] = coords[pIndex + 4];
-      coords[pIndex + 9] = coords[pIndex + 5];
-      coords[pIndex + 10] = coordsMap[ps[offset]];
-      coords[pIndex + 11] = coordsMap[ps[offset] + 1];
-      colors[cIndex + 9] = colors[cIndex + 3];
-      colors[cIndex + 10] = colors[cIndex + 4];
-      colors[cIndex + 11] = colors[cIndex + 5];
-      colors[cIndex + 12] = colors[cIndex + 6];
-      colors[cIndex + 13] = colors[cIndex + 7];
-      colors[cIndex + 14] = colors[cIndex + 8];
-      colors[cIndex + 15] = colorsMap[cs[offset]];
-      colors[cIndex + 16] = colorsMap[cs[offset] + 1];
-      colors[cIndex + 17] = colorsMap[cs[offset] + 2];
-      pIndex += 12;
-      cIndex += 18;
-     }
+  function drawFigures(width, height, backgroundColor, figures, context) {
+    if (!figuresCache) {
+      initFiguresGL();
     }
-    break;
-   case 'triangles':
-    for (var j = 0, jj = ps.length; j < jj; j++) {
-     coords[pIndex] = coordsMap[ps[j]];
-     coords[pIndex + 1] = coordsMap[ps[j] + 1];
-     colors[cIndex] = colorsMap[cs[j]];
-     colors[cIndex + 1] = colorsMap[cs[j] + 1];
-     colors[cIndex + 2] = colorsMap[cs[j] + 2];
-     pIndex += 2;
-     cIndex += 3;
+    var cache = figuresCache,
+        canvas = cache.canvas,
+        gl = cache.gl;
+    canvas.width = width;
+    canvas.height = height;
+    gl.viewport(0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight);
+    gl.uniform2f(cache.resolutionLocation, width, height);
+    var count = 0;
+    var i, ii, rows;
+    for (i = 0, ii = figures.length; i < ii; i++) {
+      switch (figures[i].type) {
+        case 'lattice':
+          rows = figures[i].coords.length / figures[i].verticesPerRow | 0;
+          count += (rows - 1) * (figures[i].verticesPerRow - 1) * 6;
+          break;
+        case 'triangles':
+          count += figures[i].coords.length;
+          break;
+      }
     }
-    break;
-   }
-  }
-  if (backgroundColor) {
-   gl.clearColor(backgroundColor[0] / 255, backgroundColor[1] / 255, backgroundColor[2] / 255, 1.0);
-  } else {
-   gl.clearColor(0, 0, 0, 0);
-  }
-  gl.clear(gl.COLOR_BUFFER_BIT);
-  var coordsBuffer = gl.createBuffer();
-  gl.bindBuffer(gl.ARRAY_BUFFER, coordsBuffer);
-  gl.bufferData(gl.ARRAY_BUFFER, coords, gl.STATIC_DRAW);
-  gl.enableVertexAttribArray(cache.positionLocation);
-  gl.vertexAttribPointer(cache.positionLocation, 2, gl.FLOAT, false, 0, 0);
-  var colorsBuffer = gl.createBuffer();
-  gl.bindBuffer(gl.ARRAY_BUFFER, colorsBuffer);
-  gl.bufferData(gl.ARRAY_BUFFER, colors, gl.STATIC_DRAW);
-  gl.enableVertexAttribArray(cache.colorLocation);
-  gl.vertexAttribPointer(cache.colorLocation, 3, gl.UNSIGNED_BYTE, false, 0, 0);
-  gl.uniform2f(cache.scaleLocation, context.scaleX, context.scaleY);
-  gl.uniform2f(cache.offsetLocation, context.offsetX, context.offsetY);
-  gl.drawArrays(gl.TRIANGLES, 0, count);
-  gl.flush();
-  gl.deleteBuffer(coordsBuffer);
-  gl.deleteBuffer(colorsBuffer);
-  return canvas;
- }
- function cleanup() {
-  if (smaskCache && smaskCache.canvas) {
-   smaskCache.canvas.width = 0;
-   smaskCache.canvas.height = 0;
+    var coords = new Float32Array(count * 2);
+    var colors = new Uint8Array(count * 3);
+    var coordsMap = context.coords,
+        colorsMap = context.colors;
+    var pIndex = 0,
+        cIndex = 0;
+    for (i = 0, ii = figures.length; i < ii; i++) {
+      var figure = figures[i],
+          ps = figure.coords,
+          cs = figure.colors;
+      switch (figure.type) {
+        case 'lattice':
+          var cols = figure.verticesPerRow;
+          rows = ps.length / cols | 0;
+          for (var row = 1; row < rows; row++) {
+            var offset = row * cols + 1;
+            for (var col = 1; col < cols; col++, offset++) {
+              coords[pIndex] = coordsMap[ps[offset - cols - 1]];
+              coords[pIndex + 1] = coordsMap[ps[offset - cols - 1] + 1];
+              coords[pIndex + 2] = coordsMap[ps[offset - cols]];
+              coords[pIndex + 3] = coordsMap[ps[offset - cols] + 1];
+              coords[pIndex + 4] = coordsMap[ps[offset - 1]];
+              coords[pIndex + 5] = coordsMap[ps[offset - 1] + 1];
+              colors[cIndex] = colorsMap[cs[offset - cols - 1]];
+              colors[cIndex + 1] = colorsMap[cs[offset - cols - 1] + 1];
+              colors[cIndex + 2] = colorsMap[cs[offset - cols - 1] + 2];
+              colors[cIndex + 3] = colorsMap[cs[offset - cols]];
+              colors[cIndex + 4] = colorsMap[cs[offset - cols] + 1];
+              colors[cIndex + 5] = colorsMap[cs[offset - cols] + 2];
+              colors[cIndex + 6] = colorsMap[cs[offset - 1]];
+              colors[cIndex + 7] = colorsMap[cs[offset - 1] + 1];
+              colors[cIndex + 8] = colorsMap[cs[offset - 1] + 2];
+              coords[pIndex + 6] = coords[pIndex + 2];
+              coords[pIndex + 7] = coords[pIndex + 3];
+              coords[pIndex + 8] = coords[pIndex + 4];
+              coords[pIndex + 9] = coords[pIndex + 5];
+              coords[pIndex + 10] = coordsMap[ps[offset]];
+              coords[pIndex + 11] = coordsMap[ps[offset] + 1];
+              colors[cIndex + 9] = colors[cIndex + 3];
+              colors[cIndex + 10] = colors[cIndex + 4];
+              colors[cIndex + 11] = colors[cIndex + 5];
+              colors[cIndex + 12] = colors[cIndex + 6];
+              colors[cIndex + 13] = colors[cIndex + 7];
+              colors[cIndex + 14] = colors[cIndex + 8];
+              colors[cIndex + 15] = colorsMap[cs[offset]];
+              colors[cIndex + 16] = colorsMap[cs[offset] + 1];
+              colors[cIndex + 17] = colorsMap[cs[offset] + 2];
+              pIndex += 12;
+              cIndex += 18;
+            }
+          }
+          break;
+        case 'triangles':
+          for (var j = 0, jj = ps.length; j < jj; j++) {
+            coords[pIndex] = coordsMap[ps[j]];
+            coords[pIndex + 1] = coordsMap[ps[j] + 1];
+            colors[cIndex] = colorsMap[cs[j]];
+            colors[cIndex + 1] = colorsMap[cs[j] + 1];
+            colors[cIndex + 2] = colorsMap[cs[j] + 2];
+            pIndex += 2;
+            cIndex += 3;
+          }
+          break;
+      }
+    }
+    if (backgroundColor) {
+      gl.clearColor(backgroundColor[0] / 255, backgroundColor[1] / 255, backgroundColor[2] / 255, 1.0);
+    } else {
+      gl.clearColor(0, 0, 0, 0);
+    }
+    gl.clear(gl.COLOR_BUFFER_BIT);
+    var coordsBuffer = gl.createBuffer();
+    gl.bindBuffer(gl.ARRAY_BUFFER, coordsBuffer);
+    gl.bufferData(gl.ARRAY_BUFFER, coords, gl.STATIC_DRAW);
+    gl.enableVertexAttribArray(cache.positionLocation);
+    gl.vertexAttribPointer(cache.positionLocation, 2, gl.FLOAT, false, 0, 0);
+    var colorsBuffer = gl.createBuffer();
+    gl.bindBuffer(gl.ARRAY_BUFFER, colorsBuffer);
+    gl.bufferData(gl.ARRAY_BUFFER, colors, gl.STATIC_DRAW);
+    gl.enableVertexAttribArray(cache.colorLocation);
+    gl.vertexAttribPointer(cache.colorLocation, 3, gl.UNSIGNED_BYTE, false, 0, 0);
+    gl.uniform2f(cache.scaleLocation, context.scaleX, context.scaleY);
+    gl.uniform2f(cache.offsetLocation, context.offsetX, context.offsetY);
+    gl.drawArrays(gl.TRIANGLES, 0, count);
+    gl.flush();
+    gl.deleteBuffer(coordsBuffer);
+    gl.deleteBuffer(colorsBuffer);
+    return canvas;
   }
-  if (figuresCache && figuresCache.canvas) {
-   figuresCache.canvas.width = 0;
-   figuresCache.canvas.height = 0;
+  function cleanup() {
+    if (smaskCache && smaskCache.canvas) {
+      smaskCache.canvas.width = 0;
+      smaskCache.canvas.height = 0;
+    }
+    if (figuresCache && figuresCache.canvas) {
+      figuresCache.canvas.width = 0;
+      figuresCache.canvas.height = 0;
+    }
+    smaskCache = null;
+    figuresCache = null;
   }
-  smaskCache = null;
-  figuresCache = null;
- }
- return {
-  get isEnabled() {
-   if (getDefaultSetting('disableWebGL')) {
-    return false;
-   }
-   var enabled = false;
-   try {
-    generateGL();
-    enabled = !!currentGL;
-   } catch (e) {
-   }
-   return shadow(this, 'isEnabled', enabled);
-  },
-  composeSMask: composeSMask,
-  drawFigures: drawFigures,
-  clear: cleanup
- };
+  return {
+    get isEnabled() {
+      if (getDefaultSetting('disableWebGL')) {
+        return false;
+      }
+      var enabled = false;
+      try {
+        generateGL();
+        enabled = !!currentGL;
+      } catch (e) {}
+      return shadow(this, 'isEnabled', enabled);
+    },
+    composeSMask: composeSMask,
+    drawFigures: drawFigures,
+    clear: cleanup
+  };
 }();
 exports.WebGLUtils = WebGLUtils;

+ 3 - 2
lib/pdf.js

@@ -13,8 +13,9 @@
  * limitations under the License.
  */
 'use strict';
-var pdfjsVersion = '1.7.389';
-var pdfjsBuild = '0423bb69';
+
+var pdfjsVersion = '1.7.391';
+var pdfjsBuild = 'cd5acf50';
 var pdfjsSharedUtil = require('./shared/util.js');
 var pdfjsDisplayGlobal = require('./display/global.js');
 var pdfjsDisplayAPI = require('./display/api.js');

+ 4 - 3
lib/pdf.worker.js

@@ -13,10 +13,11 @@
  * limitations under the License.
  */
 'use strict';
-var pdfjsVersion = '1.7.389';
-var pdfjsBuild = '0423bb69';
+
+var pdfjsVersion = '1.7.391';
+var pdfjsBuild = 'cd5acf50';
 var pdfjsCoreWorker = require('./core/worker.js');
 {
- require('./core/network.js');
+  require('./core/network.js');
 }
 exports.WorkerMessageHandler = pdfjsCoreWorker.WorkerMessageHandler;

+ 1338 - 1344
lib/shared/compatibility.js

@@ -13,1414 +13,1408 @@
  * limitations under the License.
  */
 'use strict';
+
 if (typeof PDFJS === 'undefined' || !PDFJS.compatibilityChecked) {
- var globalScope = typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : this;
- var userAgent = typeof navigator !== 'undefined' && navigator.userAgent || '';
- var isAndroid = /Android/.test(userAgent);
- var isAndroidPre3 = /Android\s[0-2][^\d]/.test(userAgent);
- var isAndroidPre5 = /Android\s[0-4][^\d]/.test(userAgent);
- var isChrome = userAgent.indexOf('Chrom') >= 0;
- var isChromeWithRangeBug = /Chrome\/(39|40)\./.test(userAgent);
- var isIOSChrome = userAgent.indexOf('CriOS') >= 0;
- var isIE = userAgent.indexOf('Trident') >= 0;
- var isIOS = /\b(iPad|iPhone|iPod)(?=;)/.test(userAgent);
- var isOpera = userAgent.indexOf('Opera') >= 0;
- var isSafari = /Safari\//.test(userAgent) && !/(Chrome\/|Android\s)/.test(userAgent);
- var hasDOM = typeof window === 'object' && typeof document === 'object';
- if (typeof PDFJS === 'undefined') {
-  globalScope.PDFJS = {};
- }
- PDFJS.compatibilityChecked = true;
- (function checkTypedArrayCompatibility() {
-  if (typeof Uint8Array !== 'undefined') {
-   if (typeof Uint8Array.prototype.subarray === 'undefined') {
-    Uint8Array.prototype.subarray = function subarray(start, end) {
-     return new Uint8Array(this.slice(start, end));
-    };
-    Float32Array.prototype.subarray = function subarray(start, end) {
-     return new Float32Array(this.slice(start, end));
-    };
-   }
-   if (typeof Float64Array === 'undefined') {
-    globalScope.Float64Array = Float32Array;
-   }
-   return;
-  }
-  function subarray(start, end) {
-   return new TypedArray(this.slice(start, end));
-  }
-  function setArrayOffset(array, offset) {
-   if (arguments.length < 2) {
-    offset = 0;
-   }
-   for (var i = 0, n = array.length; i < n; ++i, ++offset) {
-    this[offset] = array[i] & 0xFF;
-   }
+  var globalScope = typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : undefined;
+  var userAgent = typeof navigator !== 'undefined' && navigator.userAgent || '';
+  var isAndroid = /Android/.test(userAgent);
+  var isAndroidPre3 = /Android\s[0-2][^\d]/.test(userAgent);
+  var isAndroidPre5 = /Android\s[0-4][^\d]/.test(userAgent);
+  var isChrome = userAgent.indexOf('Chrom') >= 0;
+  var isChromeWithRangeBug = /Chrome\/(39|40)\./.test(userAgent);
+  var isIOSChrome = userAgent.indexOf('CriOS') >= 0;
+  var isIE = userAgent.indexOf('Trident') >= 0;
+  var isIOS = /\b(iPad|iPhone|iPod)(?=;)/.test(userAgent);
+  var isOpera = userAgent.indexOf('Opera') >= 0;
+  var isSafari = /Safari\//.test(userAgent) && !/(Chrome\/|Android\s)/.test(userAgent);
+  var hasDOM = typeof window === 'object' && typeof document === 'object';
+  if (typeof PDFJS === 'undefined') {
+    globalScope.PDFJS = {};
   }
-  function TypedArray(arg1) {
-   var result, i, n;
-   if (typeof arg1 === 'number') {
-    result = [];
-    for (i = 0; i < arg1; ++i) {
-     result[i] = 0;
-    }
-   } else if ('slice' in arg1) {
-    result = arg1.slice(0);
-   } else {
-    result = [];
-    for (i = 0, n = arg1.length; i < n; ++i) {
-     result[i] = arg1[i];
+  PDFJS.compatibilityChecked = true;
+  (function checkTypedArrayCompatibility() {
+    if (typeof Uint8Array !== 'undefined') {
+      if (typeof Uint8Array.prototype.subarray === 'undefined') {
+        Uint8Array.prototype.subarray = function subarray(start, end) {
+          return new Uint8Array(this.slice(start, end));
+        };
+        Float32Array.prototype.subarray = function subarray(start, end) {
+          return new Float32Array(this.slice(start, end));
+        };
+      }
+      if (typeof Float64Array === 'undefined') {
+        globalScope.Float64Array = Float32Array;
+      }
+      return;
     }
-   }
-   result.subarray = subarray;
-   result.buffer = result;
-   result.byteLength = result.length;
-   result.set = setArrayOffset;
-   if (typeof arg1 === 'object' && arg1.buffer) {
-    result.buffer = arg1.buffer;
-   }
-   return result;
-  }
-  globalScope.Uint8Array = TypedArray;
-  globalScope.Int8Array = TypedArray;
-  globalScope.Uint32Array = TypedArray;
-  globalScope.Int32Array = TypedArray;
-  globalScope.Uint16Array = TypedArray;
-  globalScope.Float32Array = TypedArray;
-  globalScope.Float64Array = TypedArray;
- }());
- (function normalizeURLObject() {
-  if (!globalScope.URL) {
-   globalScope.URL = globalScope.webkitURL;
-  }
- }());
- (function checkObjectDefinePropertyCompatibility() {
-  if (typeof Object.defineProperty !== 'undefined') {
-   var definePropertyPossible = true;
-   try {
-    if (hasDOM) {
-     Object.defineProperty(new Image(), 'id', { value: 'test' });
+    function subarray(start, end) {
+      return new TypedArray(this.slice(start, end));
     }
-    var Test = function Test() {
-    };
-    Test.prototype = {
-     get id() {
-     }
-    };
-    Object.defineProperty(new Test(), 'id', {
-     value: '',
-     configurable: true,
-     enumerable: true,
-     writable: false
-    });
-   } catch (e) {
-    definePropertyPossible = false;
-   }
-   if (definePropertyPossible) {
-    return;
-   }
-  }
-  Object.defineProperty = function objectDefineProperty(obj, name, def) {
-   delete obj[name];
-   if ('get' in def) {
-    obj.__defineGetter__(name, def['get']);
-   }
-   if ('set' in def) {
-    obj.__defineSetter__(name, def['set']);
-   }
-   if ('value' in def) {
-    obj.__defineSetter__(name, function objectDefinePropertySetter(value) {
-     this.__defineGetter__(name, function objectDefinePropertyGetter() {
-      return value;
-     });
-     return value;
-    });
-    obj[name] = def.value;
-   }
-  };
- }());
- (function checkXMLHttpRequestResponseCompatibility() {
-  if (typeof XMLHttpRequest === 'undefined') {
-   return;
-  }
-  var xhrPrototype = XMLHttpRequest.prototype;
-  var xhr = new XMLHttpRequest();
-  if (!('overrideMimeType' in xhr)) {
-   Object.defineProperty(xhrPrototype, 'overrideMimeType', {
-    value: function xmlHttpRequestOverrideMimeType(mimeType) {
+    function setArrayOffset(array, offset) {
+      if (arguments.length < 2) {
+        offset = 0;
+      }
+      for (var i = 0, n = array.length; i < n; ++i, ++offset) {
+        this[offset] = array[i] & 0xFF;
+      }
     }
-   });
-  }
-  if ('responseType' in xhr) {
-   return;
-  }
-  Object.defineProperty(xhrPrototype, 'responseType', {
-   get: function xmlHttpRequestGetResponseType() {
-    return this._responseType || 'text';
-   },
-   set: function xmlHttpRequestSetResponseType(value) {
-    if (value === 'text' || value === 'arraybuffer') {
-     this._responseType = value;
-     if (value === 'arraybuffer' && typeof this.overrideMimeType === 'function') {
-      this.overrideMimeType('text/plain; charset=x-user-defined');
-     }
+    function TypedArray(arg1) {
+      var result, i, n;
+      if (typeof arg1 === 'number') {
+        result = [];
+        for (i = 0; i < arg1; ++i) {
+          result[i] = 0;
+        }
+      } else if ('slice' in arg1) {
+        result = arg1.slice(0);
+      } else {
+        result = [];
+        for (i = 0, n = arg1.length; i < n; ++i) {
+          result[i] = arg1[i];
+        }
+      }
+      result.subarray = subarray;
+      result.buffer = result;
+      result.byteLength = result.length;
+      result.set = setArrayOffset;
+      if (typeof arg1 === 'object' && arg1.buffer) {
+        result.buffer = arg1.buffer;
+      }
+      return result;
     }
-   }
-  });
-  if (typeof VBArray !== 'undefined') {
-   Object.defineProperty(xhrPrototype, 'response', {
-    get: function xmlHttpRequestResponseGet() {
-     if (this.responseType === 'arraybuffer') {
-      return new Uint8Array(new VBArray(this.responseBody).toArray());
-     }
-     return this.responseText;
+    globalScope.Uint8Array = TypedArray;
+    globalScope.Int8Array = TypedArray;
+    globalScope.Uint32Array = TypedArray;
+    globalScope.Int32Array = TypedArray;
+    globalScope.Uint16Array = TypedArray;
+    globalScope.Float32Array = TypedArray;
+    globalScope.Float64Array = TypedArray;
+  })();
+  (function normalizeURLObject() {
+    if (!globalScope.URL) {
+      globalScope.URL = globalScope.webkitURL;
     }
-   });
-   return;
-  }
-  Object.defineProperty(xhrPrototype, 'response', {
-   get: function xmlHttpRequestResponseGet() {
-    if (this.responseType !== 'arraybuffer') {
-     return this.responseText;
+  })();
+  (function checkObjectDefinePropertyCompatibility() {
+    if (typeof Object.defineProperty !== 'undefined') {
+      var definePropertyPossible = true;
+      try {
+        if (hasDOM) {
+          Object.defineProperty(new Image(), 'id', { value: 'test' });
+        }
+        var Test = function Test() {};
+        Test.prototype = {
+          get id() {}
+        };
+        Object.defineProperty(new Test(), 'id', {
+          value: '',
+          configurable: true,
+          enumerable: true,
+          writable: false
+        });
+      } catch (e) {
+        definePropertyPossible = false;
+      }
+      if (definePropertyPossible) {
+        return;
+      }
     }
-    var text = this.responseText;
-    var i, n = text.length;
-    var result = new Uint8Array(n);
-    for (i = 0; i < n; ++i) {
-     result[i] = text.charCodeAt(i) & 0xFF;
+    Object.defineProperty = function objectDefineProperty(obj, name, def) {
+      delete obj[name];
+      if ('get' in def) {
+        obj.__defineGetter__(name, def['get']);
+      }
+      if ('set' in def) {
+        obj.__defineSetter__(name, def['set']);
+      }
+      if ('value' in def) {
+        obj.__defineSetter__(name, function objectDefinePropertySetter(value) {
+          this.__defineGetter__(name, function objectDefinePropertyGetter() {
+            return value;
+          });
+          return value;
+        });
+        obj[name] = def.value;
+      }
+    };
+  })();
+  (function checkXMLHttpRequestResponseCompatibility() {
+    if (typeof XMLHttpRequest === 'undefined') {
+      return;
     }
-    return result.buffer;
-   }
-  });
- }());
- (function checkWindowBtoaCompatibility() {
-  if ('btoa' in globalScope) {
-   return;
-  }
-  var digits = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
-  globalScope.btoa = function (chars) {
-   var buffer = '';
-   var i, n;
-   for (i = 0, n = chars.length; i < n; i += 3) {
-    var b1 = chars.charCodeAt(i) & 0xFF;
-    var b2 = chars.charCodeAt(i + 1) & 0xFF;
-    var b3 = chars.charCodeAt(i + 2) & 0xFF;
-    var d1 = b1 >> 2, d2 = (b1 & 3) << 4 | b2 >> 4;
-    var d3 = i + 1 < n ? (b2 & 0xF) << 2 | b3 >> 6 : 64;
-    var d4 = i + 2 < n ? b3 & 0x3F : 64;
-    buffer += digits.charAt(d1) + digits.charAt(d2) + digits.charAt(d3) + digits.charAt(d4);
-   }
-   return buffer;
-  };
- }());
- (function checkWindowAtobCompatibility() {
-  if ('atob' in globalScope) {
-   return;
-  }
-  var digits = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
-  globalScope.atob = function (input) {
-   input = input.replace(/=+$/, '');
-   if (input.length % 4 === 1) {
-    throw new Error('bad atob input');
-   }
-   for (var bc = 0, bs, buffer, idx = 0, output = ''; buffer = input.charAt(idx++); ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer, bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0) {
-    buffer = digits.indexOf(buffer);
-   }
-   return output;
-  };
- }());
- (function checkFunctionPrototypeBindCompatibility() {
-  if (typeof Function.prototype.bind !== 'undefined') {
-   return;
-  }
-  Function.prototype.bind = function functionPrototypeBind(obj) {
-   var fn = this, headArgs = Array.prototype.slice.call(arguments, 1);
-   var bound = function functionPrototypeBindBound() {
-    var args = headArgs.concat(Array.prototype.slice.call(arguments));
-    return fn.apply(obj, args);
-   };
-   return bound;
-  };
- }());
- (function checkDatasetProperty() {
-  if (!hasDOM) {
-   return;
-  }
-  var div = document.createElement('div');
-  if ('dataset' in div) {
-   return;
-  }
-  Object.defineProperty(HTMLElement.prototype, 'dataset', {
-   get: function () {
-    if (this._dataset) {
-     return this._dataset;
+    var xhrPrototype = XMLHttpRequest.prototype;
+    var xhr = new XMLHttpRequest();
+    if (!('overrideMimeType' in xhr)) {
+      Object.defineProperty(xhrPrototype, 'overrideMimeType', {
+        value: function xmlHttpRequestOverrideMimeType(mimeType) {}
+      });
     }
-    var dataset = {};
-    for (var j = 0, jj = this.attributes.length; j < jj; j++) {
-     var attribute = this.attributes[j];
-     if (attribute.name.substring(0, 5) !== 'data-') {
-      continue;
-     }
-     var key = attribute.name.substring(5).replace(/\-([a-z])/g, function (all, ch) {
-      return ch.toUpperCase();
-     });
-     dataset[key] = attribute.value;
+    if ('responseType' in xhr) {
+      return;
     }
-    Object.defineProperty(this, '_dataset', {
-     value: dataset,
-     writable: false,
-     enumerable: false
+    Object.defineProperty(xhrPrototype, 'responseType', {
+      get: function xmlHttpRequestGetResponseType() {
+        return this._responseType || 'text';
+      },
+      set: function xmlHttpRequestSetResponseType(value) {
+        if (value === 'text' || value === 'arraybuffer') {
+          this._responseType = value;
+          if (value === 'arraybuffer' && typeof this.overrideMimeType === 'function') {
+            this.overrideMimeType('text/plain; charset=x-user-defined');
+          }
+        }
+      }
     });
-    return dataset;
-   },
-   enumerable: true
-  });
- }());
- (function checkClassListProperty() {
-  function changeList(element, itemName, add, remove) {
-   var s = element.className || '';
-   var list = s.split(/\s+/g);
-   if (list[0] === '') {
-    list.shift();
-   }
-   var index = list.indexOf(itemName);
-   if (index < 0 && add) {
-    list.push(itemName);
-   }
-   if (index >= 0 && remove) {
-    list.splice(index, 1);
-   }
-   element.className = list.join(' ');
-   return index >= 0;
-  }
-  if (!hasDOM) {
-   return;
-  }
-  var div = document.createElement('div');
-  if ('classList' in div) {
-   return;
-  }
-  var classListPrototype = {
-   add: function (name) {
-    changeList(this.element, name, true, false);
-   },
-   contains: function (name) {
-    return changeList(this.element, name, false, false);
-   },
-   remove: function (name) {
-    changeList(this.element, name, false, true);
-   },
-   toggle: function (name) {
-    changeList(this.element, name, true, true);
-   }
-  };
-  Object.defineProperty(HTMLElement.prototype, 'classList', {
-   get: function () {
-    if (this._classList) {
-     return this._classList;
+    if (typeof VBArray !== 'undefined') {
+      Object.defineProperty(xhrPrototype, 'response', {
+        get: function xmlHttpRequestResponseGet() {
+          if (this.responseType === 'arraybuffer') {
+            return new Uint8Array(new VBArray(this.responseBody).toArray());
+          }
+          return this.responseText;
+        }
+      });
+      return;
     }
-    var classList = Object.create(classListPrototype, {
-     element: {
-      value: this,
-      writable: false,
-      enumerable: true
-     }
-    });
-    Object.defineProperty(this, '_classList', {
-     value: classList,
-     writable: false,
-     enumerable: false
-    });
-    return classList;
-   },
-   enumerable: true
-  });
- }());
- (function checkWorkerConsoleCompatibility() {
-  if (typeof importScripts === 'undefined' || 'console' in globalScope) {
-   return;
-  }
-  var consoleTimer = {};
-  var workerConsole = {
-   log: function log() {
-    var args = Array.prototype.slice.call(arguments);
-    globalScope.postMessage({
-     targetName: 'main',
-     action: 'console_log',
-     data: args
-    });
-   },
-   error: function error() {
-    var args = Array.prototype.slice.call(arguments);
-    globalScope.postMessage({
-     targetName: 'main',
-     action: 'console_error',
-     data: args
+    Object.defineProperty(xhrPrototype, 'response', {
+      get: function xmlHttpRequestResponseGet() {
+        if (this.responseType !== 'arraybuffer') {
+          return this.responseText;
+        }
+        var text = this.responseText;
+        var i,
+            n = text.length;
+        var result = new Uint8Array(n);
+        for (i = 0; i < n; ++i) {
+          result[i] = text.charCodeAt(i) & 0xFF;
+        }
+        return result.buffer;
+      }
     });
-   },
-   time: function time(name) {
-    consoleTimer[name] = Date.now();
-   },
-   timeEnd: function timeEnd(name) {
-    var time = consoleTimer[name];
-    if (!time) {
-     throw new Error('Unknown timer name ' + name);
-    }
-    this.log('Timer:', name, Date.now() - time);
-   }
-  };
-  globalScope.console = workerConsole;
- }());
- (function checkConsoleCompatibility() {
-  if (!hasDOM) {
-   return;
-  }
-  if (!('console' in window)) {
-   window.console = {
-    log: function () {
-    },
-    error: function () {
-    },
-    warn: function () {
+  })();
+  (function checkWindowBtoaCompatibility() {
+    if ('btoa' in globalScope) {
+      return;
     }
-   };
-   return;
-  }
-  if (!('bind' in console.log)) {
-   console.log = function (fn) {
-    return function (msg) {
-     return fn(msg);
-    };
-   }(console.log);
-   console.error = function (fn) {
-    return function (msg) {
-     return fn(msg);
-    };
-   }(console.error);
-   console.warn = function (fn) {
-    return function (msg) {
-     return fn(msg);
-    };
-   }(console.warn);
-   return;
-  }
- }());
- (function checkOnClickCompatibility() {
-  function ignoreIfTargetDisabled(event) {
-   if (isDisabled(event.target)) {
-    event.stopPropagation();
-   }
-  }
-  function isDisabled(node) {
-   return node.disabled || node.parentNode && isDisabled(node.parentNode);
-  }
-  if (isOpera) {
-   document.addEventListener('click', ignoreIfTargetDisabled, true);
-  }
- }());
- (function checkOnBlobSupport() {
-  if (isIE || isIOSChrome) {
-   PDFJS.disableCreateObjectURL = true;
-  }
- }());
- (function checkNavigatorLanguage() {
-  if (typeof navigator === 'undefined') {
-   return;
-  }
-  if ('language' in navigator) {
-   return;
-  }
-  PDFJS.locale = navigator.userLanguage || 'en-US';
- }());
- (function checkRangeRequests() {
-  if (isSafari || isAndroidPre3 || isChromeWithRangeBug || isIOS) {
-   PDFJS.disableRange = true;
-   PDFJS.disableStream = true;
-  }
- }());
- (function checkHistoryManipulation() {
-  if (!hasDOM) {
-   return;
-  }
-  if (!history.pushState || isAndroidPre3) {
-   PDFJS.disableHistory = true;
-  }
- }());
- (function checkSetPresenceInImageData() {
-  if (!hasDOM) {
-   return;
-  }
-  if (window.CanvasPixelArray) {
-   if (typeof window.CanvasPixelArray.prototype.set !== 'function') {
-    window.CanvasPixelArray.prototype.set = function (arr) {
-     for (var i = 0, ii = this.length; i < ii; i++) {
-      this[i] = arr[i];
-     }
-    };
-   }
-  } else {
-   var polyfill = false, versionMatch;
-   if (isChrome) {
-    versionMatch = userAgent.match(/Chrom(e|ium)\/([0-9]+)\./);
-    polyfill = versionMatch && parseInt(versionMatch[2]) < 21;
-   } else if (isAndroid) {
-    polyfill = isAndroidPre5;
-   } else if (isSafari) {
-    versionMatch = userAgent.match(/Version\/([0-9]+)\.([0-9]+)\.([0-9]+) Safari\//);
-    polyfill = versionMatch && parseInt(versionMatch[1]) < 6;
-   }
-   if (polyfill) {
-    var contextPrototype = window.CanvasRenderingContext2D.prototype;
-    var createImageData = contextPrototype.createImageData;
-    contextPrototype.createImageData = function (w, h) {
-     var imageData = createImageData.call(this, w, h);
-     imageData.data.set = function (arr) {
-      for (var i = 0, ii = this.length; i < ii; i++) {
-       this[i] = arr[i];
+    var digits = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
+    globalScope.btoa = function (chars) {
+      var buffer = '';
+      var i, n;
+      for (i = 0, n = chars.length; i < n; i += 3) {
+        var b1 = chars.charCodeAt(i) & 0xFF;
+        var b2 = chars.charCodeAt(i + 1) & 0xFF;
+        var b3 = chars.charCodeAt(i + 2) & 0xFF;
+        var d1 = b1 >> 2,
+            d2 = (b1 & 3) << 4 | b2 >> 4;
+        var d3 = i + 1 < n ? (b2 & 0xF) << 2 | b3 >> 6 : 64;
+        var d4 = i + 2 < n ? b3 & 0x3F : 64;
+        buffer += digits.charAt(d1) + digits.charAt(d2) + digits.charAt(d3) + digits.charAt(d4);
       }
-     };
-     return imageData;
+      return buffer;
     };
-    contextPrototype = null;
-   }
-  }
- }());
- (function checkRequestAnimationFrame() {
-  function fakeRequestAnimationFrame(callback) {
-   window.setTimeout(callback, 20);
-  }
-  if (!hasDOM) {
-   return;
-  }
-  if (isIOS) {
-   window.requestAnimationFrame = fakeRequestAnimationFrame;
-   return;
-  }
-  if ('requestAnimationFrame' in window) {
-   return;
-  }
-  window.requestAnimationFrame = window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || fakeRequestAnimationFrame;
- }());
- (function checkCanvasSizeLimitation() {
-  if (isIOS || isAndroid) {
-   PDFJS.maxCanvasPixels = 5242880;
-  }
- }());
- (function checkFullscreenSupport() {
-  if (!hasDOM) {
-   return;
-  }
-  if (isIE && window.parent !== window) {
-   PDFJS.disableFullscreen = true;
-  }
- }());
- (function checkCurrentScript() {
-  if (!hasDOM) {
-   return;
-  }
-  if ('currentScript' in document) {
-   return;
-  }
-  Object.defineProperty(document, 'currentScript', {
-   get: function () {
-    var scripts = document.getElementsByTagName('script');
-    return scripts[scripts.length - 1];
-   },
-   enumerable: true,
-   configurable: true
-  });
- }());
- (function checkInputTypeNumberAssign() {
-  if (!hasDOM) {
-   return;
-  }
-  var el = document.createElement('input');
-  try {
-   el.type = 'number';
-  } catch (ex) {
-   var inputProto = el.constructor.prototype;
-   var typeProperty = Object.getOwnPropertyDescriptor(inputProto, 'type');
-   Object.defineProperty(inputProto, 'type', {
-    get: function () {
-     return typeProperty.get.call(this);
-    },
-    set: function (value) {
-     typeProperty.set.call(this, value === 'number' ? 'text' : value);
-    },
-    enumerable: true,
-    configurable: true
-   });
-  }
- }());
- (function checkDocumentReadyState() {
-  if (!hasDOM) {
-   return;
-  }
-  if (!document.attachEvent) {
-   return;
-  }
-  var documentProto = document.constructor.prototype;
-  var readyStateProto = Object.getOwnPropertyDescriptor(documentProto, 'readyState');
-  Object.defineProperty(documentProto, 'readyState', {
-   get: function () {
-    var value = readyStateProto.get.call(this);
-    return value === 'interactive' ? 'loading' : value;
-   },
-   set: function (value) {
-    readyStateProto.set.call(this, value);
-   },
-   enumerable: true,
-   configurable: true
-  });
- }());
- (function checkChildNodeRemove() {
-  if (!hasDOM) {
-   return;
-  }
-  if (typeof Element.prototype.remove !== 'undefined') {
-   return;
-  }
-  Element.prototype.remove = function () {
-   if (this.parentNode) {
-    this.parentNode.removeChild(this);
-   }
-  };
- }());
- (function checkPromise() {
-  if (globalScope.Promise) {
-   if (typeof globalScope.Promise.all !== 'function') {
-    globalScope.Promise.all = function (iterable) {
-     var count = 0, results = [], resolve, reject;
-     var promise = new globalScope.Promise(function (resolve_, reject_) {
-      resolve = resolve_;
-      reject = reject_;
-     });
-     iterable.forEach(function (p, i) {
-      count++;
-      p.then(function (result) {
-       results[i] = result;
-       count--;
-       if (count === 0) {
-        resolve(results);
-       }
-      }, reject);
-     });
-     if (count === 0) {
-      resolve(results);
-     }
-     return promise;
-    };
-   }
-   if (typeof globalScope.Promise.resolve !== 'function') {
-    globalScope.Promise.resolve = function (value) {
-     return new globalScope.Promise(function (resolve) {
-      resolve(value);
-     });
-    };
-   }
-   if (typeof globalScope.Promise.reject !== 'function') {
-    globalScope.Promise.reject = function (reason) {
-     return new globalScope.Promise(function (resolve, reject) {
-      reject(reason);
-     });
+  })();
+  (function checkWindowAtobCompatibility() {
+    if ('atob' in globalScope) {
+      return;
+    }
+    var digits = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
+    globalScope.atob = function (input) {
+      input = input.replace(/=+$/, '');
+      if (input.length % 4 === 1) {
+        throw new Error('bad atob input');
+      }
+      for (var bc = 0, bs, buffer, idx = 0, output = ''; buffer = input.charAt(idx++); ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer, bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0) {
+        buffer = digits.indexOf(buffer);
+      }
+      return output;
     };
-   }
-   if (typeof globalScope.Promise.prototype.catch !== 'function') {
-    globalScope.Promise.prototype.catch = function (onReject) {
-     return globalScope.Promise.prototype.then(undefined, onReject);
+  })();
+  (function checkFunctionPrototypeBindCompatibility() {
+    if (typeof Function.prototype.bind !== 'undefined') {
+      return;
+    }
+    Function.prototype.bind = function functionPrototypeBind(obj) {
+      var fn = this,
+          headArgs = Array.prototype.slice.call(arguments, 1);
+      var bound = function functionPrototypeBindBound() {
+        var args = headArgs.concat(Array.prototype.slice.call(arguments));
+        return fn.apply(obj, args);
+      };
+      return bound;
     };
-   }
-   return;
-  }
-  var STATUS_PENDING = 0;
-  var STATUS_RESOLVED = 1;
-  var STATUS_REJECTED = 2;
-  var REJECTION_TIMEOUT = 500;
-  var HandlerManager = {
-   handlers: [],
-   running: false,
-   unhandledRejections: [],
-   pendingRejectionCheck: false,
-   scheduleHandlers: function scheduleHandlers(promise) {
-    if (promise._status === STATUS_PENDING) {
-     return;
+  })();
+  (function checkDatasetProperty() {
+    if (!hasDOM) {
+      return;
     }
-    this.handlers = this.handlers.concat(promise._handlers);
-    promise._handlers = [];
-    if (this.running) {
-     return;
+    var div = document.createElement('div');
+    if ('dataset' in div) {
+      return;
     }
-    this.running = true;
-    setTimeout(this.runHandlers.bind(this), 0);
-   },
-   runHandlers: function runHandlers() {
-    var RUN_TIMEOUT = 1;
-    var timeoutAt = Date.now() + RUN_TIMEOUT;
-    while (this.handlers.length > 0) {
-     var handler = this.handlers.shift();
-     var nextStatus = handler.thisPromise._status;
-     var nextValue = handler.thisPromise._value;
-     try {
-      if (nextStatus === STATUS_RESOLVED) {
-       if (typeof handler.onResolve === 'function') {
-        nextValue = handler.onResolve(nextValue);
-       }
-      } else if (typeof handler.onReject === 'function') {
-       nextValue = handler.onReject(nextValue);
-       nextStatus = STATUS_RESOLVED;
-       if (handler.thisPromise._unhandledRejection) {
-        this.removeUnhandeledRejection(handler.thisPromise);
-       }
+    Object.defineProperty(HTMLElement.prototype, 'dataset', {
+      get: function () {
+        if (this._dataset) {
+          return this._dataset;
+        }
+        var dataset = {};
+        for (var j = 0, jj = this.attributes.length; j < jj; j++) {
+          var attribute = this.attributes[j];
+          if (attribute.name.substring(0, 5) !== 'data-') {
+            continue;
+          }
+          var key = attribute.name.substring(5).replace(/\-([a-z])/g, function (all, ch) {
+            return ch.toUpperCase();
+          });
+          dataset[key] = attribute.value;
+        }
+        Object.defineProperty(this, '_dataset', {
+          value: dataset,
+          writable: false,
+          enumerable: false
+        });
+        return dataset;
+      },
+      enumerable: true
+    });
+  })();
+  (function checkClassListProperty() {
+    function changeList(element, itemName, add, remove) {
+      var s = element.className || '';
+      var list = s.split(/\s+/g);
+      if (list[0] === '') {
+        list.shift();
+      }
+      var index = list.indexOf(itemName);
+      if (index < 0 && add) {
+        list.push(itemName);
       }
-     } catch (ex) {
-      nextStatus = STATUS_REJECTED;
-      nextValue = ex;
-     }
-     handler.nextPromise._updateStatus(nextStatus, nextValue);
-     if (Date.now() >= timeoutAt) {
-      break;
-     }
+      if (index >= 0 && remove) {
+        list.splice(index, 1);
+      }
+      element.className = list.join(' ');
+      return index >= 0;
     }
-    if (this.handlers.length > 0) {
-     setTimeout(this.runHandlers.bind(this), 0);
-     return;
+    if (!hasDOM) {
+      return;
     }
-    this.running = false;
-   },
-   addUnhandledRejection: function addUnhandledRejection(promise) {
-    this.unhandledRejections.push({
-     promise: promise,
-     time: Date.now()
-    });
-    this.scheduleRejectionCheck();
-   },
-   removeUnhandeledRejection: function removeUnhandeledRejection(promise) {
-    promise._unhandledRejection = false;
-    for (var i = 0; i < this.unhandledRejections.length; i++) {
-     if (this.unhandledRejections[i].promise === promise) {
-      this.unhandledRejections.splice(i);
-      i--;
-     }
+    var div = document.createElement('div');
+    if ('classList' in div) {
+      return;
     }
-   },
-   scheduleRejectionCheck: function scheduleRejectionCheck() {
-    if (this.pendingRejectionCheck) {
-     return;
+    var classListPrototype = {
+      add: function (name) {
+        changeList(this.element, name, true, false);
+      },
+      contains: function (name) {
+        return changeList(this.element, name, false, false);
+      },
+      remove: function (name) {
+        changeList(this.element, name, false, true);
+      },
+      toggle: function (name) {
+        changeList(this.element, name, true, true);
+      }
+    };
+    Object.defineProperty(HTMLElement.prototype, 'classList', {
+      get: function () {
+        if (this._classList) {
+          return this._classList;
+        }
+        var classList = Object.create(classListPrototype, {
+          element: {
+            value: this,
+            writable: false,
+            enumerable: true
+          }
+        });
+        Object.defineProperty(this, '_classList', {
+          value: classList,
+          writable: false,
+          enumerable: false
+        });
+        return classList;
+      },
+      enumerable: true
+    });
+  })();
+  (function checkWorkerConsoleCompatibility() {
+    if (typeof importScripts === 'undefined' || 'console' in globalScope) {
+      return;
     }
-    this.pendingRejectionCheck = true;
-    setTimeout(function rejectionCheck() {
-     this.pendingRejectionCheck = false;
-     var now = Date.now();
-     for (var i = 0; i < this.unhandledRejections.length; i++) {
-      if (now - this.unhandledRejections[i].time > REJECTION_TIMEOUT) {
-       var unhandled = this.unhandledRejections[i].promise._value;
-       var msg = 'Unhandled rejection: ' + unhandled;
-       if (unhandled.stack) {
-        msg += '\n' + unhandled.stack;
-       }
-       try {
-        throw new Error(msg);
-       } catch (_) {
-        console.warn(msg);
-       }
-       this.unhandledRejections.splice(i);
-       i--;
+    var consoleTimer = {};
+    var workerConsole = {
+      log: function log() {
+        var args = Array.prototype.slice.call(arguments);
+        globalScope.postMessage({
+          targetName: 'main',
+          action: 'console_log',
+          data: args
+        });
+      },
+      error: function error() {
+        var args = Array.prototype.slice.call(arguments);
+        globalScope.postMessage({
+          targetName: 'main',
+          action: 'console_error',
+          data: args
+        });
+      },
+      time: function time(name) {
+        consoleTimer[name] = Date.now();
+      },
+      timeEnd: function timeEnd(name) {
+        var time = consoleTimer[name];
+        if (!time) {
+          throw new Error('Unknown timer name ' + name);
+        }
+        this.log('Timer:', name, Date.now() - time);
       }
-     }
-     if (this.unhandledRejections.length) {
-      this.scheduleRejectionCheck();
-     }
-    }.bind(this), REJECTION_TIMEOUT);
-   }
-  };
-  var Promise = function Promise(resolver) {
-   this._status = STATUS_PENDING;
-   this._handlers = [];
-   try {
-    resolver.call(this, this._resolve.bind(this), this._reject.bind(this));
-   } catch (e) {
-    this._reject(e);
-   }
-  };
-  Promise.all = function Promise_all(promises) {
-   var resolveAll, rejectAll;
-   var deferred = new Promise(function (resolve, reject) {
-    resolveAll = resolve;
-    rejectAll = reject;
-   });
-   var unresolved = promises.length;
-   var results = [];
-   if (unresolved === 0) {
-    resolveAll(results);
-    return deferred;
-   }
-   function reject(reason) {
-    if (deferred._status === STATUS_REJECTED) {
-     return;
+    };
+    globalScope.console = workerConsole;
+  })();
+  (function checkConsoleCompatibility() {
+    if (!hasDOM) {
+      return;
+    }
+    if (!('console' in window)) {
+      window.console = {
+        log: function () {},
+        error: function () {},
+        warn: function () {}
+      };
+      return;
     }
-    results = [];
-    rejectAll(reason);
-   }
-   for (var i = 0, ii = promises.length; i < ii; ++i) {
-    var promise = promises[i];
-    var resolve = function (i) {
-     return function (value) {
-      if (deferred._status === STATUS_REJECTED) {
-       return;
+    if (!('bind' in console.log)) {
+      console.log = function (fn) {
+        return function (msg) {
+          return fn(msg);
+        };
+      }(console.log);
+      console.error = function (fn) {
+        return function (msg) {
+          return fn(msg);
+        };
+      }(console.error);
+      console.warn = function (fn) {
+        return function (msg) {
+          return fn(msg);
+        };
+      }(console.warn);
+      return;
+    }
+  })();
+  (function checkOnClickCompatibility() {
+    function ignoreIfTargetDisabled(event) {
+      if (isDisabled(event.target)) {
+        event.stopPropagation();
       }
-      results[i] = value;
-      unresolved--;
-      if (unresolved === 0) {
-       resolveAll(results);
+    }
+    function isDisabled(node) {
+      return node.disabled || node.parentNode && isDisabled(node.parentNode);
+    }
+    if (isOpera) {
+      document.addEventListener('click', ignoreIfTargetDisabled, true);
+    }
+  })();
+  (function checkOnBlobSupport() {
+    if (isIE || isIOSChrome) {
+      PDFJS.disableCreateObjectURL = true;
+    }
+  })();
+  (function checkNavigatorLanguage() {
+    if (typeof navigator === 'undefined') {
+      return;
+    }
+    if ('language' in navigator) {
+      return;
+    }
+    PDFJS.locale = navigator.userLanguage || 'en-US';
+  })();
+  (function checkRangeRequests() {
+    if (isSafari || isAndroidPre3 || isChromeWithRangeBug || isIOS) {
+      PDFJS.disableRange = true;
+      PDFJS.disableStream = true;
+    }
+  })();
+  (function checkHistoryManipulation() {
+    if (!hasDOM) {
+      return;
+    }
+    if (!history.pushState || isAndroidPre3) {
+      PDFJS.disableHistory = true;
+    }
+  })();
+  (function checkSetPresenceInImageData() {
+    if (!hasDOM) {
+      return;
+    }
+    if (window.CanvasPixelArray) {
+      if (typeof window.CanvasPixelArray.prototype.set !== 'function') {
+        window.CanvasPixelArray.prototype.set = function (arr) {
+          for (var i = 0, ii = this.length; i < ii; i++) {
+            this[i] = arr[i];
+          }
+        };
       }
-     };
-    }(i);
-    if (Promise.isPromise(promise)) {
-     promise.then(resolve, reject);
     } else {
-     resolve(promise);
+      var polyfill = false,
+          versionMatch;
+      if (isChrome) {
+        versionMatch = userAgent.match(/Chrom(e|ium)\/([0-9]+)\./);
+        polyfill = versionMatch && parseInt(versionMatch[2]) < 21;
+      } else if (isAndroid) {
+        polyfill = isAndroidPre5;
+      } else if (isSafari) {
+        versionMatch = userAgent.match(/Version\/([0-9]+)\.([0-9]+)\.([0-9]+) Safari\//);
+        polyfill = versionMatch && parseInt(versionMatch[1]) < 6;
+      }
+      if (polyfill) {
+        var contextPrototype = window.CanvasRenderingContext2D.prototype;
+        var createImageData = contextPrototype.createImageData;
+        contextPrototype.createImageData = function (w, h) {
+          var imageData = createImageData.call(this, w, h);
+          imageData.data.set = function (arr) {
+            for (var i = 0, ii = this.length; i < ii; i++) {
+              this[i] = arr[i];
+            }
+          };
+          return imageData;
+        };
+        contextPrototype = null;
+      }
     }
-   }
-   return deferred;
-  };
-  Promise.isPromise = function Promise_isPromise(value) {
-   return value && typeof value.then === 'function';
-  };
-  Promise.resolve = function Promise_resolve(value) {
-   return new Promise(function (resolve) {
-    resolve(value);
-   });
-  };
-  Promise.reject = function Promise_reject(reason) {
-   return new Promise(function (resolve, reject) {
-    reject(reason);
-   });
-  };
-  Promise.prototype = {
-   _status: null,
-   _value: null,
-   _handlers: null,
-   _unhandledRejection: null,
-   _updateStatus: function Promise__updateStatus(status, value) {
-    if (this._status === STATUS_RESOLVED || this._status === STATUS_REJECTED) {
-     return;
+  })();
+  (function checkRequestAnimationFrame() {
+    function fakeRequestAnimationFrame(callback) {
+      window.setTimeout(callback, 20);
     }
-    if (status === STATUS_RESOLVED && Promise.isPromise(value)) {
-     value.then(this._updateStatus.bind(this, STATUS_RESOLVED), this._updateStatus.bind(this, STATUS_REJECTED));
-     return;
+    if (!hasDOM) {
+      return;
     }
-    this._status = status;
-    this._value = value;
-    if (status === STATUS_REJECTED && this._handlers.length === 0) {
-     this._unhandledRejection = true;
-     HandlerManager.addUnhandledRejection(this);
+    if (isIOS) {
+      window.requestAnimationFrame = fakeRequestAnimationFrame;
+      return;
     }
-    HandlerManager.scheduleHandlers(this);
-   },
-   _resolve: function Promise_resolve(value) {
-    this._updateStatus(STATUS_RESOLVED, value);
-   },
-   _reject: function Promise_reject(reason) {
-    this._updateStatus(STATUS_REJECTED, reason);
-   },
-   then: function Promise_then(onResolve, onReject) {
-    var nextPromise = new Promise(function (resolve, reject) {
-     this.resolve = resolve;
-     this.reject = reject;
-    });
-    this._handlers.push({
-     thisPromise: this,
-     onResolve: onResolve,
-     onReject: onReject,
-     nextPromise: nextPromise
+    if ('requestAnimationFrame' in window) {
+      return;
+    }
+    window.requestAnimationFrame = window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || fakeRequestAnimationFrame;
+  })();
+  (function checkCanvasSizeLimitation() {
+    if (isIOS || isAndroid) {
+      PDFJS.maxCanvasPixels = 5242880;
+    }
+  })();
+  (function checkFullscreenSupport() {
+    if (!hasDOM) {
+      return;
+    }
+    if (isIE && window.parent !== window) {
+      PDFJS.disableFullscreen = true;
+    }
+  })();
+  (function checkCurrentScript() {
+    if (!hasDOM) {
+      return;
+    }
+    if ('currentScript' in document) {
+      return;
+    }
+    Object.defineProperty(document, 'currentScript', {
+      get: function () {
+        var scripts = document.getElementsByTagName('script');
+        return scripts[scripts.length - 1];
+      },
+      enumerable: true,
+      configurable: true
     });
-    HandlerManager.scheduleHandlers(this);
-    return nextPromise;
-   },
-   catch: function Promise_catch(onReject) {
-    return this.then(undefined, onReject);
-   }
-  };
-  globalScope.Promise = Promise;
- }());
- (function checkWeakMap() {
-  if (globalScope.WeakMap) {
-   return;
-  }
-  var id = 0;
-  function WeakMap() {
-   this.id = '$weakmap' + id++;
-  }
-  WeakMap.prototype = {
-   has: function (obj) {
-    return !!Object.getOwnPropertyDescriptor(obj, this.id);
-   },
-   get: function (obj, defaultValue) {
-    return this.has(obj) ? obj[this.id] : defaultValue;
-   },
-   set: function (obj, value) {
-    Object.defineProperty(obj, this.id, {
-     value: value,
-     enumerable: false,
-     configurable: true
+  })();
+  (function checkInputTypeNumberAssign() {
+    if (!hasDOM) {
+      return;
+    }
+    var el = document.createElement('input');
+    try {
+      el.type = 'number';
+    } catch (ex) {
+      var inputProto = el.constructor.prototype;
+      var typeProperty = Object.getOwnPropertyDescriptor(inputProto, 'type');
+      Object.defineProperty(inputProto, 'type', {
+        get: function () {
+          return typeProperty.get.call(this);
+        },
+        set: function (value) {
+          typeProperty.set.call(this, value === 'number' ? 'text' : value);
+        },
+        enumerable: true,
+        configurable: true
+      });
+    }
+  })();
+  (function checkDocumentReadyState() {
+    if (!hasDOM) {
+      return;
+    }
+    if (!document.attachEvent) {
+      return;
+    }
+    var documentProto = document.constructor.prototype;
+    var readyStateProto = Object.getOwnPropertyDescriptor(documentProto, 'readyState');
+    Object.defineProperty(documentProto, 'readyState', {
+      get: function () {
+        var value = readyStateProto.get.call(this);
+        return value === 'interactive' ? 'loading' : value;
+      },
+      set: function (value) {
+        readyStateProto.set.call(this, value);
+      },
+      enumerable: true,
+      configurable: true
     });
-   },
-   delete: function (obj) {
-    delete obj[this.id];
-   }
-  };
-  globalScope.WeakMap = WeakMap;
- }());
- (function checkURLConstructor() {
-  var hasWorkingUrl = false;
-  try {
-   if (typeof URL === 'function' && typeof URL.prototype === 'object' && 'origin' in URL.prototype) {
-    var u = new URL('b', 'http://a');
-    u.pathname = 'c%20d';
-    hasWorkingUrl = u.href === 'http://a/c%20d';
-   }
-  } catch (e) {
-  }
-  if (hasWorkingUrl) {
-   return;
-  }
-  var relative = Object.create(null);
-  relative['ftp'] = 21;
-  relative['file'] = 0;
-  relative['gopher'] = 70;
-  relative['http'] = 80;
-  relative['https'] = 443;
-  relative['ws'] = 80;
-  relative['wss'] = 443;
-  var relativePathDotMapping = Object.create(null);
-  relativePathDotMapping['%2e'] = '.';
-  relativePathDotMapping['.%2e'] = '..';
-  relativePathDotMapping['%2e.'] = '..';
-  relativePathDotMapping['%2e%2e'] = '..';
-  function isRelativeScheme(scheme) {
-   return relative[scheme] !== undefined;
-  }
-  function invalid() {
-   clear.call(this);
-   this._isInvalid = true;
-  }
-  function IDNAToASCII(h) {
-   if (h === '') {
-    invalid.call(this);
-   }
-   return h.toLowerCase();
-  }
-  function percentEscape(c) {
-   var unicode = c.charCodeAt(0);
-   if (unicode > 0x20 && unicode < 0x7F && [
-     0x22,
-     0x23,
-     0x3C,
-     0x3E,
-     0x3F,
-     0x60
-    ].indexOf(unicode) === -1) {
-    return c;
-   }
-   return encodeURIComponent(c);
-  }
-  function percentEscapeQuery(c) {
-   var unicode = c.charCodeAt(0);
-   if (unicode > 0x20 && unicode < 0x7F && [
-     0x22,
-     0x23,
-     0x3C,
-     0x3E,
-     0x60
-    ].indexOf(unicode) === -1) {
-    return c;
-   }
-   return encodeURIComponent(c);
-  }
-  var EOF, ALPHA = /[a-zA-Z]/, ALPHANUMERIC = /[a-zA-Z0-9\+\-\.]/;
-  function parse(input, stateOverride, base) {
-   function err(message) {
-    errors.push(message);
-   }
-   var state = stateOverride || 'scheme start', cursor = 0, buffer = '', seenAt = false, seenBracket = false, errors = [];
-   loop:
-    while ((input[cursor - 1] !== EOF || cursor === 0) && !this._isInvalid) {
-     var c = input[cursor];
-     switch (state) {
-     case 'scheme start':
-      if (c && ALPHA.test(c)) {
-       buffer += c.toLowerCase();
-       state = 'scheme';
-      } else if (!stateOverride) {
-       buffer = '';
-       state = 'no scheme';
-       continue;
-      } else {
-       err('Invalid scheme.');
-       break loop;
-      }
-      break;
-     case 'scheme':
-      if (c && ALPHANUMERIC.test(c)) {
-       buffer += c.toLowerCase();
-      } else if (c === ':') {
-       this._scheme = buffer;
-       buffer = '';
-       if (stateOverride) {
-        break loop;
-       }
-       if (isRelativeScheme(this._scheme)) {
-        this._isRelative = true;
-       }
-       if (this._scheme === 'file') {
-        state = 'relative';
-       } else if (this._isRelative && base && base._scheme === this._scheme) {
-        state = 'relative or authority';
-       } else if (this._isRelative) {
-        state = 'authority first slash';
-       } else {
-        state = 'scheme data';
-       }
-      } else if (!stateOverride) {
-       buffer = '';
-       cursor = 0;
-       state = 'no scheme';
-       continue;
-      } else if (c === EOF) {
-       break loop;
-      } else {
-       err('Code point not allowed in scheme: ' + c);
-       break loop;
-      }
-      break;
-     case 'scheme data':
-      if (c === '?') {
-       this._query = '?';
-       state = 'query';
-      } else if (c === '#') {
-       this._fragment = '#';
-       state = 'fragment';
-      } else {
-       if (c !== EOF && c !== '\t' && c !== '\n' && c !== '\r') {
-        this._schemeData += percentEscape(c);
-       }
-      }
-      break;
-     case 'no scheme':
-      if (!base || !isRelativeScheme(base._scheme)) {
-       err('Missing scheme.');
-       invalid.call(this);
-      } else {
-       state = 'relative';
-       continue;
-      }
-      break;
-     case 'relative or authority':
-      if (c === '/' && input[cursor + 1] === '/') {
-       state = 'authority ignore slashes';
-      } else {
-       err('Expected /, got: ' + c);
-       state = 'relative';
-       continue;
-      }
-      break;
-     case 'relative':
-      this._isRelative = true;
-      if (this._scheme !== 'file') {
-       this._scheme = base._scheme;
-      }
-      if (c === EOF) {
-       this._host = base._host;
-       this._port = base._port;
-       this._path = base._path.slice();
-       this._query = base._query;
-       this._username = base._username;
-       this._password = base._password;
-       break loop;
-      } else if (c === '/' || c === '\\') {
-       if (c === '\\') {
-        err('\\ is an invalid code point.');
-       }
-       state = 'relative slash';
-      } else if (c === '?') {
-       this._host = base._host;
-       this._port = base._port;
-       this._path = base._path.slice();
-       this._query = '?';
-       this._username = base._username;
-       this._password = base._password;
-       state = 'query';
-      } else if (c === '#') {
-       this._host = base._host;
-       this._port = base._port;
-       this._path = base._path.slice();
-       this._query = base._query;
-       this._fragment = '#';
-       this._username = base._username;
-       this._password = base._password;
-       state = 'fragment';
-      } else {
-       var nextC = input[cursor + 1];
-       var nextNextC = input[cursor + 2];
-       if (this._scheme !== 'file' || !ALPHA.test(c) || nextC !== ':' && nextC !== '|' || nextNextC !== EOF && nextNextC !== '/' && nextNextC !== '\\' && nextNextC !== '?' && nextNextC !== '#') {
-        this._host = base._host;
-        this._port = base._port;
-        this._username = base._username;
-        this._password = base._password;
-        this._path = base._path.slice();
-        this._path.pop();
-       }
-       state = 'relative path';
-       continue;
+  })();
+  (function checkChildNodeRemove() {
+    if (!hasDOM) {
+      return;
+    }
+    if (typeof Element.prototype.remove !== 'undefined') {
+      return;
+    }
+    Element.prototype.remove = function () {
+      if (this.parentNode) {
+        this.parentNode.removeChild(this);
       }
-      break;
-     case 'relative slash':
-      if (c === '/' || c === '\\') {
-       if (c === '\\') {
-        err('\\ is an invalid code point.');
-       }
-       if (this._scheme === 'file') {
-        state = 'file host';
-       } else {
-        state = 'authority ignore slashes';
-       }
-      } else {
-       if (this._scheme !== 'file') {
-        this._host = base._host;
-        this._port = base._port;
-        this._username = base._username;
-        this._password = base._password;
-       }
-       state = 'relative path';
-       continue;
+    };
+  })();
+  (function checkPromise() {
+    if (globalScope.Promise) {
+      if (typeof globalScope.Promise.all !== 'function') {
+        globalScope.Promise.all = function (iterable) {
+          var count = 0,
+              results = [],
+              resolve,
+              reject;
+          var promise = new globalScope.Promise(function (resolve_, reject_) {
+            resolve = resolve_;
+            reject = reject_;
+          });
+          iterable.forEach(function (p, i) {
+            count++;
+            p.then(function (result) {
+              results[i] = result;
+              count--;
+              if (count === 0) {
+                resolve(results);
+              }
+            }, reject);
+          });
+          if (count === 0) {
+            resolve(results);
+          }
+          return promise;
+        };
       }
-      break;
-     case 'authority first slash':
-      if (c === '/') {
-       state = 'authority second slash';
-      } else {
-       err('Expected \'/\', got: ' + c);
-       state = 'authority ignore slashes';
-       continue;
+      if (typeof globalScope.Promise.resolve !== 'function') {
+        globalScope.Promise.resolve = function (value) {
+          return new globalScope.Promise(function (resolve) {
+            resolve(value);
+          });
+        };
       }
-      break;
-     case 'authority second slash':
-      state = 'authority ignore slashes';
-      if (c !== '/') {
-       err('Expected \'/\', got: ' + c);
-       continue;
+      if (typeof globalScope.Promise.reject !== 'function') {
+        globalScope.Promise.reject = function (reason) {
+          return new globalScope.Promise(function (resolve, reject) {
+            reject(reason);
+          });
+        };
       }
-      break;
-     case 'authority ignore slashes':
-      if (c !== '/' && c !== '\\') {
-       state = 'authority';
-       continue;
-      } else {
-       err('Expected authority, got: ' + c);
+      if (typeof globalScope.Promise.prototype.catch !== 'function') {
+        globalScope.Promise.prototype.catch = function (onReject) {
+          return globalScope.Promise.prototype.then(undefined, onReject);
+        };
       }
-      break;
-     case 'authority':
-      if (c === '@') {
-       if (seenAt) {
-        err('@ already seen.');
-        buffer += '%40';
-       }
-       seenAt = true;
-       for (var i = 0; i < buffer.length; i++) {
-        var cp = buffer[i];
-        if (cp === '\t' || cp === '\n' || cp === '\r') {
-         err('Invalid whitespace in authority.');
-         continue;
+      return;
+    }
+    var STATUS_PENDING = 0;
+    var STATUS_RESOLVED = 1;
+    var STATUS_REJECTED = 2;
+    var REJECTION_TIMEOUT = 500;
+    var HandlerManager = {
+      handlers: [],
+      running: false,
+      unhandledRejections: [],
+      pendingRejectionCheck: false,
+      scheduleHandlers: function scheduleHandlers(promise) {
+        if (promise._status === STATUS_PENDING) {
+          return;
         }
-        if (cp === ':' && this._password === null) {
-         this._password = '';
-         continue;
+        this.handlers = this.handlers.concat(promise._handlers);
+        promise._handlers = [];
+        if (this.running) {
+          return;
         }
-        var tempC = percentEscape(cp);
-        if (this._password !== null) {
-         this._password += tempC;
-        } else {
-         this._username += tempC;
+        this.running = true;
+        setTimeout(this.runHandlers.bind(this), 0);
+      },
+      runHandlers: function runHandlers() {
+        var RUN_TIMEOUT = 1;
+        var timeoutAt = Date.now() + RUN_TIMEOUT;
+        while (this.handlers.length > 0) {
+          var handler = this.handlers.shift();
+          var nextStatus = handler.thisPromise._status;
+          var nextValue = handler.thisPromise._value;
+          try {
+            if (nextStatus === STATUS_RESOLVED) {
+              if (typeof handler.onResolve === 'function') {
+                nextValue = handler.onResolve(nextValue);
+              }
+            } else if (typeof handler.onReject === 'function') {
+              nextValue = handler.onReject(nextValue);
+              nextStatus = STATUS_RESOLVED;
+              if (handler.thisPromise._unhandledRejection) {
+                this.removeUnhandeledRejection(handler.thisPromise);
+              }
+            }
+          } catch (ex) {
+            nextStatus = STATUS_REJECTED;
+            nextValue = ex;
+          }
+          handler.nextPromise._updateStatus(nextStatus, nextValue);
+          if (Date.now() >= timeoutAt) {
+            break;
+          }
         }
-       }
-       buffer = '';
-      } else if (c === EOF || c === '/' || c === '\\' || c === '?' || c === '#') {
-       cursor -= buffer.length;
-       buffer = '';
-       state = 'host';
-       continue;
-      } else {
-       buffer += c;
+        if (this.handlers.length > 0) {
+          setTimeout(this.runHandlers.bind(this), 0);
+          return;
+        }
+        this.running = false;
+      },
+      addUnhandledRejection: function addUnhandledRejection(promise) {
+        this.unhandledRejections.push({
+          promise: promise,
+          time: Date.now()
+        });
+        this.scheduleRejectionCheck();
+      },
+      removeUnhandeledRejection: function removeUnhandeledRejection(promise) {
+        promise._unhandledRejection = false;
+        for (var i = 0; i < this.unhandledRejections.length; i++) {
+          if (this.unhandledRejections[i].promise === promise) {
+            this.unhandledRejections.splice(i);
+            i--;
+          }
+        }
+      },
+      scheduleRejectionCheck: function scheduleRejectionCheck() {
+        if (this.pendingRejectionCheck) {
+          return;
+        }
+        this.pendingRejectionCheck = true;
+        setTimeout(function rejectionCheck() {
+          this.pendingRejectionCheck = false;
+          var now = Date.now();
+          for (var i = 0; i < this.unhandledRejections.length; i++) {
+            if (now - this.unhandledRejections[i].time > REJECTION_TIMEOUT) {
+              var unhandled = this.unhandledRejections[i].promise._value;
+              var msg = 'Unhandled rejection: ' + unhandled;
+              if (unhandled.stack) {
+                msg += '\n' + unhandled.stack;
+              }
+              try {
+                throw new Error(msg);
+              } catch (_) {
+                console.warn(msg);
+              }
+              this.unhandledRejections.splice(i);
+              i--;
+            }
+          }
+          if (this.unhandledRejections.length) {
+            this.scheduleRejectionCheck();
+          }
+        }.bind(this), REJECTION_TIMEOUT);
       }
-      break;
-     case 'file host':
-      if (c === EOF || c === '/' || c === '\\' || c === '?' || c === '#') {
-       if (buffer.length === 2 && ALPHA.test(buffer[0]) && (buffer[1] === ':' || buffer[1] === '|')) {
-        state = 'relative path';
-       } else if (buffer.length === 0) {
-        state = 'relative path start';
-       } else {
-        this._host = IDNAToASCII.call(this, buffer);
-        buffer = '';
-        state = 'relative path start';
-       }
-       continue;
-      } else if (c === '\t' || c === '\n' || c === '\r') {
-       err('Invalid whitespace in file host.');
-      } else {
-       buffer += c;
+    };
+    var Promise = function Promise(resolver) {
+      this._status = STATUS_PENDING;
+      this._handlers = [];
+      try {
+        resolver.call(this, this._resolve.bind(this), this._reject.bind(this));
+      } catch (e) {
+        this._reject(e);
       }
-      break;
-     case 'host':
-     case 'hostname':
-      if (c === ':' && !seenBracket) {
-       this._host = IDNAToASCII.call(this, buffer);
-       buffer = '';
-       state = 'port';
-       if (stateOverride === 'hostname') {
-        break loop;
-       }
-      } else if (c === EOF || c === '/' || c === '\\' || c === '?' || c === '#') {
-       this._host = IDNAToASCII.call(this, buffer);
-       buffer = '';
-       state = 'relative path start';
-       if (stateOverride) {
-        break loop;
-       }
-       continue;
-      } else if (c !== '\t' && c !== '\n' && c !== '\r') {
-       if (c === '[') {
-        seenBracket = true;
-       } else if (c === ']') {
-        seenBracket = false;
-       }
-       buffer += c;
-      } else {
-       err('Invalid code point in host/hostname: ' + c);
+    };
+    Promise.all = function Promise_all(promises) {
+      var resolveAll, rejectAll;
+      var deferred = new Promise(function (resolve, reject) {
+        resolveAll = resolve;
+        rejectAll = reject;
+      });
+      var unresolved = promises.length;
+      var results = [];
+      if (unresolved === 0) {
+        resolveAll(results);
+        return deferred;
       }
-      break;
-     case 'port':
-      if (/[0-9]/.test(c)) {
-       buffer += c;
-      } else if (c === EOF || c === '/' || c === '\\' || c === '?' || c === '#' || stateOverride) {
-       if (buffer !== '') {
-        var temp = parseInt(buffer, 10);
-        if (temp !== relative[this._scheme]) {
-         this._port = temp + '';
+      function reject(reason) {
+        if (deferred._status === STATUS_REJECTED) {
+          return;
         }
-        buffer = '';
-       }
-       if (stateOverride) {
-        break loop;
-       }
-       state = 'relative path start';
-       continue;
-      } else if (c === '\t' || c === '\n' || c === '\r') {
-       err('Invalid code point in port: ' + c);
-      } else {
-       invalid.call(this);
-      }
-      break;
-     case 'relative path start':
-      if (c === '\\') {
-       err('\'\\\' not allowed in path.');
+        results = [];
+        rejectAll(reason);
       }
-      state = 'relative path';
-      if (c !== '/' && c !== '\\') {
-       continue;
+      for (var i = 0, ii = promises.length; i < ii; ++i) {
+        var promise = promises[i];
+        var resolve = function (i) {
+          return function (value) {
+            if (deferred._status === STATUS_REJECTED) {
+              return;
+            }
+            results[i] = value;
+            unresolved--;
+            if (unresolved === 0) {
+              resolveAll(results);
+            }
+          };
+        }(i);
+        if (Promise.isPromise(promise)) {
+          promise.then(resolve, reject);
+        } else {
+          resolve(promise);
+        }
       }
-      break;
-     case 'relative path':
-      if (c === EOF || c === '/' || c === '\\' || !stateOverride && (c === '?' || c === '#')) {
-       if (c === '\\') {
-        err('\\ not allowed in relative path.');
-       }
-       var tmp;
-       if (tmp = relativePathDotMapping[buffer.toLowerCase()]) {
-        buffer = tmp;
-       }
-       if (buffer === '..') {
-        this._path.pop();
-        if (c !== '/' && c !== '\\') {
-         this._path.push('');
+      return deferred;
+    };
+    Promise.isPromise = function Promise_isPromise(value) {
+      return value && typeof value.then === 'function';
+    };
+    Promise.resolve = function Promise_resolve(value) {
+      return new Promise(function (resolve) {
+        resolve(value);
+      });
+    };
+    Promise.reject = function Promise_reject(reason) {
+      return new Promise(function (resolve, reject) {
+        reject(reason);
+      });
+    };
+    Promise.prototype = {
+      _status: null,
+      _value: null,
+      _handlers: null,
+      _unhandledRejection: null,
+      _updateStatus: function Promise__updateStatus(status, value) {
+        if (this._status === STATUS_RESOLVED || this._status === STATUS_REJECTED) {
+          return;
         }
-       } else if (buffer === '.' && c !== '/' && c !== '\\') {
-        this._path.push('');
-       } else if (buffer !== '.') {
-        if (this._scheme === 'file' && this._path.length === 0 && buffer.length === 2 && ALPHA.test(buffer[0]) && buffer[1] === '|') {
-         buffer = buffer[0] + ':';
+        if (status === STATUS_RESOLVED && Promise.isPromise(value)) {
+          value.then(this._updateStatus.bind(this, STATUS_RESOLVED), this._updateStatus.bind(this, STATUS_REJECTED));
+          return;
         }
-        this._path.push(buffer);
-       }
-       buffer = '';
-       if (c === '?') {
-        this._query = '?';
-        state = 'query';
-       } else if (c === '#') {
-        this._fragment = '#';
-        state = 'fragment';
-       }
-      } else if (c !== '\t' && c !== '\n' && c !== '\r') {
-       buffer += percentEscape(c);
-      }
-      break;
-     case 'query':
-      if (!stateOverride && c === '#') {
-       this._fragment = '#';
-       state = 'fragment';
-      } else if (c !== EOF && c !== '\t' && c !== '\n' && c !== '\r') {
-       this._query += percentEscapeQuery(c);
-      }
-      break;
-     case 'fragment':
-      if (c !== EOF && c !== '\t' && c !== '\n' && c !== '\r') {
-       this._fragment += c;
+        this._status = status;
+        this._value = value;
+        if (status === STATUS_REJECTED && this._handlers.length === 0) {
+          this._unhandledRejection = true;
+          HandlerManager.addUnhandledRejection(this);
+        }
+        HandlerManager.scheduleHandlers(this);
+      },
+      _resolve: function Promise_resolve(value) {
+        this._updateStatus(STATUS_RESOLVED, value);
+      },
+      _reject: function Promise_reject(reason) {
+        this._updateStatus(STATUS_REJECTED, reason);
+      },
+      then: function Promise_then(onResolve, onReject) {
+        var nextPromise = new Promise(function (resolve, reject) {
+          this.resolve = resolve;
+          this.reject = reject;
+        });
+        this._handlers.push({
+          thisPromise: this,
+          onResolve: onResolve,
+          onReject: onReject,
+          nextPromise: nextPromise
+        });
+        HandlerManager.scheduleHandlers(this);
+        return nextPromise;
+      },
+      catch: function Promise_catch(onReject) {
+        return this.then(undefined, onReject);
       }
-      break;
-     }
-     cursor++;
-    }
-  }
-  function clear() {
-   this._scheme = '';
-   this._schemeData = '';
-   this._username = '';
-   this._password = null;
-   this._host = '';
-   this._port = '';
-   this._path = [];
-   this._query = '';
-   this._fragment = '';
-   this._isInvalid = false;
-   this._isRelative = false;
-  }
-  function JURL(url, base) {
-   if (base !== undefined && !(base instanceof JURL)) {
-    base = new JURL(String(base));
-   }
-   this._url = url;
-   clear.call(this);
-   var input = url.replace(/^[ \t\r\n\f]+|[ \t\r\n\f]+$/g, '');
-   parse.call(this, input, null, base);
-  }
-  JURL.prototype = {
-   toString: function () {
-    return this.href;
-   },
-   get href() {
-    if (this._isInvalid) {
-     return this._url;
-    }
-    var authority = '';
-    if (this._username !== '' || this._password !== null) {
-     authority = this._username + (this._password !== null ? ':' + this._password : '') + '@';
-    }
-    return this.protocol + (this._isRelative ? '//' + authority + this.host : '') + this.pathname + this._query + this._fragment;
-   },
-   set href(href) {
-    clear.call(this);
-    parse.call(this, href);
-   },
-   get protocol() {
-    return this._scheme + ':';
-   },
-   set protocol(protocol) {
-    if (this._isInvalid) {
-     return;
+    };
+    globalScope.Promise = Promise;
+  })();
+  (function checkWeakMap() {
+    if (globalScope.WeakMap) {
+      return;
     }
-    parse.call(this, protocol + ':', 'scheme start');
-   },
-   get host() {
-    return this._isInvalid ? '' : this._port ? this._host + ':' + this._port : this._host;
-   },
-   set host(host) {
-    if (this._isInvalid || !this._isRelative) {
-     return;
+    var id = 0;
+    function WeakMap() {
+      this.id = '$weakmap' + id++;
     }
-    parse.call(this, host, 'host');
-   },
-   get hostname() {
-    return this._host;
-   },
-   set hostname(hostname) {
-    if (this._isInvalid || !this._isRelative) {
-     return;
+    WeakMap.prototype = {
+      has: function (obj) {
+        return !!Object.getOwnPropertyDescriptor(obj, this.id);
+      },
+      get: function (obj, defaultValue) {
+        return this.has(obj) ? obj[this.id] : defaultValue;
+      },
+      set: function (obj, value) {
+        Object.defineProperty(obj, this.id, {
+          value: value,
+          enumerable: false,
+          configurable: true
+        });
+      },
+      delete: function (obj) {
+        delete obj[this.id];
+      }
+    };
+    globalScope.WeakMap = WeakMap;
+  })();
+  (function checkURLConstructor() {
+    var hasWorkingUrl = false;
+    try {
+      if (typeof URL === 'function' && typeof URL.prototype === 'object' && 'origin' in URL.prototype) {
+        var u = new URL('b', 'http://a');
+        u.pathname = 'c%20d';
+        hasWorkingUrl = u.href === 'http://a/c%20d';
+      }
+    } catch (e) {}
+    if (hasWorkingUrl) {
+      return;
     }
-    parse.call(this, hostname, 'hostname');
-   },
-   get port() {
-    return this._port;
-   },
-   set port(port) {
-    if (this._isInvalid || !this._isRelative) {
-     return;
+    var relative = Object.create(null);
+    relative['ftp'] = 21;
+    relative['file'] = 0;
+    relative['gopher'] = 70;
+    relative['http'] = 80;
+    relative['https'] = 443;
+    relative['ws'] = 80;
+    relative['wss'] = 443;
+    var relativePathDotMapping = Object.create(null);
+    relativePathDotMapping['%2e'] = '.';
+    relativePathDotMapping['.%2e'] = '..';
+    relativePathDotMapping['%2e.'] = '..';
+    relativePathDotMapping['%2e%2e'] = '..';
+    function isRelativeScheme(scheme) {
+      return relative[scheme] !== undefined;
     }
-    parse.call(this, port, 'port');
-   },
-   get pathname() {
-    return this._isInvalid ? '' : this._isRelative ? '/' + this._path.join('/') : this._schemeData;
-   },
-   set pathname(pathname) {
-    if (this._isInvalid || !this._isRelative) {
-     return;
+    function invalid() {
+      clear.call(this);
+      this._isInvalid = true;
     }
-    this._path = [];
-    parse.call(this, pathname, 'relative path start');
-   },
-   get search() {
-    return this._isInvalid || !this._query || this._query === '?' ? '' : this._query;
-   },
-   set search(search) {
-    if (this._isInvalid || !this._isRelative) {
-     return;
+    function IDNAToASCII(h) {
+      if (h === '') {
+        invalid.call(this);
+      }
+      return h.toLowerCase();
     }
-    this._query = '?';
-    if (search[0] === '?') {
-     search = search.slice(1);
+    function percentEscape(c) {
+      var unicode = c.charCodeAt(0);
+      if (unicode > 0x20 && unicode < 0x7F && [0x22, 0x23, 0x3C, 0x3E, 0x3F, 0x60].indexOf(unicode) === -1) {
+        return c;
+      }
+      return encodeURIComponent(c);
     }
-    parse.call(this, search, 'query');
-   },
-   get hash() {
-    return this._isInvalid || !this._fragment || this._fragment === '#' ? '' : this._fragment;
-   },
-   set hash(hash) {
-    if (this._isInvalid) {
-     return;
+    function percentEscapeQuery(c) {
+      var unicode = c.charCodeAt(0);
+      if (unicode > 0x20 && unicode < 0x7F && [0x22, 0x23, 0x3C, 0x3E, 0x60].indexOf(unicode) === -1) {
+        return c;
+      }
+      return encodeURIComponent(c);
     }
-    this._fragment = '#';
-    if (hash[0] === '#') {
-     hash = hash.slice(1);
+    var EOF,
+        ALPHA = /[a-zA-Z]/,
+        ALPHANUMERIC = /[a-zA-Z0-9\+\-\.]/;
+    function parse(input, stateOverride, base) {
+      function err(message) {
+        errors.push(message);
+      }
+      var state = stateOverride || 'scheme start',
+          cursor = 0,
+          buffer = '',
+          seenAt = false,
+          seenBracket = false,
+          errors = [];
+      loop: while ((input[cursor - 1] !== EOF || cursor === 0) && !this._isInvalid) {
+        var c = input[cursor];
+        switch (state) {
+          case 'scheme start':
+            if (c && ALPHA.test(c)) {
+              buffer += c.toLowerCase();
+              state = 'scheme';
+            } else if (!stateOverride) {
+              buffer = '';
+              state = 'no scheme';
+              continue;
+            } else {
+              err('Invalid scheme.');
+              break loop;
+            }
+            break;
+          case 'scheme':
+            if (c && ALPHANUMERIC.test(c)) {
+              buffer += c.toLowerCase();
+            } else if (c === ':') {
+              this._scheme = buffer;
+              buffer = '';
+              if (stateOverride) {
+                break loop;
+              }
+              if (isRelativeScheme(this._scheme)) {
+                this._isRelative = true;
+              }
+              if (this._scheme === 'file') {
+                state = 'relative';
+              } else if (this._isRelative && base && base._scheme === this._scheme) {
+                state = 'relative or authority';
+              } else if (this._isRelative) {
+                state = 'authority first slash';
+              } else {
+                state = 'scheme data';
+              }
+            } else if (!stateOverride) {
+              buffer = '';
+              cursor = 0;
+              state = 'no scheme';
+              continue;
+            } else if (c === EOF) {
+              break loop;
+            } else {
+              err('Code point not allowed in scheme: ' + c);
+              break loop;
+            }
+            break;
+          case 'scheme data':
+            if (c === '?') {
+              this._query = '?';
+              state = 'query';
+            } else if (c === '#') {
+              this._fragment = '#';
+              state = 'fragment';
+            } else {
+              if (c !== EOF && c !== '\t' && c !== '\n' && c !== '\r') {
+                this._schemeData += percentEscape(c);
+              }
+            }
+            break;
+          case 'no scheme':
+            if (!base || !isRelativeScheme(base._scheme)) {
+              err('Missing scheme.');
+              invalid.call(this);
+            } else {
+              state = 'relative';
+              continue;
+            }
+            break;
+          case 'relative or authority':
+            if (c === '/' && input[cursor + 1] === '/') {
+              state = 'authority ignore slashes';
+            } else {
+              err('Expected /, got: ' + c);
+              state = 'relative';
+              continue;
+            }
+            break;
+          case 'relative':
+            this._isRelative = true;
+            if (this._scheme !== 'file') {
+              this._scheme = base._scheme;
+            }
+            if (c === EOF) {
+              this._host = base._host;
+              this._port = base._port;
+              this._path = base._path.slice();
+              this._query = base._query;
+              this._username = base._username;
+              this._password = base._password;
+              break loop;
+            } else if (c === '/' || c === '\\') {
+              if (c === '\\') {
+                err('\\ is an invalid code point.');
+              }
+              state = 'relative slash';
+            } else if (c === '?') {
+              this._host = base._host;
+              this._port = base._port;
+              this._path = base._path.slice();
+              this._query = '?';
+              this._username = base._username;
+              this._password = base._password;
+              state = 'query';
+            } else if (c === '#') {
+              this._host = base._host;
+              this._port = base._port;
+              this._path = base._path.slice();
+              this._query = base._query;
+              this._fragment = '#';
+              this._username = base._username;
+              this._password = base._password;
+              state = 'fragment';
+            } else {
+              var nextC = input[cursor + 1];
+              var nextNextC = input[cursor + 2];
+              if (this._scheme !== 'file' || !ALPHA.test(c) || nextC !== ':' && nextC !== '|' || nextNextC !== EOF && nextNextC !== '/' && nextNextC !== '\\' && nextNextC !== '?' && nextNextC !== '#') {
+                this._host = base._host;
+                this._port = base._port;
+                this._username = base._username;
+                this._password = base._password;
+                this._path = base._path.slice();
+                this._path.pop();
+              }
+              state = 'relative path';
+              continue;
+            }
+            break;
+          case 'relative slash':
+            if (c === '/' || c === '\\') {
+              if (c === '\\') {
+                err('\\ is an invalid code point.');
+              }
+              if (this._scheme === 'file') {
+                state = 'file host';
+              } else {
+                state = 'authority ignore slashes';
+              }
+            } else {
+              if (this._scheme !== 'file') {
+                this._host = base._host;
+                this._port = base._port;
+                this._username = base._username;
+                this._password = base._password;
+              }
+              state = 'relative path';
+              continue;
+            }
+            break;
+          case 'authority first slash':
+            if (c === '/') {
+              state = 'authority second slash';
+            } else {
+              err('Expected \'/\', got: ' + c);
+              state = 'authority ignore slashes';
+              continue;
+            }
+            break;
+          case 'authority second slash':
+            state = 'authority ignore slashes';
+            if (c !== '/') {
+              err('Expected \'/\', got: ' + c);
+              continue;
+            }
+            break;
+          case 'authority ignore slashes':
+            if (c !== '/' && c !== '\\') {
+              state = 'authority';
+              continue;
+            } else {
+              err('Expected authority, got: ' + c);
+            }
+            break;
+          case 'authority':
+            if (c === '@') {
+              if (seenAt) {
+                err('@ already seen.');
+                buffer += '%40';
+              }
+              seenAt = true;
+              for (var i = 0; i < buffer.length; i++) {
+                var cp = buffer[i];
+                if (cp === '\t' || cp === '\n' || cp === '\r') {
+                  err('Invalid whitespace in authority.');
+                  continue;
+                }
+                if (cp === ':' && this._password === null) {
+                  this._password = '';
+                  continue;
+                }
+                var tempC = percentEscape(cp);
+                if (this._password !== null) {
+                  this._password += tempC;
+                } else {
+                  this._username += tempC;
+                }
+              }
+              buffer = '';
+            } else if (c === EOF || c === '/' || c === '\\' || c === '?' || c === '#') {
+              cursor -= buffer.length;
+              buffer = '';
+              state = 'host';
+              continue;
+            } else {
+              buffer += c;
+            }
+            break;
+          case 'file host':
+            if (c === EOF || c === '/' || c === '\\' || c === '?' || c === '#') {
+              if (buffer.length === 2 && ALPHA.test(buffer[0]) && (buffer[1] === ':' || buffer[1] === '|')) {
+                state = 'relative path';
+              } else if (buffer.length === 0) {
+                state = 'relative path start';
+              } else {
+                this._host = IDNAToASCII.call(this, buffer);
+                buffer = '';
+                state = 'relative path start';
+              }
+              continue;
+            } else if (c === '\t' || c === '\n' || c === '\r') {
+              err('Invalid whitespace in file host.');
+            } else {
+              buffer += c;
+            }
+            break;
+          case 'host':
+          case 'hostname':
+            if (c === ':' && !seenBracket) {
+              this._host = IDNAToASCII.call(this, buffer);
+              buffer = '';
+              state = 'port';
+              if (stateOverride === 'hostname') {
+                break loop;
+              }
+            } else if (c === EOF || c === '/' || c === '\\' || c === '?' || c === '#') {
+              this._host = IDNAToASCII.call(this, buffer);
+              buffer = '';
+              state = 'relative path start';
+              if (stateOverride) {
+                break loop;
+              }
+              continue;
+            } else if (c !== '\t' && c !== '\n' && c !== '\r') {
+              if (c === '[') {
+                seenBracket = true;
+              } else if (c === ']') {
+                seenBracket = false;
+              }
+              buffer += c;
+            } else {
+              err('Invalid code point in host/hostname: ' + c);
+            }
+            break;
+          case 'port':
+            if (/[0-9]/.test(c)) {
+              buffer += c;
+            } else if (c === EOF || c === '/' || c === '\\' || c === '?' || c === '#' || stateOverride) {
+              if (buffer !== '') {
+                var temp = parseInt(buffer, 10);
+                if (temp !== relative[this._scheme]) {
+                  this._port = temp + '';
+                }
+                buffer = '';
+              }
+              if (stateOverride) {
+                break loop;
+              }
+              state = 'relative path start';
+              continue;
+            } else if (c === '\t' || c === '\n' || c === '\r') {
+              err('Invalid code point in port: ' + c);
+            } else {
+              invalid.call(this);
+            }
+            break;
+          case 'relative path start':
+            if (c === '\\') {
+              err('\'\\\' not allowed in path.');
+            }
+            state = 'relative path';
+            if (c !== '/' && c !== '\\') {
+              continue;
+            }
+            break;
+          case 'relative path':
+            if (c === EOF || c === '/' || c === '\\' || !stateOverride && (c === '?' || c === '#')) {
+              if (c === '\\') {
+                err('\\ not allowed in relative path.');
+              }
+              var tmp;
+              if (tmp = relativePathDotMapping[buffer.toLowerCase()]) {
+                buffer = tmp;
+              }
+              if (buffer === '..') {
+                this._path.pop();
+                if (c !== '/' && c !== '\\') {
+                  this._path.push('');
+                }
+              } else if (buffer === '.' && c !== '/' && c !== '\\') {
+                this._path.push('');
+              } else if (buffer !== '.') {
+                if (this._scheme === 'file' && this._path.length === 0 && buffer.length === 2 && ALPHA.test(buffer[0]) && buffer[1] === '|') {
+                  buffer = buffer[0] + ':';
+                }
+                this._path.push(buffer);
+              }
+              buffer = '';
+              if (c === '?') {
+                this._query = '?';
+                state = 'query';
+              } else if (c === '#') {
+                this._fragment = '#';
+                state = 'fragment';
+              }
+            } else if (c !== '\t' && c !== '\n' && c !== '\r') {
+              buffer += percentEscape(c);
+            }
+            break;
+          case 'query':
+            if (!stateOverride && c === '#') {
+              this._fragment = '#';
+              state = 'fragment';
+            } else if (c !== EOF && c !== '\t' && c !== '\n' && c !== '\r') {
+              this._query += percentEscapeQuery(c);
+            }
+            break;
+          case 'fragment':
+            if (c !== EOF && c !== '\t' && c !== '\n' && c !== '\r') {
+              this._fragment += c;
+            }
+            break;
+        }
+        cursor++;
+      }
     }
-    parse.call(this, hash, 'fragment');
-   },
-   get origin() {
-    var host;
-    if (this._isInvalid || !this._scheme) {
-     return '';
+    function clear() {
+      this._scheme = '';
+      this._schemeData = '';
+      this._username = '';
+      this._password = null;
+      this._host = '';
+      this._port = '';
+      this._path = [];
+      this._query = '';
+      this._fragment = '';
+      this._isInvalid = false;
+      this._isRelative = false;
     }
-    switch (this._scheme) {
-    case 'data':
-    case 'file':
-    case 'javascript':
-    case 'mailto':
-     return 'null';
+    function JURL(url, base) {
+      if (base !== undefined && !(base instanceof JURL)) {
+        base = new JURL(String(base));
+      }
+      this._url = url;
+      clear.call(this);
+      var input = url.replace(/^[ \t\r\n\f]+|[ \t\r\n\f]+$/g, '');
+      parse.call(this, input, null, base);
     }
-    host = this.host;
-    if (!host) {
-     return '';
+    JURL.prototype = {
+      toString: function () {
+        return this.href;
+      },
+      get href() {
+        if (this._isInvalid) {
+          return this._url;
+        }
+        var authority = '';
+        if (this._username !== '' || this._password !== null) {
+          authority = this._username + (this._password !== null ? ':' + this._password : '') + '@';
+        }
+        return this.protocol + (this._isRelative ? '//' + authority + this.host : '') + this.pathname + this._query + this._fragment;
+      },
+      set href(href) {
+        clear.call(this);
+        parse.call(this, href);
+      },
+      get protocol() {
+        return this._scheme + ':';
+      },
+      set protocol(protocol) {
+        if (this._isInvalid) {
+          return;
+        }
+        parse.call(this, protocol + ':', 'scheme start');
+      },
+      get host() {
+        return this._isInvalid ? '' : this._port ? this._host + ':' + this._port : this._host;
+      },
+      set host(host) {
+        if (this._isInvalid || !this._isRelative) {
+          return;
+        }
+        parse.call(this, host, 'host');
+      },
+      get hostname() {
+        return this._host;
+      },
+      set hostname(hostname) {
+        if (this._isInvalid || !this._isRelative) {
+          return;
+        }
+        parse.call(this, hostname, 'hostname');
+      },
+      get port() {
+        return this._port;
+      },
+      set port(port) {
+        if (this._isInvalid || !this._isRelative) {
+          return;
+        }
+        parse.call(this, port, 'port');
+      },
+      get pathname() {
+        return this._isInvalid ? '' : this._isRelative ? '/' + this._path.join('/') : this._schemeData;
+      },
+      set pathname(pathname) {
+        if (this._isInvalid || !this._isRelative) {
+          return;
+        }
+        this._path = [];
+        parse.call(this, pathname, 'relative path start');
+      },
+      get search() {
+        return this._isInvalid || !this._query || this._query === '?' ? '' : this._query;
+      },
+      set search(search) {
+        if (this._isInvalid || !this._isRelative) {
+          return;
+        }
+        this._query = '?';
+        if (search[0] === '?') {
+          search = search.slice(1);
+        }
+        parse.call(this, search, 'query');
+      },
+      get hash() {
+        return this._isInvalid || !this._fragment || this._fragment === '#' ? '' : this._fragment;
+      },
+      set hash(hash) {
+        if (this._isInvalid) {
+          return;
+        }
+        this._fragment = '#';
+        if (hash[0] === '#') {
+          hash = hash.slice(1);
+        }
+        parse.call(this, hash, 'fragment');
+      },
+      get origin() {
+        var host;
+        if (this._isInvalid || !this._scheme) {
+          return '';
+        }
+        switch (this._scheme) {
+          case 'data':
+          case 'file':
+          case 'javascript':
+          case 'mailto':
+            return 'null';
+        }
+        host = this.host;
+        if (!host) {
+          return '';
+        }
+        return this._scheme + '://' + host;
+      }
+    };
+    var OriginalURL = globalScope.URL;
+    if (OriginalURL) {
+      JURL.createObjectURL = function (blob) {
+        return OriginalURL.createObjectURL.apply(OriginalURL, arguments);
+      };
+      JURL.revokeObjectURL = function (url) {
+        OriginalURL.revokeObjectURL(url);
+      };
     }
-    return this._scheme + '://' + host;
-   }
-  };
-  var OriginalURL = globalScope.URL;
-  if (OriginalURL) {
-   JURL.createObjectURL = function (blob) {
-    return OriginalURL.createObjectURL.apply(OriginalURL, arguments);
-   };
-   JURL.revokeObjectURL = function (url) {
-    OriginalURL.revokeObjectURL(url);
-   };
-  }
-  globalScope.URL = JURL;
- }());
+    globalScope.URL = JURL;
+  })();
 }

+ 889 - 1175
lib/shared/util.js

@@ -13,1332 +13,1046 @@
  * limitations under the License.
  */
 'use strict';
+
 var compatibility = require('./compatibility.js');
-var globalScope = typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : this;
-var FONT_IDENTITY_MATRIX = [
- 0.001,
- 0,
- 0,
- 0.001,
- 0,
- 0
-];
+var globalScope = typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : undefined;
+var FONT_IDENTITY_MATRIX = [0.001, 0, 0, 0.001, 0, 0];
 var TextRenderingMode = {
- FILL: 0,
- STROKE: 1,
- FILL_STROKE: 2,
- INVISIBLE: 3,
- FILL_ADD_TO_PATH: 4,
- STROKE_ADD_TO_PATH: 5,
- FILL_STROKE_ADD_TO_PATH: 6,
- ADD_TO_PATH: 7,
- FILL_STROKE_MASK: 3,
- ADD_TO_PATH_FLAG: 4
+  FILL: 0,
+  STROKE: 1,
+  FILL_STROKE: 2,
+  INVISIBLE: 3,
+  FILL_ADD_TO_PATH: 4,
+  STROKE_ADD_TO_PATH: 5,
+  FILL_STROKE_ADD_TO_PATH: 6,
+  ADD_TO_PATH: 7,
+  FILL_STROKE_MASK: 3,
+  ADD_TO_PATH_FLAG: 4
 };
 var ImageKind = {
- GRAYSCALE_1BPP: 1,
- RGB_24BPP: 2,
- RGBA_32BPP: 3
+  GRAYSCALE_1BPP: 1,
+  RGB_24BPP: 2,
+  RGBA_32BPP: 3
 };
 var AnnotationType = {
- TEXT: 1,
- LINK: 2,
- FREETEXT: 3,
- LINE: 4,
- SQUARE: 5,
- CIRCLE: 6,
- POLYGON: 7,
- POLYLINE: 8,
- HIGHLIGHT: 9,
- UNDERLINE: 10,
- SQUIGGLY: 11,
- STRIKEOUT: 12,
- STAMP: 13,
- CARET: 14,
- INK: 15,
- POPUP: 16,
- FILEATTACHMENT: 17,
- SOUND: 18,
- MOVIE: 19,
- WIDGET: 20,
- SCREEN: 21,
- PRINTERMARK: 22,
- TRAPNET: 23,
- WATERMARK: 24,
- THREED: 25,
- REDACT: 26
+  TEXT: 1,
+  LINK: 2,
+  FREETEXT: 3,
+  LINE: 4,
+  SQUARE: 5,
+  CIRCLE: 6,
+  POLYGON: 7,
+  POLYLINE: 8,
+  HIGHLIGHT: 9,
+  UNDERLINE: 10,
+  SQUIGGLY: 11,
+  STRIKEOUT: 12,
+  STAMP: 13,
+  CARET: 14,
+  INK: 15,
+  POPUP: 16,
+  FILEATTACHMENT: 17,
+  SOUND: 18,
+  MOVIE: 19,
+  WIDGET: 20,
+  SCREEN: 21,
+  PRINTERMARK: 22,
+  TRAPNET: 23,
+  WATERMARK: 24,
+  THREED: 25,
+  REDACT: 26
 };
 var AnnotationFlag = {
- INVISIBLE: 0x01,
- HIDDEN: 0x02,
- PRINT: 0x04,
- NOZOOM: 0x08,
- NOROTATE: 0x10,
- NOVIEW: 0x20,
- READONLY: 0x40,
- LOCKED: 0x80,
- TOGGLENOVIEW: 0x100,
- LOCKEDCONTENTS: 0x200
+  INVISIBLE: 0x01,
+  HIDDEN: 0x02,
+  PRINT: 0x04,
+  NOZOOM: 0x08,
+  NOROTATE: 0x10,
+  NOVIEW: 0x20,
+  READONLY: 0x40,
+  LOCKED: 0x80,
+  TOGGLENOVIEW: 0x100,
+  LOCKEDCONTENTS: 0x200
 };
 var AnnotationFieldFlag = {
- READONLY: 0x0000001,
- REQUIRED: 0x0000002,
- NOEXPORT: 0x0000004,
- MULTILINE: 0x0001000,
- PASSWORD: 0x0002000,
- NOTOGGLETOOFF: 0x0004000,
- RADIO: 0x0008000,
- PUSHBUTTON: 0x0010000,
- COMBO: 0x0020000,
- EDIT: 0x0040000,
- SORT: 0x0080000,
- FILESELECT: 0x0100000,
- MULTISELECT: 0x0200000,
- DONOTSPELLCHECK: 0x0400000,
- DONOTSCROLL: 0x0800000,
- COMB: 0x1000000,
- RICHTEXT: 0x2000000,
- RADIOSINUNISON: 0x2000000,
- COMMITONSELCHANGE: 0x4000000
+  READONLY: 0x0000001,
+  REQUIRED: 0x0000002,
+  NOEXPORT: 0x0000004,
+  MULTILINE: 0x0001000,
+  PASSWORD: 0x0002000,
+  NOTOGGLETOOFF: 0x0004000,
+  RADIO: 0x0008000,
+  PUSHBUTTON: 0x0010000,
+  COMBO: 0x0020000,
+  EDIT: 0x0040000,
+  SORT: 0x0080000,
+  FILESELECT: 0x0100000,
+  MULTISELECT: 0x0200000,
+  DONOTSPELLCHECK: 0x0400000,
+  DONOTSCROLL: 0x0800000,
+  COMB: 0x1000000,
+  RICHTEXT: 0x2000000,
+  RADIOSINUNISON: 0x2000000,
+  COMMITONSELCHANGE: 0x4000000
 };
 var AnnotationBorderStyleType = {
- SOLID: 1,
- DASHED: 2,
- BEVELED: 3,
- INSET: 4,
- UNDERLINE: 5
+  SOLID: 1,
+  DASHED: 2,
+  BEVELED: 3,
+  INSET: 4,
+  UNDERLINE: 5
 };
 var StreamType = {
- UNKNOWN: 0,
- FLATE: 1,
- LZW: 2,
- DCT: 3,
- JPX: 4,
- JBIG: 5,
- A85: 6,
- AHX: 7,
- CCF: 8,
- RL: 9
+  UNKNOWN: 0,
+  FLATE: 1,
+  LZW: 2,
+  DCT: 3,
+  JPX: 4,
+  JBIG: 5,
+  A85: 6,
+  AHX: 7,
+  CCF: 8,
+  RL: 9
 };
 var FontType = {
- UNKNOWN: 0,
- TYPE1: 1,
- TYPE1C: 2,
- CIDFONTTYPE0: 3,
- CIDFONTTYPE0C: 4,
- TRUETYPE: 5,
- CIDFONTTYPE2: 6,
- TYPE3: 7,
- OPENTYPE: 8,
- TYPE0: 9,
- MMTYPE1: 10
+  UNKNOWN: 0,
+  TYPE1: 1,
+  TYPE1C: 2,
+  CIDFONTTYPE0: 3,
+  CIDFONTTYPE0C: 4,
+  TRUETYPE: 5,
+  CIDFONTTYPE2: 6,
+  TYPE3: 7,
+  OPENTYPE: 8,
+  TYPE0: 9,
+  MMTYPE1: 10
 };
 var VERBOSITY_LEVELS = {
- errors: 0,
- warnings: 1,
- infos: 5
+  errors: 0,
+  warnings: 1,
+  infos: 5
 };
 var CMapCompressionType = {
- NONE: 0,
- BINARY: 1,
- STREAM: 2
+  NONE: 0,
+  BINARY: 1,
+  STREAM: 2
 };
 var OPS = {
- dependency: 1,
- setLineWidth: 2,
- setLineCap: 3,
- setLineJoin: 4,
- setMiterLimit: 5,
- setDash: 6,
- setRenderingIntent: 7,
- setFlatness: 8,
- setGState: 9,
- save: 10,
- restore: 11,
- transform: 12,
- moveTo: 13,
- lineTo: 14,
- curveTo: 15,
- curveTo2: 16,
- curveTo3: 17,
- closePath: 18,
- rectangle: 19,
- stroke: 20,
- closeStroke: 21,
- fill: 22,
- eoFill: 23,
- fillStroke: 24,
- eoFillStroke: 25,
- closeFillStroke: 26,
- closeEOFillStroke: 27,
- endPath: 28,
- clip: 29,
- eoClip: 30,
- beginText: 31,
- endText: 32,
- setCharSpacing: 33,
- setWordSpacing: 34,
- setHScale: 35,
- setLeading: 36,
- setFont: 37,
- setTextRenderingMode: 38,
- setTextRise: 39,
- moveText: 40,
- setLeadingMoveText: 41,
- setTextMatrix: 42,
- nextLine: 43,
- showText: 44,
- showSpacedText: 45,
- nextLineShowText: 46,
- nextLineSetSpacingShowText: 47,
- setCharWidth: 48,
- setCharWidthAndBounds: 49,
- setStrokeColorSpace: 50,
- setFillColorSpace: 51,
- setStrokeColor: 52,
- setStrokeColorN: 53,
- setFillColor: 54,
- setFillColorN: 55,
- setStrokeGray: 56,
- setFillGray: 57,
- setStrokeRGBColor: 58,
- setFillRGBColor: 59,
- setStrokeCMYKColor: 60,
- setFillCMYKColor: 61,
- shadingFill: 62,
- beginInlineImage: 63,
- beginImageData: 64,
- endInlineImage: 65,
- paintXObject: 66,
- markPoint: 67,
- markPointProps: 68,
- beginMarkedContent: 69,
- beginMarkedContentProps: 70,
- endMarkedContent: 71,
- beginCompat: 72,
- endCompat: 73,
- paintFormXObjectBegin: 74,
- paintFormXObjectEnd: 75,
- beginGroup: 76,
- endGroup: 77,
- beginAnnotations: 78,
- endAnnotations: 79,
- beginAnnotation: 80,
- endAnnotation: 81,
- paintJpegXObject: 82,
- paintImageMaskXObject: 83,
- paintImageMaskXObjectGroup: 84,
- paintImageXObject: 85,
- paintInlineImageXObject: 86,
- paintInlineImageXObjectGroup: 87,
- paintImageXObjectRepeat: 88,
- paintImageMaskXObjectRepeat: 89,
- paintSolidColorImageMask: 90,
- constructPath: 91
+  dependency: 1,
+  setLineWidth: 2,
+  setLineCap: 3,
+  setLineJoin: 4,
+  setMiterLimit: 5,
+  setDash: 6,
+  setRenderingIntent: 7,
+  setFlatness: 8,
+  setGState: 9,
+  save: 10,
+  restore: 11,
+  transform: 12,
+  moveTo: 13,
+  lineTo: 14,
+  curveTo: 15,
+  curveTo2: 16,
+  curveTo3: 17,
+  closePath: 18,
+  rectangle: 19,
+  stroke: 20,
+  closeStroke: 21,
+  fill: 22,
+  eoFill: 23,
+  fillStroke: 24,
+  eoFillStroke: 25,
+  closeFillStroke: 26,
+  closeEOFillStroke: 27,
+  endPath: 28,
+  clip: 29,
+  eoClip: 30,
+  beginText: 31,
+  endText: 32,
+  setCharSpacing: 33,
+  setWordSpacing: 34,
+  setHScale: 35,
+  setLeading: 36,
+  setFont: 37,
+  setTextRenderingMode: 38,
+  setTextRise: 39,
+  moveText: 40,
+  setLeadingMoveText: 41,
+  setTextMatrix: 42,
+  nextLine: 43,
+  showText: 44,
+  showSpacedText: 45,
+  nextLineShowText: 46,
+  nextLineSetSpacingShowText: 47,
+  setCharWidth: 48,
+  setCharWidthAndBounds: 49,
+  setStrokeColorSpace: 50,
+  setFillColorSpace: 51,
+  setStrokeColor: 52,
+  setStrokeColorN: 53,
+  setFillColor: 54,
+  setFillColorN: 55,
+  setStrokeGray: 56,
+  setFillGray: 57,
+  setStrokeRGBColor: 58,
+  setFillRGBColor: 59,
+  setStrokeCMYKColor: 60,
+  setFillCMYKColor: 61,
+  shadingFill: 62,
+  beginInlineImage: 63,
+  beginImageData: 64,
+  endInlineImage: 65,
+  paintXObject: 66,
+  markPoint: 67,
+  markPointProps: 68,
+  beginMarkedContent: 69,
+  beginMarkedContentProps: 70,
+  endMarkedContent: 71,
+  beginCompat: 72,
+  endCompat: 73,
+  paintFormXObjectBegin: 74,
+  paintFormXObjectEnd: 75,
+  beginGroup: 76,
+  endGroup: 77,
+  beginAnnotations: 78,
+  endAnnotations: 79,
+  beginAnnotation: 80,
+  endAnnotation: 81,
+  paintJpegXObject: 82,
+  paintImageMaskXObject: 83,
+  paintImageMaskXObjectGroup: 84,
+  paintImageXObject: 85,
+  paintInlineImageXObject: 86,
+  paintInlineImageXObjectGroup: 87,
+  paintImageXObjectRepeat: 88,
+  paintImageMaskXObjectRepeat: 89,
+  paintSolidColorImageMask: 90,
+  constructPath: 91
 };
 var verbosity = VERBOSITY_LEVELS.warnings;
 function setVerbosityLevel(level) {
- verbosity = level;
+  verbosity = level;
 }
 function getVerbosityLevel() {
- return verbosity;
+  return verbosity;
 }
 function info(msg) {
- if (verbosity >= VERBOSITY_LEVELS.infos) {
-  console.log('Info: ' + msg);
- }
+  if (verbosity >= VERBOSITY_LEVELS.infos) {
+    console.log('Info: ' + msg);
+  }
 }
 function warn(msg) {
- if (verbosity >= VERBOSITY_LEVELS.warnings) {
-  console.log('Warning: ' + msg);
- }
+  if (verbosity >= VERBOSITY_LEVELS.warnings) {
+    console.log('Warning: ' + msg);
+  }
 }
 function deprecated(details) {
- console.log('Deprecated API usage: ' + details);
+  console.log('Deprecated API usage: ' + details);
 }
 function error(msg) {
- if (verbosity >= VERBOSITY_LEVELS.errors) {
-  console.log('Error: ' + msg);
-  console.log(backtrace());
- }
- throw new Error(msg);
+  if (verbosity >= VERBOSITY_LEVELS.errors) {
+    console.log('Error: ' + msg);
+    console.log(backtrace());
+  }
+  throw new Error(msg);
 }
 function backtrace() {
- try {
-  throw new Error();
- } catch (e) {
-  return e.stack ? e.stack.split('\n').slice(2).join('\n') : '';
- }
+  try {
+    throw new Error();
+  } catch (e) {
+    return e.stack ? e.stack.split('\n').slice(2).join('\n') : '';
+  }
 }
 function assert(cond, msg) {
- if (!cond) {
-  error(msg);
- }
+  if (!cond) {
+    error(msg);
+  }
 }
 var UNSUPPORTED_FEATURES = {
- unknown: 'unknown',
- forms: 'forms',
- javaScript: 'javaScript',
- smask: 'smask',
- shadingPattern: 'shadingPattern',
- font: 'font'
+  unknown: 'unknown',
+  forms: 'forms',
+  javaScript: 'javaScript',
+  smask: 'smask',
+  shadingPattern: 'shadingPattern',
+  font: 'font'
 };
 function isSameOrigin(baseUrl, otherUrl) {
- try {
-  var base = new URL(baseUrl);
-  if (!base.origin || base.origin === 'null') {
-   return false;
+  try {
+    var base = new URL(baseUrl);
+    if (!base.origin || base.origin === 'null') {
+      return false;
+    }
+  } catch (e) {
+    return false;
   }
- } catch (e) {
-  return false;
- }
- var other = new URL(otherUrl, base);
- return base.origin === other.origin;
+  var other = new URL(otherUrl, base);
+  return base.origin === other.origin;
 }
 function isValidProtocol(url) {
- if (!url) {
-  return false;
- }
- switch (url.protocol) {
- case 'http:':
- case 'https:':
- case 'ftp:':
- case 'mailto:':
- case 'tel:':
-  return true;
- default:
-  return false;
- }
+  if (!url) {
+    return false;
+  }
+  switch (url.protocol) {
+    case 'http:':
+    case 'https:':
+    case 'ftp:':
+    case 'mailto:':
+    case 'tel:':
+      return true;
+    default:
+      return false;
+  }
 }
 function createValidAbsoluteUrl(url, baseUrl) {
- if (!url) {
-  return null;
- }
- try {
-  var absoluteUrl = baseUrl ? new URL(url, baseUrl) : new URL(url);
-  if (isValidProtocol(absoluteUrl)) {
-   return absoluteUrl;
+  if (!url) {
+    return null;
   }
- } catch (ex) {
- }
- return null;
+  try {
+    var absoluteUrl = baseUrl ? new URL(url, baseUrl) : new URL(url);
+    if (isValidProtocol(absoluteUrl)) {
+      return absoluteUrl;
+    }
+  } catch (ex) {}
+  return null;
 }
 function shadow(obj, prop, value) {
- Object.defineProperty(obj, prop, {
-  value: value,
-  enumerable: true,
-  configurable: true,
-  writable: false
- });
- return value;
+  Object.defineProperty(obj, prop, {
+    value: value,
+    enumerable: true,
+    configurable: true,
+    writable: false
+  });
+  return value;
 }
 function getLookupTableFactory(initializer) {
- var lookup;
- return function () {
-  if (initializer) {
-   lookup = Object.create(null);
-   initializer(lookup);
-   initializer = null;
-  }
-  return lookup;
- };
+  var lookup;
+  return function () {
+    if (initializer) {
+      lookup = Object.create(null);
+      initializer(lookup);
+      initializer = null;
+    }
+    return lookup;
+  };
 }
 var PasswordResponses = {
- NEED_PASSWORD: 1,
- INCORRECT_PASSWORD: 2
+  NEED_PASSWORD: 1,
+  INCORRECT_PASSWORD: 2
 };
 var PasswordException = function PasswordExceptionClosure() {
- function PasswordException(msg, code) {
-  this.name = 'PasswordException';
-  this.message = msg;
-  this.code = code;
- }
- PasswordException.prototype = new Error();
- PasswordException.constructor = PasswordException;
- return PasswordException;
+  function PasswordException(msg, code) {
+    this.name = 'PasswordException';
+    this.message = msg;
+    this.code = code;
+  }
+  PasswordException.prototype = new Error();
+  PasswordException.constructor = PasswordException;
+  return PasswordException;
 }();
 var UnknownErrorException = function UnknownErrorExceptionClosure() {
- function UnknownErrorException(msg, details) {
-  this.name = 'UnknownErrorException';
-  this.message = msg;
-  this.details = details;
- }
- UnknownErrorException.prototype = new Error();
- UnknownErrorException.constructor = UnknownErrorException;
- return UnknownErrorException;
+  function UnknownErrorException(msg, details) {
+    this.name = 'UnknownErrorException';
+    this.message = msg;
+    this.details = details;
+  }
+  UnknownErrorException.prototype = new Error();
+  UnknownErrorException.constructor = UnknownErrorException;
+  return UnknownErrorException;
 }();
 var InvalidPDFException = function InvalidPDFExceptionClosure() {
- function InvalidPDFException(msg) {
-  this.name = 'InvalidPDFException';
-  this.message = msg;
- }
- InvalidPDFException.prototype = new Error();
- InvalidPDFException.constructor = InvalidPDFException;
- return InvalidPDFException;
+  function InvalidPDFException(msg) {
+    this.name = 'InvalidPDFException';
+    this.message = msg;
+  }
+  InvalidPDFException.prototype = new Error();
+  InvalidPDFException.constructor = InvalidPDFException;
+  return InvalidPDFException;
 }();
 var MissingPDFException = function MissingPDFExceptionClosure() {
- function MissingPDFException(msg) {
-  this.name = 'MissingPDFException';
-  this.message = msg;
- }
- MissingPDFException.prototype = new Error();
- MissingPDFException.constructor = MissingPDFException;
- return MissingPDFException;
+  function MissingPDFException(msg) {
+    this.name = 'MissingPDFException';
+    this.message = msg;
+  }
+  MissingPDFException.prototype = new Error();
+  MissingPDFException.constructor = MissingPDFException;
+  return MissingPDFException;
 }();
 var UnexpectedResponseException = function UnexpectedResponseExceptionClosure() {
- function UnexpectedResponseException(msg, status) {
-  this.name = 'UnexpectedResponseException';
-  this.message = msg;
-  this.status = status;
- }
- UnexpectedResponseException.prototype = new Error();
- UnexpectedResponseException.constructor = UnexpectedResponseException;
- return UnexpectedResponseException;
+  function UnexpectedResponseException(msg, status) {
+    this.name = 'UnexpectedResponseException';
+    this.message = msg;
+    this.status = status;
+  }
+  UnexpectedResponseException.prototype = new Error();
+  UnexpectedResponseException.constructor = UnexpectedResponseException;
+  return UnexpectedResponseException;
 }();
 var NotImplementedException = function NotImplementedExceptionClosure() {
- function NotImplementedException(msg) {
-  this.message = msg;
- }
- NotImplementedException.prototype = new Error();
- NotImplementedException.prototype.name = 'NotImplementedException';
- NotImplementedException.constructor = NotImplementedException;
- return NotImplementedException;
+  function NotImplementedException(msg) {
+    this.message = msg;
+  }
+  NotImplementedException.prototype = new Error();
+  NotImplementedException.prototype.name = 'NotImplementedException';
+  NotImplementedException.constructor = NotImplementedException;
+  return NotImplementedException;
 }();
 var MissingDataException = function MissingDataExceptionClosure() {
- function MissingDataException(begin, end) {
-  this.begin = begin;
-  this.end = end;
-  this.message = 'Missing data [' + begin + ', ' + end + ')';
- }
- MissingDataException.prototype = new Error();
- MissingDataException.prototype.name = 'MissingDataException';
- MissingDataException.constructor = MissingDataException;
- return MissingDataException;
+  function MissingDataException(begin, end) {
+    this.begin = begin;
+    this.end = end;
+    this.message = 'Missing data [' + begin + ', ' + end + ')';
+  }
+  MissingDataException.prototype = new Error();
+  MissingDataException.prototype.name = 'MissingDataException';
+  MissingDataException.constructor = MissingDataException;
+  return MissingDataException;
 }();
 var XRefParseException = function XRefParseExceptionClosure() {
- function XRefParseException(msg) {
-  this.message = msg;
- }
- XRefParseException.prototype = new Error();
- XRefParseException.prototype.name = 'XRefParseException';
- XRefParseException.constructor = XRefParseException;
- return XRefParseException;
+  function XRefParseException(msg) {
+    this.message = msg;
+  }
+  XRefParseException.prototype = new Error();
+  XRefParseException.prototype.name = 'XRefParseException';
+  XRefParseException.constructor = XRefParseException;
+  return XRefParseException;
 }();
 var NullCharactersRegExp = /\x00/g;
 function removeNullCharacters(str) {
- if (typeof str !== 'string') {
-  warn('The argument for removeNullCharacters must be a string.');
-  return str;
- }
- return str.replace(NullCharactersRegExp, '');
+  if (typeof str !== 'string') {
+    warn('The argument for removeNullCharacters must be a string.');
+    return str;
+  }
+  return str.replace(NullCharactersRegExp, '');
 }
 function bytesToString(bytes) {
- assert(bytes !== null && typeof bytes === 'object' && bytes.length !== undefined, 'Invalid argument for bytesToString');
- var length = bytes.length;
- var MAX_ARGUMENT_COUNT = 8192;
- if (length < MAX_ARGUMENT_COUNT) {
-  return String.fromCharCode.apply(null, bytes);
- }
- var strBuf = [];
- for (var i = 0; i < length; i += MAX_ARGUMENT_COUNT) {
-  var chunkEnd = Math.min(i + MAX_ARGUMENT_COUNT, length);
-  var chunk = bytes.subarray(i, chunkEnd);
-  strBuf.push(String.fromCharCode.apply(null, chunk));
- }
- return strBuf.join('');
+  assert(bytes !== null && typeof bytes === 'object' && bytes.length !== undefined, 'Invalid argument for bytesToString');
+  var length = bytes.length;
+  var MAX_ARGUMENT_COUNT = 8192;
+  if (length < MAX_ARGUMENT_COUNT) {
+    return String.fromCharCode.apply(null, bytes);
+  }
+  var strBuf = [];
+  for (var i = 0; i < length; i += MAX_ARGUMENT_COUNT) {
+    var chunkEnd = Math.min(i + MAX_ARGUMENT_COUNT, length);
+    var chunk = bytes.subarray(i, chunkEnd);
+    strBuf.push(String.fromCharCode.apply(null, chunk));
+  }
+  return strBuf.join('');
 }
 function stringToBytes(str) {
- assert(typeof str === 'string', 'Invalid argument for stringToBytes');
- var length = str.length;
- var bytes = new Uint8Array(length);
- for (var i = 0; i < length; ++i) {
-  bytes[i] = str.charCodeAt(i) & 0xFF;
- }
- return bytes;
+  assert(typeof str === 'string', 'Invalid argument for stringToBytes');
+  var length = str.length;
+  var bytes = new Uint8Array(length);
+  for (var i = 0; i < length; ++i) {
+    bytes[i] = str.charCodeAt(i) & 0xFF;
+  }
+  return bytes;
 }
 function arrayByteLength(arr) {
- if (arr.length !== undefined) {
-  return arr.length;
- }
- assert(arr.byteLength !== undefined);
- return arr.byteLength;
+  if (arr.length !== undefined) {
+    return arr.length;
+  }
+  assert(arr.byteLength !== undefined);
+  return arr.byteLength;
 }
 function arraysToBytes(arr) {
- if (arr.length === 1 && arr[0] instanceof Uint8Array) {
-  return arr[0];
- }
- var resultLength = 0;
- var i, ii = arr.length;
- var item, itemLength;
- for (i = 0; i < ii; i++) {
-  item = arr[i];
-  itemLength = arrayByteLength(item);
-  resultLength += itemLength;
- }
- var pos = 0;
- var data = new Uint8Array(resultLength);
- for (i = 0; i < ii; i++) {
-  item = arr[i];
-  if (!(item instanceof Uint8Array)) {
-   if (typeof item === 'string') {
-    item = stringToBytes(item);
-   } else {
-    item = new Uint8Array(item);
-   }
+  if (arr.length === 1 && arr[0] instanceof Uint8Array) {
+    return arr[0];
+  }
+  var resultLength = 0;
+  var i,
+      ii = arr.length;
+  var item, itemLength;
+  for (i = 0; i < ii; i++) {
+    item = arr[i];
+    itemLength = arrayByteLength(item);
+    resultLength += itemLength;
   }
-  itemLength = item.byteLength;
-  data.set(item, pos);
-  pos += itemLength;
- }
- return data;
+  var pos = 0;
+  var data = new Uint8Array(resultLength);
+  for (i = 0; i < ii; i++) {
+    item = arr[i];
+    if (!(item instanceof Uint8Array)) {
+      if (typeof item === 'string') {
+        item = stringToBytes(item);
+      } else {
+        item = new Uint8Array(item);
+      }
+    }
+    itemLength = item.byteLength;
+    data.set(item, pos);
+    pos += itemLength;
+  }
+  return data;
 }
 function string32(value) {
- return String.fromCharCode(value >> 24 & 0xff, value >> 16 & 0xff, value >> 8 & 0xff, value & 0xff);
+  return String.fromCharCode(value >> 24 & 0xff, value >> 16 & 0xff, value >> 8 & 0xff, value & 0xff);
 }
 function log2(x) {
- var n = 1, i = 0;
- while (x > n) {
-  n <<= 1;
-  i++;
- }
- return i;
+  var n = 1,
+      i = 0;
+  while (x > n) {
+    n <<= 1;
+    i++;
+  }
+  return i;
 }
 function readInt8(data, start) {
- return data[start] << 24 >> 24;
+  return data[start] << 24 >> 24;
 }
 function readUint16(data, offset) {
- return data[offset] << 8 | data[offset + 1];
+  return data[offset] << 8 | data[offset + 1];
 }
 function readUint32(data, offset) {
- return (data[offset] << 24 | data[offset + 1] << 16 | data[offset + 2] << 8 | data[offset + 3]) >>> 0;
+  return (data[offset] << 24 | data[offset + 1] << 16 | data[offset + 2] << 8 | data[offset + 3]) >>> 0;
 }
 function isLittleEndian() {
- var buffer8 = new Uint8Array(2);
- buffer8[0] = 1;
- var buffer16 = new Uint16Array(buffer8.buffer);
- return buffer16[0] === 1;
+  var buffer8 = new Uint8Array(2);
+  buffer8[0] = 1;
+  var buffer16 = new Uint16Array(buffer8.buffer);
+  return buffer16[0] === 1;
 }
 function isEvalSupported() {
- try {
-  new Function('');
-  return true;
- } catch (e) {
-  return false;
- }
+  try {
+    new Function('');
+    return true;
+  } catch (e) {
+    return false;
+  }
 }
 var Uint32ArrayView = function Uint32ArrayViewClosure() {
- function Uint32ArrayView(buffer, length) {
-  this.buffer = buffer;
-  this.byteLength = buffer.length;
-  this.length = length === undefined ? this.byteLength >> 2 : length;
-  ensureUint32ArrayViewProps(this.length);
- }
- Uint32ArrayView.prototype = Object.create(null);
- var uint32ArrayViewSetters = 0;
- function createUint32ArrayProp(index) {
-  return {
-   get: function () {
-    var buffer = this.buffer, offset = index << 2;
-    return (buffer[offset] | buffer[offset + 1] << 8 | buffer[offset + 2] << 16 | buffer[offset + 3] << 24) >>> 0;
-   },
-   set: function (value) {
-    var buffer = this.buffer, offset = index << 2;
-    buffer[offset] = value & 255;
-    buffer[offset + 1] = value >> 8 & 255;
-    buffer[offset + 2] = value >> 16 & 255;
-    buffer[offset + 3] = value >>> 24 & 255;
-   }
-  };
- }
- function ensureUint32ArrayViewProps(length) {
-  while (uint32ArrayViewSetters < length) {
-   Object.defineProperty(Uint32ArrayView.prototype, uint32ArrayViewSetters, createUint32ArrayProp(uint32ArrayViewSetters));
-   uint32ArrayViewSetters++;
+  function Uint32ArrayView(buffer, length) {
+    this.buffer = buffer;
+    this.byteLength = buffer.length;
+    this.length = length === undefined ? this.byteLength >> 2 : length;
+    ensureUint32ArrayViewProps(this.length);
+  }
+  Uint32ArrayView.prototype = Object.create(null);
+  var uint32ArrayViewSetters = 0;
+  function createUint32ArrayProp(index) {
+    return {
+      get: function () {
+        var buffer = this.buffer,
+            offset = index << 2;
+        return (buffer[offset] | buffer[offset + 1] << 8 | buffer[offset + 2] << 16 | buffer[offset + 3] << 24) >>> 0;
+      },
+      set: function (value) {
+        var buffer = this.buffer,
+            offset = index << 2;
+        buffer[offset] = value & 255;
+        buffer[offset + 1] = value >> 8 & 255;
+        buffer[offset + 2] = value >> 16 & 255;
+        buffer[offset + 3] = value >>> 24 & 255;
+      }
+    };
   }
- }
- return Uint32ArrayView;
+  function ensureUint32ArrayViewProps(length) {
+    while (uint32ArrayViewSetters < length) {
+      Object.defineProperty(Uint32ArrayView.prototype, uint32ArrayViewSetters, createUint32ArrayProp(uint32ArrayViewSetters));
+      uint32ArrayViewSetters++;
+    }
+  }
+  return Uint32ArrayView;
 }();
 exports.Uint32ArrayView = Uint32ArrayView;
-var IDENTITY_MATRIX = [
- 1,
- 0,
- 0,
- 1,
- 0,
- 0
-];
+var IDENTITY_MATRIX = [1, 0, 0, 1, 0, 0];
 var Util = function UtilClosure() {
- function Util() {
- }
- var rgbBuf = [
-  'rgb(',
-  0,
-  ',',
-  0,
-  ',',
-  0,
-  ')'
- ];
- Util.makeCssRgb = function Util_makeCssRgb(r, g, b) {
-  rgbBuf[1] = r;
-  rgbBuf[3] = g;
-  rgbBuf[5] = b;
-  return rgbBuf.join('');
- };
- Util.transform = function Util_transform(m1, m2) {
-  return [
-   m1[0] * m2[0] + m1[2] * m2[1],
-   m1[1] * m2[0] + m1[3] * m2[1],
-   m1[0] * m2[2] + m1[2] * m2[3],
-   m1[1] * m2[2] + m1[3] * m2[3],
-   m1[0] * m2[4] + m1[2] * m2[5] + m1[4],
-   m1[1] * m2[4] + m1[3] * m2[5] + m1[5]
-  ];
- };
- Util.applyTransform = function Util_applyTransform(p, m) {
-  var xt = p[0] * m[0] + p[1] * m[2] + m[4];
-  var yt = p[0] * m[1] + p[1] * m[3] + m[5];
-  return [
-   xt,
-   yt
-  ];
- };
- Util.applyInverseTransform = function Util_applyInverseTransform(p, m) {
-  var d = m[0] * m[3] - m[1] * m[2];
-  var xt = (p[0] * m[3] - p[1] * m[2] + m[2] * m[5] - m[4] * m[3]) / d;
-  var yt = (-p[0] * m[1] + p[1] * m[0] + m[4] * m[1] - m[5] * m[0]) / d;
-  return [
-   xt,
-   yt
-  ];
- };
- Util.getAxialAlignedBoundingBox = function Util_getAxialAlignedBoundingBox(r, m) {
-  var p1 = Util.applyTransform(r, m);
-  var p2 = Util.applyTransform(r.slice(2, 4), m);
-  var p3 = Util.applyTransform([
-   r[0],
-   r[3]
-  ], m);
-  var p4 = Util.applyTransform([
-   r[2],
-   r[1]
-  ], m);
-  return [
-   Math.min(p1[0], p2[0], p3[0], p4[0]),
-   Math.min(p1[1], p2[1], p3[1], p4[1]),
-   Math.max(p1[0], p2[0], p3[0], p4[0]),
-   Math.max(p1[1], p2[1], p3[1], p4[1])
-  ];
- };
- Util.inverseTransform = function Util_inverseTransform(m) {
-  var d = m[0] * m[3] - m[1] * m[2];
-  return [
-   m[3] / d,
-   -m[1] / d,
-   -m[2] / d,
-   m[0] / d,
-   (m[2] * m[5] - m[4] * m[3]) / d,
-   (m[4] * m[1] - m[5] * m[0]) / d
-  ];
- };
- Util.apply3dTransform = function Util_apply3dTransform(m, v) {
-  return [
-   m[0] * v[0] + m[1] * v[1] + m[2] * v[2],
-   m[3] * v[0] + m[4] * v[1] + m[5] * v[2],
-   m[6] * v[0] + m[7] * v[1] + m[8] * v[2]
-  ];
- };
- Util.singularValueDecompose2dScale = function Util_singularValueDecompose2dScale(m) {
-  var transpose = [
-   m[0],
-   m[2],
-   m[1],
-   m[3]
-  ];
-  var a = m[0] * transpose[0] + m[1] * transpose[2];
-  var b = m[0] * transpose[1] + m[1] * transpose[3];
-  var c = m[2] * transpose[0] + m[3] * transpose[2];
-  var d = m[2] * transpose[1] + m[3] * transpose[3];
-  var first = (a + d) / 2;
-  var second = Math.sqrt((a + d) * (a + d) - 4 * (a * d - c * b)) / 2;
-  var sx = first + second || 1;
-  var sy = first - second || 1;
-  return [
-   Math.sqrt(sx),
-   Math.sqrt(sy)
-  ];
- };
- Util.normalizeRect = function Util_normalizeRect(rect) {
-  var r = rect.slice(0);
-  if (rect[0] > rect[2]) {
-   r[0] = rect[2];
-   r[2] = rect[0];
-  }
-  if (rect[1] > rect[3]) {
-   r[1] = rect[3];
-   r[3] = rect[1];
-  }
-  return r;
- };
- Util.intersect = function Util_intersect(rect1, rect2) {
-  function compare(a, b) {
-   return a - b;
-  }
-  var orderedX = [
-    rect1[0],
-    rect1[2],
-    rect2[0],
-    rect2[2]
-   ].sort(compare), orderedY = [
-    rect1[1],
-    rect1[3],
-    rect2[1],
-    rect2[3]
-   ].sort(compare), result = [];
-  rect1 = Util.normalizeRect(rect1);
-  rect2 = Util.normalizeRect(rect2);
-  if (orderedX[0] === rect1[0] && orderedX[1] === rect2[0] || orderedX[0] === rect2[0] && orderedX[1] === rect1[0]) {
-   result[0] = orderedX[1];
-   result[2] = orderedX[2];
-  } else {
-   return false;
-  }
-  if (orderedY[0] === rect1[1] && orderedY[1] === rect2[1] || orderedY[0] === rect2[1] && orderedY[1] === rect1[1]) {
-   result[1] = orderedY[1];
-   result[3] = orderedY[2];
-  } else {
-   return false;
-  }
-  return result;
- };
- Util.sign = function Util_sign(num) {
-  return num < 0 ? -1 : 1;
- };
- var ROMAN_NUMBER_MAP = [
-  '',
-  'C',
-  'CC',
-  'CCC',
-  'CD',
-  'D',
-  'DC',
-  'DCC',
-  'DCCC',
-  'CM',
-  '',
-  'X',
-  'XX',
-  'XXX',
-  'XL',
-  'L',
-  'LX',
-  'LXX',
-  'LXXX',
-  'XC',
-  '',
-  'I',
-  'II',
-  'III',
-  'IV',
-  'V',
-  'VI',
-  'VII',
-  'VIII',
-  'IX'
- ];
- Util.toRoman = function Util_toRoman(number, lowerCase) {
-  assert(isInt(number) && number > 0, 'The number should be a positive integer.');
-  var pos, romanBuf = [];
-  while (number >= 1000) {
-   number -= 1000;
-   romanBuf.push('M');
-  }
-  pos = number / 100 | 0;
-  number %= 100;
-  romanBuf.push(ROMAN_NUMBER_MAP[pos]);
-  pos = number / 10 | 0;
-  number %= 10;
-  romanBuf.push(ROMAN_NUMBER_MAP[10 + pos]);
-  romanBuf.push(ROMAN_NUMBER_MAP[20 + number]);
-  var romanStr = romanBuf.join('');
-  return lowerCase ? romanStr.toLowerCase() : romanStr;
- };
- Util.appendToArray = function Util_appendToArray(arr1, arr2) {
-  Array.prototype.push.apply(arr1, arr2);
- };
- Util.prependToArray = function Util_prependToArray(arr1, arr2) {
-  Array.prototype.unshift.apply(arr1, arr2);
- };
- Util.extendObj = function extendObj(obj1, obj2) {
-  for (var key in obj2) {
-   obj1[key] = obj2[key];
-  }
- };
- Util.getInheritableProperty = function Util_getInheritableProperty(dict, name, getArray) {
-  while (dict && !dict.has(name)) {
-   dict = dict.get('Parent');
-  }
-  if (!dict) {
-   return null;
-  }
-  return getArray ? dict.getArray(name) : dict.get(name);
- };
- Util.inherit = function Util_inherit(sub, base, prototype) {
-  sub.prototype = Object.create(base.prototype);
-  sub.prototype.constructor = sub;
-  for (var prop in prototype) {
-   sub.prototype[prop] = prototype[prop];
-  }
- };
- Util.loadScript = function Util_loadScript(src, callback) {
-  var script = document.createElement('script');
-  var loaded = false;
-  script.setAttribute('src', src);
-  if (callback) {
-   script.onload = function () {
-    if (!loaded) {
-     callback();
+  function Util() {}
+  var rgbBuf = ['rgb(', 0, ',', 0, ',', 0, ')'];
+  Util.makeCssRgb = function Util_makeCssRgb(r, g, b) {
+    rgbBuf[1] = r;
+    rgbBuf[3] = g;
+    rgbBuf[5] = b;
+    return rgbBuf.join('');
+  };
+  Util.transform = function Util_transform(m1, m2) {
+    return [m1[0] * m2[0] + m1[2] * m2[1], m1[1] * m2[0] + m1[3] * m2[1], m1[0] * m2[2] + m1[2] * m2[3], m1[1] * m2[2] + m1[3] * m2[3], m1[0] * m2[4] + m1[2] * m2[5] + m1[4], m1[1] * m2[4] + m1[3] * m2[5] + m1[5]];
+  };
+  Util.applyTransform = function Util_applyTransform(p, m) {
+    var xt = p[0] * m[0] + p[1] * m[2] + m[4];
+    var yt = p[0] * m[1] + p[1] * m[3] + m[5];
+    return [xt, yt];
+  };
+  Util.applyInverseTransform = function Util_applyInverseTransform(p, m) {
+    var d = m[0] * m[3] - m[1] * m[2];
+    var xt = (p[0] * m[3] - p[1] * m[2] + m[2] * m[5] - m[4] * m[3]) / d;
+    var yt = (-p[0] * m[1] + p[1] * m[0] + m[4] * m[1] - m[5] * m[0]) / d;
+    return [xt, yt];
+  };
+  Util.getAxialAlignedBoundingBox = function Util_getAxialAlignedBoundingBox(r, m) {
+    var p1 = Util.applyTransform(r, m);
+    var p2 = Util.applyTransform(r.slice(2, 4), m);
+    var p3 = Util.applyTransform([r[0], r[3]], m);
+    var p4 = Util.applyTransform([r[2], r[1]], m);
+    return [Math.min(p1[0], p2[0], p3[0], p4[0]), Math.min(p1[1], p2[1], p3[1], p4[1]), Math.max(p1[0], p2[0], p3[0], p4[0]), Math.max(p1[1], p2[1], p3[1], p4[1])];
+  };
+  Util.inverseTransform = function Util_inverseTransform(m) {
+    var d = m[0] * m[3] - m[1] * m[2];
+    return [m[3] / d, -m[1] / d, -m[2] / d, m[0] / d, (m[2] * m[5] - m[4] * m[3]) / d, (m[4] * m[1] - m[5] * m[0]) / d];
+  };
+  Util.apply3dTransform = function Util_apply3dTransform(m, v) {
+    return [m[0] * v[0] + m[1] * v[1] + m[2] * v[2], m[3] * v[0] + m[4] * v[1] + m[5] * v[2], m[6] * v[0] + m[7] * v[1] + m[8] * v[2]];
+  };
+  Util.singularValueDecompose2dScale = function Util_singularValueDecompose2dScale(m) {
+    var transpose = [m[0], m[2], m[1], m[3]];
+    var a = m[0] * transpose[0] + m[1] * transpose[2];
+    var b = m[0] * transpose[1] + m[1] * transpose[3];
+    var c = m[2] * transpose[0] + m[3] * transpose[2];
+    var d = m[2] * transpose[1] + m[3] * transpose[3];
+    var first = (a + d) / 2;
+    var second = Math.sqrt((a + d) * (a + d) - 4 * (a * d - c * b)) / 2;
+    var sx = first + second || 1;
+    var sy = first - second || 1;
+    return [Math.sqrt(sx), Math.sqrt(sy)];
+  };
+  Util.normalizeRect = function Util_normalizeRect(rect) {
+    var r = rect.slice(0);
+    if (rect[0] > rect[2]) {
+      r[0] = rect[2];
+      r[2] = rect[0];
     }
-    loaded = true;
-   };
-  }
-  document.getElementsByTagName('head')[0].appendChild(script);
- };
- return Util;
+    if (rect[1] > rect[3]) {
+      r[1] = rect[3];
+      r[3] = rect[1];
+    }
+    return r;
+  };
+  Util.intersect = function Util_intersect(rect1, rect2) {
+    function compare(a, b) {
+      return a - b;
+    }
+    var orderedX = [rect1[0], rect1[2], rect2[0], rect2[2]].sort(compare),
+        orderedY = [rect1[1], rect1[3], rect2[1], rect2[3]].sort(compare),
+        result = [];
+    rect1 = Util.normalizeRect(rect1);
+    rect2 = Util.normalizeRect(rect2);
+    if (orderedX[0] === rect1[0] && orderedX[1] === rect2[0] || orderedX[0] === rect2[0] && orderedX[1] === rect1[0]) {
+      result[0] = orderedX[1];
+      result[2] = orderedX[2];
+    } else {
+      return false;
+    }
+    if (orderedY[0] === rect1[1] && orderedY[1] === rect2[1] || orderedY[0] === rect2[1] && orderedY[1] === rect1[1]) {
+      result[1] = orderedY[1];
+      result[3] = orderedY[2];
+    } else {
+      return false;
+    }
+    return result;
+  };
+  Util.sign = function Util_sign(num) {
+    return num < 0 ? -1 : 1;
+  };
+  var ROMAN_NUMBER_MAP = ['', 'C', 'CC', 'CCC', 'CD', 'D', 'DC', 'DCC', 'DCCC', 'CM', '', 'X', 'XX', 'XXX', 'XL', 'L', 'LX', 'LXX', 'LXXX', 'XC', '', 'I', 'II', 'III', 'IV', 'V', 'VI', 'VII', 'VIII', 'IX'];
+  Util.toRoman = function Util_toRoman(number, lowerCase) {
+    assert(isInt(number) && number > 0, 'The number should be a positive integer.');
+    var pos,
+        romanBuf = [];
+    while (number >= 1000) {
+      number -= 1000;
+      romanBuf.push('M');
+    }
+    pos = number / 100 | 0;
+    number %= 100;
+    romanBuf.push(ROMAN_NUMBER_MAP[pos]);
+    pos = number / 10 | 0;
+    number %= 10;
+    romanBuf.push(ROMAN_NUMBER_MAP[10 + pos]);
+    romanBuf.push(ROMAN_NUMBER_MAP[20 + number]);
+    var romanStr = romanBuf.join('');
+    return lowerCase ? romanStr.toLowerCase() : romanStr;
+  };
+  Util.appendToArray = function Util_appendToArray(arr1, arr2) {
+    Array.prototype.push.apply(arr1, arr2);
+  };
+  Util.prependToArray = function Util_prependToArray(arr1, arr2) {
+    Array.prototype.unshift.apply(arr1, arr2);
+  };
+  Util.extendObj = function extendObj(obj1, obj2) {
+    for (var key in obj2) {
+      obj1[key] = obj2[key];
+    }
+  };
+  Util.getInheritableProperty = function Util_getInheritableProperty(dict, name, getArray) {
+    while (dict && !dict.has(name)) {
+      dict = dict.get('Parent');
+    }
+    if (!dict) {
+      return null;
+    }
+    return getArray ? dict.getArray(name) : dict.get(name);
+  };
+  Util.inherit = function Util_inherit(sub, base, prototype) {
+    sub.prototype = Object.create(base.prototype);
+    sub.prototype.constructor = sub;
+    for (var prop in prototype) {
+      sub.prototype[prop] = prototype[prop];
+    }
+  };
+  Util.loadScript = function Util_loadScript(src, callback) {
+    var script = document.createElement('script');
+    var loaded = false;
+    script.setAttribute('src', src);
+    if (callback) {
+      script.onload = function () {
+        if (!loaded) {
+          callback();
+        }
+        loaded = true;
+      };
+    }
+    document.getElementsByTagName('head')[0].appendChild(script);
+  };
+  return Util;
 }();
 var PageViewport = function PageViewportClosure() {
- function PageViewport(viewBox, scale, rotation, offsetX, offsetY, dontFlip) {
-  this.viewBox = viewBox;
-  this.scale = scale;
-  this.rotation = rotation;
-  this.offsetX = offsetX;
-  this.offsetY = offsetY;
-  var centerX = (viewBox[2] + viewBox[0]) / 2;
-  var centerY = (viewBox[3] + viewBox[1]) / 2;
-  var rotateA, rotateB, rotateC, rotateD;
-  rotation = rotation % 360;
-  rotation = rotation < 0 ? rotation + 360 : rotation;
-  switch (rotation) {
-  case 180:
-   rotateA = -1;
-   rotateB = 0;
-   rotateC = 0;
-   rotateD = 1;
-   break;
-  case 90:
-   rotateA = 0;
-   rotateB = 1;
-   rotateC = 1;
-   rotateD = 0;
-   break;
-  case 270:
-   rotateA = 0;
-   rotateB = -1;
-   rotateC = -1;
-   rotateD = 0;
-   break;
-  default:
-   rotateA = 1;
-   rotateB = 0;
-   rotateC = 0;
-   rotateD = -1;
-   break;
-  }
-  if (dontFlip) {
-   rotateC = -rotateC;
-   rotateD = -rotateD;
-  }
-  var offsetCanvasX, offsetCanvasY;
-  var width, height;
-  if (rotateA === 0) {
-   offsetCanvasX = Math.abs(centerY - viewBox[1]) * scale + offsetX;
-   offsetCanvasY = Math.abs(centerX - viewBox[0]) * scale + offsetY;
-   width = Math.abs(viewBox[3] - viewBox[1]) * scale;
-   height = Math.abs(viewBox[2] - viewBox[0]) * scale;
-  } else {
-   offsetCanvasX = Math.abs(centerX - viewBox[0]) * scale + offsetX;
-   offsetCanvasY = Math.abs(centerY - viewBox[1]) * scale + offsetY;
-   width = Math.abs(viewBox[2] - viewBox[0]) * scale;
-   height = Math.abs(viewBox[3] - viewBox[1]) * scale;
-  }
-  this.transform = [
-   rotateA * scale,
-   rotateB * scale,
-   rotateC * scale,
-   rotateD * scale,
-   offsetCanvasX - rotateA * scale * centerX - rotateC * scale * centerY,
-   offsetCanvasY - rotateB * scale * centerX - rotateD * scale * centerY
-  ];
-  this.width = width;
-  this.height = height;
-  this.fontScale = scale;
- }
- PageViewport.prototype = {
-  clone: function PageViewPort_clone(args) {
-   args = args || {};
-   var scale = 'scale' in args ? args.scale : this.scale;
-   var rotation = 'rotation' in args ? args.rotation : this.rotation;
-   return new PageViewport(this.viewBox.slice(), scale, rotation, this.offsetX, this.offsetY, args.dontFlip);
-  },
-  convertToViewportPoint: function PageViewport_convertToViewportPoint(x, y) {
-   return Util.applyTransform([
-    x,
-    y
-   ], this.transform);
-  },
-  convertToViewportRectangle: function PageViewport_convertToViewportRectangle(rect) {
-   var tl = Util.applyTransform([
-    rect[0],
-    rect[1]
-   ], this.transform);
-   var br = Util.applyTransform([
-    rect[2],
-    rect[3]
-   ], this.transform);
-   return [
-    tl[0],
-    tl[1],
-    br[0],
-    br[1]
-   ];
-  },
-  convertToPdfPoint: function PageViewport_convertToPdfPoint(x, y) {
-   return Util.applyInverseTransform([
-    x,
-    y
-   ], this.transform);
+  function PageViewport(viewBox, scale, rotation, offsetX, offsetY, dontFlip) {
+    this.viewBox = viewBox;
+    this.scale = scale;
+    this.rotation = rotation;
+    this.offsetX = offsetX;
+    this.offsetY = offsetY;
+    var centerX = (viewBox[2] + viewBox[0]) / 2;
+    var centerY = (viewBox[3] + viewBox[1]) / 2;
+    var rotateA, rotateB, rotateC, rotateD;
+    rotation = rotation % 360;
+    rotation = rotation < 0 ? rotation + 360 : rotation;
+    switch (rotation) {
+      case 180:
+        rotateA = -1;
+        rotateB = 0;
+        rotateC = 0;
+        rotateD = 1;
+        break;
+      case 90:
+        rotateA = 0;
+        rotateB = 1;
+        rotateC = 1;
+        rotateD = 0;
+        break;
+      case 270:
+        rotateA = 0;
+        rotateB = -1;
+        rotateC = -1;
+        rotateD = 0;
+        break;
+      default:
+        rotateA = 1;
+        rotateB = 0;
+        rotateC = 0;
+        rotateD = -1;
+        break;
+    }
+    if (dontFlip) {
+      rotateC = -rotateC;
+      rotateD = -rotateD;
+    }
+    var offsetCanvasX, offsetCanvasY;
+    var width, height;
+    if (rotateA === 0) {
+      offsetCanvasX = Math.abs(centerY - viewBox[1]) * scale + offsetX;
+      offsetCanvasY = Math.abs(centerX - viewBox[0]) * scale + offsetY;
+      width = Math.abs(viewBox[3] - viewBox[1]) * scale;
+      height = Math.abs(viewBox[2] - viewBox[0]) * scale;
+    } else {
+      offsetCanvasX = Math.abs(centerX - viewBox[0]) * scale + offsetX;
+      offsetCanvasY = Math.abs(centerY - viewBox[1]) * scale + offsetY;
+      width = Math.abs(viewBox[2] - viewBox[0]) * scale;
+      height = Math.abs(viewBox[3] - viewBox[1]) * scale;
+    }
+    this.transform = [rotateA * scale, rotateB * scale, rotateC * scale, rotateD * scale, offsetCanvasX - rotateA * scale * centerX - rotateC * scale * centerY, offsetCanvasY - rotateB * scale * centerX - rotateD * scale * centerY];
+    this.width = width;
+    this.height = height;
+    this.fontScale = scale;
   }
- };
- return PageViewport;
+  PageViewport.prototype = {
+    clone: function PageViewPort_clone(args) {
+      args = args || {};
+      var scale = 'scale' in args ? args.scale : this.scale;
+      var rotation = 'rotation' in args ? args.rotation : this.rotation;
+      return new PageViewport(this.viewBox.slice(), scale, rotation, this.offsetX, this.offsetY, args.dontFlip);
+    },
+    convertToViewportPoint: function PageViewport_convertToViewportPoint(x, y) {
+      return Util.applyTransform([x, y], this.transform);
+    },
+    convertToViewportRectangle: function PageViewport_convertToViewportRectangle(rect) {
+      var tl = Util.applyTransform([rect[0], rect[1]], this.transform);
+      var br = Util.applyTransform([rect[2], rect[3]], this.transform);
+      return [tl[0], tl[1], br[0], br[1]];
+    },
+    convertToPdfPoint: function PageViewport_convertToPdfPoint(x, y) {
+      return Util.applyInverseTransform([x, y], this.transform);
+    }
+  };
+  return PageViewport;
 }();
-var PDFStringTranslateTable = [
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0x2D8,
- 0x2C7,
- 0x2C6,
- 0x2D9,
- 0x2DD,
- 0x2DB,
- 0x2DA,
- 0x2DC,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0,
- 0x2022,
- 0x2020,
- 0x2021,
- 0x2026,
- 0x2014,
- 0x2013,
- 0x192,
- 0x2044,
- 0x2039,
- 0x203A,
- 0x2212,
- 0x2030,
- 0x201E,
- 0x201C,
- 0x201D,
- 0x2018,
- 0x2019,
- 0x201A,
- 0x2122,
- 0xFB01,
- 0xFB02,
- 0x141,
- 0x152,
- 0x160,
- 0x178,
- 0x17D,
- 0x131,
- 0x142,
- 0x153,
- 0x161,
- 0x17E,
- 0,
- 0x20AC
-];
+var PDFStringTranslateTable = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x2D8, 0x2C7, 0x2C6, 0x2D9, 0x2DD, 0x2DB, 0x2DA, 0x2DC, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x2022, 0x2020, 0x2021, 0x2026, 0x2014, 0x2013, 0x192, 0x2044, 0x2039, 0x203A, 0x2212, 0x2030, 0x201E, 0x201C, 0x201D, 0x2018, 0x2019, 0x201A, 0x2122, 0xFB01, 0xFB02, 0x141, 0x152, 0x160, 0x178, 0x17D, 0x131, 0x142, 0x153, 0x161, 0x17E, 0, 0x20AC];
 function stringToPDFString(str) {
- var i, n = str.length, strBuf = [];
- if (str[0] === '\xFE' && str[1] === '\xFF') {
-  for (i = 2; i < n; i += 2) {
-   strBuf.push(String.fromCharCode(str.charCodeAt(i) << 8 | str.charCodeAt(i + 1)));
-  }
- } else {
-  for (i = 0; i < n; ++i) {
-   var code = PDFStringTranslateTable[str.charCodeAt(i)];
-   strBuf.push(code ? String.fromCharCode(code) : str.charAt(i));
+  var i,
+      n = str.length,
+      strBuf = [];
+  if (str[0] === '\xFE' && str[1] === '\xFF') {
+    for (i = 2; i < n; i += 2) {
+      strBuf.push(String.fromCharCode(str.charCodeAt(i) << 8 | str.charCodeAt(i + 1)));
+    }
+  } else {
+    for (i = 0; i < n; ++i) {
+      var code = PDFStringTranslateTable[str.charCodeAt(i)];
+      strBuf.push(code ? String.fromCharCode(code) : str.charAt(i));
+    }
   }
- }
- return strBuf.join('');
+  return strBuf.join('');
 }
 function stringToUTF8String(str) {
- return decodeURIComponent(escape(str));
+  return decodeURIComponent(escape(str));
 }
 function utf8StringToString(str) {
- return unescape(encodeURIComponent(str));
+  return unescape(encodeURIComponent(str));
 }
 function isEmptyObj(obj) {
- for (var key in obj) {
-  return false;
- }
- return true;
+  for (var key in obj) {
+    return false;
+  }
+  return true;
 }
 function isBool(v) {
- return typeof v === 'boolean';
+  return typeof v === 'boolean';
 }
 function isInt(v) {
- return typeof v === 'number' && (v | 0) === v;
+  return typeof v === 'number' && (v | 0) === v;
 }
 function isNum(v) {
- return typeof v === 'number';
+  return typeof v === 'number';
 }
 function isString(v) {
- return typeof v === 'string';
+  return typeof v === 'string';
 }
 function isArray(v) {
- return v instanceof Array;
+  return v instanceof Array;
 }
 function isArrayBuffer(v) {
- return typeof v === 'object' && v !== null && v.byteLength !== undefined;
+  return typeof v === 'object' && v !== null && v.byteLength !== undefined;
 }
 function isSpace(ch) {
- return ch === 0x20 || ch === 0x09 || ch === 0x0D || ch === 0x0A;
+  return ch === 0x20 || ch === 0x09 || ch === 0x0D || ch === 0x0A;
 }
 function isNodeJS() {
- if (typeof __pdfjsdev_webpack__ === 'undefined') {
-  return typeof process === 'object' && process + '' === '[object process]';
- }
- return false;
+  if (typeof __pdfjsdev_webpack__ === 'undefined') {
+    return typeof process === 'object' && process + '' === '[object process]';
+  }
+  return false;
 }
 function createPromiseCapability() {
- var capability = {};
- capability.promise = new Promise(function (resolve, reject) {
-  capability.resolve = resolve;
-  capability.reject = reject;
- });
- return capability;
+  var capability = {};
+  capability.promise = new Promise(function (resolve, reject) {
+    capability.resolve = resolve;
+    capability.reject = reject;
+  });
+  return capability;
 }
 var StatTimer = function StatTimerClosure() {
- function rpad(str, pad, length) {
-  while (str.length < length) {
-   str += pad;
-  }
-  return str;
- }
- function StatTimer() {
-  this.started = Object.create(null);
-  this.times = [];
-  this.enabled = true;
- }
- StatTimer.prototype = {
-  time: function StatTimer_time(name) {
-   if (!this.enabled) {
-    return;
-   }
-   if (name in this.started) {
-    warn('Timer is already running for ' + name);
-   }
-   this.started[name] = Date.now();
-  },
-  timeEnd: function StatTimer_timeEnd(name) {
-   if (!this.enabled) {
-    return;
-   }
-   if (!(name in this.started)) {
-    warn('Timer has not been started for ' + name);
-   }
-   this.times.push({
-    'name': name,
-    'start': this.started[name],
-    'end': Date.now()
-   });
-   delete this.started[name];
-  },
-  toString: function StatTimer_toString() {
-   var i, ii;
-   var times = this.times;
-   var out = '';
-   var longest = 0;
-   for (i = 0, ii = times.length; i < ii; ++i) {
-    var name = times[i]['name'];
-    if (name.length > longest) {
-     longest = name.length;
+  function rpad(str, pad, length) {
+    while (str.length < length) {
+      str += pad;
     }
-   }
-   for (i = 0, ii = times.length; i < ii; ++i) {
-    var span = times[i];
-    var duration = span.end - span.start;
-    out += rpad(span['name'], ' ', longest) + ' ' + duration + 'ms\n';
-   }
-   return out;
+    return str;
+  }
+  function StatTimer() {
+    this.started = Object.create(null);
+    this.times = [];
+    this.enabled = true;
   }
- };
- return StatTimer;
+  StatTimer.prototype = {
+    time: function StatTimer_time(name) {
+      if (!this.enabled) {
+        return;
+      }
+      if (name in this.started) {
+        warn('Timer is already running for ' + name);
+      }
+      this.started[name] = Date.now();
+    },
+    timeEnd: function StatTimer_timeEnd(name) {
+      if (!this.enabled) {
+        return;
+      }
+      if (!(name in this.started)) {
+        warn('Timer has not been started for ' + name);
+      }
+      this.times.push({
+        'name': name,
+        'start': this.started[name],
+        'end': Date.now()
+      });
+      delete this.started[name];
+    },
+    toString: function StatTimer_toString() {
+      var i, ii;
+      var times = this.times;
+      var out = '';
+      var longest = 0;
+      for (i = 0, ii = times.length; i < ii; ++i) {
+        var name = times[i]['name'];
+        if (name.length > longest) {
+          longest = name.length;
+        }
+      }
+      for (i = 0, ii = times.length; i < ii; ++i) {
+        var span = times[i];
+        var duration = span.end - span.start;
+        out += rpad(span['name'], ' ', longest) + ' ' + duration + 'ms\n';
+      }
+      return out;
+    }
+  };
+  return StatTimer;
 }();
 var createBlob = function createBlob(data, contentType) {
- if (typeof Blob !== 'undefined') {
-  return new Blob([data], { type: contentType });
- }
- warn('The "Blob" constructor is not supported.');
+  if (typeof Blob !== 'undefined') {
+    return new Blob([data], { type: contentType });
+  }
+  warn('The "Blob" constructor is not supported.');
 };
 var createObjectURL = function createObjectURLClosure() {
- var digits = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
- return function createObjectURL(data, contentType, forceDataSchema) {
-  if (!forceDataSchema && typeof URL !== 'undefined' && URL.createObjectURL) {
-   var blob = createBlob(data, contentType);
-   return URL.createObjectURL(blob);
-  }
-  var buffer = 'data:' + contentType + ';base64,';
-  for (var i = 0, ii = data.length; i < ii; i += 3) {
-   var b1 = data[i] & 0xFF;
-   var b2 = data[i + 1] & 0xFF;
-   var b3 = data[i + 2] & 0xFF;
-   var d1 = b1 >> 2, d2 = (b1 & 3) << 4 | b2 >> 4;
-   var d3 = i + 1 < ii ? (b2 & 0xF) << 2 | b3 >> 6 : 64;
-   var d4 = i + 2 < ii ? b3 & 0x3F : 64;
-   buffer += digits[d1] + digits[d2] + digits[d3] + digits[d4];
-  }
-  return buffer;
- };
+  var digits = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
+  return function createObjectURL(data, contentType, forceDataSchema) {
+    if (!forceDataSchema && typeof URL !== 'undefined' && URL.createObjectURL) {
+      var blob = createBlob(data, contentType);
+      return URL.createObjectURL(blob);
+    }
+    var buffer = 'data:' + contentType + ';base64,';
+    for (var i = 0, ii = data.length; i < ii; i += 3) {
+      var b1 = data[i] & 0xFF;
+      var b2 = data[i + 1] & 0xFF;
+      var b3 = data[i + 2] & 0xFF;
+      var d1 = b1 >> 2,
+          d2 = (b1 & 3) << 4 | b2 >> 4;
+      var d3 = i + 1 < ii ? (b2 & 0xF) << 2 | b3 >> 6 : 64;
+      var d4 = i + 2 < ii ? b3 & 0x3F : 64;
+      buffer += digits[d1] + digits[d2] + digits[d3] + digits[d4];
+    }
+    return buffer;
+  };
 }();
 function MessageHandler(sourceName, targetName, comObj) {
- this.sourceName = sourceName;
- this.targetName = targetName;
- this.comObj = comObj;
- this.callbackIndex = 1;
- this.postMessageTransfers = true;
- var callbacksCapabilities = this.callbacksCapabilities = Object.create(null);
- var ah = this.actionHandler = Object.create(null);
- this._onComObjOnMessage = function messageHandlerComObjOnMessage(event) {
-  var data = event.data;
-  if (data.targetName !== this.sourceName) {
-   return;
-  }
-  if (data.isReply) {
-   var callbackId = data.callbackId;
-   if (data.callbackId in callbacksCapabilities) {
-    var callback = callbacksCapabilities[callbackId];
-    delete callbacksCapabilities[callbackId];
-    if ('error' in data) {
-     callback.reject(data.error);
+  this.sourceName = sourceName;
+  this.targetName = targetName;
+  this.comObj = comObj;
+  this.callbackIndex = 1;
+  this.postMessageTransfers = true;
+  var callbacksCapabilities = this.callbacksCapabilities = Object.create(null);
+  var ah = this.actionHandler = Object.create(null);
+  this._onComObjOnMessage = function messageHandlerComObjOnMessage(event) {
+    var data = event.data;
+    if (data.targetName !== this.sourceName) {
+      return;
+    }
+    if (data.isReply) {
+      var callbackId = data.callbackId;
+      if (data.callbackId in callbacksCapabilities) {
+        var callback = callbacksCapabilities[callbackId];
+        delete callbacksCapabilities[callbackId];
+        if ('error' in data) {
+          callback.reject(data.error);
+        } else {
+          callback.resolve(data.data);
+        }
+      } else {
+        error('Cannot resolve callback ' + callbackId);
+      }
+    } else if (data.action in ah) {
+      var action = ah[data.action];
+      if (data.callbackId) {
+        var sourceName = this.sourceName;
+        var targetName = data.sourceName;
+        Promise.resolve().then(function () {
+          return action[0].call(action[1], data.data);
+        }).then(function (result) {
+          comObj.postMessage({
+            sourceName: sourceName,
+            targetName: targetName,
+            isReply: true,
+            callbackId: data.callbackId,
+            data: result
+          });
+        }, function (reason) {
+          if (reason instanceof Error) {
+            reason = reason + '';
+          }
+          comObj.postMessage({
+            sourceName: sourceName,
+            targetName: targetName,
+            isReply: true,
+            callbackId: data.callbackId,
+            error: reason
+          });
+        });
+      } else {
+        action[0].call(action[1], data.data);
+      }
     } else {
-     callback.resolve(data.data);
+      error('Unknown action from worker: ' + data.action);
     }
-   } else {
-    error('Cannot resolve callback ' + callbackId);
-   }
-  } else if (data.action in ah) {
-   var action = ah[data.action];
-   if (data.callbackId) {
-    var sourceName = this.sourceName;
-    var targetName = data.sourceName;
-    Promise.resolve().then(function () {
-     return action[0].call(action[1], data.data);
-    }).then(function (result) {
-     comObj.postMessage({
-      sourceName: sourceName,
-      targetName: targetName,
-      isReply: true,
-      callbackId: data.callbackId,
-      data: result
-     });
-    }, function (reason) {
-     if (reason instanceof Error) {
-      reason = reason + '';
-     }
-     comObj.postMessage({
-      sourceName: sourceName,
-      targetName: targetName,
-      isReply: true,
-      callbackId: data.callbackId,
-      error: reason
-     });
-    });
-   } else {
-    action[0].call(action[1], data.data);
-   }
-  } else {
-   error('Unknown action from worker: ' + data.action);
-  }
- }.bind(this);
- comObj.addEventListener('message', this._onComObjOnMessage);
+  }.bind(this);
+  comObj.addEventListener('message', this._onComObjOnMessage);
 }
 MessageHandler.prototype = {
- on: function messageHandlerOn(actionName, handler, scope) {
-  var ah = this.actionHandler;
-  if (ah[actionName]) {
-   error('There is already an actionName called "' + actionName + '"');
-  }
-  ah[actionName] = [
-   handler,
-   scope
-  ];
- },
- send: function messageHandlerSend(actionName, data, transfers) {
-  var message = {
-   sourceName: this.sourceName,
-   targetName: this.targetName,
-   action: actionName,
-   data: data
-  };
-  this.postMessage(message, transfers);
- },
- sendWithPromise: function messageHandlerSendWithPromise(actionName, data, transfers) {
-  var callbackId = this.callbackIndex++;
-  var message = {
-   sourceName: this.sourceName,
-   targetName: this.targetName,
-   action: actionName,
-   data: data,
-   callbackId: callbackId
-  };
-  var capability = createPromiseCapability();
-  this.callbacksCapabilities[callbackId] = capability;
-  try {
-   this.postMessage(message, transfers);
-  } catch (e) {
-   capability.reject(e);
-  }
-  return capability.promise;
- },
- postMessage: function (message, transfers) {
-  if (transfers && this.postMessageTransfers) {
-   this.comObj.postMessage(message, transfers);
-  } else {
-   this.comObj.postMessage(message);
+  on: function messageHandlerOn(actionName, handler, scope) {
+    var ah = this.actionHandler;
+    if (ah[actionName]) {
+      error('There is already an actionName called "' + actionName + '"');
+    }
+    ah[actionName] = [handler, scope];
+  },
+  send: function messageHandlerSend(actionName, data, transfers) {
+    var message = {
+      sourceName: this.sourceName,
+      targetName: this.targetName,
+      action: actionName,
+      data: data
+    };
+    this.postMessage(message, transfers);
+  },
+  sendWithPromise: function messageHandlerSendWithPromise(actionName, data, transfers) {
+    var callbackId = this.callbackIndex++;
+    var message = {
+      sourceName: this.sourceName,
+      targetName: this.targetName,
+      action: actionName,
+      data: data,
+      callbackId: callbackId
+    };
+    var capability = createPromiseCapability();
+    this.callbacksCapabilities[callbackId] = capability;
+    try {
+      this.postMessage(message, transfers);
+    } catch (e) {
+      capability.reject(e);
+    }
+    return capability.promise;
+  },
+  postMessage: function (message, transfers) {
+    if (transfers && this.postMessageTransfers) {
+      this.comObj.postMessage(message, transfers);
+    } else {
+      this.comObj.postMessage(message);
+    }
+  },
+  destroy: function () {
+    this.comObj.removeEventListener('message', this._onComObjOnMessage);
   }
- },
- destroy: function () {
-  this.comObj.removeEventListener('message', this._onComObjOnMessage);
- }
 };
 function loadJpegStream(id, imageUrl, objs) {
- var img = new Image();
- img.onload = function loadJpegStream_onloadClosure() {
-  objs.resolve(id, img);
- };
- img.onerror = function loadJpegStream_onerrorClosure() {
-  objs.resolve(id, null);
-  warn('Error during JPEG image loading');
- };
- img.src = imageUrl;
+  var img = new Image();
+  img.onload = function loadJpegStream_onloadClosure() {
+    objs.resolve(id, img);
+  };
+  img.onerror = function loadJpegStream_onerrorClosure() {
+    objs.resolve(id, null);
+    warn('Error during JPEG image loading');
+  };
+  img.src = imageUrl;
 }
 exports.FONT_IDENTITY_MATRIX = FONT_IDENTITY_MATRIX;
 exports.IDENTITY_MATRIX = IDENTITY_MATRIX;

+ 1025 - 1150
lib/test/unit/annotation_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var corePrimitives = require('../../core/primitives.js');
 var coreAnnotation = require('../../core/annotation.js');
 var coreStream = require('../../core/stream.js');
@@ -35,1160 +36,1034 @@ var AnnotationFieldFlag = sharedUtil.AnnotationFieldFlag;
 var stringToBytes = sharedUtil.stringToBytes;
 var stringToUTF8String = sharedUtil.stringToUTF8String;
 describe('annotation', function () {
- function XRefMock(array) {
-  this.map = Object.create(null);
-  for (var elem in array) {
-   var obj = array[elem];
-   var ref = obj.ref, data = obj.data;
-   this.map[ref.toString()] = data;
-  }
- }
- XRefMock.prototype = {
-  fetch: function (ref) {
-   return this.map[ref.toString()];
-  },
-  fetchIfRef: function (obj) {
-   if (!isRef(obj)) {
-    return obj;
-   }
-   return this.fetch(obj);
+  function XRefMock(array) {
+    this.map = Object.create(null);
+    for (var elem in array) {
+      var obj = array[elem];
+      var ref = obj.ref,
+          data = obj.data;
+      this.map[ref.toString()] = data;
+    }
   }
- };
- function PDFManagerMock(params) {
-  this.docBaseUrl = params.docBaseUrl || null;
- }
- PDFManagerMock.prototype = {};
- function IdFactoryMock(params) {
-  var uniquePrefix = params.prefix || 'p0_';
-  var idCounters = { obj: params.startObjId || 0 };
-  return {
-   createObjId: function () {
-    return uniquePrefix + ++idCounters.obj;
-   }
+  XRefMock.prototype = {
+    fetch: function (ref) {
+      return this.map[ref.toString()];
+    },
+    fetchIfRef: function (obj) {
+      if (!isRef(obj)) {
+        return obj;
+      }
+      return this.fetch(obj);
+    }
   };
- }
- IdFactoryMock.prototype = {};
- var annotationFactory, pdfManagerMock, idFactoryMock;
- beforeAll(function (done) {
-  annotationFactory = new AnnotationFactory();
-  pdfManagerMock = new PDFManagerMock({ docBaseUrl: null });
-  idFactoryMock = new IdFactoryMock({});
-  done();
- });
- afterAll(function () {
-  annotationFactory = null;
-  pdfManagerMock = null;
-  idFactoryMock = null;
- });
- describe('AnnotationFactory', function () {
-  it('should get id for annotation', function () {
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   var annotationRef = new Ref(10, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.id).toEqual('10R');
-  });
-  it('should handle, and get fallback id\'s for, annotations that are not ' + 'indirect objects (issue 7569)', function () {
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   var xref = new XRefMock();
-   var idFactory = new IdFactoryMock({
-    prefix: 'p0_',
-    startObjId: 0
-   });
-   var annotation1 = annotationFactory.create(xref, annotationDict, pdfManagerMock, idFactory);
-   var annotation2 = annotationFactory.create(xref, annotationDict, pdfManagerMock, idFactory);
-   var data1 = annotation1.data, data2 = annotation2.data;
-   expect(data1.annotationType).toEqual(AnnotationType.LINK);
-   expect(data2.annotationType).toEqual(AnnotationType.LINK);
-   expect(data1.id).toEqual('annot_p0_1');
-   expect(data2.id).toEqual('annot_p0_2');
-  });
-  it('should handle missing /Subtype', function () {
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   var annotationRef = new Ref(1, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toBeUndefined();
-  });
- });
- describe('Annotation', function () {
-  var dict, ref;
+  function PDFManagerMock(params) {
+    this.docBaseUrl = params.docBaseUrl || null;
+  }
+  PDFManagerMock.prototype = {};
+  function IdFactoryMock(params) {
+    var uniquePrefix = params.prefix || 'p0_';
+    var idCounters = { obj: params.startObjId || 0 };
+    return {
+      createObjId: function () {
+        return uniquePrefix + ++idCounters.obj;
+      }
+    };
+  }
+  IdFactoryMock.prototype = {};
+  var annotationFactory, pdfManagerMock, idFactoryMock;
   beforeAll(function (done) {
-   dict = new Dict();
-   ref = new Ref(1, 0);
-   done();
+    annotationFactory = new AnnotationFactory();
+    pdfManagerMock = new PDFManagerMock({ docBaseUrl: null });
+    idFactoryMock = new IdFactoryMock({});
+    done();
   });
   afterAll(function () {
-   dict = ref = null;
-  });
-  it('should set and get flags', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   annotation.setFlags(13);
-   expect(annotation.hasFlag(AnnotationFlag.INVISIBLE)).toEqual(true);
-   expect(annotation.hasFlag(AnnotationFlag.NOZOOM)).toEqual(true);
-   expect(annotation.hasFlag(AnnotationFlag.PRINT)).toEqual(true);
-   expect(annotation.hasFlag(AnnotationFlag.READONLY)).toEqual(false);
-  });
-  it('should be viewable and not printable by default', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   expect(annotation.viewable).toEqual(true);
-   expect(annotation.printable).toEqual(false);
-  });
-  it('should set and get a valid rectangle', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   annotation.setRectangle([
-    117,
-    694,
-    164.298,
-    720
-   ]);
-   expect(annotation.rectangle).toEqual([
-    117,
-    694,
-    164.298,
-    720
-   ]);
-  });
-  it('should not set and get an invalid rectangle', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   annotation.setRectangle([
-    117,
-    694,
-    164.298
-   ]);
-   expect(annotation.rectangle).toEqual([
-    0,
-    0,
-    0,
-    0
-   ]);
-  });
-  it('should reject a color if it is not an array', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   annotation.setColor('red');
-   expect(annotation.color).toEqual(new Uint8Array([
-    0,
-    0,
-    0
-   ]));
-  });
-  it('should set and get a transparent color', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   annotation.setColor([]);
-   expect(annotation.color).toEqual(null);
-  });
-  it('should set and get a grayscale color', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   annotation.setColor([0.4]);
-   expect(annotation.color).toEqual(new Uint8Array([
-    102,
-    102,
-    102
-   ]));
-  });
-  it('should set and get an RGB color', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   annotation.setColor([
-    0,
-    0,
-    1
-   ]);
-   expect(annotation.color).toEqual(new Uint8Array([
-    0,
-    0,
-    255
-   ]));
-  });
-  it('should set and get a CMYK color', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   annotation.setColor([
-    0.1,
-    0.92,
-    0.84,
-    0.02
-   ]);
-   expect(annotation.color).toEqual(new Uint8Array([
-    233,
-    59,
-    47
-   ]));
-  });
-  it('should not set and get an invalid color', function () {
-   var annotation = new Annotation({
-    dict: dict,
-    ref: ref
-   });
-   annotation.setColor([
-    0.4,
-    0.6
-   ]);
-   expect(annotation.color).toEqual(new Uint8Array([
-    0,
-    0,
-    0
-   ]));
-  });
- });
- describe('AnnotationBorderStyle', function () {
-  it('should set and get a valid width', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setWidth(3);
-   expect(borderStyle.width).toEqual(3);
-  });
-  it('should not set and get an invalid width', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setWidth('three');
-   expect(borderStyle.width).toEqual(1);
-  });
-  it('should set and get a valid style', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setStyle(Name.get('D'));
-   expect(borderStyle.style).toEqual(AnnotationBorderStyleType.DASHED);
-  });
-  it('should not set and get an invalid style', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setStyle('Dashed');
-   expect(borderStyle.style).toEqual(AnnotationBorderStyleType.SOLID);
-  });
-  it('should set and get a valid dash array', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setDashArray([
-    1,
-    2,
-    3
-   ]);
-   expect(borderStyle.dashArray).toEqual([
-    1,
-    2,
-    3
-   ]);
-  });
-  it('should not set and get an invalid dash array', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setDashArray([
-    0,
-    0
-   ]);
-   expect(borderStyle.dashArray).toEqual([3]);
-  });
-  it('should set and get a valid horizontal corner radius', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setHorizontalCornerRadius(3);
-   expect(borderStyle.horizontalCornerRadius).toEqual(3);
-  });
-  it('should not set and get an invalid horizontal corner radius', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setHorizontalCornerRadius('three');
-   expect(borderStyle.horizontalCornerRadius).toEqual(0);
-  });
-  it('should set and get a valid vertical corner radius', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setVerticalCornerRadius(3);
-   expect(borderStyle.verticalCornerRadius).toEqual(3);
-  });
-  it('should not set and get an invalid horizontal corner radius', function () {
-   var borderStyle = new AnnotationBorderStyle();
-   borderStyle.setVerticalCornerRadius('three');
-   expect(borderStyle.verticalCornerRadius).toEqual(0);
-  });
- });
- describe('LinkAnnotation', function () {
-  it('should correctly parse a URI action', function () {
-   var actionDict = new Dict();
-   actionDict.set('Type', Name.get('Action'));
-   actionDict.set('S', Name.get('URI'));
-   actionDict.set('URI', 'http://www.ctan.org/tex-archive/info/lshort');
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(820, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toEqual('http://www.ctan.org/tex-archive/info/lshort');
-   expect(data.unsafeUrl).toEqual('http://www.ctan.org/tex-archive/info/lshort');
-   expect(data.dest).toBeUndefined();
-  });
-  it('should correctly parse a URI action, where the URI entry ' + 'is missing a protocol', function () {
-   var actionDict = new Dict();
-   actionDict.set('Type', Name.get('Action'));
-   actionDict.set('S', Name.get('URI'));
-   actionDict.set('URI', 'www.hmrc.gov.uk');
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(353, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toEqual('http://www.hmrc.gov.uk/');
-   expect(data.unsafeUrl).toEqual('http://www.hmrc.gov.uk');
-   expect(data.dest).toBeUndefined();
-  });
-  it('should correctly parse a URI action, where the URI entry ' + 'has an incorrect encoding (bug 1122280)', function () {
-   var actionStream = new StringStream('<<\n' + '/Type /Action\n' + '/S /URI\n' + '/URI (http://www.example.com/\\303\\274\\303\\266\\303\\244)\n' + '>>\n');
-   var lexer = new Lexer(actionStream);
-   var parser = new Parser(lexer);
-   var actionDict = parser.getObj();
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(8, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toEqual(new URL(stringToUTF8String('http://www.example.com/\xC3\xBC\xC3\xB6\xC3\xA4')).href);
-   expect(data.unsafeUrl).toEqual(stringToUTF8String('http://www.example.com/\xC3\xBC\xC3\xB6\xC3\xA4'));
-   expect(data.dest).toBeUndefined();
-  });
-  it('should correctly parse a GoTo action', function () {
-   var actionDict = new Dict();
-   actionDict.set('Type', Name.get('Action'));
-   actionDict.set('S', Name.get('GoTo'));
-   actionDict.set('D', 'page.157');
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(798, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toBeUndefined();
-   expect(data.unsafeUrl).toBeUndefined();
-   expect(data.dest).toEqual('page.157');
-  });
-  it('should correctly parse a GoToR action, where the FileSpec entry ' + 'is a string containing a relative URL', function () {
-   var actionDict = new Dict();
-   actionDict.set('Type', Name.get('Action'));
-   actionDict.set('S', Name.get('GoToR'));
-   actionDict.set('F', '../../0013/001346/134685E.pdf');
-   actionDict.set('D', '4.3');
-   actionDict.set('NewWindow', true);
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(489, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toBeUndefined();
-   expect(data.unsafeUrl).toEqual('../../0013/001346/134685E.pdf#4.3');
-   expect(data.dest).toBeUndefined();
-   expect(data.newWindow).toEqual(true);
-  });
-  it('should correctly parse a GoToR action, containing a relative URL, ' + 'with the "docBaseUrl" parameter specified', function () {
-   var actionDict = new Dict();
-   actionDict.set('Type', Name.get('Action'));
-   actionDict.set('S', Name.get('GoToR'));
-   actionDict.set('F', '../../0013/001346/134685E.pdf');
-   actionDict.set('D', '4.3');
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(489, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var pdfManager = new PDFManagerMock({ docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf' });
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManager, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toEqual('http://www.example.com/0013/001346/134685E.pdf#4.3');
-   expect(data.unsafeUrl).toEqual('../../0013/001346/134685E.pdf#4.3');
-   expect(data.dest).toBeUndefined();
-  });
-  it('should correctly parse a GoToR action, with named destination', function () {
-   var actionDict = new Dict();
-   actionDict.set('Type', Name.get('Action'));
-   actionDict.set('S', Name.get('GoToR'));
-   actionDict.set('F', 'http://www.example.com/test.pdf');
-   actionDict.set('D', '15');
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(495, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toEqual('http://www.example.com/test.pdf#nameddest=15');
-   expect(data.unsafeUrl).toEqual('http://www.example.com/test.pdf#nameddest=15');
-   expect(data.dest).toBeUndefined();
-   expect(data.newWindow).toBeFalsy();
-  });
-  it('should correctly parse a GoToR action, with explicit destination array', function () {
-   var actionDict = new Dict();
-   actionDict.set('Type', Name.get('Action'));
-   actionDict.set('S', Name.get('GoToR'));
-   actionDict.set('F', 'http://www.example.com/test.pdf');
-   actionDict.set('D', [
-    14,
-    Name.get('XYZ'),
-    null,
-    298.043,
-    null
-   ]);
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(489, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toEqual(new URL('http://www.example.com/test.pdf#' + '[14,{"name":"XYZ"},null,298.043,null]').href);
-   expect(data.unsafeUrl).toEqual('http://www.example.com/test.pdf#' + '[14,{"name":"XYZ"},null,298.043,null]');
-   expect(data.dest).toBeUndefined();
-   expect(data.newWindow).toBeFalsy();
-  });
-  it('should correctly parse a Launch action, where the FileSpec dict ' + 'contains a relative URL, with the "docBaseUrl" parameter specified', function () {
-   var fileSpecDict = new Dict();
-   fileSpecDict.set('Type', Name.get('FileSpec'));
-   fileSpecDict.set('F', 'Part II/Part II.pdf');
-   fileSpecDict.set('UF', 'Part II/Part II.pdf');
-   var actionDict = new Dict();
-   actionDict.set('Type', Name.get('Action'));
-   actionDict.set('S', Name.get('Launch'));
-   actionDict.set('F', fileSpecDict);
-   actionDict.set('NewWindow', true);
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(88, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var pdfManager = new PDFManagerMock({ docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf' });
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManager, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toEqual(new URL('http://www.example.com/test/pdfs/Part II/Part II.pdf').href);
-   expect(data.unsafeUrl).toEqual('Part II/Part II.pdf');
-   expect(data.dest).toBeUndefined();
-   expect(data.newWindow).toEqual(true);
-  });
-  it('should recover valid URLs from JavaScript actions having certain ' + 'white-listed formats', function () {
-   function checkJsAction(params) {
-    var jsEntry = params.jsEntry;
-    var expectedUrl = params.expectedUrl;
-    var expectedUnsafeUrl = params.expectedUnsafeUrl;
-    var expectedNewWindow = params.expectedNewWindow;
-    var actionDict = new Dict();
-    actionDict.set('Type', Name.get('Action'));
-    actionDict.set('S', Name.get('JavaScript'));
-    actionDict.set('JS', jsEntry);
-    var annotationDict = new Dict();
-    annotationDict.set('Type', Name.get('Annot'));
-    annotationDict.set('Subtype', Name.get('Link'));
-    annotationDict.set('A', actionDict);
-    var annotationRef = new Ref(46, 0);
-    var xref = new XRefMock([{
-      ref: annotationRef,
-      data: annotationDict
-     }]);
-    var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-    var data = annotation.data;
-    expect(data.annotationType).toEqual(AnnotationType.LINK);
-    expect(data.url).toEqual(expectedUrl);
-    expect(data.unsafeUrl).toEqual(expectedUnsafeUrl);
-    expect(data.dest).toBeUndefined();
-    expect(data.newWindow).toEqual(expectedNewWindow);
-   }
-   checkJsAction({
-    jsEntry: 'function someFun() { return "qwerty"; } someFun();',
-    expectedUrl: undefined,
-    expectedUnsafeUrl: undefined,
-    expectedNewWindow: undefined
-   });
-   checkJsAction({
-    jsEntry: 'window.open(\'http://www.example.com/test.pdf\')',
-    expectedUrl: new URL('http://www.example.com/test.pdf').href,
-    expectedUnsafeUrl: 'http://www.example.com/test.pdf',
-    expectedNewWindow: undefined
-   });
-   checkJsAction({
-    jsEntry: new StringStream('app.launchURL("http://www.example.com/test.pdf", true)'),
-    expectedUrl: new URL('http://www.example.com/test.pdf').href,
-    expectedUnsafeUrl: 'http://www.example.com/test.pdf',
-    expectedNewWindow: true
-   });
-  });
-  it('should correctly parse a Named action', function () {
-   var actionDict = new Dict();
-   actionDict.set('Type', Name.get('Action'));
-   actionDict.set('S', Name.get('Named'));
-   actionDict.set('N', Name.get('GoToPage'));
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('A', actionDict);
-   var annotationRef = new Ref(12, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toBeUndefined();
-   expect(data.unsafeUrl).toBeUndefined();
-   expect(data.action).toEqual('GoToPage');
-  });
-  it('should correctly parse a simple Dest', function () {
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('Dest', Name.get('LI0'));
-   var annotationRef = new Ref(583, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toBeUndefined();
-   expect(data.unsafeUrl).toBeUndefined();
-   expect(data.dest).toEqual('LI0');
-  });
-  it('should correctly parse a simple Dest, with explicit destination array', function () {
-   var annotationDict = new Dict();
-   annotationDict.set('Type', Name.get('Annot'));
-   annotationDict.set('Subtype', Name.get('Link'));
-   annotationDict.set('Dest', [
-    new Ref(17, 0),
-    Name.get('XYZ'),
-    0,
-    841.89,
-    null
-   ]);
-   var annotationRef = new Ref(10, 0);
-   var xref = new XRefMock([{
-     ref: annotationRef,
-     data: annotationDict
-    }]);
-   var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.LINK);
-   expect(data.url).toBeUndefined();
-   expect(data.unsafeUrl).toBeUndefined();
-   expect(data.dest).toEqual([
-    {
-     num: 17,
-     gen: 0
-    },
-    { name: 'XYZ' },
-    0,
-    841.89,
-    null
-   ]);
-  });
- });
- describe('WidgetAnnotation', function () {
-  var widgetDict;
-  beforeEach(function (done) {
-   widgetDict = new Dict();
-   widgetDict.set('Type', Name.get('Annot'));
-   widgetDict.set('Subtype', Name.get('Widget'));
-   done();
-  });
-  afterEach(function () {
-   widgetDict = null;
-  });
-  it('should handle unknown field names', function () {
-   var widgetRef = new Ref(20, 0);
-   var xref = new XRefMock([{
-     ref: widgetRef,
-     data: widgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.fieldName).toEqual('');
-  });
-  it('should construct the field name when there are no ancestors', function () {
-   widgetDict.set('T', 'foo');
-   var widgetRef = new Ref(21, 0);
-   var xref = new XRefMock([{
-     ref: widgetRef,
-     data: widgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.fieldName).toEqual('foo');
-  });
-  it('should construct the field name when there are ancestors', function () {
-   var firstParent = new Dict();
-   firstParent.set('T', 'foo');
-   var secondParent = new Dict();
-   secondParent.set('Parent', firstParent);
-   secondParent.set('T', 'bar');
-   widgetDict.set('Parent', secondParent);
-   widgetDict.set('T', 'baz');
-   var widgetRef = new Ref(22, 0);
-   var xref = new XRefMock([{
-     ref: widgetRef,
-     data: widgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.fieldName).toEqual('foo.bar.baz');
-  });
-  it('should construct the field name if a parent is not a dictionary ' + '(issue 8143)', function () {
-   var parentDict = new Dict();
-   parentDict.set('Parent', null);
-   parentDict.set('T', 'foo');
-   widgetDict.set('Parent', parentDict);
-   widgetDict.set('T', 'bar');
-   var widgetRef = new Ref(22, 0);
-   var xref = new XRefMock([{
-     ref: widgetRef,
-     data: widgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.fieldName).toEqual('foo.bar');
-  });
- });
- describe('TextWidgetAnnotation', function () {
-  var textWidgetDict;
-  beforeEach(function (done) {
-   textWidgetDict = new Dict();
-   textWidgetDict.set('Type', Name.get('Annot'));
-   textWidgetDict.set('Subtype', Name.get('Widget'));
-   textWidgetDict.set('FT', Name.get('Tx'));
-   done();
-  });
-  afterEach(function () {
-   textWidgetDict = null;
-  });
-  it('should handle unknown text alignment, maximum length and flags', function () {
-   var textWidgetRef = new Ref(124, 0);
-   var xref = new XRefMock([{
-     ref: textWidgetRef,
-     data: textWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.textAlignment).toEqual(null);
-   expect(data.maxLen).toEqual(null);
-   expect(data.readOnly).toEqual(false);
-   expect(data.multiLine).toEqual(false);
-   expect(data.comb).toEqual(false);
-  });
-  it('should not set invalid text alignment, maximum length and flags', function () {
-   textWidgetDict.set('Q', 'center');
-   textWidgetDict.set('MaxLen', 'five');
-   textWidgetDict.set('Ff', 'readonly');
-   var textWidgetRef = new Ref(43, 0);
-   var xref = new XRefMock([{
-     ref: textWidgetRef,
-     data: textWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.textAlignment).toEqual(null);
-   expect(data.maxLen).toEqual(null);
-   expect(data.readOnly).toEqual(false);
-   expect(data.multiLine).toEqual(false);
-   expect(data.comb).toEqual(false);
-  });
-  it('should set valid text alignment, maximum length and flags', function () {
-   textWidgetDict.set('Q', 1);
-   textWidgetDict.set('MaxLen', 20);
-   textWidgetDict.set('Ff', AnnotationFieldFlag.READONLY + AnnotationFieldFlag.MULTILINE);
-   var textWidgetRef = new Ref(84, 0);
-   var xref = new XRefMock([{
-     ref: textWidgetRef,
-     data: textWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.textAlignment).toEqual(1);
-   expect(data.maxLen).toEqual(20);
-   expect(data.readOnly).toEqual(true);
-   expect(data.multiLine).toEqual(true);
-  });
-  it('should reject comb fields without a maximum length', function () {
-   textWidgetDict.set('Ff', AnnotationFieldFlag.COMB);
-   var textWidgetRef = new Ref(46, 0);
-   var xref = new XRefMock([{
-     ref: textWidgetRef,
-     data: textWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.comb).toEqual(false);
-  });
-  it('should accept comb fields with a maximum length', function () {
-   textWidgetDict.set('MaxLen', 20);
-   textWidgetDict.set('Ff', AnnotationFieldFlag.COMB);
-   var textWidgetRef = new Ref(46, 0);
-   var xref = new XRefMock([{
-     ref: textWidgetRef,
-     data: textWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.comb).toEqual(true);
-  });
-  it('should only accept comb fields when the flags are valid', function () {
-   var invalidFieldFlags = [
-    AnnotationFieldFlag.MULTILINE,
-    AnnotationFieldFlag.PASSWORD,
-    AnnotationFieldFlag.FILESELECT
-   ];
-   var flags = AnnotationFieldFlag.COMB + AnnotationFieldFlag.MULTILINE + AnnotationFieldFlag.PASSWORD + AnnotationFieldFlag.FILESELECT;
-   for (var i = 0, ii = invalidFieldFlags.length; i <= ii; i++) {
-    textWidgetDict.set('MaxLen', 20);
-    textWidgetDict.set('Ff', flags);
-    var textWidgetRef = new Ref(93, 0);
-    var xref = new XRefMock([{
-      ref: textWidgetRef,
-      data: textWidgetDict
-     }]);
-    var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
-    var data = annotation.data;
-    expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-    var valid = invalidFieldFlags.length === 0;
-    expect(data.comb).toEqual(valid);
-    if (!valid) {
-     flags -= invalidFieldFlags.splice(-1, 1);
-    }
-   }
-  });
- });
- describe('ButtonWidgetAnnotation', function () {
-  var buttonWidgetDict;
-  beforeEach(function (done) {
-   buttonWidgetDict = new Dict();
-   buttonWidgetDict.set('Type', Name.get('Annot'));
-   buttonWidgetDict.set('Subtype', Name.get('Widget'));
-   buttonWidgetDict.set('FT', Name.get('Btn'));
-   done();
-  });
-  afterEach(function () {
-   buttonWidgetDict = null;
-  });
-  it('should handle checkboxes', function () {
-   buttonWidgetDict.set('V', Name.get('1'));
-   var buttonWidgetRef = new Ref(124, 0);
-   var xref = new XRefMock([{
-     ref: buttonWidgetRef,
-     data: buttonWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, buttonWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.checkBox).toEqual(true);
-   expect(data.fieldValue).toEqual('1');
-   expect(data.radioButton).toEqual(false);
-  });
-  it('should handle radio buttons with a field value', function () {
-   var parentDict = new Dict();
-   parentDict.set('V', Name.get('1'));
-   var normalAppearanceStateDict = new Dict();
-   normalAppearanceStateDict.set('2', null);
-   var appearanceStatesDict = new Dict();
-   appearanceStatesDict.set('N', normalAppearanceStateDict);
-   buttonWidgetDict.set('Ff', AnnotationFieldFlag.RADIO);
-   buttonWidgetDict.set('Parent', parentDict);
-   buttonWidgetDict.set('AP', appearanceStatesDict);
-   var buttonWidgetRef = new Ref(124, 0);
-   var xref = new XRefMock([{
-     ref: buttonWidgetRef,
-     data: buttonWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, buttonWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.checkBox).toEqual(false);
-   expect(data.radioButton).toEqual(true);
-   expect(data.fieldValue).toEqual('1');
-   expect(data.buttonValue).toEqual('2');
-  });
-  it('should handle radio buttons without a field value', function () {
-   var normalAppearanceStateDict = new Dict();
-   normalAppearanceStateDict.set('2', null);
-   var appearanceStatesDict = new Dict();
-   appearanceStatesDict.set('N', normalAppearanceStateDict);
-   buttonWidgetDict.set('Ff', AnnotationFieldFlag.RADIO);
-   buttonWidgetDict.set('AP', appearanceStatesDict);
-   var buttonWidgetRef = new Ref(124, 0);
-   var xref = new XRefMock([{
-     ref: buttonWidgetRef,
-     data: buttonWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, buttonWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.checkBox).toEqual(false);
-   expect(data.radioButton).toEqual(true);
-   expect(data.fieldValue).toEqual(null);
-   expect(data.buttonValue).toEqual('2');
-  });
- });
- describe('ChoiceWidgetAnnotation', function () {
-  var choiceWidgetDict;
-  beforeEach(function (done) {
-   choiceWidgetDict = new Dict();
-   choiceWidgetDict.set('Type', Name.get('Annot'));
-   choiceWidgetDict.set('Subtype', Name.get('Widget'));
-   choiceWidgetDict.set('FT', Name.get('Ch'));
-   done();
-  });
-  afterEach(function () {
-   choiceWidgetDict = null;
-  });
-  it('should handle missing option arrays', function () {
-   var choiceWidgetRef = new Ref(122, 0);
-   var xref = new XRefMock([{
-     ref: choiceWidgetRef,
-     data: choiceWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.options).toEqual([]);
-  });
-  it('should handle option arrays with array elements', function () {
-   var optionBarRef = new Ref(20, 0);
-   var optionBarStr = 'Bar';
-   var optionOneRef = new Ref(10, 0);
-   var optionOneArr = [
-    'bar_export',
-    optionBarRef
-   ];
-   var options = [
-    [
-     'foo_export',
-     'Foo'
-    ],
-    optionOneRef
-   ];
-   var expected = [
-    {
-     exportValue: 'foo_export',
-     displayValue: 'Foo'
-    },
-    {
-     exportValue: 'bar_export',
-     displayValue: 'Bar'
-    }
-   ];
-   choiceWidgetDict.set('Opt', options);
-   var choiceWidgetRef = new Ref(123, 0);
-   var xref = new XRefMock([
-    {
-     ref: choiceWidgetRef,
-     data: choiceWidgetDict
-    },
-    {
-     ref: optionBarRef,
-     data: optionBarStr
-    },
-    {
-     ref: optionOneRef,
-     data: optionOneArr
-    }
-   ]);
-   var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.options).toEqual(expected);
-  });
-  it('should handle option arrays with string elements', function () {
-   var optionBarRef = new Ref(10, 0);
-   var optionBarStr = 'Bar';
-   var options = [
-    'Foo',
-    optionBarRef
-   ];
-   var expected = [
-    {
-     exportValue: 'Foo',
-     displayValue: 'Foo'
-    },
-    {
-     exportValue: 'Bar',
-     displayValue: 'Bar'
-    }
-   ];
-   choiceWidgetDict.set('Opt', options);
-   var choiceWidgetRef = new Ref(981, 0);
-   var xref = new XRefMock([
-    {
-     ref: choiceWidgetRef,
-     data: choiceWidgetDict
-    },
-    {
-     ref: optionBarRef,
-     data: optionBarStr
-    }
-   ]);
-   var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.options).toEqual(expected);
-  });
-  it('should handle inherited option arrays (issue 8094)', function () {
-   var options = [
-    [
-     'Value1',
-     'Description1'
-    ],
-    [
-     'Value2',
-     'Description2'
-    ]
-   ];
-   var expected = [
-    {
-     exportValue: 'Value1',
-     displayValue: 'Description1'
-    },
-    {
-     exportValue: 'Value2',
-     displayValue: 'Description2'
-    }
-   ];
-   var parentDict = new Dict();
-   parentDict.set('Opt', options);
-   choiceWidgetDict.set('Parent', parentDict);
-   var choiceWidgetRef = new Ref(123, 0);
-   var xref = new XRefMock([{
-     ref: choiceWidgetRef,
-     data: choiceWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.options).toEqual(expected);
-  });
-  it('should handle array field values', function () {
-   var fieldValue = [
-    'Foo',
-    'Bar'
-   ];
-   choiceWidgetDict.set('V', fieldValue);
-   var choiceWidgetRef = new Ref(968, 0);
-   var xref = new XRefMock([{
-     ref: choiceWidgetRef,
-     data: choiceWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.fieldValue).toEqual(fieldValue);
-  });
-  it('should handle string field values', function () {
-   var fieldValue = 'Foo';
-   choiceWidgetDict.set('V', fieldValue);
-   var choiceWidgetRef = new Ref(978, 0);
-   var xref = new XRefMock([{
-     ref: choiceWidgetRef,
-     data: choiceWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.fieldValue).toEqual([fieldValue]);
-  });
-  it('should handle unknown flags', function () {
-   var choiceWidgetRef = new Ref(166, 0);
-   var xref = new XRefMock([{
-     ref: choiceWidgetRef,
-     data: choiceWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.readOnly).toEqual(false);
-   expect(data.combo).toEqual(false);
-   expect(data.multiSelect).toEqual(false);
-  });
-  it('should not set invalid flags', function () {
-   choiceWidgetDict.set('Ff', 'readonly');
-   var choiceWidgetRef = new Ref(165, 0);
-   var xref = new XRefMock([{
-     ref: choiceWidgetRef,
-     data: choiceWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.readOnly).toEqual(false);
-   expect(data.combo).toEqual(false);
-   expect(data.multiSelect).toEqual(false);
-  });
-  it('should set valid flags', function () {
-   choiceWidgetDict.set('Ff', AnnotationFieldFlag.READONLY + AnnotationFieldFlag.COMBO + AnnotationFieldFlag.MULTISELECT);
-   var choiceWidgetRef = new Ref(512, 0);
-   var xref = new XRefMock([{
-     ref: choiceWidgetRef,
-     data: choiceWidgetDict
-    }]);
-   var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.WIDGET);
-   expect(data.readOnly).toEqual(true);
-   expect(data.combo).toEqual(true);
-   expect(data.multiSelect).toEqual(true);
-  });
- });
- describe('FileAttachmentAnnotation', function () {
-  it('should correctly parse a file attachment', function () {
-   var fileStream = new StringStream('<<\n' + '/Type /EmbeddedFile\n' + '/Subtype /text#2Fplain\n' + '>>\n' + 'stream\n' + 'Test attachment' + 'endstream\n');
-   var lexer = new Lexer(fileStream);
-   var parser = new Parser(lexer, true);
-   var fileStreamRef = new Ref(18, 0);
-   var fileStreamDict = parser.getObj();
-   var embeddedFileDict = new Dict();
-   embeddedFileDict.set('F', fileStreamRef);
-   var fileSpecRef = new Ref(19, 0);
-   var fileSpecDict = new Dict();
-   fileSpecDict.set('Type', Name.get('Filespec'));
-   fileSpecDict.set('Desc', '');
-   fileSpecDict.set('EF', embeddedFileDict);
-   fileSpecDict.set('UF', 'Test.txt');
-   var fileAttachmentRef = new Ref(20, 0);
-   var fileAttachmentDict = new Dict();
-   fileAttachmentDict.set('Type', Name.get('Annot'));
-   fileAttachmentDict.set('Subtype', Name.get('FileAttachment'));
-   fileAttachmentDict.set('FS', fileSpecRef);
-   fileAttachmentDict.set('T', 'Topic');
-   fileAttachmentDict.set('Contents', 'Test.txt');
-   var xref = new XRefMock([
-    {
-     ref: fileStreamRef,
-     data: fileStreamDict
-    },
-    {
-     ref: fileSpecRef,
-     data: fileSpecDict
-    },
-    {
-     ref: fileAttachmentRef,
-     data: fileAttachmentDict
-    }
-   ]);
-   embeddedFileDict.assignXref(xref);
-   fileSpecDict.assignXref(xref);
-   fileAttachmentDict.assignXref(xref);
-   var annotation = annotationFactory.create(xref, fileAttachmentRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.FILEATTACHMENT);
-   expect(data.file.filename).toEqual('Test.txt');
-   expect(data.file.content).toEqual(stringToBytes('Test attachment'));
-  });
- });
- describe('PopupAnnotation', function () {
-  it('should inherit the parent flags when the Popup is not viewable, ' + 'but the parent is (PR 7352)', function () {
-   var parentDict = new Dict();
-   parentDict.set('Type', Name.get('Annot'));
-   parentDict.set('Subtype', Name.get('Text'));
-   parentDict.set('F', 28);
-   var popupDict = new Dict();
-   popupDict.set('Type', Name.get('Annot'));
-   popupDict.set('Subtype', Name.get('Popup'));
-   popupDict.set('F', 25);
-   popupDict.set('Parent', parentDict);
-   var popupRef = new Ref(13, 0);
-   var xref = new XRefMock([{
-     ref: popupRef,
-     data: popupDict
-    }]);
-   var annotation = annotationFactory.create(xref, popupRef, pdfManagerMock, idFactoryMock);
-   var data = annotation.data;
-   expect(data.annotationType).toEqual(AnnotationType.POPUP);
-   expect(data.annotationFlags).toEqual(25);
-   expect(annotation.viewable).toEqual(true);
+    annotationFactory = null;
+    pdfManagerMock = null;
+    idFactoryMock = null;
+  });
+  describe('AnnotationFactory', function () {
+    it('should get id for annotation', function () {
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      var annotationRef = new Ref(10, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.id).toEqual('10R');
+    });
+    it('should handle, and get fallback id\'s for, annotations that are not ' + 'indirect objects (issue 7569)', function () {
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      var xref = new XRefMock();
+      var idFactory = new IdFactoryMock({
+        prefix: 'p0_',
+        startObjId: 0
+      });
+      var annotation1 = annotationFactory.create(xref, annotationDict, pdfManagerMock, idFactory);
+      var annotation2 = annotationFactory.create(xref, annotationDict, pdfManagerMock, idFactory);
+      var data1 = annotation1.data,
+          data2 = annotation2.data;
+      expect(data1.annotationType).toEqual(AnnotationType.LINK);
+      expect(data2.annotationType).toEqual(AnnotationType.LINK);
+      expect(data1.id).toEqual('annot_p0_1');
+      expect(data2.id).toEqual('annot_p0_2');
+    });
+    it('should handle missing /Subtype', function () {
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      var annotationRef = new Ref(1, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toBeUndefined();
+    });
+  });
+  describe('Annotation', function () {
+    var dict, ref;
+    beforeAll(function (done) {
+      dict = new Dict();
+      ref = new Ref(1, 0);
+      done();
+    });
+    afterAll(function () {
+      dict = ref = null;
+    });
+    it('should set and get flags', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      annotation.setFlags(13);
+      expect(annotation.hasFlag(AnnotationFlag.INVISIBLE)).toEqual(true);
+      expect(annotation.hasFlag(AnnotationFlag.NOZOOM)).toEqual(true);
+      expect(annotation.hasFlag(AnnotationFlag.PRINT)).toEqual(true);
+      expect(annotation.hasFlag(AnnotationFlag.READONLY)).toEqual(false);
+    });
+    it('should be viewable and not printable by default', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      expect(annotation.viewable).toEqual(true);
+      expect(annotation.printable).toEqual(false);
+    });
+    it('should set and get a valid rectangle', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      annotation.setRectangle([117, 694, 164.298, 720]);
+      expect(annotation.rectangle).toEqual([117, 694, 164.298, 720]);
+    });
+    it('should not set and get an invalid rectangle', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      annotation.setRectangle([117, 694, 164.298]);
+      expect(annotation.rectangle).toEqual([0, 0, 0, 0]);
+    });
+    it('should reject a color if it is not an array', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      annotation.setColor('red');
+      expect(annotation.color).toEqual(new Uint8Array([0, 0, 0]));
+    });
+    it('should set and get a transparent color', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      annotation.setColor([]);
+      expect(annotation.color).toEqual(null);
+    });
+    it('should set and get a grayscale color', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      annotation.setColor([0.4]);
+      expect(annotation.color).toEqual(new Uint8Array([102, 102, 102]));
+    });
+    it('should set and get an RGB color', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      annotation.setColor([0, 0, 1]);
+      expect(annotation.color).toEqual(new Uint8Array([0, 0, 255]));
+    });
+    it('should set and get a CMYK color', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      annotation.setColor([0.1, 0.92, 0.84, 0.02]);
+      expect(annotation.color).toEqual(new Uint8Array([233, 59, 47]));
+    });
+    it('should not set and get an invalid color', function () {
+      var annotation = new Annotation({
+        dict: dict,
+        ref: ref
+      });
+      annotation.setColor([0.4, 0.6]);
+      expect(annotation.color).toEqual(new Uint8Array([0, 0, 0]));
+    });
+  });
+  describe('AnnotationBorderStyle', function () {
+    it('should set and get a valid width', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setWidth(3);
+      expect(borderStyle.width).toEqual(3);
+    });
+    it('should not set and get an invalid width', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setWidth('three');
+      expect(borderStyle.width).toEqual(1);
+    });
+    it('should set and get a valid style', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setStyle(Name.get('D'));
+      expect(borderStyle.style).toEqual(AnnotationBorderStyleType.DASHED);
+    });
+    it('should not set and get an invalid style', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setStyle('Dashed');
+      expect(borderStyle.style).toEqual(AnnotationBorderStyleType.SOLID);
+    });
+    it('should set and get a valid dash array', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setDashArray([1, 2, 3]);
+      expect(borderStyle.dashArray).toEqual([1, 2, 3]);
+    });
+    it('should not set and get an invalid dash array', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setDashArray([0, 0]);
+      expect(borderStyle.dashArray).toEqual([3]);
+    });
+    it('should set and get a valid horizontal corner radius', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setHorizontalCornerRadius(3);
+      expect(borderStyle.horizontalCornerRadius).toEqual(3);
+    });
+    it('should not set and get an invalid horizontal corner radius', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setHorizontalCornerRadius('three');
+      expect(borderStyle.horizontalCornerRadius).toEqual(0);
+    });
+    it('should set and get a valid vertical corner radius', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setVerticalCornerRadius(3);
+      expect(borderStyle.verticalCornerRadius).toEqual(3);
+    });
+    it('should not set and get an invalid horizontal corner radius', function () {
+      var borderStyle = new AnnotationBorderStyle();
+      borderStyle.setVerticalCornerRadius('three');
+      expect(borderStyle.verticalCornerRadius).toEqual(0);
+    });
+  });
+  describe('LinkAnnotation', function () {
+    it('should correctly parse a URI action', function () {
+      var actionDict = new Dict();
+      actionDict.set('Type', Name.get('Action'));
+      actionDict.set('S', Name.get('URI'));
+      actionDict.set('URI', 'http://www.ctan.org/tex-archive/info/lshort');
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(820, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toEqual('http://www.ctan.org/tex-archive/info/lshort');
+      expect(data.unsafeUrl).toEqual('http://www.ctan.org/tex-archive/info/lshort');
+      expect(data.dest).toBeUndefined();
+    });
+    it('should correctly parse a URI action, where the URI entry ' + 'is missing a protocol', function () {
+      var actionDict = new Dict();
+      actionDict.set('Type', Name.get('Action'));
+      actionDict.set('S', Name.get('URI'));
+      actionDict.set('URI', 'www.hmrc.gov.uk');
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(353, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toEqual('http://www.hmrc.gov.uk/');
+      expect(data.unsafeUrl).toEqual('http://www.hmrc.gov.uk');
+      expect(data.dest).toBeUndefined();
+    });
+    it('should correctly parse a URI action, where the URI entry ' + 'has an incorrect encoding (bug 1122280)', function () {
+      var actionStream = new StringStream('<<\n' + '/Type /Action\n' + '/S /URI\n' + '/URI (http://www.example.com/\\303\\274\\303\\266\\303\\244)\n' + '>>\n');
+      var lexer = new Lexer(actionStream);
+      var parser = new Parser(lexer);
+      var actionDict = parser.getObj();
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(8, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toEqual(new URL(stringToUTF8String('http://www.example.com/\xC3\xBC\xC3\xB6\xC3\xA4')).href);
+      expect(data.unsafeUrl).toEqual(stringToUTF8String('http://www.example.com/\xC3\xBC\xC3\xB6\xC3\xA4'));
+      expect(data.dest).toBeUndefined();
+    });
+    it('should correctly parse a GoTo action', function () {
+      var actionDict = new Dict();
+      actionDict.set('Type', Name.get('Action'));
+      actionDict.set('S', Name.get('GoTo'));
+      actionDict.set('D', 'page.157');
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(798, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toBeUndefined();
+      expect(data.unsafeUrl).toBeUndefined();
+      expect(data.dest).toEqual('page.157');
+    });
+    it('should correctly parse a GoToR action, where the FileSpec entry ' + 'is a string containing a relative URL', function () {
+      var actionDict = new Dict();
+      actionDict.set('Type', Name.get('Action'));
+      actionDict.set('S', Name.get('GoToR'));
+      actionDict.set('F', '../../0013/001346/134685E.pdf');
+      actionDict.set('D', '4.3');
+      actionDict.set('NewWindow', true);
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(489, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toBeUndefined();
+      expect(data.unsafeUrl).toEqual('../../0013/001346/134685E.pdf#4.3');
+      expect(data.dest).toBeUndefined();
+      expect(data.newWindow).toEqual(true);
+    });
+    it('should correctly parse a GoToR action, containing a relative URL, ' + 'with the "docBaseUrl" parameter specified', function () {
+      var actionDict = new Dict();
+      actionDict.set('Type', Name.get('Action'));
+      actionDict.set('S', Name.get('GoToR'));
+      actionDict.set('F', '../../0013/001346/134685E.pdf');
+      actionDict.set('D', '4.3');
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(489, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var pdfManager = new PDFManagerMock({ docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf' });
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManager, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toEqual('http://www.example.com/0013/001346/134685E.pdf#4.3');
+      expect(data.unsafeUrl).toEqual('../../0013/001346/134685E.pdf#4.3');
+      expect(data.dest).toBeUndefined();
+    });
+    it('should correctly parse a GoToR action, with named destination', function () {
+      var actionDict = new Dict();
+      actionDict.set('Type', Name.get('Action'));
+      actionDict.set('S', Name.get('GoToR'));
+      actionDict.set('F', 'http://www.example.com/test.pdf');
+      actionDict.set('D', '15');
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(495, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toEqual('http://www.example.com/test.pdf#nameddest=15');
+      expect(data.unsafeUrl).toEqual('http://www.example.com/test.pdf#nameddest=15');
+      expect(data.dest).toBeUndefined();
+      expect(data.newWindow).toBeFalsy();
+    });
+    it('should correctly parse a GoToR action, with explicit destination array', function () {
+      var actionDict = new Dict();
+      actionDict.set('Type', Name.get('Action'));
+      actionDict.set('S', Name.get('GoToR'));
+      actionDict.set('F', 'http://www.example.com/test.pdf');
+      actionDict.set('D', [14, Name.get('XYZ'), null, 298.043, null]);
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(489, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toEqual(new URL('http://www.example.com/test.pdf#' + '[14,{"name":"XYZ"},null,298.043,null]').href);
+      expect(data.unsafeUrl).toEqual('http://www.example.com/test.pdf#' + '[14,{"name":"XYZ"},null,298.043,null]');
+      expect(data.dest).toBeUndefined();
+      expect(data.newWindow).toBeFalsy();
+    });
+    it('should correctly parse a Launch action, where the FileSpec dict ' + 'contains a relative URL, with the "docBaseUrl" parameter specified', function () {
+      var fileSpecDict = new Dict();
+      fileSpecDict.set('Type', Name.get('FileSpec'));
+      fileSpecDict.set('F', 'Part II/Part II.pdf');
+      fileSpecDict.set('UF', 'Part II/Part II.pdf');
+      var actionDict = new Dict();
+      actionDict.set('Type', Name.get('Action'));
+      actionDict.set('S', Name.get('Launch'));
+      actionDict.set('F', fileSpecDict);
+      actionDict.set('NewWindow', true);
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(88, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var pdfManager = new PDFManagerMock({ docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf' });
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManager, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toEqual(new URL('http://www.example.com/test/pdfs/Part II/Part II.pdf').href);
+      expect(data.unsafeUrl).toEqual('Part II/Part II.pdf');
+      expect(data.dest).toBeUndefined();
+      expect(data.newWindow).toEqual(true);
+    });
+    it('should recover valid URLs from JavaScript actions having certain ' + 'white-listed formats', function () {
+      function checkJsAction(params) {
+        var jsEntry = params.jsEntry;
+        var expectedUrl = params.expectedUrl;
+        var expectedUnsafeUrl = params.expectedUnsafeUrl;
+        var expectedNewWindow = params.expectedNewWindow;
+        var actionDict = new Dict();
+        actionDict.set('Type', Name.get('Action'));
+        actionDict.set('S', Name.get('JavaScript'));
+        actionDict.set('JS', jsEntry);
+        var annotationDict = new Dict();
+        annotationDict.set('Type', Name.get('Annot'));
+        annotationDict.set('Subtype', Name.get('Link'));
+        annotationDict.set('A', actionDict);
+        var annotationRef = new Ref(46, 0);
+        var xref = new XRefMock([{
+          ref: annotationRef,
+          data: annotationDict
+        }]);
+        var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+        var data = annotation.data;
+        expect(data.annotationType).toEqual(AnnotationType.LINK);
+        expect(data.url).toEqual(expectedUrl);
+        expect(data.unsafeUrl).toEqual(expectedUnsafeUrl);
+        expect(data.dest).toBeUndefined();
+        expect(data.newWindow).toEqual(expectedNewWindow);
+      }
+      checkJsAction({
+        jsEntry: 'function someFun() { return "qwerty"; } someFun();',
+        expectedUrl: undefined,
+        expectedUnsafeUrl: undefined,
+        expectedNewWindow: undefined
+      });
+      checkJsAction({
+        jsEntry: 'window.open(\'http://www.example.com/test.pdf\')',
+        expectedUrl: new URL('http://www.example.com/test.pdf').href,
+        expectedUnsafeUrl: 'http://www.example.com/test.pdf',
+        expectedNewWindow: undefined
+      });
+      checkJsAction({
+        jsEntry: new StringStream('app.launchURL("http://www.example.com/test.pdf", true)'),
+        expectedUrl: new URL('http://www.example.com/test.pdf').href,
+        expectedUnsafeUrl: 'http://www.example.com/test.pdf',
+        expectedNewWindow: true
+      });
+    });
+    it('should correctly parse a Named action', function () {
+      var actionDict = new Dict();
+      actionDict.set('Type', Name.get('Action'));
+      actionDict.set('S', Name.get('Named'));
+      actionDict.set('N', Name.get('GoToPage'));
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('A', actionDict);
+      var annotationRef = new Ref(12, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toBeUndefined();
+      expect(data.unsafeUrl).toBeUndefined();
+      expect(data.action).toEqual('GoToPage');
+    });
+    it('should correctly parse a simple Dest', function () {
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('Dest', Name.get('LI0'));
+      var annotationRef = new Ref(583, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toBeUndefined();
+      expect(data.unsafeUrl).toBeUndefined();
+      expect(data.dest).toEqual('LI0');
+    });
+    it('should correctly parse a simple Dest, with explicit destination array', function () {
+      var annotationDict = new Dict();
+      annotationDict.set('Type', Name.get('Annot'));
+      annotationDict.set('Subtype', Name.get('Link'));
+      annotationDict.set('Dest', [new Ref(17, 0), Name.get('XYZ'), 0, 841.89, null]);
+      var annotationRef = new Ref(10, 0);
+      var xref = new XRefMock([{
+        ref: annotationRef,
+        data: annotationDict
+      }]);
+      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.LINK);
+      expect(data.url).toBeUndefined();
+      expect(data.unsafeUrl).toBeUndefined();
+      expect(data.dest).toEqual([{
+        num: 17,
+        gen: 0
+      }, { name: 'XYZ' }, 0, 841.89, null]);
+    });
+  });
+  describe('WidgetAnnotation', function () {
+    var widgetDict;
+    beforeEach(function (done) {
+      widgetDict = new Dict();
+      widgetDict.set('Type', Name.get('Annot'));
+      widgetDict.set('Subtype', Name.get('Widget'));
+      done();
+    });
+    afterEach(function () {
+      widgetDict = null;
+    });
+    it('should handle unknown field names', function () {
+      var widgetRef = new Ref(20, 0);
+      var xref = new XRefMock([{
+        ref: widgetRef,
+        data: widgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.fieldName).toEqual('');
+    });
+    it('should construct the field name when there are no ancestors', function () {
+      widgetDict.set('T', 'foo');
+      var widgetRef = new Ref(21, 0);
+      var xref = new XRefMock([{
+        ref: widgetRef,
+        data: widgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.fieldName).toEqual('foo');
+    });
+    it('should construct the field name when there are ancestors', function () {
+      var firstParent = new Dict();
+      firstParent.set('T', 'foo');
+      var secondParent = new Dict();
+      secondParent.set('Parent', firstParent);
+      secondParent.set('T', 'bar');
+      widgetDict.set('Parent', secondParent);
+      widgetDict.set('T', 'baz');
+      var widgetRef = new Ref(22, 0);
+      var xref = new XRefMock([{
+        ref: widgetRef,
+        data: widgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.fieldName).toEqual('foo.bar.baz');
+    });
+    it('should construct the field name if a parent is not a dictionary ' + '(issue 8143)', function () {
+      var parentDict = new Dict();
+      parentDict.set('Parent', null);
+      parentDict.set('T', 'foo');
+      widgetDict.set('Parent', parentDict);
+      widgetDict.set('T', 'bar');
+      var widgetRef = new Ref(22, 0);
+      var xref = new XRefMock([{
+        ref: widgetRef,
+        data: widgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.fieldName).toEqual('foo.bar');
+    });
+  });
+  describe('TextWidgetAnnotation', function () {
+    var textWidgetDict;
+    beforeEach(function (done) {
+      textWidgetDict = new Dict();
+      textWidgetDict.set('Type', Name.get('Annot'));
+      textWidgetDict.set('Subtype', Name.get('Widget'));
+      textWidgetDict.set('FT', Name.get('Tx'));
+      done();
+    });
+    afterEach(function () {
+      textWidgetDict = null;
+    });
+    it('should handle unknown text alignment, maximum length and flags', function () {
+      var textWidgetRef = new Ref(124, 0);
+      var xref = new XRefMock([{
+        ref: textWidgetRef,
+        data: textWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.textAlignment).toEqual(null);
+      expect(data.maxLen).toEqual(null);
+      expect(data.readOnly).toEqual(false);
+      expect(data.multiLine).toEqual(false);
+      expect(data.comb).toEqual(false);
+    });
+    it('should not set invalid text alignment, maximum length and flags', function () {
+      textWidgetDict.set('Q', 'center');
+      textWidgetDict.set('MaxLen', 'five');
+      textWidgetDict.set('Ff', 'readonly');
+      var textWidgetRef = new Ref(43, 0);
+      var xref = new XRefMock([{
+        ref: textWidgetRef,
+        data: textWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.textAlignment).toEqual(null);
+      expect(data.maxLen).toEqual(null);
+      expect(data.readOnly).toEqual(false);
+      expect(data.multiLine).toEqual(false);
+      expect(data.comb).toEqual(false);
+    });
+    it('should set valid text alignment, maximum length and flags', function () {
+      textWidgetDict.set('Q', 1);
+      textWidgetDict.set('MaxLen', 20);
+      textWidgetDict.set('Ff', AnnotationFieldFlag.READONLY + AnnotationFieldFlag.MULTILINE);
+      var textWidgetRef = new Ref(84, 0);
+      var xref = new XRefMock([{
+        ref: textWidgetRef,
+        data: textWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.textAlignment).toEqual(1);
+      expect(data.maxLen).toEqual(20);
+      expect(data.readOnly).toEqual(true);
+      expect(data.multiLine).toEqual(true);
+    });
+    it('should reject comb fields without a maximum length', function () {
+      textWidgetDict.set('Ff', AnnotationFieldFlag.COMB);
+      var textWidgetRef = new Ref(46, 0);
+      var xref = new XRefMock([{
+        ref: textWidgetRef,
+        data: textWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.comb).toEqual(false);
+    });
+    it('should accept comb fields with a maximum length', function () {
+      textWidgetDict.set('MaxLen', 20);
+      textWidgetDict.set('Ff', AnnotationFieldFlag.COMB);
+      var textWidgetRef = new Ref(46, 0);
+      var xref = new XRefMock([{
+        ref: textWidgetRef,
+        data: textWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.comb).toEqual(true);
+    });
+    it('should only accept comb fields when the flags are valid', function () {
+      var invalidFieldFlags = [AnnotationFieldFlag.MULTILINE, AnnotationFieldFlag.PASSWORD, AnnotationFieldFlag.FILESELECT];
+      var flags = AnnotationFieldFlag.COMB + AnnotationFieldFlag.MULTILINE + AnnotationFieldFlag.PASSWORD + AnnotationFieldFlag.FILESELECT;
+      for (var i = 0, ii = invalidFieldFlags.length; i <= ii; i++) {
+        textWidgetDict.set('MaxLen', 20);
+        textWidgetDict.set('Ff', flags);
+        var textWidgetRef = new Ref(93, 0);
+        var xref = new XRefMock([{
+          ref: textWidgetRef,
+          data: textWidgetDict
+        }]);
+        var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);
+        var data = annotation.data;
+        expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+        var valid = invalidFieldFlags.length === 0;
+        expect(data.comb).toEqual(valid);
+        if (!valid) {
+          flags -= invalidFieldFlags.splice(-1, 1);
+        }
+      }
+    });
+  });
+  describe('ButtonWidgetAnnotation', function () {
+    var buttonWidgetDict;
+    beforeEach(function (done) {
+      buttonWidgetDict = new Dict();
+      buttonWidgetDict.set('Type', Name.get('Annot'));
+      buttonWidgetDict.set('Subtype', Name.get('Widget'));
+      buttonWidgetDict.set('FT', Name.get('Btn'));
+      done();
+    });
+    afterEach(function () {
+      buttonWidgetDict = null;
+    });
+    it('should handle checkboxes', function () {
+      buttonWidgetDict.set('V', Name.get('1'));
+      var buttonWidgetRef = new Ref(124, 0);
+      var xref = new XRefMock([{
+        ref: buttonWidgetRef,
+        data: buttonWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, buttonWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.checkBox).toEqual(true);
+      expect(data.fieldValue).toEqual('1');
+      expect(data.radioButton).toEqual(false);
+    });
+    it('should handle radio buttons with a field value', function () {
+      var parentDict = new Dict();
+      parentDict.set('V', Name.get('1'));
+      var normalAppearanceStateDict = new Dict();
+      normalAppearanceStateDict.set('2', null);
+      var appearanceStatesDict = new Dict();
+      appearanceStatesDict.set('N', normalAppearanceStateDict);
+      buttonWidgetDict.set('Ff', AnnotationFieldFlag.RADIO);
+      buttonWidgetDict.set('Parent', parentDict);
+      buttonWidgetDict.set('AP', appearanceStatesDict);
+      var buttonWidgetRef = new Ref(124, 0);
+      var xref = new XRefMock([{
+        ref: buttonWidgetRef,
+        data: buttonWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, buttonWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.checkBox).toEqual(false);
+      expect(data.radioButton).toEqual(true);
+      expect(data.fieldValue).toEqual('1');
+      expect(data.buttonValue).toEqual('2');
+    });
+    it('should handle radio buttons without a field value', function () {
+      var normalAppearanceStateDict = new Dict();
+      normalAppearanceStateDict.set('2', null);
+      var appearanceStatesDict = new Dict();
+      appearanceStatesDict.set('N', normalAppearanceStateDict);
+      buttonWidgetDict.set('Ff', AnnotationFieldFlag.RADIO);
+      buttonWidgetDict.set('AP', appearanceStatesDict);
+      var buttonWidgetRef = new Ref(124, 0);
+      var xref = new XRefMock([{
+        ref: buttonWidgetRef,
+        data: buttonWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, buttonWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.checkBox).toEqual(false);
+      expect(data.radioButton).toEqual(true);
+      expect(data.fieldValue).toEqual(null);
+      expect(data.buttonValue).toEqual('2');
+    });
+  });
+  describe('ChoiceWidgetAnnotation', function () {
+    var choiceWidgetDict;
+    beforeEach(function (done) {
+      choiceWidgetDict = new Dict();
+      choiceWidgetDict.set('Type', Name.get('Annot'));
+      choiceWidgetDict.set('Subtype', Name.get('Widget'));
+      choiceWidgetDict.set('FT', Name.get('Ch'));
+      done();
+    });
+    afterEach(function () {
+      choiceWidgetDict = null;
+    });
+    it('should handle missing option arrays', function () {
+      var choiceWidgetRef = new Ref(122, 0);
+      var xref = new XRefMock([{
+        ref: choiceWidgetRef,
+        data: choiceWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.options).toEqual([]);
+    });
+    it('should handle option arrays with array elements', function () {
+      var optionBarRef = new Ref(20, 0);
+      var optionBarStr = 'Bar';
+      var optionOneRef = new Ref(10, 0);
+      var optionOneArr = ['bar_export', optionBarRef];
+      var options = [['foo_export', 'Foo'], optionOneRef];
+      var expected = [{
+        exportValue: 'foo_export',
+        displayValue: 'Foo'
+      }, {
+        exportValue: 'bar_export',
+        displayValue: 'Bar'
+      }];
+      choiceWidgetDict.set('Opt', options);
+      var choiceWidgetRef = new Ref(123, 0);
+      var xref = new XRefMock([{
+        ref: choiceWidgetRef,
+        data: choiceWidgetDict
+      }, {
+        ref: optionBarRef,
+        data: optionBarStr
+      }, {
+        ref: optionOneRef,
+        data: optionOneArr
+      }]);
+      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.options).toEqual(expected);
+    });
+    it('should handle option arrays with string elements', function () {
+      var optionBarRef = new Ref(10, 0);
+      var optionBarStr = 'Bar';
+      var options = ['Foo', optionBarRef];
+      var expected = [{
+        exportValue: 'Foo',
+        displayValue: 'Foo'
+      }, {
+        exportValue: 'Bar',
+        displayValue: 'Bar'
+      }];
+      choiceWidgetDict.set('Opt', options);
+      var choiceWidgetRef = new Ref(981, 0);
+      var xref = new XRefMock([{
+        ref: choiceWidgetRef,
+        data: choiceWidgetDict
+      }, {
+        ref: optionBarRef,
+        data: optionBarStr
+      }]);
+      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.options).toEqual(expected);
+    });
+    it('should handle inherited option arrays (issue 8094)', function () {
+      var options = [['Value1', 'Description1'], ['Value2', 'Description2']];
+      var expected = [{
+        exportValue: 'Value1',
+        displayValue: 'Description1'
+      }, {
+        exportValue: 'Value2',
+        displayValue: 'Description2'
+      }];
+      var parentDict = new Dict();
+      parentDict.set('Opt', options);
+      choiceWidgetDict.set('Parent', parentDict);
+      var choiceWidgetRef = new Ref(123, 0);
+      var xref = new XRefMock([{
+        ref: choiceWidgetRef,
+        data: choiceWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.options).toEqual(expected);
+    });
+    it('should handle array field values', function () {
+      var fieldValue = ['Foo', 'Bar'];
+      choiceWidgetDict.set('V', fieldValue);
+      var choiceWidgetRef = new Ref(968, 0);
+      var xref = new XRefMock([{
+        ref: choiceWidgetRef,
+        data: choiceWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.fieldValue).toEqual(fieldValue);
+    });
+    it('should handle string field values', function () {
+      var fieldValue = 'Foo';
+      choiceWidgetDict.set('V', fieldValue);
+      var choiceWidgetRef = new Ref(978, 0);
+      var xref = new XRefMock([{
+        ref: choiceWidgetRef,
+        data: choiceWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.fieldValue).toEqual([fieldValue]);
+    });
+    it('should handle unknown flags', function () {
+      var choiceWidgetRef = new Ref(166, 0);
+      var xref = new XRefMock([{
+        ref: choiceWidgetRef,
+        data: choiceWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.readOnly).toEqual(false);
+      expect(data.combo).toEqual(false);
+      expect(data.multiSelect).toEqual(false);
+    });
+    it('should not set invalid flags', function () {
+      choiceWidgetDict.set('Ff', 'readonly');
+      var choiceWidgetRef = new Ref(165, 0);
+      var xref = new XRefMock([{
+        ref: choiceWidgetRef,
+        data: choiceWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.readOnly).toEqual(false);
+      expect(data.combo).toEqual(false);
+      expect(data.multiSelect).toEqual(false);
+    });
+    it('should set valid flags', function () {
+      choiceWidgetDict.set('Ff', AnnotationFieldFlag.READONLY + AnnotationFieldFlag.COMBO + AnnotationFieldFlag.MULTISELECT);
+      var choiceWidgetRef = new Ref(512, 0);
+      var xref = new XRefMock([{
+        ref: choiceWidgetRef,
+        data: choiceWidgetDict
+      }]);
+      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.WIDGET);
+      expect(data.readOnly).toEqual(true);
+      expect(data.combo).toEqual(true);
+      expect(data.multiSelect).toEqual(true);
+    });
+  });
+  describe('FileAttachmentAnnotation', function () {
+    it('should correctly parse a file attachment', function () {
+      var fileStream = new StringStream('<<\n' + '/Type /EmbeddedFile\n' + '/Subtype /text#2Fplain\n' + '>>\n' + 'stream\n' + 'Test attachment' + 'endstream\n');
+      var lexer = new Lexer(fileStream);
+      var parser = new Parser(lexer, true);
+      var fileStreamRef = new Ref(18, 0);
+      var fileStreamDict = parser.getObj();
+      var embeddedFileDict = new Dict();
+      embeddedFileDict.set('F', fileStreamRef);
+      var fileSpecRef = new Ref(19, 0);
+      var fileSpecDict = new Dict();
+      fileSpecDict.set('Type', Name.get('Filespec'));
+      fileSpecDict.set('Desc', '');
+      fileSpecDict.set('EF', embeddedFileDict);
+      fileSpecDict.set('UF', 'Test.txt');
+      var fileAttachmentRef = new Ref(20, 0);
+      var fileAttachmentDict = new Dict();
+      fileAttachmentDict.set('Type', Name.get('Annot'));
+      fileAttachmentDict.set('Subtype', Name.get('FileAttachment'));
+      fileAttachmentDict.set('FS', fileSpecRef);
+      fileAttachmentDict.set('T', 'Topic');
+      fileAttachmentDict.set('Contents', 'Test.txt');
+      var xref = new XRefMock([{
+        ref: fileStreamRef,
+        data: fileStreamDict
+      }, {
+        ref: fileSpecRef,
+        data: fileSpecDict
+      }, {
+        ref: fileAttachmentRef,
+        data: fileAttachmentDict
+      }]);
+      embeddedFileDict.assignXref(xref);
+      fileSpecDict.assignXref(xref);
+      fileAttachmentDict.assignXref(xref);
+      var annotation = annotationFactory.create(xref, fileAttachmentRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.FILEATTACHMENT);
+      expect(data.file.filename).toEqual('Test.txt');
+      expect(data.file.content).toEqual(stringToBytes('Test attachment'));
+    });
+  });
+  describe('PopupAnnotation', function () {
+    it('should inherit the parent flags when the Popup is not viewable, ' + 'but the parent is (PR 7352)', function () {
+      var parentDict = new Dict();
+      parentDict.set('Type', Name.get('Annot'));
+      parentDict.set('Subtype', Name.get('Text'));
+      parentDict.set('F', 28);
+      var popupDict = new Dict();
+      popupDict.set('Type', Name.get('Annot'));
+      popupDict.set('Subtype', Name.get('Popup'));
+      popupDict.set('F', 25);
+      popupDict.set('Parent', parentDict);
+      var popupRef = new Ref(13, 0);
+      var xref = new XRefMock([{
+        ref: popupRef,
+        data: popupDict
+      }]);
+      var annotation = annotationFactory.create(xref, popupRef, pdfManagerMock, idFactoryMock);
+      var data = annotation.data;
+      expect(data.annotationType).toEqual(AnnotationType.POPUP);
+      expect(data.annotationFlags).toEqual(25);
+      expect(annotation.viewable).toEqual(true);
+    });
   });
- });
 });

+ 972 - 1074
lib/test/unit/api_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../../shared/util.js');
 var displayGlobal = require('../../display/global.js');
 var displayApi = require('../../display/api.js');
@@ -27,1111 +28,1008 @@ var PDFPageProxy = displayApi.PDFPageProxy;
 var StreamType = sharedUtil.StreamType;
 var FontType = sharedUtil.FontType;
 describe('api', function () {
- var basicApiUrl = new URL('../pdfs/basicapi.pdf', window.location).href;
- var basicApiFileLength = 105779;
- function waitSome(callback) {
-  var WAIT_TIMEOUT = 10;
-  setTimeout(function () {
-   callback();
-  }, WAIT_TIMEOUT);
- }
- describe('PDFJS', function () {
-  describe('getDocument', function () {
-   it('creates pdf doc from URL', function (done) {
-    var loadingTask = PDFJS.getDocument(basicApiUrl);
-    var isProgressReportedResolved = false;
-    var progressReportedCapability = createPromiseCapability();
-    loadingTask.onProgress = function (progressData) {
-     if (!isProgressReportedResolved) {
-      isProgressReportedResolved = true;
-      progressReportedCapability.resolve(progressData);
-     }
-    };
-    var promises = [
-     progressReportedCapability.promise,
-     loadingTask.promise
-    ];
-    Promise.all(promises).then(function (data) {
-     expect(data[0].loaded / data[0].total > 0).toEqual(true);
-     expect(data[1] instanceof PDFDocumentProxy).toEqual(true);
-     expect(loadingTask).toEqual(data[1].loadingTask);
-     loadingTask.destroy();
-     done();
-    }).catch(function (reason) {
-     done.fail(reason);
+  var basicApiUrl = new URL('../pdfs/basicapi.pdf', window.location).href;
+  var basicApiFileLength = 105779;
+  function waitSome(callback) {
+    var WAIT_TIMEOUT = 10;
+    setTimeout(function () {
+      callback();
+    }, WAIT_TIMEOUT);
+  }
+  describe('PDFJS', function () {
+    describe('getDocument', function () {
+      it('creates pdf doc from URL', function (done) {
+        var loadingTask = PDFJS.getDocument(basicApiUrl);
+        var isProgressReportedResolved = false;
+        var progressReportedCapability = createPromiseCapability();
+        loadingTask.onProgress = function (progressData) {
+          if (!isProgressReportedResolved) {
+            isProgressReportedResolved = true;
+            progressReportedCapability.resolve(progressData);
+          }
+        };
+        var promises = [progressReportedCapability.promise, loadingTask.promise];
+        Promise.all(promises).then(function (data) {
+          expect(data[0].loaded / data[0].total > 0).toEqual(true);
+          expect(data[1] instanceof PDFDocumentProxy).toEqual(true);
+          expect(loadingTask).toEqual(data[1].loadingTask);
+          loadingTask.destroy();
+          done();
+        }).catch(function (reason) {
+          done.fail(reason);
+        });
+      });
+      it('creates pdf doc from URL and aborts before worker initialized', function (done) {
+        var loadingTask = PDFJS.getDocument(basicApiUrl);
+        loadingTask.destroy();
+        loadingTask.promise.then(function (reason) {
+          done.fail('shall fail loading');
+        }).catch(function (reason) {
+          expect(true).toEqual(true);
+          done();
+        });
+      });
+      it('creates pdf doc from URL and aborts loading after worker initialized', function (done) {
+        var loadingTask = PDFJS.getDocument(basicApiUrl);
+        var destroyed = loadingTask._worker.promise.then(function () {
+          return loadingTask.destroy();
+        });
+        destroyed.then(function (data) {
+          expect(true).toEqual(true);
+          done();
+        }).catch(function (reason) {
+          done.fail(reason);
+        });
+      });
+      it('creates pdf doc from typed array', function (done) {
+        var nonBinaryRequest = PDFJS.disableWorker;
+        var request = new XMLHttpRequest();
+        request.open('GET', basicApiUrl, false);
+        if (!nonBinaryRequest) {
+          try {
+            request.responseType = 'arraybuffer';
+            nonBinaryRequest = request.responseType !== 'arraybuffer';
+          } catch (e) {
+            nonBinaryRequest = true;
+          }
+        }
+        if (nonBinaryRequest && request.overrideMimeType) {
+          request.overrideMimeType('text/plain; charset=x-user-defined');
+        }
+        request.send(null);
+        var typedArrayPdf;
+        if (nonBinaryRequest) {
+          var data = Array.prototype.map.call(request.responseText, function (ch) {
+            return ch.charCodeAt(0) & 0xFF;
+          });
+          typedArrayPdf = new Uint8Array(data);
+        } else {
+          typedArrayPdf = new Uint8Array(request.response);
+        }
+        expect(typedArrayPdf.length).toEqual(basicApiFileLength);
+        var loadingTask = PDFJS.getDocument(typedArrayPdf);
+        loadingTask.promise.then(function (data) {
+          expect(data instanceof PDFDocumentProxy).toEqual(true);
+          loadingTask.destroy();
+          done();
+        }).catch(function (reason) {
+          done.fail(reason);
+        });
+      });
+      it('creates pdf doc from invalid PDF file', function (done) {
+        var url = new URL('../pdfs/bug1020226.pdf', window.location).href;
+        var loadingTask = PDFJS.getDocument(url);
+        loadingTask.promise.then(function () {
+          done.fail('shall fail loading');
+        }).catch(function (error) {
+          expect(error instanceof InvalidPDFException).toEqual(true);
+          loadingTask.destroy();
+          done();
+        });
+      });
+      it('creates pdf doc from non-existent URL', function (done) {
+        var nonExistentUrl = new URL('../pdfs/non-existent.pdf', window.location).href;
+        var loadingTask = PDFJS.getDocument(nonExistentUrl);
+        loadingTask.promise.then(function (error) {
+          done.fail('shall fail loading');
+        }).catch(function (error) {
+          expect(error instanceof MissingPDFException).toEqual(true);
+          loadingTask.destroy();
+          done();
+        });
+      });
+      it('creates pdf doc from PDF file protected with user and owner password', function (done) {
+        var url = new URL('../pdfs/pr6531_1.pdf', window.location).href;
+        var loadingTask = PDFJS.getDocument(url);
+        var isPasswordNeededResolved = false;
+        var passwordNeededCapability = createPromiseCapability();
+        var isPasswordIncorrectResolved = false;
+        var passwordIncorrectCapability = createPromiseCapability();
+        loadingTask.onPassword = function (updatePassword, reason) {
+          if (reason === PasswordResponses.NEED_PASSWORD && !isPasswordNeededResolved) {
+            isPasswordNeededResolved = true;
+            passwordNeededCapability.resolve();
+            updatePassword('qwerty');
+            return;
+          }
+          if (reason === PasswordResponses.INCORRECT_PASSWORD && !isPasswordIncorrectResolved) {
+            isPasswordIncorrectResolved = true;
+            passwordIncorrectCapability.resolve();
+            updatePassword('asdfasdf');
+            return;
+          }
+          expect(false).toEqual(true);
+        };
+        var promises = [passwordNeededCapability.promise, passwordIncorrectCapability.promise, loadingTask.promise];
+        Promise.all(promises).then(function (data) {
+          expect(data[2] instanceof PDFDocumentProxy).toEqual(true);
+          loadingTask.destroy();
+          done();
+        }).catch(function (reason) {
+          done.fail(reason);
+        });
+      });
+      it('creates pdf doc from PDF file protected with only a user password', function (done) {
+        var url = new URL('../pdfs/pr6531_2.pdf', window.location).href;
+        var passwordNeededLoadingTask = PDFJS.getDocument({
+          url: url,
+          password: ''
+        });
+        var result1 = passwordNeededLoadingTask.promise.then(function () {
+          done.fail('shall fail with no password');
+          return Promise.reject(new Error('loadingTask should be rejected'));
+        }, function (data) {
+          expect(data instanceof PasswordException).toEqual(true);
+          expect(data.code).toEqual(PasswordResponses.NEED_PASSWORD);
+          return passwordNeededLoadingTask.destroy();
+        });
+        var passwordIncorrectLoadingTask = PDFJS.getDocument({
+          url: url,
+          password: 'qwerty'
+        });
+        var result2 = passwordIncorrectLoadingTask.promise.then(function () {
+          done.fail('shall fail with wrong password');
+          return Promise.reject(new Error('loadingTask should be rejected'));
+        }, function (data) {
+          expect(data instanceof PasswordException).toEqual(true);
+          expect(data.code).toEqual(PasswordResponses.INCORRECT_PASSWORD);
+          return passwordIncorrectLoadingTask.destroy();
+        });
+        var passwordAcceptedLoadingTask = PDFJS.getDocument({
+          url: url,
+          password: 'asdfasdf'
+        });
+        var result3 = passwordAcceptedLoadingTask.promise.then(function (data) {
+          expect(data instanceof PDFDocumentProxy).toEqual(true);
+          return passwordAcceptedLoadingTask.destroy();
+        });
+        Promise.all([result1, result2, result3]).then(function () {
+          done();
+        }).catch(function (reason) {
+          done.fail(reason);
+        });
+      });
+      it('creates pdf doc from password protected PDF file and aborts/throws ' + 'in the onPassword callback (issue 7806)', function (done) {
+        var url = new URL('../pdfs/issue3371.pdf', window.location).href;
+        var passwordNeededLoadingTask = PDFJS.getDocument(url);
+        var passwordIncorrectLoadingTask = PDFJS.getDocument({
+          url: url,
+          password: 'qwerty'
+        });
+        passwordNeededLoadingTask.onPassword = function (callback, reason) {
+          if (reason === PasswordResponses.NEED_PASSWORD) {
+            passwordNeededLoadingTask.destroy();
+            return;
+          }
+          expect(false).toEqual(true);
+        };
+        var result1 = passwordNeededLoadingTask.promise.then(function () {
+          done.fail('shall fail since the loadingTask should be destroyed');
+          return Promise.reject(new Error('loadingTask should be rejected'));
+        }, function (reason) {
+          expect(reason instanceof PasswordException).toEqual(true);
+          expect(reason.code).toEqual(PasswordResponses.NEED_PASSWORD);
+        });
+        passwordIncorrectLoadingTask.onPassword = function (callback, reason) {
+          if (reason === PasswordResponses.INCORRECT_PASSWORD) {
+            throw new Error('Incorrect password');
+          }
+          expect(false).toEqual(true);
+        };
+        var result2 = passwordIncorrectLoadingTask.promise.then(function () {
+          done.fail('shall fail since the onPassword callback should throw');
+          return Promise.reject(new Error('loadingTask should be rejected'));
+        }, function (reason) {
+          expect(reason instanceof PasswordException).toEqual(true);
+          expect(reason.code).toEqual(PasswordResponses.INCORRECT_PASSWORD);
+          return passwordIncorrectLoadingTask.destroy();
+        });
+        Promise.all([result1, result2]).then(function () {
+          done();
+        }).catch(function (reason) {
+          done.fail(reason);
+        });
+      });
     });
-   });
-   it('creates pdf doc from URL and aborts before worker initialized', function (done) {
-    var loadingTask = PDFJS.getDocument(basicApiUrl);
-    loadingTask.destroy();
-    loadingTask.promise.then(function (reason) {
-     done.fail('shall fail loading');
-    }).catch(function (reason) {
-     expect(true).toEqual(true);
-     done();
+  });
+  describe('PDFWorker', function () {
+    it('worker created or destroyed', function (done) {
+      var worker = new PDFJS.PDFWorker('test1');
+      worker.promise.then(function () {
+        expect(worker.name).toEqual('test1');
+        expect(!!worker.port).toEqual(true);
+        expect(worker.destroyed).toEqual(false);
+        expect(!!worker._webWorker).toEqual(true);
+        expect(worker.port === worker._webWorker).toEqual(true);
+        worker.destroy();
+        expect(!!worker.port).toEqual(false);
+        expect(worker.destroyed).toEqual(true);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   });
-   it('creates pdf doc from URL and aborts loading after worker initialized', function (done) {
-    var loadingTask = PDFJS.getDocument(basicApiUrl);
-    var destroyed = loadingTask._worker.promise.then(function () {
-     return loadingTask.destroy();
+    it('worker created or destroyed by getDocument', function (done) {
+      var loadingTask = PDFJS.getDocument(basicApiUrl);
+      var worker;
+      loadingTask.promise.then(function () {
+        worker = loadingTask._worker;
+        expect(!!worker).toEqual(true);
+      });
+      var destroyPromise = loadingTask.promise.then(function () {
+        return loadingTask.destroy();
+      });
+      destroyPromise.then(function () {
+        var destroyedWorker = loadingTask._worker;
+        expect(!!destroyedWorker).toEqual(false);
+        expect(worker.destroyed).toEqual(true);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    destroyed.then(function (data) {
-     expect(true).toEqual(true);
-     done();
-    }).catch(function (reason) {
-     done.fail(reason);
+    it('worker created and can be used in getDocument', function (done) {
+      var worker = new PDFJS.PDFWorker('test1');
+      var loadingTask = PDFJS.getDocument({
+        url: basicApiUrl,
+        worker: worker
+      });
+      loadingTask.promise.then(function () {
+        var docWorker = loadingTask._worker;
+        expect(!!docWorker).toEqual(false);
+        var messageHandlerPort = loadingTask._transport.messageHandler.comObj;
+        expect(messageHandlerPort === worker.port).toEqual(true);
+      });
+      var destroyPromise = loadingTask.promise.then(function () {
+        return loadingTask.destroy();
+      });
+      destroyPromise.then(function () {
+        expect(worker.destroyed).toEqual(false);
+        worker.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   });
-   it('creates pdf doc from typed array', function (done) {
-    var nonBinaryRequest = PDFJS.disableWorker;
-    var request = new XMLHttpRequest();
-    request.open('GET', basicApiUrl, false);
-    if (!nonBinaryRequest) {
-     try {
-      request.responseType = 'arraybuffer';
-      nonBinaryRequest = request.responseType !== 'arraybuffer';
-     } catch (e) {
-      nonBinaryRequest = true;
-     }
-    }
-    if (nonBinaryRequest && request.overrideMimeType) {
-     request.overrideMimeType('text/plain; charset=x-user-defined');
-    }
-    request.send(null);
-    var typedArrayPdf;
-    if (nonBinaryRequest) {
-     var data = Array.prototype.map.call(request.responseText, function (ch) {
-      return ch.charCodeAt(0) & 0xFF;
-     });
-     typedArrayPdf = new Uint8Array(data);
-    } else {
-     typedArrayPdf = new Uint8Array(request.response);
-    }
-    expect(typedArrayPdf.length).toEqual(basicApiFileLength);
-    var loadingTask = PDFJS.getDocument(typedArrayPdf);
-    loadingTask.promise.then(function (data) {
-     expect(data instanceof PDFDocumentProxy).toEqual(true);
-     loadingTask.destroy();
-     done();
-    }).catch(function (reason) {
-     done.fail(reason);
+    it('creates more than one worker', function (done) {
+      var worker1 = new PDFJS.PDFWorker('test1');
+      var worker2 = new PDFJS.PDFWorker('test2');
+      var worker3 = new PDFJS.PDFWorker('test3');
+      var ready = Promise.all([worker1.promise, worker2.promise, worker3.promise]);
+      ready.then(function () {
+        expect(worker1.port !== worker2.port && worker1.port !== worker3.port && worker2.port !== worker3.port).toEqual(true);
+        worker1.destroy();
+        worker2.destroy();
+        worker3.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   });
-   it('creates pdf doc from invalid PDF file', function (done) {
-    var url = new URL('../pdfs/bug1020226.pdf', window.location).href;
-    var loadingTask = PDFJS.getDocument(url);
-    loadingTask.promise.then(function () {
-     done.fail('shall fail loading');
-    }).catch(function (error) {
-     expect(error instanceof InvalidPDFException).toEqual(true);
-     loadingTask.destroy();
-     done();
+  });
+  describe('PDFDocument', function () {
+    var loadingTask;
+    var doc;
+    beforeAll(function (done) {
+      loadingTask = PDFJS.getDocument(basicApiUrl);
+      loadingTask.promise.then(function (data) {
+        doc = data;
+        done();
+      });
     });
-   });
-   it('creates pdf doc from non-existent URL', function (done) {
-    var nonExistentUrl = new URL('../pdfs/non-existent.pdf', window.location).href;
-    var loadingTask = PDFJS.getDocument(nonExistentUrl);
-    loadingTask.promise.then(function (error) {
-     done.fail('shall fail loading');
-    }).catch(function (error) {
-     expect(error instanceof MissingPDFException).toEqual(true);
-     loadingTask.destroy();
-     done();
+    afterAll(function () {
+      loadingTask.destroy();
     });
-   });
-   it('creates pdf doc from PDF file protected with user and owner password', function (done) {
-    var url = new URL('../pdfs/pr6531_1.pdf', window.location).href;
-    var loadingTask = PDFJS.getDocument(url);
-    var isPasswordNeededResolved = false;
-    var passwordNeededCapability = createPromiseCapability();
-    var isPasswordIncorrectResolved = false;
-    var passwordIncorrectCapability = createPromiseCapability();
-    loadingTask.onPassword = function (updatePassword, reason) {
-     if (reason === PasswordResponses.NEED_PASSWORD && !isPasswordNeededResolved) {
-      isPasswordNeededResolved = true;
-      passwordNeededCapability.resolve();
-      updatePassword('qwerty');
-      return;
-     }
-     if (reason === PasswordResponses.INCORRECT_PASSWORD && !isPasswordIncorrectResolved) {
-      isPasswordIncorrectResolved = true;
-      passwordIncorrectCapability.resolve();
-      updatePassword('asdfasdf');
-      return;
-     }
-     expect(false).toEqual(true);
-    };
-    var promises = [
-     passwordNeededCapability.promise,
-     passwordIncorrectCapability.promise,
-     loadingTask.promise
-    ];
-    Promise.all(promises).then(function (data) {
-     expect(data[2] instanceof PDFDocumentProxy).toEqual(true);
-     loadingTask.destroy();
-     done();
-    }).catch(function (reason) {
-     done.fail(reason);
+    it('gets number of pages', function () {
+      expect(doc.numPages).toEqual(3);
     });
-   });
-   it('creates pdf doc from PDF file protected with only a user password', function (done) {
-    var url = new URL('../pdfs/pr6531_2.pdf', window.location).href;
-    var passwordNeededLoadingTask = PDFJS.getDocument({
-     url: url,
-     password: ''
+    it('gets fingerprint', function () {
+      var fingerprint = doc.fingerprint;
+      expect(typeof fingerprint).toEqual('string');
+      expect(fingerprint.length > 0).toEqual(true);
     });
-    var result1 = passwordNeededLoadingTask.promise.then(function () {
-     done.fail('shall fail with no password');
-     return Promise.reject(new Error('loadingTask should be rejected'));
-    }, function (data) {
-     expect(data instanceof PasswordException).toEqual(true);
-     expect(data.code).toEqual(PasswordResponses.NEED_PASSWORD);
-     return passwordNeededLoadingTask.destroy();
+    it('gets page', function (done) {
+      var promise = doc.getPage(1);
+      promise.then(function (data) {
+        expect(data instanceof PDFPageProxy).toEqual(true);
+        expect(data.pageIndex).toEqual(0);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    var passwordIncorrectLoadingTask = PDFJS.getDocument({
-     url: url,
-     password: 'qwerty'
+    it('gets non-existent page', function (done) {
+      var outOfRangePromise = doc.getPage(100);
+      var nonIntegerPromise = doc.getPage(2.5);
+      var nonNumberPromise = doc.getPage('1');
+      outOfRangePromise = outOfRangePromise.then(function () {
+        throw new Error('shall fail for out-of-range pageNumber parameter');
+      }, function (reason) {
+        expect(reason instanceof Error).toEqual(true);
+      });
+      nonIntegerPromise = nonIntegerPromise.then(function () {
+        throw new Error('shall fail for non-integer pageNumber parameter');
+      }, function (reason) {
+        expect(reason instanceof Error).toEqual(true);
+      });
+      nonNumberPromise = nonNumberPromise.then(function () {
+        throw new Error('shall fail for non-number pageNumber parameter');
+      }, function (reason) {
+        expect(reason instanceof Error).toEqual(true);
+      });
+      Promise.all([outOfRangePromise, nonIntegerPromise, nonNumberPromise]).then(function () {
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    var result2 = passwordIncorrectLoadingTask.promise.then(function () {
-     done.fail('shall fail with wrong password');
-     return Promise.reject(new Error('loadingTask should be rejected'));
-    }, function (data) {
-     expect(data instanceof PasswordException).toEqual(true);
-     expect(data.code).toEqual(PasswordResponses.INCORRECT_PASSWORD);
-     return passwordIncorrectLoadingTask.destroy();
+    it('gets page index', function (done) {
+      var ref = {
+        num: 17,
+        gen: 0
+      };
+      var promise = doc.getPageIndex(ref);
+      promise.then(function (pageIndex) {
+        expect(pageIndex).toEqual(1);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    var passwordAcceptedLoadingTask = PDFJS.getDocument({
-     url: url,
-     password: 'asdfasdf'
+    it('gets invalid page index', function (done) {
+      var ref = {
+        num: 3,
+        gen: 0
+      };
+      var promise = doc.getPageIndex(ref);
+      promise.then(function () {
+        done.fail('shall fail for invalid page reference.');
+      }).catch(function (reason) {
+        expect(reason instanceof Error).toEqual(true);
+        done();
+      });
     });
-    var result3 = passwordAcceptedLoadingTask.promise.then(function (data) {
-     expect(data instanceof PDFDocumentProxy).toEqual(true);
-     return passwordAcceptedLoadingTask.destroy();
+    it('gets destinations, from /Dests dictionary', function (done) {
+      var promise = doc.getDestinations();
+      promise.then(function (data) {
+        expect(data).toEqual({
+          chapter1: [{
+            gen: 0,
+            num: 17
+          }, { name: 'XYZ' }, 0, 841.89, null]
+        });
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    Promise.all([
-     result1,
-     result2,
-     result3
-    ]).then(function () {
-     done();
-    }).catch(function (reason) {
-     done.fail(reason);
+    it('gets a destination, from /Dests dictionary', function (done) {
+      var promise = doc.getDestination('chapter1');
+      promise.then(function (data) {
+        expect(data).toEqual([{
+          gen: 0,
+          num: 17
+        }, { name: 'XYZ' }, 0, 841.89, null]);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   });
-   it('creates pdf doc from password protected PDF file and aborts/throws ' + 'in the onPassword callback (issue 7806)', function (done) {
-    var url = new URL('../pdfs/issue3371.pdf', window.location).href;
-    var passwordNeededLoadingTask = PDFJS.getDocument(url);
-    var passwordIncorrectLoadingTask = PDFJS.getDocument({
-     url: url,
-     password: 'qwerty'
+    it('gets a non-existent destination, from /Dests dictionary', function (done) {
+      var promise = doc.getDestination('non-existent-named-destination');
+      promise.then(function (data) {
+        expect(data).toEqual(null);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    passwordNeededLoadingTask.onPassword = function (callback, reason) {
-     if (reason === PasswordResponses.NEED_PASSWORD) {
-      passwordNeededLoadingTask.destroy();
-      return;
-     }
-     expect(false).toEqual(true);
-    };
-    var result1 = passwordNeededLoadingTask.promise.then(function () {
-     done.fail('shall fail since the loadingTask should be destroyed');
-     return Promise.reject(new Error('loadingTask should be rejected'));
-    }, function (reason) {
-     expect(reason instanceof PasswordException).toEqual(true);
-     expect(reason.code).toEqual(PasswordResponses.NEED_PASSWORD);
+    it('gets destinations, from /Names (NameTree) dictionary', function (done) {
+      var url = new URL('../pdfs/issue6204.pdf', window.location).href;
+      var loadingTask = PDFJS.getDocument(url);
+      var promise = loadingTask.promise.then(function (pdfDocument) {
+        return pdfDocument.getDestinations();
+      });
+      promise.then(function (destinations) {
+        expect(destinations).toEqual({
+          'Page.1': [{
+            num: 1,
+            gen: 0
+          }, { name: 'XYZ' }, 0, 375, null],
+          'Page.2': [{
+            num: 6,
+            gen: 0
+          }, { name: 'XYZ' }, 0, 375, null]
+        });
+        loadingTask.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    passwordIncorrectLoadingTask.onPassword = function (callback, reason) {
-     if (reason === PasswordResponses.INCORRECT_PASSWORD) {
-      throw new Error('Incorrect password');
-     }
-     expect(false).toEqual(true);
-    };
-    var result2 = passwordIncorrectLoadingTask.promise.then(function () {
-     done.fail('shall fail since the onPassword callback should throw');
-     return Promise.reject(new Error('loadingTask should be rejected'));
-    }, function (reason) {
-     expect(reason instanceof PasswordException).toEqual(true);
-     expect(reason.code).toEqual(PasswordResponses.INCORRECT_PASSWORD);
-     return passwordIncorrectLoadingTask.destroy();
+    it('gets a destination, from /Names (NameTree) dictionary', function (done) {
+      var url = new URL('../pdfs/issue6204.pdf', window.location).href;
+      var loadingTask = PDFJS.getDocument(url);
+      var promise = loadingTask.promise.then(function (pdfDocument) {
+        return pdfDocument.getDestination('Page.1');
+      });
+      promise.then(function (destination) {
+        expect(destination).toEqual([{
+          num: 1,
+          gen: 0
+        }, { name: 'XYZ' }, 0, 375, null]);
+        loadingTask.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    Promise.all([
-     result1,
-     result2
-    ]).then(function () {
-     done();
-    }).catch(function (reason) {
-     done.fail(reason);
+    it('gets a non-existent destination, from /Names (NameTree) dictionary', function (done) {
+      var url = new URL('../pdfs/issue6204.pdf', window.location).href;
+      var loadingTask = PDFJS.getDocument(url);
+      var promise = loadingTask.promise.then(function (pdfDocument) {
+        return pdfDocument.getDestination('non-existent-named-destination');
+      });
+      promise.then(function (destination) {
+        expect(destination).toEqual(null);
+        loadingTask.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   });
-  });
- });
- describe('PDFWorker', function () {
-  it('worker created or destroyed', function (done) {
-   var worker = new PDFJS.PDFWorker('test1');
-   worker.promise.then(function () {
-    expect(worker.name).toEqual('test1');
-    expect(!!worker.port).toEqual(true);
-    expect(worker.destroyed).toEqual(false);
-    expect(!!worker._webWorker).toEqual(true);
-    expect(worker.port === worker._webWorker).toEqual(true);
-    worker.destroy();
-    expect(!!worker.port).toEqual(false);
-    expect(worker.destroyed).toEqual(true);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('worker created or destroyed by getDocument', function (done) {
-   var loadingTask = PDFJS.getDocument(basicApiUrl);
-   var worker;
-   loadingTask.promise.then(function () {
-    worker = loadingTask._worker;
-    expect(!!worker).toEqual(true);
-   });
-   var destroyPromise = loadingTask.promise.then(function () {
-    return loadingTask.destroy();
-   });
-   destroyPromise.then(function () {
-    var destroyedWorker = loadingTask._worker;
-    expect(!!destroyedWorker).toEqual(false);
-    expect(worker.destroyed).toEqual(true);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('worker created and can be used in getDocument', function (done) {
-   var worker = new PDFJS.PDFWorker('test1');
-   var loadingTask = PDFJS.getDocument({
-    url: basicApiUrl,
-    worker: worker
-   });
-   loadingTask.promise.then(function () {
-    var docWorker = loadingTask._worker;
-    expect(!!docWorker).toEqual(false);
-    var messageHandlerPort = loadingTask._transport.messageHandler.comObj;
-    expect(messageHandlerPort === worker.port).toEqual(true);
-   });
-   var destroyPromise = loadingTask.promise.then(function () {
-    return loadingTask.destroy();
-   });
-   destroyPromise.then(function () {
-    expect(worker.destroyed).toEqual(false);
-    worker.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('creates more than one worker', function (done) {
-   var worker1 = new PDFJS.PDFWorker('test1');
-   var worker2 = new PDFJS.PDFWorker('test2');
-   var worker3 = new PDFJS.PDFWorker('test3');
-   var ready = Promise.all([
-    worker1.promise,
-    worker2.promise,
-    worker3.promise
-   ]);
-   ready.then(function () {
-    expect(worker1.port !== worker2.port && worker1.port !== worker3.port && worker2.port !== worker3.port).toEqual(true);
-    worker1.destroy();
-    worker2.destroy();
-    worker3.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
- });
- describe('PDFDocument', function () {
-  var loadingTask;
-  var doc;
-  beforeAll(function (done) {
-   loadingTask = PDFJS.getDocument(basicApiUrl);
-   loadingTask.promise.then(function (data) {
-    doc = data;
-    done();
-   });
-  });
-  afterAll(function () {
-   loadingTask.destroy();
-  });
-  it('gets number of pages', function () {
-   expect(doc.numPages).toEqual(3);
-  });
-  it('gets fingerprint', function () {
-   var fingerprint = doc.fingerprint;
-   expect(typeof fingerprint).toEqual('string');
-   expect(fingerprint.length > 0).toEqual(true);
-  });
-  it('gets page', function (done) {
-   var promise = doc.getPage(1);
-   promise.then(function (data) {
-    expect(data instanceof PDFPageProxy).toEqual(true);
-    expect(data.pageIndex).toEqual(0);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets non-existent page', function (done) {
-   var outOfRangePromise = doc.getPage(100);
-   var nonIntegerPromise = doc.getPage(2.5);
-   var nonNumberPromise = doc.getPage('1');
-   outOfRangePromise = outOfRangePromise.then(function () {
-    throw new Error('shall fail for out-of-range pageNumber parameter');
-   }, function (reason) {
-    expect(reason instanceof Error).toEqual(true);
-   });
-   nonIntegerPromise = nonIntegerPromise.then(function () {
-    throw new Error('shall fail for non-integer pageNumber parameter');
-   }, function (reason) {
-    expect(reason instanceof Error).toEqual(true);
-   });
-   nonNumberPromise = nonNumberPromise.then(function () {
-    throw new Error('shall fail for non-number pageNumber parameter');
-   }, function (reason) {
-    expect(reason instanceof Error).toEqual(true);
-   });
-   Promise.all([
-    outOfRangePromise,
-    nonIntegerPromise,
-    nonNumberPromise
-   ]).then(function () {
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets page index', function (done) {
-   var ref = {
-    num: 17,
-    gen: 0
-   };
-   var promise = doc.getPageIndex(ref);
-   promise.then(function (pageIndex) {
-    expect(pageIndex).toEqual(1);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets invalid page index', function (done) {
-   var ref = {
-    num: 3,
-    gen: 0
-   };
-   var promise = doc.getPageIndex(ref);
-   promise.then(function () {
-    done.fail('shall fail for invalid page reference.');
-   }).catch(function (reason) {
-    expect(reason instanceof Error).toEqual(true);
-    done();
-   });
-  });
-  it('gets destinations, from /Dests dictionary', function (done) {
-   var promise = doc.getDestinations();
-   promise.then(function (data) {
-    expect(data).toEqual({
-     chapter1: [
-      {
-       gen: 0,
-       num: 17
-      },
-      { name: 'XYZ' },
-      0,
-      841.89,
-      null
-     ]
+    it('gets non-existent page labels', function (done) {
+      var promise = doc.getPageLabels();
+      promise.then(function (data) {
+        expect(data).toEqual(null);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets a destination, from /Dests dictionary', function (done) {
-   var promise = doc.getDestination('chapter1');
-   promise.then(function (data) {
-    expect(data).toEqual([
-     {
-      gen: 0,
-      num: 17
-     },
-     { name: 'XYZ' },
-     0,
-     841.89,
-     null
-    ]);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets a non-existent destination, from /Dests dictionary', function (done) {
-   var promise = doc.getDestination('non-existent-named-destination');
-   promise.then(function (data) {
-    expect(data).toEqual(null);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets destinations, from /Names (NameTree) dictionary', function (done) {
-   var url = new URL('../pdfs/issue6204.pdf', window.location).href;
-   var loadingTask = PDFJS.getDocument(url);
-   var promise = loadingTask.promise.then(function (pdfDocument) {
-    return pdfDocument.getDestinations();
-   });
-   promise.then(function (destinations) {
-    expect(destinations).toEqual({
-     'Page.1': [
-      {
-       num: 1,
-       gen: 0
-      },
-      { name: 'XYZ' },
-      0,
-      375,
-      null
-     ],
-     'Page.2': [
-      {
-       num: 6,
-       gen: 0
-      },
-      { name: 'XYZ' },
-      0,
-      375,
-      null
-     ]
+    it('gets page labels', function (done) {
+      var url0 = new URL('../pdfs/bug793632.pdf', window.location).href;
+      var loadingTask0 = PDFJS.getDocument(url0);
+      var promise0 = loadingTask0.promise.then(function (pdfDoc) {
+        return pdfDoc.getPageLabels();
+      });
+      var url1 = new URL('../pdfs/issue1453.pdf', window.location).href;
+      var loadingTask1 = PDFJS.getDocument(url1);
+      var promise1 = loadingTask1.promise.then(function (pdfDoc) {
+        return pdfDoc.getPageLabels();
+      });
+      var url2 = new URL('../pdfs/rotation.pdf', window.location).href;
+      var loadingTask2 = PDFJS.getDocument(url2);
+      var promise2 = loadingTask2.promise.then(function (pdfDoc) {
+        return pdfDoc.getPageLabels();
+      });
+      var url3 = new URL('../pdfs/bad-PageLabels.pdf', window.location).href;
+      var loadingTask3 = PDFJS.getDocument(url3);
+      var promise3 = loadingTask3.promise.then(function (pdfDoc) {
+        return pdfDoc.getPageLabels();
+      });
+      Promise.all([promise0, promise1, promise2, promise3]).then(function (pageLabels) {
+        expect(pageLabels[0]).toEqual(['i', 'ii', 'iii', '1']);
+        expect(pageLabels[1]).toEqual(['Front Page1']);
+        expect(pageLabels[2]).toEqual(['1', '2']);
+        expect(pageLabels[3]).toEqual(['X3']);
+        loadingTask0.destroy();
+        loadingTask1.destroy();
+        loadingTask2.destroy();
+        loadingTask3.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    loadingTask.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets a destination, from /Names (NameTree) dictionary', function (done) {
-   var url = new URL('../pdfs/issue6204.pdf', window.location).href;
-   var loadingTask = PDFJS.getDocument(url);
-   var promise = loadingTask.promise.then(function (pdfDocument) {
-    return pdfDocument.getDestination('Page.1');
-   });
-   promise.then(function (destination) {
-    expect(destination).toEqual([
-     {
-      num: 1,
-      gen: 0
-     },
-     { name: 'XYZ' },
-     0,
-     375,
-     null
-    ]);
-    loadingTask.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets a non-existent destination, from /Names (NameTree) dictionary', function (done) {
-   var url = new URL('../pdfs/issue6204.pdf', window.location).href;
-   var loadingTask = PDFJS.getDocument(url);
-   var promise = loadingTask.promise.then(function (pdfDocument) {
-    return pdfDocument.getDestination('non-existent-named-destination');
-   });
-   promise.then(function (destination) {
-    expect(destination).toEqual(null);
-    loadingTask.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets non-existent page labels', function (done) {
-   var promise = doc.getPageLabels();
-   promise.then(function (data) {
-    expect(data).toEqual(null);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets page labels', function (done) {
-   var url0 = new URL('../pdfs/bug793632.pdf', window.location).href;
-   var loadingTask0 = PDFJS.getDocument(url0);
-   var promise0 = loadingTask0.promise.then(function (pdfDoc) {
-    return pdfDoc.getPageLabels();
-   });
-   var url1 = new URL('../pdfs/issue1453.pdf', window.location).href;
-   var loadingTask1 = PDFJS.getDocument(url1);
-   var promise1 = loadingTask1.promise.then(function (pdfDoc) {
-    return pdfDoc.getPageLabels();
-   });
-   var url2 = new URL('../pdfs/rotation.pdf', window.location).href;
-   var loadingTask2 = PDFJS.getDocument(url2);
-   var promise2 = loadingTask2.promise.then(function (pdfDoc) {
-    return pdfDoc.getPageLabels();
-   });
-   var url3 = new URL('../pdfs/bad-PageLabels.pdf', window.location).href;
-   var loadingTask3 = PDFJS.getDocument(url3);
-   var promise3 = loadingTask3.promise.then(function (pdfDoc) {
-    return pdfDoc.getPageLabels();
-   });
-   Promise.all([
-    promise0,
-    promise1,
-    promise2,
-    promise3
-   ]).then(function (pageLabels) {
-    expect(pageLabels[0]).toEqual([
-     'i',
-     'ii',
-     'iii',
-     '1'
-    ]);
-    expect(pageLabels[1]).toEqual(['Front Page1']);
-    expect(pageLabels[2]).toEqual([
-     '1',
-     '2'
-    ]);
-    expect(pageLabels[3]).toEqual(['X3']);
-    loadingTask0.destroy();
-    loadingTask1.destroy();
-    loadingTask2.destroy();
-    loadingTask3.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets non-existent attachments', function (done) {
-   var promise = doc.getAttachments();
-   promise.then(function (data) {
-    expect(data).toEqual(null);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets attachments', function (done) {
-   var url = new URL('../pdfs/bug766138.pdf', window.location).href;
-   var loadingTask = PDFJS.getDocument(url);
-   var promise = loadingTask.promise.then(function (pdfDoc) {
-    return pdfDoc.getAttachments();
-   });
-   promise.then(function (data) {
-    var attachment = data['Press Quality.joboptions'];
-    expect(attachment.filename).toEqual('Press Quality.joboptions');
-    expect(attachment.content instanceof Uint8Array).toBeTruthy();
-    expect(attachment.content.length).toEqual(30098);
-    loadingTask.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets javascript', function (done) {
-   var promise = doc.getJavaScript();
-   promise.then(function (data) {
-    expect(data).toEqual([]);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  var viewerPrintRegExp = /\bprint\s*\(/;
-  it('gets javascript with printing instructions (Print action)', function (done) {
-   var pdfUrl = new URL('../pdfs/bug1001080.pdf', window.location).href;
-   var loadingTask = PDFJS.getDocument(pdfUrl);
-   var promise = loadingTask.promise.then(function (doc) {
-    return doc.getJavaScript();
-   });
-   promise.then(function (data) {
-    expect(data).toEqual(['print({});']);
-    expect(data[0]).toMatch(viewerPrintRegExp);
-    loadingTask.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets javascript with printing instructions (JS action)', function (done) {
-   var pdfUrl = new URL('../pdfs/issue6106.pdf', window.location).href;
-   var loadingTask = PDFJS.getDocument(pdfUrl);
-   var promise = loadingTask.promise.then(function (doc) {
-    return doc.getJavaScript();
-   });
-   promise.then(function (data) {
-    expect(data).toEqual(['this.print({bUI:true,bSilent:false,bShrinkToFit:true});']);
-    expect(data[0]).toMatch(viewerPrintRegExp);
-    loadingTask.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets non-existent outline', function (done) {
-   var url = new URL('../pdfs/tracemonkey.pdf', window.location).href;
-   var loadingTask = PDFJS.getDocument(url);
-   var promise = loadingTask.promise.then(function (pdfDocument) {
-    return pdfDocument.getOutline();
-   });
-   promise.then(function (outline) {
-    expect(outline).toEqual(null);
-    loadingTask.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets outline', function (done) {
-   var promise = doc.getOutline();
-   promise.then(function (outline) {
-    expect(outline instanceof Array).toEqual(true);
-    expect(outline.length).toEqual(2);
-    var outlineItem = outline[1];
-    expect(outlineItem.title).toEqual('Chapter 1');
-    expect(outlineItem.dest instanceof Array).toEqual(true);
-    expect(outlineItem.url).toEqual(null);
-    expect(outlineItem.unsafeUrl).toBeUndefined();
-    expect(outlineItem.newWindow).toBeUndefined();
-    expect(outlineItem.bold).toEqual(true);
-    expect(outlineItem.italic).toEqual(false);
-    expect(outlineItem.color).toEqual(new Uint8Array([
-     0,
-     64,
-     128
-    ]));
-    expect(outlineItem.items.length).toEqual(1);
-    expect(outlineItem.items[0].title).toEqual('Paragraph 1.1');
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets outline containing a url', function (done) {
-   var pdfUrl = new URL('../pdfs/issue3214.pdf', window.location).href;
-   var loadingTask = PDFJS.getDocument(pdfUrl);
-   loadingTask.promise.then(function (pdfDocument) {
-    pdfDocument.getOutline().then(function (outline) {
-     expect(outline instanceof Array).toEqual(true);
-     expect(outline.length).toEqual(5);
-     var outlineItemTwo = outline[2];
-     expect(typeof outlineItemTwo.title).toEqual('string');
-     expect(outlineItemTwo.dest).toEqual(null);
-     expect(outlineItemTwo.url).toEqual('http://google.com/');
-     expect(outlineItemTwo.unsafeUrl).toEqual('http://google.com');
-     expect(outlineItemTwo.newWindow).toBeUndefined();
-     var outlineItemOne = outline[1];
-     expect(outlineItemOne.bold).toEqual(false);
-     expect(outlineItemOne.italic).toEqual(true);
-     expect(outlineItemOne.color).toEqual(new Uint8Array([
-      0,
-      0,
-      0
-     ]));
-     loadingTask.destroy();
-     done();
+    it('gets non-existent attachments', function (done) {
+      var promise = doc.getAttachments();
+      promise.then(function (data) {
+        expect(data).toEqual(null);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets metadata', function (done) {
-   var promise = doc.getMetadata();
-   promise.then(function (metadata) {
-    expect(metadata.info['Title']).toEqual('Basic API Test');
-    expect(metadata.info['PDFFormatVersion']).toEqual('1.7');
-    expect(metadata.metadata.get('dc:title')).toEqual('Basic API Test');
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets data', function (done) {
-   var promise = doc.getData();
-   promise.then(function (data) {
-    expect(data instanceof Uint8Array).toEqual(true);
-    expect(data.length).toEqual(basicApiFileLength);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets download info', function (done) {
-   var promise = doc.getDownloadInfo();
-   promise.then(function (data) {
-    expect(data).toEqual({ length: basicApiFileLength });
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets stats', function (done) {
-   var promise = doc.getStats();
-   promise.then(function (stats) {
-    expect(stats).toEqual({
-     streamTypes: [],
-     fontTypes: []
+    it('gets attachments', function (done) {
+      var url = new URL('../pdfs/bug766138.pdf', window.location).href;
+      var loadingTask = PDFJS.getDocument(url);
+      var promise = loadingTask.promise.then(function (pdfDoc) {
+        return pdfDoc.getAttachments();
+      });
+      promise.then(function (data) {
+        var attachment = data['Press Quality.joboptions'];
+        expect(attachment.filename).toEqual('Press Quality.joboptions');
+        expect(attachment.content instanceof Uint8Array).toBeTruthy();
+        expect(attachment.content.length).toEqual(30098);
+        loadingTask.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('checks that fingerprints are unique', function (done) {
-   var url1 = new URL('../pdfs/issue4436r.pdf', window.location).href;
-   var loadingTask1 = PDFJS.getDocument(url1);
-   var url2 = new URL('../pdfs/issue4575.pdf', window.location).href;
-   var loadingTask2 = PDFJS.getDocument(url2);
-   var promises = [
-    loadingTask1.promise,
-    loadingTask2.promise
-   ];
-   Promise.all(promises).then(function (data) {
-    var fingerprint1 = data[0].fingerprint;
-    expect(typeof fingerprint1).toEqual('string');
-    expect(fingerprint1.length > 0).toEqual(true);
-    var fingerprint2 = data[1].fingerprint;
-    expect(typeof fingerprint2).toEqual('string');
-    expect(fingerprint2.length > 0).toEqual(true);
-    expect(fingerprint1).not.toEqual(fingerprint2);
-    loadingTask1.destroy();
-    loadingTask2.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
- });
- describe('Page', function () {
-  var loadingTask;
-  var pdfDocument, page;
-  beforeAll(function (done) {
-   loadingTask = PDFJS.getDocument(basicApiUrl);
-   loadingTask.promise.then(function (doc) {
-    pdfDocument = doc;
-    pdfDocument.getPage(1).then(function (data) {
-     page = data;
-     done();
+    it('gets javascript', function (done) {
+      var promise = doc.getJavaScript();
+      promise.then(function (data) {
+        expect(data).toEqual([]);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  afterAll(function () {
-   loadingTask.destroy();
-  });
-  it('gets page number', function () {
-   expect(page.pageNumber).toEqual(1);
-  });
-  it('gets rotate', function () {
-   expect(page.rotate).toEqual(0);
-  });
-  it('gets ref', function () {
-   expect(page.ref).toEqual({
-    num: 15,
-    gen: 0
-   });
-  });
-  it('gets userUnit', function () {
-   expect(page.userUnit).toEqual(1.0);
-  });
-  it('gets view', function () {
-   expect(page.view).toEqual([
-    0,
-    0,
-    595.28,
-    841.89
-   ]);
-  });
-  it('gets viewport', function () {
-   var viewport = page.getViewport(1.5, 90);
-   expect(viewport.viewBox).toEqual(page.view);
-   expect(viewport.scale).toEqual(1.5);
-   expect(viewport.rotation).toEqual(90);
-   expect(viewport.transform).toEqual([
-    0,
-    1.5,
-    1.5,
-    0,
-    0,
-    0
-   ]);
-   expect(viewport.width).toEqual(1262.835);
-   expect(viewport.height).toEqual(892.92);
-  });
-  it('gets annotations', function (done) {
-   var defaultPromise = page.getAnnotations().then(function (data) {
-    expect(data.length).toEqual(4);
-   });
-   var displayPromise = page.getAnnotations({ intent: 'display' }).then(function (data) {
-    expect(data.length).toEqual(4);
-   });
-   var printPromise = page.getAnnotations({ intent: 'print' }).then(function (data) {
-    expect(data.length).toEqual(4);
-   });
-   Promise.all([
-    defaultPromise,
-    displayPromise,
-    printPromise
-   ]).then(function () {
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets annotations containing relative URLs (bug 766086)', function (done) {
-   var url = new URL('../pdfs/bug766086.pdf', window.location).href;
-   var defaultLoadingTask = PDFJS.getDocument(url);
-   var defaultPromise = defaultLoadingTask.promise.then(function (pdfDoc) {
-    return pdfDoc.getPage(1).then(function (pdfPage) {
-     return pdfPage.getAnnotations();
+    var viewerPrintRegExp = /\bprint\s*\(/;
+    it('gets javascript with printing instructions (Print action)', function (done) {
+      var pdfUrl = new URL('../pdfs/bug1001080.pdf', window.location).href;
+      var loadingTask = PDFJS.getDocument(pdfUrl);
+      var promise = loadingTask.promise.then(function (doc) {
+        return doc.getJavaScript();
+      });
+      promise.then(function (data) {
+        expect(data).toEqual(['print({});']);
+        expect(data[0]).toMatch(viewerPrintRegExp);
+        loadingTask.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   });
-   var docBaseUrlLoadingTask = PDFJS.getDocument({
-    url: url,
-    docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf'
-   });
-   var docBaseUrlPromise = docBaseUrlLoadingTask.promise.then(function (pdfDoc) {
-    return pdfDoc.getPage(1).then(function (pdfPage) {
-     return pdfPage.getAnnotations();
+    it('gets javascript with printing instructions (JS action)', function (done) {
+      var pdfUrl = new URL('../pdfs/issue6106.pdf', window.location).href;
+      var loadingTask = PDFJS.getDocument(pdfUrl);
+      var promise = loadingTask.promise.then(function (doc) {
+        return doc.getJavaScript();
+      });
+      promise.then(function (data) {
+        expect(data).toEqual(['this.print({bUI:true,bSilent:false,bShrinkToFit:true});']);
+        expect(data[0]).toMatch(viewerPrintRegExp);
+        loadingTask.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   });
-   var invalidDocBaseUrlLoadingTask = PDFJS.getDocument({
-    url: url,
-    docBaseUrl: 'qwerty.pdf'
-   });
-   var invalidDocBaseUrlPromise = invalidDocBaseUrlLoadingTask.promise.then(function (pdfDoc) {
-    return pdfDoc.getPage(1).then(function (pdfPage) {
-     return pdfPage.getAnnotations();
+    it('gets non-existent outline', function (done) {
+      var url = new URL('../pdfs/tracemonkey.pdf', window.location).href;
+      var loadingTask = PDFJS.getDocument(url);
+      var promise = loadingTask.promise.then(function (pdfDocument) {
+        return pdfDocument.getOutline();
+      });
+      promise.then(function (outline) {
+        expect(outline).toEqual(null);
+        loadingTask.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   });
-   Promise.all([
-    defaultPromise,
-    docBaseUrlPromise,
-    invalidDocBaseUrlPromise
-   ]).then(function (data) {
-    var defaultAnnotations = data[0];
-    var docBaseUrlAnnotations = data[1];
-    var invalidDocBaseUrlAnnotations = data[2];
-    expect(defaultAnnotations[0].url).toBeUndefined();
-    expect(defaultAnnotations[0].unsafeUrl).toEqual('../../0021/002156/215675E.pdf#nameddest=15');
-    expect(docBaseUrlAnnotations[0].url).toEqual('http://www.example.com/0021/002156/215675E.pdf#nameddest=15');
-    expect(docBaseUrlAnnotations[0].unsafeUrl).toEqual('../../0021/002156/215675E.pdf#nameddest=15');
-    expect(invalidDocBaseUrlAnnotations[0].url).toBeUndefined();
-    expect(invalidDocBaseUrlAnnotations[0].unsafeUrl).toEqual('../../0021/002156/215675E.pdf#nameddest=15');
-    defaultLoadingTask.destroy();
-    docBaseUrlLoadingTask.destroy();
-    invalidDocBaseUrlLoadingTask.destroy();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets text content', function (done) {
-   var defaultPromise = page.getTextContent();
-   var parametersPromise = page.getTextContent({
-    normalizeWhitespace: true,
-    disableCombineTextItems: true
-   });
-   var promises = [
-    defaultPromise,
-    parametersPromise
-   ];
-   Promise.all(promises).then(function (data) {
-    expect(!!data[0].items).toEqual(true);
-    expect(data[0].items.length).toEqual(7);
-    expect(!!data[0].styles).toEqual(true);
-    expect(JSON.stringify(data[0])).toEqual(JSON.stringify(data[1]));
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets operator list', function (done) {
-   var promise = page.getOperatorList();
-   promise.then(function (oplist) {
-    expect(!!oplist.fnArray).toEqual(true);
-    expect(!!oplist.argsArray).toEqual(true);
-    expect(oplist.lastChunk).toEqual(true);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('gets stats after parsing page', function (done) {
-   var promise = page.getOperatorList().then(function () {
-    return pdfDocument.getStats();
-   });
-   var expectedStreamTypes = [];
-   expectedStreamTypes[StreamType.FLATE] = true;
-   var expectedFontTypes = [];
-   expectedFontTypes[FontType.TYPE1] = true;
-   expectedFontTypes[FontType.CIDFONTTYPE2] = true;
-   promise.then(function (stats) {
-    expect(stats).toEqual({
-     streamTypes: expectedStreamTypes,
-     fontTypes: expectedFontTypes
+    it('gets outline', function (done) {
+      var promise = doc.getOutline();
+      promise.then(function (outline) {
+        expect(outline instanceof Array).toEqual(true);
+        expect(outline.length).toEqual(2);
+        var outlineItem = outline[1];
+        expect(outlineItem.title).toEqual('Chapter 1');
+        expect(outlineItem.dest instanceof Array).toEqual(true);
+        expect(outlineItem.url).toEqual(null);
+        expect(outlineItem.unsafeUrl).toBeUndefined();
+        expect(outlineItem.newWindow).toBeUndefined();
+        expect(outlineItem.bold).toEqual(true);
+        expect(outlineItem.italic).toEqual(false);
+        expect(outlineItem.color).toEqual(new Uint8Array([0, 64, 128]));
+        expect(outlineItem.items.length).toEqual(1);
+        expect(outlineItem.items[0].title).toEqual('Paragraph 1.1');
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
- });
- describe('Multiple PDFJS instances', function () {
-  var pdf1 = new URL('../pdfs/tracemonkey.pdf', window.location).href;
-  var pdf2 = new URL('../pdfs/TAMReview.pdf', window.location).href;
-  var pdf3 = new URL('../pdfs/issue6068.pdf', window.location).href;
-  var loadingTasks = [];
-  var pdfDocuments = [];
-  function renderPDF(filename) {
-   var loadingTask = PDFJS.getDocument(filename);
-   loadingTasks.push(loadingTask);
-   return loadingTask.promise.then(function (pdf) {
-    pdfDocuments.push(pdf);
-    return pdf.getPage(1);
-   }).then(function (page) {
-    var c = document.createElement('canvas');
-    var v = page.getViewport(1.2);
-    c.width = v.width;
-    c.height = v.height;
-    return page.render({
-     canvasContext: c.getContext('2d'),
-     viewport: v
-    }).then(function () {
-     return c.toDataURL();
+    it('gets outline containing a url', function (done) {
+      var pdfUrl = new URL('../pdfs/issue3214.pdf', window.location).href;
+      var loadingTask = PDFJS.getDocument(pdfUrl);
+      loadingTask.promise.then(function (pdfDocument) {
+        pdfDocument.getOutline().then(function (outline) {
+          expect(outline instanceof Array).toEqual(true);
+          expect(outline.length).toEqual(5);
+          var outlineItemTwo = outline[2];
+          expect(typeof outlineItemTwo.title).toEqual('string');
+          expect(outlineItemTwo.dest).toEqual(null);
+          expect(outlineItemTwo.url).toEqual('http://google.com/');
+          expect(outlineItemTwo.unsafeUrl).toEqual('http://google.com');
+          expect(outlineItemTwo.newWindow).toBeUndefined();
+          var outlineItemOne = outline[1];
+          expect(outlineItemOne.bold).toEqual(false);
+          expect(outlineItemOne.italic).toEqual(true);
+          expect(outlineItemOne.color).toEqual(new Uint8Array([0, 0, 0]));
+          loadingTask.destroy();
+          done();
+        });
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('gets metadata', function (done) {
+      var promise = doc.getMetadata();
+      promise.then(function (metadata) {
+        expect(metadata.info['Title']).toEqual('Basic API Test');
+        expect(metadata.info['PDFFormatVersion']).toEqual('1.7');
+        expect(metadata.metadata.get('dc:title')).toEqual('Basic API Test');
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('gets data', function (done) {
+      var promise = doc.getData();
+      promise.then(function (data) {
+        expect(data instanceof Uint8Array).toEqual(true);
+        expect(data.length).toEqual(basicApiFileLength);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('gets download info', function (done) {
+      var promise = doc.getDownloadInfo();
+      promise.then(function (data) {
+        expect(data).toEqual({ length: basicApiFileLength });
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('gets stats', function (done) {
+      var promise = doc.getStats();
+      promise.then(function (stats) {
+        expect(stats).toEqual({
+          streamTypes: [],
+          fontTypes: []
+        });
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('checks that fingerprints are unique', function (done) {
+      var url1 = new URL('../pdfs/issue4436r.pdf', window.location).href;
+      var loadingTask1 = PDFJS.getDocument(url1);
+      var url2 = new URL('../pdfs/issue4575.pdf', window.location).href;
+      var loadingTask2 = PDFJS.getDocument(url2);
+      var promises = [loadingTask1.promise, loadingTask2.promise];
+      Promise.all(promises).then(function (data) {
+        var fingerprint1 = data[0].fingerprint;
+        expect(typeof fingerprint1).toEqual('string');
+        expect(fingerprint1.length > 0).toEqual(true);
+        var fingerprint2 = data[1].fingerprint;
+        expect(typeof fingerprint2).toEqual('string');
+        expect(fingerprint2.length > 0).toEqual(true);
+        expect(fingerprint1).not.toEqual(fingerprint2);
+        loadingTask1.destroy();
+        loadingTask2.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
     });
-   });
-  }
-  afterEach(function (done) {
-   var destroyPromises = pdfDocuments.map(function (pdfDocument) {
-    return pdfDocument.destroy();
-   });
-   var destroyPromises2 = loadingTasks.map(function (loadingTask) {
-    return loadingTask.destroy();
-   });
-   Promise.all(destroyPromises.concat(destroyPromises2)).then(function () {
-    done();
-   });
   });
-  it('should correctly render PDFs in parallel', function (done) {
-   var baseline1, baseline2, baseline3;
-   var promiseDone = renderPDF(pdf1).then(function (data1) {
-    baseline1 = data1;
-    return renderPDF(pdf2);
-   }).then(function (data2) {
-    baseline2 = data2;
-    return renderPDF(pdf3);
-   }).then(function (data3) {
-    baseline3 = data3;
-    return Promise.all([
-     renderPDF(pdf1),
-     renderPDF(pdf2),
-     renderPDF(pdf3)
-    ]);
-   }).then(function (dataUrls) {
-    expect(dataUrls[0]).toEqual(baseline1);
-    expect(dataUrls[1]).toEqual(baseline2);
-    expect(dataUrls[2]).toEqual(baseline3);
-    return true;
-   });
-   promiseDone.then(function () {
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
+  describe('Page', function () {
+    var loadingTask;
+    var pdfDocument, page;
+    beforeAll(function (done) {
+      loadingTask = PDFJS.getDocument(basicApiUrl);
+      loadingTask.promise.then(function (doc) {
+        pdfDocument = doc;
+        pdfDocument.getPage(1).then(function (data) {
+          page = data;
+          done();
+        });
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    afterAll(function () {
+      loadingTask.destroy();
+    });
+    it('gets page number', function () {
+      expect(page.pageNumber).toEqual(1);
+    });
+    it('gets rotate', function () {
+      expect(page.rotate).toEqual(0);
+    });
+    it('gets ref', function () {
+      expect(page.ref).toEqual({
+        num: 15,
+        gen: 0
+      });
+    });
+    it('gets userUnit', function () {
+      expect(page.userUnit).toEqual(1.0);
+    });
+    it('gets view', function () {
+      expect(page.view).toEqual([0, 0, 595.28, 841.89]);
+    });
+    it('gets viewport', function () {
+      var viewport = page.getViewport(1.5, 90);
+      expect(viewport.viewBox).toEqual(page.view);
+      expect(viewport.scale).toEqual(1.5);
+      expect(viewport.rotation).toEqual(90);
+      expect(viewport.transform).toEqual([0, 1.5, 1.5, 0, 0, 0]);
+      expect(viewport.width).toEqual(1262.835);
+      expect(viewport.height).toEqual(892.92);
+    });
+    it('gets annotations', function (done) {
+      var defaultPromise = page.getAnnotations().then(function (data) {
+        expect(data.length).toEqual(4);
+      });
+      var displayPromise = page.getAnnotations({ intent: 'display' }).then(function (data) {
+        expect(data.length).toEqual(4);
+      });
+      var printPromise = page.getAnnotations({ intent: 'print' }).then(function (data) {
+        expect(data.length).toEqual(4);
+      });
+      Promise.all([defaultPromise, displayPromise, printPromise]).then(function () {
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('gets annotations containing relative URLs (bug 766086)', function (done) {
+      var url = new URL('../pdfs/bug766086.pdf', window.location).href;
+      var defaultLoadingTask = PDFJS.getDocument(url);
+      var defaultPromise = defaultLoadingTask.promise.then(function (pdfDoc) {
+        return pdfDoc.getPage(1).then(function (pdfPage) {
+          return pdfPage.getAnnotations();
+        });
+      });
+      var docBaseUrlLoadingTask = PDFJS.getDocument({
+        url: url,
+        docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf'
+      });
+      var docBaseUrlPromise = docBaseUrlLoadingTask.promise.then(function (pdfDoc) {
+        return pdfDoc.getPage(1).then(function (pdfPage) {
+          return pdfPage.getAnnotations();
+        });
+      });
+      var invalidDocBaseUrlLoadingTask = PDFJS.getDocument({
+        url: url,
+        docBaseUrl: 'qwerty.pdf'
+      });
+      var invalidDocBaseUrlPromise = invalidDocBaseUrlLoadingTask.promise.then(function (pdfDoc) {
+        return pdfDoc.getPage(1).then(function (pdfPage) {
+          return pdfPage.getAnnotations();
+        });
+      });
+      Promise.all([defaultPromise, docBaseUrlPromise, invalidDocBaseUrlPromise]).then(function (data) {
+        var defaultAnnotations = data[0];
+        var docBaseUrlAnnotations = data[1];
+        var invalidDocBaseUrlAnnotations = data[2];
+        expect(defaultAnnotations[0].url).toBeUndefined();
+        expect(defaultAnnotations[0].unsafeUrl).toEqual('../../0021/002156/215675E.pdf#nameddest=15');
+        expect(docBaseUrlAnnotations[0].url).toEqual('http://www.example.com/0021/002156/215675E.pdf#nameddest=15');
+        expect(docBaseUrlAnnotations[0].unsafeUrl).toEqual('../../0021/002156/215675E.pdf#nameddest=15');
+        expect(invalidDocBaseUrlAnnotations[0].url).toBeUndefined();
+        expect(invalidDocBaseUrlAnnotations[0].unsafeUrl).toEqual('../../0021/002156/215675E.pdf#nameddest=15');
+        defaultLoadingTask.destroy();
+        docBaseUrlLoadingTask.destroy();
+        invalidDocBaseUrlLoadingTask.destroy();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('gets text content', function (done) {
+      var defaultPromise = page.getTextContent();
+      var parametersPromise = page.getTextContent({
+        normalizeWhitespace: true,
+        disableCombineTextItems: true
+      });
+      var promises = [defaultPromise, parametersPromise];
+      Promise.all(promises).then(function (data) {
+        expect(!!data[0].items).toEqual(true);
+        expect(data[0].items.length).toEqual(7);
+        expect(!!data[0].styles).toEqual(true);
+        expect(JSON.stringify(data[0])).toEqual(JSON.stringify(data[1]));
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('gets operator list', function (done) {
+      var promise = page.getOperatorList();
+      promise.then(function (oplist) {
+        expect(!!oplist.fnArray).toEqual(true);
+        expect(!!oplist.argsArray).toEqual(true);
+        expect(oplist.lastChunk).toEqual(true);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('gets stats after parsing page', function (done) {
+      var promise = page.getOperatorList().then(function () {
+        return pdfDocument.getStats();
+      });
+      var expectedStreamTypes = [];
+      expectedStreamTypes[StreamType.FLATE] = true;
+      var expectedFontTypes = [];
+      expectedFontTypes[FontType.TYPE1] = true;
+      expectedFontTypes[FontType.CIDFONTTYPE2] = true;
+      promise.then(function (stats) {
+        expect(stats).toEqual({
+          streamTypes: expectedStreamTypes,
+          fontTypes: expectedFontTypes
+        });
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
   });
- });
- describe('PDFDataRangeTransport', function () {
-  var pdfPath = new URL('../pdfs/tracemonkey.pdf', window.location).href;
-  var loadPromise;
-  function getDocumentData() {
-   if (loadPromise) {
-    return loadPromise;
-   }
-   loadPromise = new Promise(function (resolve, reject) {
-    var xhr = new XMLHttpRequest(pdfPath);
-    xhr.open('GET', pdfPath);
-    xhr.responseType = 'arraybuffer';
-    xhr.onload = function () {
-     resolve(new Uint8Array(xhr.response));
-    };
-    xhr.onerror = function () {
-     reject(new Error('PDF is not loaded'));
-    };
-    xhr.send();
-   });
-   return loadPromise;
-  }
-  it('should fetch document info and page using ranges', function (done) {
-   var transport;
-   var initialDataLength = 4000;
-   var fetches = 0;
-   var getDocumentPromise = getDocumentData().then(function (data) {
-    var initialData = data.subarray(0, initialDataLength);
-    transport = new PDFJS.PDFDataRangeTransport(data.length, initialData);
-    transport.requestDataRange = function (begin, end) {
-     fetches++;
-     waitSome(function () {
-      transport.onDataProgress(4000);
-      transport.onDataRange(begin, data.subarray(begin, end));
-     });
-    };
-    var loadingTask = PDFJS.getDocument(transport);
-    return loadingTask.promise;
-   });
-   var pdfDocument;
-   var getPagePromise = getDocumentPromise.then(function (pdfDocument_) {
-    pdfDocument = pdfDocument_;
-    var pagePromise = pdfDocument.getPage(10);
-    return pagePromise;
-   });
-   getPagePromise.then(function (page) {
-    expect(pdfDocument.numPages).toEqual(14);
-    expect(page.rotate).toEqual(0);
-    expect(fetches).toBeGreaterThan(2);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
+  describe('Multiple PDFJS instances', function () {
+    var pdf1 = new URL('../pdfs/tracemonkey.pdf', window.location).href;
+    var pdf2 = new URL('../pdfs/TAMReview.pdf', window.location).href;
+    var pdf3 = new URL('../pdfs/issue6068.pdf', window.location).href;
+    var loadingTasks = [];
+    var pdfDocuments = [];
+    function renderPDF(filename) {
+      var loadingTask = PDFJS.getDocument(filename);
+      loadingTasks.push(loadingTask);
+      return loadingTask.promise.then(function (pdf) {
+        pdfDocuments.push(pdf);
+        return pdf.getPage(1);
+      }).then(function (page) {
+        var c = document.createElement('canvas');
+        var v = page.getViewport(1.2);
+        c.width = v.width;
+        c.height = v.height;
+        return page.render({
+          canvasContext: c.getContext('2d'),
+          viewport: v
+        }).then(function () {
+          return c.toDataURL();
+        });
+      });
+    }
+    afterEach(function (done) {
+      var destroyPromises = pdfDocuments.map(function (pdfDocument) {
+        return pdfDocument.destroy();
+      });
+      var destroyPromises2 = loadingTasks.map(function (loadingTask) {
+        return loadingTask.destroy();
+      });
+      Promise.all(destroyPromises.concat(destroyPromises2)).then(function () {
+        done();
+      });
+    });
+    it('should correctly render PDFs in parallel', function (done) {
+      var baseline1, baseline2, baseline3;
+      var promiseDone = renderPDF(pdf1).then(function (data1) {
+        baseline1 = data1;
+        return renderPDF(pdf2);
+      }).then(function (data2) {
+        baseline2 = data2;
+        return renderPDF(pdf3);
+      }).then(function (data3) {
+        baseline3 = data3;
+        return Promise.all([renderPDF(pdf1), renderPDF(pdf2), renderPDF(pdf3)]);
+      }).then(function (dataUrls) {
+        expect(dataUrls[0]).toEqual(baseline1);
+        expect(dataUrls[1]).toEqual(baseline2);
+        expect(dataUrls[2]).toEqual(baseline3);
+        return true;
+      });
+      promiseDone.then(function () {
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
   });
-  it('should fetch document info and page using range and streaming', function (done) {
-   var transport;
-   var initialDataLength = 4000;
-   var fetches = 0;
-   var getDocumentPromise = getDocumentData().then(function (data) {
-    var initialData = data.subarray(0, initialDataLength);
-    transport = new PDFJS.PDFDataRangeTransport(data.length, initialData);
-    transport.requestDataRange = function (begin, end) {
-     fetches++;
-     if (fetches === 1) {
-      transport.onDataProgressiveRead(data.subarray(initialDataLength));
-     }
-     waitSome(function () {
-      transport.onDataRange(begin, data.subarray(begin, end));
-     });
-    };
-    var loadingTask = PDFJS.getDocument(transport);
-    return loadingTask.promise;
-   });
-   var pdfDocument;
-   var getPagePromise = getDocumentPromise.then(function (pdfDocument_) {
-    pdfDocument = pdfDocument_;
-    var pagePromise = pdfDocument.getPage(10);
-    return pagePromise;
-   });
-   getPagePromise.then(function (page) {
-    expect(pdfDocument.numPages).toEqual(14);
-    expect(page.rotate).toEqual(0);
-    expect(fetches).toEqual(1);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
+  describe('PDFDataRangeTransport', function () {
+    var pdfPath = new URL('../pdfs/tracemonkey.pdf', window.location).href;
+    var loadPromise;
+    function getDocumentData() {
+      if (loadPromise) {
+        return loadPromise;
+      }
+      loadPromise = new Promise(function (resolve, reject) {
+        var xhr = new XMLHttpRequest(pdfPath);
+        xhr.open('GET', pdfPath);
+        xhr.responseType = 'arraybuffer';
+        xhr.onload = function () {
+          resolve(new Uint8Array(xhr.response));
+        };
+        xhr.onerror = function () {
+          reject(new Error('PDF is not loaded'));
+        };
+        xhr.send();
+      });
+      return loadPromise;
+    }
+    it('should fetch document info and page using ranges', function (done) {
+      var transport;
+      var initialDataLength = 4000;
+      var fetches = 0;
+      var getDocumentPromise = getDocumentData().then(function (data) {
+        var initialData = data.subarray(0, initialDataLength);
+        transport = new PDFJS.PDFDataRangeTransport(data.length, initialData);
+        transport.requestDataRange = function (begin, end) {
+          fetches++;
+          waitSome(function () {
+            transport.onDataProgress(4000);
+            transport.onDataRange(begin, data.subarray(begin, end));
+          });
+        };
+        var loadingTask = PDFJS.getDocument(transport);
+        return loadingTask.promise;
+      });
+      var pdfDocument;
+      var getPagePromise = getDocumentPromise.then(function (pdfDocument_) {
+        pdfDocument = pdfDocument_;
+        var pagePromise = pdfDocument.getPage(10);
+        return pagePromise;
+      });
+      getPagePromise.then(function (page) {
+        expect(pdfDocument.numPages).toEqual(14);
+        expect(page.rotate).toEqual(0);
+        expect(fetches).toBeGreaterThan(2);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('should fetch document info and page using range and streaming', function (done) {
+      var transport;
+      var initialDataLength = 4000;
+      var fetches = 0;
+      var getDocumentPromise = getDocumentData().then(function (data) {
+        var initialData = data.subarray(0, initialDataLength);
+        transport = new PDFJS.PDFDataRangeTransport(data.length, initialData);
+        transport.requestDataRange = function (begin, end) {
+          fetches++;
+          if (fetches === 1) {
+            transport.onDataProgressiveRead(data.subarray(initialDataLength));
+          }
+          waitSome(function () {
+            transport.onDataRange(begin, data.subarray(begin, end));
+          });
+        };
+        var loadingTask = PDFJS.getDocument(transport);
+        return loadingTask.promise;
+      });
+      var pdfDocument;
+      var getPagePromise = getDocumentPromise.then(function (pdfDocument_) {
+        pdfDocument = pdfDocument_;
+        var pagePromise = pdfDocument.getPage(10);
+        return pagePromise;
+      });
+      getPagePromise.then(function (page) {
+        expect(pdfDocument.numPages).toEqual(14);
+        expect(page.rotate).toEqual(0);
+        expect(fetches).toEqual(1);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
   });
- });
 });

+ 15 - 14
lib/test/unit/bidi_spec.js

@@ -13,21 +13,22 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreBidi = require('../../core/bidi.js');
 var bidi = coreBidi.bidi;
 describe('bidi', function () {
- it('should mark text as RTL if more than 30% of text is RTL', function () {
-  var test = '\u0645\u0635\u0631 Egypt';
-  var result = 'Egypt \u0631\u0635\u0645';
-  var bidiText = bidi(test, -1, false);
-  expect(bidiText.str).toEqual(result);
-  expect(bidiText.dir).toEqual('rtl');
- });
- it('should mark text as LTR if less than 30% of text is RTL', function () {
-  var test = 'Egypt is known as \u0645\u0635\u0631 in Arabic.';
-  var result = 'Egypt is known as \u0631\u0635\u0645 in Arabic.';
-  var bidiText = bidi(test, -1, false);
-  expect(bidiText.str).toEqual(result);
-  expect(bidiText.dir).toEqual('ltr');
- });
+  it('should mark text as RTL if more than 30% of text is RTL', function () {
+    var test = '\u0645\u0635\u0631 Egypt';
+    var result = 'Egypt \u0631\u0635\u0645';
+    var bidiText = bidi(test, -1, false);
+    expect(bidiText.str).toEqual(result);
+    expect(bidiText.dir).toEqual('rtl');
+  });
+  it('should mark text as LTR if less than 30% of text is RTL', function () {
+    var test = 'Egypt is known as \u0645\u0635\u0631 in Arabic.';
+    var result = 'Egypt is known as \u0631\u0635\u0645 in Arabic.';
+    var bidiText = bidi(test, -1, false);
+    expect(bidiText.str).toEqual(result);
+    expect(bidiText.dir).toEqual('ltr');
+  });
 });

+ 217 - 455
lib/test/unit/cff_parser_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreCFFParser = require('../../core/cff_parser.js');
 var coreFonts = require('../../core/fonts.js');
 var coreStream = require('../../core/stream.js');
@@ -23,462 +24,223 @@ var CFFCompiler = coreCFFParser.CFFCompiler;
 var SEAC_ANALYSIS_ENABLED = coreFonts.SEAC_ANALYSIS_ENABLED;
 var Stream = coreStream.Stream;
 describe('CFFParser', function () {
- function createWithNullProto(obj) {
-  var result = Object.create(null);
-  for (var i in obj) {
-   result[i] = obj[i];
+  function createWithNullProto(obj) {
+    var result = Object.create(null);
+    for (var i in obj) {
+      result[i] = obj[i];
+    }
+    return result;
   }
-  return result;
- }
- var fontData, parser, cff;
- beforeAll(function (done) {
-  var exampleFont = '0100040100010101134142434445462b' + '54696d65732d526f6d616e000101011f' + 'f81b00f81c02f81d03f819041c6f000d' + 'fb3cfb6efa7cfa1605e911b8f1120003' + '01010813183030312e30303754696d65' + '7320526f6d616e54696d657300000002' + '010102030e0e7d99f92a99fb7695f773' + '8b06f79a93fc7c8c077d99f85695f75e' + '9908fb6e8cf87393f7108b09a70adf0b' + 'f78e14';
-  var fontArr = [];
-  for (var i = 0, ii = exampleFont.length; i < ii; i += 2) {
-   var hex = exampleFont.substr(i, 2);
-   fontArr.push(parseInt(hex, 16));
-  }
-  fontData = new Stream(fontArr);
-  done();
- });
- afterAll(function () {
-  fontData = null;
- });
- beforeEach(function (done) {
-  parser = new CFFParser(fontData, {}, SEAC_ANALYSIS_ENABLED);
-  cff = parser.parse();
-  done();
- });
- afterEach(function (done) {
-  parser = cff = null;
-  done();
- });
- it('parses header', function () {
-  var header = cff.header;
-  expect(header.major).toEqual(1);
-  expect(header.minor).toEqual(0);
-  expect(header.hdrSize).toEqual(4);
-  expect(header.offSize).toEqual(1);
- });
- it('parses name index', function () {
-  var names = cff.names;
-  expect(names.length).toEqual(1);
-  expect(names[0]).toEqual('ABCDEF+Times-Roman');
- });
- it('sanitizes name index', function () {
-  var index = new CFFIndex();
-  index.add([
-   '['.charCodeAt(0),
-   'a'.charCodeAt(0)
-  ]);
-  var names = parser.parseNameIndex(index);
-  expect(names).toEqual(['_a']);
-  index = new CFFIndex();
-  var longName = [];
-  for (var i = 0; i < 129; i++) {
-   longName.push(0);
-  }
-  index.add(longName);
-  names = parser.parseNameIndex(index);
-  expect(names[0].length).toEqual(127);
- });
- it('parses string index', function () {
-  var strings = cff.strings;
-  expect(strings.count).toEqual(3);
-  expect(strings.get(0)).toEqual('.notdef');
-  expect(strings.get(391)).toEqual('001.007');
- });
- it('parses top dict', function () {
-  var topDict = cff.topDict;
-  expect(topDict.getByName('version')).toEqual(391);
-  expect(topDict.getByName('FullName')).toEqual(392);
-  expect(topDict.getByName('FamilyName')).toEqual(393);
-  expect(topDict.getByName('Weight')).toEqual(389);
-  expect(topDict.getByName('UniqueID')).toEqual(28416);
-  expect(topDict.getByName('FontBBox')).toEqual([
-   -168,
-   -218,
-   1000,
-   898
-  ]);
-  expect(topDict.getByName('CharStrings')).toEqual(94);
-  expect(topDict.getByName('Private')).toEqual([
-   45,
-   102
-  ]);
- });
- it('refuses to add topDict key with invalid value (bug 1068432)', function () {
-  var topDict = cff.topDict;
-  var defaultValue = topDict.getByName('UnderlinePosition');
-  topDict.setByKey(3075, [NaN]);
-  expect(topDict.getByName('UnderlinePosition')).toEqual(defaultValue);
- });
- it('ignores reserved commands in parseDict, and refuses to add privateDict ' + 'keys with invalid values (bug 1308536)', function () {
-  var bytes = new Uint8Array([
-   64,
-   39,
-   31,
-   30,
-   252,
-   114,
-   137,
-   115,
-   79,
-   30,
-   197,
-   119,
-   2,
-   99,
-   127,
-   6
-  ]);
-  parser.bytes = bytes;
-  var topDict = cff.topDict;
-  topDict.setByName('Private', [
-   bytes.length,
-   0
-  ]);
-  var parsePrivateDict = function () {
-   parser.parsePrivateDict(topDict);
-  };
-  expect(parsePrivateDict).not.toThrow();
-  var privateDict = topDict.privateDict;
-  expect(privateDict.getByName('BlueValues')).toBeNull();
- });
- it('parses a CharString having cntrmask', function () {
-  var bytes = new Uint8Array([
-   0,
-   1,
-   1,
-   0,
-   38,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   1,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   149,
-   3,
-   20,
-   22,
-   22,
-   14
-  ]);
-  parser.bytes = bytes;
-  var charStringsIndex = parser.parseIndex(0).obj;
-  var charStrings = parser.parseCharStrings(charStringsIndex).charStrings;
-  expect(charStrings.count).toEqual(1);
-  expect(charStrings.get(0).length).toEqual(38);
- });
- it('parses a CharString endchar with 4 args w/seac enabled', function () {
-  var parser = new CFFParser(fontData, {}, true);
-  parser.parse();
-  var bytes = new Uint8Array([
-   0,
-   1,
-   1,
-   0,
-   237,
-   247,
-   22,
-   247,
-   72,
-   204,
-   247,
-   86,
-   14
-  ]);
-  parser.bytes = bytes;
-  var charStringsIndex = parser.parseIndex(0).obj;
-  var result = parser.parseCharStrings(charStringsIndex);
-  expect(result.charStrings.count).toEqual(1);
-  expect(result.charStrings.get(0).length).toEqual(1);
-  expect(result.seacs.length).toEqual(1);
-  expect(result.seacs[0].length).toEqual(4);
-  expect(result.seacs[0][0]).toEqual(130);
-  expect(result.seacs[0][1]).toEqual(180);
-  expect(result.seacs[0][2]).toEqual(65);
-  expect(result.seacs[0][3]).toEqual(194);
- });
- it('parses a CharString endchar with 4 args w/seac disabled', function () {
-  var parser = new CFFParser(fontData, {}, false);
-  parser.parse();
-  var bytes = new Uint8Array([
-   0,
-   1,
-   1,
-   0,
-   237,
-   247,
-   22,
-   247,
-   72,
-   204,
-   247,
-   86,
-   14
-  ]);
-  parser.bytes = bytes;
-  var charStringsIndex = parser.parseIndex(0).obj;
-  var result = parser.parseCharStrings(charStringsIndex);
-  expect(result.charStrings.count).toEqual(1);
-  expect(result.charStrings.get(0).length).toEqual(9);
-  expect(result.seacs.length).toEqual(0);
- });
- it('parses a CharString endchar no args', function () {
-  var bytes = new Uint8Array([
-   0,
-   1,
-   1,
-   0,
-   14
-  ]);
-  parser.bytes = bytes;
-  var charStringsIndex = parser.parseIndex(0).obj;
-  var result = parser.parseCharStrings(charStringsIndex);
-  expect(result.charStrings.count).toEqual(1);
-  expect(result.charStrings.get(0)[0]).toEqual(14);
-  expect(result.seacs.length).toEqual(0);
- });
- it('parses predefined charsets', function () {
-  var charset = parser.parseCharsets(0, 0, null, true);
-  expect(charset.predefined).toEqual(true);
- });
- it('parses charset format 0', function () {
-  var bytes = new Uint8Array([
-   0x00,
-   0x00,
-   0x00,
-   0x00,
-   0x00,
-   0x02
-  ]);
-  parser.bytes = bytes;
-  var charset = parser.parseCharsets(3, 2, new CFFStrings(), false);
-  expect(charset.charset[1]).toEqual('exclam');
-  charset = parser.parseCharsets(3, 2, new CFFStrings(), true);
-  expect(charset.charset[1]).toEqual(2);
- });
- it('parses charset format 1', function () {
-  var bytes = new Uint8Array([
-   0x00,
-   0x00,
-   0x00,
-   0x01,
-   0x00,
-   0x08,
-   0x01
-  ]);
-  parser.bytes = bytes;
-  var charset = parser.parseCharsets(3, 2, new CFFStrings(), false);
-  expect(charset.charset).toEqual([
-   '.notdef',
-   'quoteright',
-   'parenleft'
-  ]);
-  charset = parser.parseCharsets(3, 2, new CFFStrings(), true);
-  expect(charset.charset).toEqual([
-   '.notdef',
-   8,
-   9
-  ]);
- });
- it('parses charset format 2', function () {
-  var bytes = new Uint8Array([
-   0x00,
-   0x00,
-   0x00,
-   0x02,
-   0x00,
-   0x08,
-   0x00,
-   0x01
-  ]);
-  parser.bytes = bytes;
-  var charset = parser.parseCharsets(3, 2, new CFFStrings(), false);
-  expect(charset.charset).toEqual([
-   '.notdef',
-   'quoteright',
-   'parenleft'
-  ]);
-  charset = parser.parseCharsets(3, 2, new CFFStrings(), true);
-  expect(charset.charset).toEqual([
-   '.notdef',
-   8,
-   9
-  ]);
- });
- it('parses encoding format 0', function () {
-  var bytes = new Uint8Array([
-   0x00,
-   0x00,
-   0x00,
-   0x01,
-   0x08
-  ]);
-  parser.bytes = bytes;
-  var encoding = parser.parseEncoding(2, {}, new CFFStrings(), null);
-  expect(encoding.encoding).toEqual(createWithNullProto({ 0x8: 1 }));
- });
- it('parses encoding format 1', function () {
-  var bytes = new Uint8Array([
-   0x00,
-   0x00,
-   0x01,
-   0x01,
-   0x07,
-   0x01
-  ]);
-  parser.bytes = bytes;
-  var encoding = parser.parseEncoding(2, {}, new CFFStrings(), null);
-  expect(encoding.encoding).toEqual(createWithNullProto({
-   0x7: 0x01,
-   0x08: 0x02
-  }));
- });
- it('parses fdselect format 0', function () {
-  var bytes = new Uint8Array([
-   0x00,
-   0x00,
-   0x01
-  ]);
-  parser.bytes = bytes.slice();
-  var fdSelect = parser.parseFDSelect(0, 2);
-  expect(fdSelect.fdSelect).toEqual([
-   0,
-   1
-  ]);
-  expect(fdSelect.raw).toEqual(bytes);
- });
- it('parses fdselect format 3', function () {
-  var bytes = new Uint8Array([
-   0x03,
-   0x00,
-   0x02,
-   0x00,
-   0x00,
-   0x09,
-   0x00,
-   0x02,
-   0x0a,
-   0x00,
-   0x04
-  ]);
-  parser.bytes = bytes.slice();
-  var fdSelect = parser.parseFDSelect(0, 4);
-  expect(fdSelect.fdSelect).toEqual([
-   9,
-   9,
-   0xa,
-   0xa
-  ]);
-  expect(fdSelect.raw).toEqual(bytes);
- });
- it('parses invalid fdselect format 3 (bug 1146106)', function () {
-  var bytes = new Uint8Array([
-   0x03,
-   0x00,
-   0x02,
-   0x00,
-   0x01,
-   0x09,
-   0x00,
-   0x02,
-   0x0a,
-   0x00,
-   0x04
-  ]);
-  parser.bytes = bytes.slice();
-  var fdSelect = parser.parseFDSelect(0, 4);
-  expect(fdSelect.fdSelect).toEqual([
-   9,
-   9,
-   0xa,
-   0xa
-  ]);
-  bytes[3] = bytes[4] = 0x00;
-  expect(fdSelect.raw).toEqual(bytes);
- });
+  var fontData, parser, cff;
+  beforeAll(function (done) {
+    var exampleFont = '0100040100010101134142434445462b' + '54696d65732d526f6d616e000101011f' + 'f81b00f81c02f81d03f819041c6f000d' + 'fb3cfb6efa7cfa1605e911b8f1120003' + '01010813183030312e30303754696d65' + '7320526f6d616e54696d657300000002' + '010102030e0e7d99f92a99fb7695f773' + '8b06f79a93fc7c8c077d99f85695f75e' + '9908fb6e8cf87393f7108b09a70adf0b' + 'f78e14';
+    var fontArr = [];
+    for (var i = 0, ii = exampleFont.length; i < ii; i += 2) {
+      var hex = exampleFont.substr(i, 2);
+      fontArr.push(parseInt(hex, 16));
+    }
+    fontData = new Stream(fontArr);
+    done();
+  });
+  afterAll(function () {
+    fontData = null;
+  });
+  beforeEach(function (done) {
+    parser = new CFFParser(fontData, {}, SEAC_ANALYSIS_ENABLED);
+    cff = parser.parse();
+    done();
+  });
+  afterEach(function (done) {
+    parser = cff = null;
+    done();
+  });
+  it('parses header', function () {
+    var header = cff.header;
+    expect(header.major).toEqual(1);
+    expect(header.minor).toEqual(0);
+    expect(header.hdrSize).toEqual(4);
+    expect(header.offSize).toEqual(1);
+  });
+  it('parses name index', function () {
+    var names = cff.names;
+    expect(names.length).toEqual(1);
+    expect(names[0]).toEqual('ABCDEF+Times-Roman');
+  });
+  it('sanitizes name index', function () {
+    var index = new CFFIndex();
+    index.add(['['.charCodeAt(0), 'a'.charCodeAt(0)]);
+    var names = parser.parseNameIndex(index);
+    expect(names).toEqual(['_a']);
+    index = new CFFIndex();
+    var longName = [];
+    for (var i = 0; i < 129; i++) {
+      longName.push(0);
+    }
+    index.add(longName);
+    names = parser.parseNameIndex(index);
+    expect(names[0].length).toEqual(127);
+  });
+  it('parses string index', function () {
+    var strings = cff.strings;
+    expect(strings.count).toEqual(3);
+    expect(strings.get(0)).toEqual('.notdef');
+    expect(strings.get(391)).toEqual('001.007');
+  });
+  it('parses top dict', function () {
+    var topDict = cff.topDict;
+    expect(topDict.getByName('version')).toEqual(391);
+    expect(topDict.getByName('FullName')).toEqual(392);
+    expect(topDict.getByName('FamilyName')).toEqual(393);
+    expect(topDict.getByName('Weight')).toEqual(389);
+    expect(topDict.getByName('UniqueID')).toEqual(28416);
+    expect(topDict.getByName('FontBBox')).toEqual([-168, -218, 1000, 898]);
+    expect(topDict.getByName('CharStrings')).toEqual(94);
+    expect(topDict.getByName('Private')).toEqual([45, 102]);
+  });
+  it('refuses to add topDict key with invalid value (bug 1068432)', function () {
+    var topDict = cff.topDict;
+    var defaultValue = topDict.getByName('UnderlinePosition');
+    topDict.setByKey(3075, [NaN]);
+    expect(topDict.getByName('UnderlinePosition')).toEqual(defaultValue);
+  });
+  it('ignores reserved commands in parseDict, and refuses to add privateDict ' + 'keys with invalid values (bug 1308536)', function () {
+    var bytes = new Uint8Array([64, 39, 31, 30, 252, 114, 137, 115, 79, 30, 197, 119, 2, 99, 127, 6]);
+    parser.bytes = bytes;
+    var topDict = cff.topDict;
+    topDict.setByName('Private', [bytes.length, 0]);
+    var parsePrivateDict = function () {
+      parser.parsePrivateDict(topDict);
+    };
+    expect(parsePrivateDict).not.toThrow();
+    var privateDict = topDict.privateDict;
+    expect(privateDict.getByName('BlueValues')).toBeNull();
+  });
+  it('parses a CharString having cntrmask', function () {
+    var bytes = new Uint8Array([0, 1, 1, 0, 38, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 1, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 149, 3, 20, 22, 22, 14]);
+    parser.bytes = bytes;
+    var charStringsIndex = parser.parseIndex(0).obj;
+    var charStrings = parser.parseCharStrings(charStringsIndex).charStrings;
+    expect(charStrings.count).toEqual(1);
+    expect(charStrings.get(0).length).toEqual(38);
+  });
+  it('parses a CharString endchar with 4 args w/seac enabled', function () {
+    var parser = new CFFParser(fontData, {}, true);
+    parser.parse();
+    var bytes = new Uint8Array([0, 1, 1, 0, 237, 247, 22, 247, 72, 204, 247, 86, 14]);
+    parser.bytes = bytes;
+    var charStringsIndex = parser.parseIndex(0).obj;
+    var result = parser.parseCharStrings(charStringsIndex);
+    expect(result.charStrings.count).toEqual(1);
+    expect(result.charStrings.get(0).length).toEqual(1);
+    expect(result.seacs.length).toEqual(1);
+    expect(result.seacs[0].length).toEqual(4);
+    expect(result.seacs[0][0]).toEqual(130);
+    expect(result.seacs[0][1]).toEqual(180);
+    expect(result.seacs[0][2]).toEqual(65);
+    expect(result.seacs[0][3]).toEqual(194);
+  });
+  it('parses a CharString endchar with 4 args w/seac disabled', function () {
+    var parser = new CFFParser(fontData, {}, false);
+    parser.parse();
+    var bytes = new Uint8Array([0, 1, 1, 0, 237, 247, 22, 247, 72, 204, 247, 86, 14]);
+    parser.bytes = bytes;
+    var charStringsIndex = parser.parseIndex(0).obj;
+    var result = parser.parseCharStrings(charStringsIndex);
+    expect(result.charStrings.count).toEqual(1);
+    expect(result.charStrings.get(0).length).toEqual(9);
+    expect(result.seacs.length).toEqual(0);
+  });
+  it('parses a CharString endchar no args', function () {
+    var bytes = new Uint8Array([0, 1, 1, 0, 14]);
+    parser.bytes = bytes;
+    var charStringsIndex = parser.parseIndex(0).obj;
+    var result = parser.parseCharStrings(charStringsIndex);
+    expect(result.charStrings.count).toEqual(1);
+    expect(result.charStrings.get(0)[0]).toEqual(14);
+    expect(result.seacs.length).toEqual(0);
+  });
+  it('parses predefined charsets', function () {
+    var charset = parser.parseCharsets(0, 0, null, true);
+    expect(charset.predefined).toEqual(true);
+  });
+  it('parses charset format 0', function () {
+    var bytes = new Uint8Array([0x00, 0x00, 0x00, 0x00, 0x00, 0x02]);
+    parser.bytes = bytes;
+    var charset = parser.parseCharsets(3, 2, new CFFStrings(), false);
+    expect(charset.charset[1]).toEqual('exclam');
+    charset = parser.parseCharsets(3, 2, new CFFStrings(), true);
+    expect(charset.charset[1]).toEqual(2);
+  });
+  it('parses charset format 1', function () {
+    var bytes = new Uint8Array([0x00, 0x00, 0x00, 0x01, 0x00, 0x08, 0x01]);
+    parser.bytes = bytes;
+    var charset = parser.parseCharsets(3, 2, new CFFStrings(), false);
+    expect(charset.charset).toEqual(['.notdef', 'quoteright', 'parenleft']);
+    charset = parser.parseCharsets(3, 2, new CFFStrings(), true);
+    expect(charset.charset).toEqual(['.notdef', 8, 9]);
+  });
+  it('parses charset format 2', function () {
+    var bytes = new Uint8Array([0x00, 0x00, 0x00, 0x02, 0x00, 0x08, 0x00, 0x01]);
+    parser.bytes = bytes;
+    var charset = parser.parseCharsets(3, 2, new CFFStrings(), false);
+    expect(charset.charset).toEqual(['.notdef', 'quoteright', 'parenleft']);
+    charset = parser.parseCharsets(3, 2, new CFFStrings(), true);
+    expect(charset.charset).toEqual(['.notdef', 8, 9]);
+  });
+  it('parses encoding format 0', function () {
+    var bytes = new Uint8Array([0x00, 0x00, 0x00, 0x01, 0x08]);
+    parser.bytes = bytes;
+    var encoding = parser.parseEncoding(2, {}, new CFFStrings(), null);
+    expect(encoding.encoding).toEqual(createWithNullProto({ 0x8: 1 }));
+  });
+  it('parses encoding format 1', function () {
+    var bytes = new Uint8Array([0x00, 0x00, 0x01, 0x01, 0x07, 0x01]);
+    parser.bytes = bytes;
+    var encoding = parser.parseEncoding(2, {}, new CFFStrings(), null);
+    expect(encoding.encoding).toEqual(createWithNullProto({
+      0x7: 0x01,
+      0x08: 0x02
+    }));
+  });
+  it('parses fdselect format 0', function () {
+    var bytes = new Uint8Array([0x00, 0x00, 0x01]);
+    parser.bytes = bytes.slice();
+    var fdSelect = parser.parseFDSelect(0, 2);
+    expect(fdSelect.fdSelect).toEqual([0, 1]);
+    expect(fdSelect.raw).toEqual(bytes);
+  });
+  it('parses fdselect format 3', function () {
+    var bytes = new Uint8Array([0x03, 0x00, 0x02, 0x00, 0x00, 0x09, 0x00, 0x02, 0x0a, 0x00, 0x04]);
+    parser.bytes = bytes.slice();
+    var fdSelect = parser.parseFDSelect(0, 4);
+    expect(fdSelect.fdSelect).toEqual([9, 9, 0xa, 0xa]);
+    expect(fdSelect.raw).toEqual(bytes);
+  });
+  it('parses invalid fdselect format 3 (bug 1146106)', function () {
+    var bytes = new Uint8Array([0x03, 0x00, 0x02, 0x00, 0x01, 0x09, 0x00, 0x02, 0x0a, 0x00, 0x04]);
+    parser.bytes = bytes.slice();
+    var fdSelect = parser.parseFDSelect(0, 4);
+    expect(fdSelect.fdSelect).toEqual([9, 9, 0xa, 0xa]);
+    bytes[3] = bytes[4] = 0x00;
+    expect(fdSelect.raw).toEqual(bytes);
+  });
 });
 describe('CFFCompiler', function () {
- it('encodes integers', function () {
-  var c = new CFFCompiler();
-  expect(c.encodeInteger(0)).toEqual([0x8b]);
-  expect(c.encodeInteger(100)).toEqual([0xef]);
-  expect(c.encodeInteger(-100)).toEqual([0x27]);
-  expect(c.encodeInteger(1000)).toEqual([
-   0xfa,
-   0x7c
-  ]);
-  expect(c.encodeInteger(-1000)).toEqual([
-   0xfe,
-   0x7c
-  ]);
-  expect(c.encodeInteger(10000)).toEqual([
-   0x1c,
-   0x27,
-   0x10
-  ]);
-  expect(c.encodeInteger(-10000)).toEqual([
-   0x1c,
-   0xd8,
-   0xf0
-  ]);
-  expect(c.encodeInteger(100000)).toEqual([
-   0x1d,
-   0x00,
-   0x01,
-   0x86,
-   0xa0
-  ]);
-  expect(c.encodeInteger(-100000)).toEqual([
-   0x1d,
-   0xff,
-   0xfe,
-   0x79,
-   0x60
-  ]);
- });
- it('encodes floats', function () {
-  var c = new CFFCompiler();
-  expect(c.encodeFloat(-2.25)).toEqual([
-   0x1e,
-   0xe2,
-   0xa2,
-   0x5f
-  ]);
-  expect(c.encodeFloat(5e-11)).toEqual([
-   0x1e,
-   0x5c,
-   0x11,
-   0xff
-  ]);
- });
+  it('encodes integers', function () {
+    var c = new CFFCompiler();
+    expect(c.encodeInteger(0)).toEqual([0x8b]);
+    expect(c.encodeInteger(100)).toEqual([0xef]);
+    expect(c.encodeInteger(-100)).toEqual([0x27]);
+    expect(c.encodeInteger(1000)).toEqual([0xfa, 0x7c]);
+    expect(c.encodeInteger(-1000)).toEqual([0xfe, 0x7c]);
+    expect(c.encodeInteger(10000)).toEqual([0x1c, 0x27, 0x10]);
+    expect(c.encodeInteger(-10000)).toEqual([0x1c, 0xd8, 0xf0]);
+    expect(c.encodeInteger(100000)).toEqual([0x1d, 0x00, 0x01, 0x86, 0xa0]);
+    expect(c.encodeInteger(-100000)).toEqual([0x1d, 0xff, 0xfe, 0x79, 0x60]);
+  });
+  it('encodes floats', function () {
+    var c = new CFFCompiler();
+    expect(c.encodeFloat(-2.25)).toEqual([0x1e, 0xe2, 0xa2, 0x5f]);
+    expect(c.encodeFloat(5e-11)).toEqual([0x1e, 0x5c, 0x11, 0xff]);
+  });
 });

+ 185 - 184
lib/test/unit/cmap_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreCMap = require('../../core/cmap.js');
 var corePrimitives = require('../../core/primitives.js');
 var coreStream = require('../../core/stream.js');
@@ -28,205 +29,205 @@ var DOMCMapReaderFactory = displayDOMUtils.DOMCMapReaderFactory;
 var isNodeJS = sharedUtil.isNodeJS;
 var NodeCMapReaderFactory = testUnitTestUtils.NodeCMapReaderFactory;
 var cMapUrl = {
- dom: '../../external/bcmaps/',
- node: './external/bcmaps/'
+  dom: '../../external/bcmaps/',
+  node: './external/bcmaps/'
 };
 var cMapPacked = true;
 describe('cmap', function () {
- var fetchBuiltInCMap;
- beforeAll(function (done) {
-  var CMapReaderFactory;
-  if (isNodeJS()) {
-   CMapReaderFactory = new NodeCMapReaderFactory({
-    baseUrl: cMapUrl.node,
-    isCompressed: cMapPacked
-   });
-  } else {
-   CMapReaderFactory = new DOMCMapReaderFactory({
-    baseUrl: cMapUrl.dom,
-    isCompressed: cMapPacked
-   });
-  }
-  fetchBuiltInCMap = function (name) {
-   return CMapReaderFactory.fetch({ name: name });
-  };
-  done();
- });
- afterAll(function () {
-  fetchBuiltInCMap = null;
- });
- it('parses beginbfchar', function (done) {
-  var str = '2 beginbfchar\n' + '<03> <00>\n' + '<04> <01>\n' + 'endbfchar\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({ encoding: stream });
-  cmapPromise.then(function (cmap) {
-   expect(cmap.lookup(0x03)).toEqual(String.fromCharCode(0x00));
-   expect(cmap.lookup(0x04)).toEqual(String.fromCharCode(0x01));
-   expect(cmap.lookup(0x05)).toBeUndefined();
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  var fetchBuiltInCMap;
+  beforeAll(function (done) {
+    var CMapReaderFactory;
+    if (isNodeJS()) {
+      CMapReaderFactory = new NodeCMapReaderFactory({
+        baseUrl: cMapUrl.node,
+        isCompressed: cMapPacked
+      });
+    } else {
+      CMapReaderFactory = new DOMCMapReaderFactory({
+        baseUrl: cMapUrl.dom,
+        isCompressed: cMapPacked
+      });
+    }
+    fetchBuiltInCMap = function (name) {
+      return CMapReaderFactory.fetch({ name: name });
+    };
+    done();
   });
- });
- it('parses beginbfrange with range', function (done) {
-  var str = '1 beginbfrange\n' + '<06> <0B> 0\n' + 'endbfrange\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({ encoding: stream });
-  cmapPromise.then(function (cmap) {
-   expect(cmap.lookup(0x05)).toBeUndefined();
-   expect(cmap.lookup(0x06)).toEqual(String.fromCharCode(0x00));
-   expect(cmap.lookup(0x0B)).toEqual(String.fromCharCode(0x05));
-   expect(cmap.lookup(0x0C)).toBeUndefined();
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  afterAll(function () {
+    fetchBuiltInCMap = null;
   });
- });
- it('parses beginbfrange with array', function (done) {
-  var str = '1 beginbfrange\n' + '<0D> <12> [ 0 1 2 3 4 5 ]\n' + 'endbfrange\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({ encoding: stream });
-  cmapPromise.then(function (cmap) {
-   expect(cmap.lookup(0x0C)).toBeUndefined();
-   expect(cmap.lookup(0x0D)).toEqual(0x00);
-   expect(cmap.lookup(0x12)).toEqual(0x05);
-   expect(cmap.lookup(0x13)).toBeUndefined();
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('parses beginbfchar', function (done) {
+    var str = '2 beginbfchar\n' + '<03> <00>\n' + '<04> <01>\n' + 'endbfchar\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({ encoding: stream });
+    cmapPromise.then(function (cmap) {
+      expect(cmap.lookup(0x03)).toEqual(String.fromCharCode(0x00));
+      expect(cmap.lookup(0x04)).toEqual(String.fromCharCode(0x01));
+      expect(cmap.lookup(0x05)).toBeUndefined();
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
- it('parses begincidchar', function (done) {
-  var str = '1 begincidchar\n' + '<14> 0\n' + 'endcidchar\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({ encoding: stream });
-  cmapPromise.then(function (cmap) {
-   expect(cmap.lookup(0x14)).toEqual(0x00);
-   expect(cmap.lookup(0x15)).toBeUndefined();
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('parses beginbfrange with range', function (done) {
+    var str = '1 beginbfrange\n' + '<06> <0B> 0\n' + 'endbfrange\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({ encoding: stream });
+    cmapPromise.then(function (cmap) {
+      expect(cmap.lookup(0x05)).toBeUndefined();
+      expect(cmap.lookup(0x06)).toEqual(String.fromCharCode(0x00));
+      expect(cmap.lookup(0x0B)).toEqual(String.fromCharCode(0x05));
+      expect(cmap.lookup(0x0C)).toBeUndefined();
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
- it('parses begincidrange', function (done) {
-  var str = '1 begincidrange\n' + '<0016> <001B>   0\n' + 'endcidrange\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({ encoding: stream });
-  cmapPromise.then(function (cmap) {
-   expect(cmap.lookup(0x15)).toBeUndefined();
-   expect(cmap.lookup(0x16)).toEqual(0x00);
-   expect(cmap.lookup(0x1B)).toEqual(0x05);
-   expect(cmap.lookup(0x1C)).toBeUndefined();
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('parses beginbfrange with array', function (done) {
+    var str = '1 beginbfrange\n' + '<0D> <12> [ 0 1 2 3 4 5 ]\n' + 'endbfrange\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({ encoding: stream });
+    cmapPromise.then(function (cmap) {
+      expect(cmap.lookup(0x0C)).toBeUndefined();
+      expect(cmap.lookup(0x0D)).toEqual(0x00);
+      expect(cmap.lookup(0x12)).toEqual(0x05);
+      expect(cmap.lookup(0x13)).toBeUndefined();
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
- it('decodes codespace ranges', function (done) {
-  var str = '1 begincodespacerange\n' + '<01> <02>\n' + '<00000003> <00000004>\n' + 'endcodespacerange\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({ encoding: stream });
-  cmapPromise.then(function (cmap) {
-   var c = {};
-   cmap.readCharCode(String.fromCharCode(1), 0, c);
-   expect(c.charcode).toEqual(1);
-   expect(c.length).toEqual(1);
-   cmap.readCharCode(String.fromCharCode(0, 0, 0, 3), 0, c);
-   expect(c.charcode).toEqual(3);
-   expect(c.length).toEqual(4);
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('parses begincidchar', function (done) {
+    var str = '1 begincidchar\n' + '<14> 0\n' + 'endcidchar\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({ encoding: stream });
+    cmapPromise.then(function (cmap) {
+      expect(cmap.lookup(0x14)).toEqual(0x00);
+      expect(cmap.lookup(0x15)).toBeUndefined();
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
- it('decodes 4 byte codespace ranges', function (done) {
-  var str = '1 begincodespacerange\n' + '<8EA1A1A1> <8EA1FEFE>\n' + 'endcodespacerange\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({ encoding: stream });
-  cmapPromise.then(function (cmap) {
-   var c = {};
-   cmap.readCharCode(String.fromCharCode(0x8E, 0xA1, 0xA1, 0xA1), 0, c);
-   expect(c.charcode).toEqual(0x8EA1A1A1);
-   expect(c.length).toEqual(4);
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('parses begincidrange', function (done) {
+    var str = '1 begincidrange\n' + '<0016> <001B>   0\n' + 'endcidrange\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({ encoding: stream });
+    cmapPromise.then(function (cmap) {
+      expect(cmap.lookup(0x15)).toBeUndefined();
+      expect(cmap.lookup(0x16)).toEqual(0x00);
+      expect(cmap.lookup(0x1B)).toEqual(0x05);
+      expect(cmap.lookup(0x1C)).toBeUndefined();
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
- it('read usecmap', function (done) {
-  var str = '/Adobe-Japan1-1 usecmap\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({
-   encoding: stream,
-   fetchBuiltInCMap: fetchBuiltInCMap,
-   useCMap: null
+  it('decodes codespace ranges', function (done) {
+    var str = '1 begincodespacerange\n' + '<01> <02>\n' + '<00000003> <00000004>\n' + 'endcodespacerange\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({ encoding: stream });
+    cmapPromise.then(function (cmap) {
+      var c = {};
+      cmap.readCharCode(String.fromCharCode(1), 0, c);
+      expect(c.charcode).toEqual(1);
+      expect(c.length).toEqual(1);
+      cmap.readCharCode(String.fromCharCode(0, 0, 0, 3), 0, c);
+      expect(c.charcode).toEqual(3);
+      expect(c.length).toEqual(4);
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
-  cmapPromise.then(function (cmap) {
-   expect(cmap instanceof CMap).toEqual(true);
-   expect(cmap.useCMap).not.toBeNull();
-   expect(cmap.builtInCMap).toBeFalsy();
-   expect(cmap.length).toEqual(0x20A7);
-   expect(cmap.isIdentityCMap).toEqual(false);
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('decodes 4 byte codespace ranges', function (done) {
+    var str = '1 begincodespacerange\n' + '<8EA1A1A1> <8EA1FEFE>\n' + 'endcodespacerange\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({ encoding: stream });
+    cmapPromise.then(function (cmap) {
+      var c = {};
+      cmap.readCharCode(String.fromCharCode(0x8E, 0xA1, 0xA1, 0xA1), 0, c);
+      expect(c.charcode).toEqual(0x8EA1A1A1);
+      expect(c.length).toEqual(4);
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
- it('parses cmapname', function (done) {
-  var str = '/CMapName /Identity-H def\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({ encoding: stream });
-  cmapPromise.then(function (cmap) {
-   expect(cmap.name).toEqual('Identity-H');
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('read usecmap', function (done) {
+    var str = '/Adobe-Japan1-1 usecmap\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({
+      encoding: stream,
+      fetchBuiltInCMap: fetchBuiltInCMap,
+      useCMap: null
+    });
+    cmapPromise.then(function (cmap) {
+      expect(cmap instanceof CMap).toEqual(true);
+      expect(cmap.useCMap).not.toBeNull();
+      expect(cmap.builtInCMap).toBeFalsy();
+      expect(cmap.length).toEqual(0x20A7);
+      expect(cmap.isIdentityCMap).toEqual(false);
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
- it('parses wmode', function (done) {
-  var str = '/WMode 1 def\n';
-  var stream = new StringStream(str);
-  var cmapPromise = CMapFactory.create({ encoding: stream });
-  cmapPromise.then(function (cmap) {
-   expect(cmap.vertical).toEqual(true);
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('parses cmapname', function (done) {
+    var str = '/CMapName /Identity-H def\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({ encoding: stream });
+    cmapPromise.then(function (cmap) {
+      expect(cmap.name).toEqual('Identity-H');
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
- it('loads built in cmap', function (done) {
-  var cmapPromise = CMapFactory.create({
-   encoding: Name.get('Adobe-Japan1-1'),
-   fetchBuiltInCMap: fetchBuiltInCMap,
-   useCMap: null
+  it('parses wmode', function (done) {
+    var str = '/WMode 1 def\n';
+    var stream = new StringStream(str);
+    var cmapPromise = CMapFactory.create({ encoding: stream });
+    cmapPromise.then(function (cmap) {
+      expect(cmap.vertical).toEqual(true);
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
-  cmapPromise.then(function (cmap) {
-   expect(cmap instanceof CMap).toEqual(true);
-   expect(cmap.useCMap).toBeNull();
-   expect(cmap.builtInCMap).toBeTruthy();
-   expect(cmap.length).toEqual(0x20A7);
-   expect(cmap.isIdentityCMap).toEqual(false);
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('loads built in cmap', function (done) {
+    var cmapPromise = CMapFactory.create({
+      encoding: Name.get('Adobe-Japan1-1'),
+      fetchBuiltInCMap: fetchBuiltInCMap,
+      useCMap: null
+    });
+    cmapPromise.then(function (cmap) {
+      expect(cmap instanceof CMap).toEqual(true);
+      expect(cmap.useCMap).toBeNull();
+      expect(cmap.builtInCMap).toBeTruthy();
+      expect(cmap.length).toEqual(0x20A7);
+      expect(cmap.isIdentityCMap).toEqual(false);
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
- it('loads built in identity cmap', function (done) {
-  var cmapPromise = CMapFactory.create({
-   encoding: Name.get('Identity-H'),
-   fetchBuiltInCMap: fetchBuiltInCMap,
-   useCMap: null
+  it('loads built in identity cmap', function (done) {
+    var cmapPromise = CMapFactory.create({
+      encoding: Name.get('Identity-H'),
+      fetchBuiltInCMap: fetchBuiltInCMap,
+      useCMap: null
+    });
+    cmapPromise.then(function (cmap) {
+      expect(cmap instanceof IdentityCMap).toEqual(true);
+      expect(cmap.vertical).toEqual(false);
+      expect(cmap.length).toEqual(0x10000);
+      expect(function () {
+        return cmap.isIdentityCMap;
+      }).toThrow(new Error('should not access .isIdentityCMap'));
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
-  cmapPromise.then(function (cmap) {
-   expect(cmap instanceof IdentityCMap).toEqual(true);
-   expect(cmap.vertical).toEqual(false);
-   expect(cmap.length).toEqual(0x10000);
-   expect(function () {
-    return cmap.isIdentityCMap;
-   }).toThrow(new Error('should not access .isIdentityCMap'));
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
-  });
- });
 });

+ 486 - 1191
lib/test/unit/crypto_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreCrypto = require('../../core/crypto.js');
 var corePrimitives = require('../../core/primitives.js');
 var sharedUtil = require('../../shared/util.js');
@@ -32,1204 +33,498 @@ var stringToBytes = sharedUtil.stringToBytes;
 var PasswordException = sharedUtil.PasswordException;
 var PasswordResponses = sharedUtil.PasswordResponses;
 describe('crypto', function () {
- function hex2binary(s) {
-  var digits = '0123456789ABCDEF';
-  s = s.toUpperCase();
-  var n = s.length >> 1, i, j;
-  var result = new Uint8Array(n);
-  for (i = 0, j = 0; i < n; ++i) {
-   var d1 = s.charAt(j++);
-   var d2 = s.charAt(j++);
-   var value = digits.indexOf(d1) << 4 | digits.indexOf(d2);
-   result[i] = value;
+  function hex2binary(s) {
+    var digits = '0123456789ABCDEF';
+    s = s.toUpperCase();
+    var n = s.length >> 1,
+        i,
+        j;
+    var result = new Uint8Array(n);
+    for (i = 0, j = 0; i < n; ++i) {
+      var d1 = s.charAt(j++);
+      var d2 = s.charAt(j++);
+      var value = digits.indexOf(d1) << 4 | digits.indexOf(d2);
+      result[i] = value;
+    }
+    return result;
   }
-  return result;
- }
- describe('calculateMD5', function () {
-  it('should pass RFC 1321 test #1', function () {
-   var input, result, expected;
-   input = stringToBytes('');
-   result = calculateMD5(input, 0, input.length);
-   expected = hex2binary('d41d8cd98f00b204e9800998ecf8427e');
-   expect(result).toEqual(expected);
+  describe('calculateMD5', function () {
+    it('should pass RFC 1321 test #1', function () {
+      var input, result, expected;
+      input = stringToBytes('');
+      result = calculateMD5(input, 0, input.length);
+      expected = hex2binary('d41d8cd98f00b204e9800998ecf8427e');
+      expect(result).toEqual(expected);
+    });
+    it('should pass RFC 1321 test #2', function () {
+      var input, result, expected;
+      input = stringToBytes('a');
+      result = calculateMD5(input, 0, input.length);
+      expected = hex2binary('0cc175b9c0f1b6a831c399e269772661');
+      expect(result).toEqual(expected);
+    });
+    it('should pass RFC 1321 test #3', function () {
+      var input, result, expected;
+      input = stringToBytes('abc');
+      result = calculateMD5(input, 0, input.length);
+      expected = hex2binary('900150983cd24fb0d6963f7d28e17f72');
+      expect(result).toEqual(expected);
+    });
+    it('should pass RFC 1321 test #4', function () {
+      var input, result, expected;
+      input = stringToBytes('message digest');
+      result = calculateMD5(input, 0, input.length);
+      expected = hex2binary('f96b697d7cb7938d525a2f31aaf161d0');
+      expect(result).toEqual(expected);
+    });
+    it('should pass RFC 1321 test #5', function () {
+      var input, result, expected;
+      input = stringToBytes('abcdefghijklmnopqrstuvwxyz');
+      result = calculateMD5(input, 0, input.length);
+      expected = hex2binary('c3fcd3d76192e4007dfb496cca67e13b');
+      expect(result).toEqual(expected);
+    });
+    it('should pass RFC 1321 test #6', function () {
+      var input, result, expected;
+      input = stringToBytes('ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuv' + 'wxyz0123456789');
+      result = calculateMD5(input, 0, input.length);
+      expected = hex2binary('d174ab98d277d9f5a5611c2c9f419d9f');
+      expect(result).toEqual(expected);
+    });
+    it('should pass RFC 1321 test #7', function () {
+      var input, result, expected;
+      input = stringToBytes('123456789012345678901234567890123456789012345678' + '90123456789012345678901234567890');
+      result = calculateMD5(input, 0, input.length);
+      expected = hex2binary('57edf4a22be3c955ac49da2e2107b67a');
+      expect(result).toEqual(expected);
+    });
+  });
+  describe('ARCFourCipher', function () {
+    it('should pass test #1', function () {
+      var key, input, result, expected, cipher;
+      key = hex2binary('0123456789abcdef');
+      input = hex2binary('0123456789abcdef');
+      cipher = new ARCFourCipher(key);
+      result = cipher.encryptBlock(input);
+      expected = hex2binary('75b7878099e0c596');
+      expect(result).toEqual(expected);
+    });
+    it('should pass test #2', function () {
+      var key, input, result, expected, cipher;
+      key = hex2binary('0123456789abcdef');
+      input = hex2binary('0000000000000000');
+      cipher = new ARCFourCipher(key);
+      result = cipher.encryptBlock(input);
+      expected = hex2binary('7494c2e7104b0879');
+      expect(result).toEqual(expected);
+    });
+    it('should pass test #3', function () {
+      var key, input, result, expected, cipher;
+      key = hex2binary('0000000000000000');
+      input = hex2binary('0000000000000000');
+      cipher = new ARCFourCipher(key);
+      result = cipher.encryptBlock(input);
+      expected = hex2binary('de188941a3375d3a');
+      expect(result).toEqual(expected);
+    });
+    it('should pass test #4', function () {
+      var key, input, result, expected, cipher;
+      key = hex2binary('ef012345');
+      input = hex2binary('00000000000000000000');
+      cipher = new ARCFourCipher(key);
+      result = cipher.encryptBlock(input);
+      expected = hex2binary('d6a141a7ec3c38dfbd61');
+      expect(result).toEqual(expected);
+    });
+    it('should pass test #5', function () {
+      var key, input, result, expected, cipher;
+      key = hex2binary('0123456789abcdef');
+      input = hex2binary('010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '101010101010101010101');
+      cipher = new ARCFourCipher(key);
+      result = cipher.encryptBlock(input);
+      expected = hex2binary('7595c3e6114a09780c4ad452338e1ffd9a1be9498f813d76' + '533449b6778dcad8c78a8d2ba9ac66085d0e53d59c26c2d1c490c1ebbe0ce66d1b6b' + '1b13b6b919b847c25a91447a95e75e4ef16779cde8bf0a95850e32af9689444fd377' + '108f98fdcbd4e726567500990bcc7e0ca3c4aaa304a387d20f3b8fbbcd42a1bd311d' + '7a4303dda5ab078896ae80c18b0af66dff319616eb784e495ad2ce90d7f772a81747' + 'b65f62093b1e0db9e5ba532fafec47508323e671327df9444432cb7367cec82f5d44' + 'c0d00b67d650a075cd4b70dedd77eb9b10231b6b5b741347396d62897421d43df9b4' + '2e446e358e9c11a9b2184ecbef0cd8e7a877ef968f1390ec9b3d35a5585cb009290e' + '2fcde7b5ec66d9084be44055a619d9dd7fc3166f9487f7cb272912426445998514c1' + '5d53a18c864ce3a2b7555793988126520eacf2e3066e230c91bee4dd5304f5fd0405' + 'b35bd99c73135d3d9bc335ee049ef69b3867bf2d7bd1eaa595d8bfc0066ff8d31509' + 'eb0c6caa006c807a623ef84c3d33c195d23ee320c40de0558157c822d4b8c569d849' + 'aed59d4e0fd7f379586b4b7ff684ed6a189f7486d49b9c4bad9ba24b96abf924372c' + '8a8fffb10d55354900a77a3db5f205e1b99fcd8660863a159ad4abe40fa48934163d' + 'dde542a6585540fd683cbfd8c00f12129a284deacc4cdefe58be7137541c047126c8' + 'd49e2755ab181ab7e940b0c0');
+      expect(result).toEqual(expected);
+    });
+    it('should pass test #6', function () {
+      var key, input, result, expected, cipher;
+      key = hex2binary('fb029e3031323334');
+      input = hex2binary('aaaa0300000008004500004e661a00008011be640a0001220af' + 'fffff00890089003a000080a601100001000000000000204543454a4548454346434' + '550464545494546464343414341434143414341414100002000011bd0b604');
+      cipher = new ARCFourCipher(key);
+      result = cipher.encryptBlock(input);
+      expected = hex2binary('f69c5806bd6ce84626bcbefb9474650aad1f7909b0f64d5f' + '58a503a258b7ed22eb0ea64930d3a056a55742fcce141d485f8aa836dea18df42c53' + '80805ad0c61a5d6f58f41040b24b7d1a693856ed0d4398e7aee3bf0e2a2ca8f7');
+      expect(result).toEqual(expected);
+    });
+    it('should pass test #7', function () {
+      var key, input, result, expected, cipher;
+      key = hex2binary('0123456789abcdef');
+      input = hex2binary('123456789abcdef0123456789abcdef0123456789abcdef0123' + '45678');
+      cipher = new ARCFourCipher(key);
+      result = cipher.encryptBlock(input);
+      expected = hex2binary('66a0949f8af7d6891f7f832ba833c00c892ebe30143ce287' + '40011ecf');
+      expect(result).toEqual(expected);
+    });
+  });
+  describe('calculateSHA256', function () {
+    it('should properly hash abc', function () {
+      var input, result, expected;
+      input = stringToBytes('abc');
+      result = calculateSHA256(input, 0, input.length);
+      expected = hex2binary('BA7816BF8F01CFEA414140DE5DAE2223B00361A396177A9C' + 'B410FF61F20015AD');
+      expect(result).toEqual(expected);
+    });
+    it('should properly hash a multiblock input', function () {
+      var input, result, expected;
+      input = stringToBytes('abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmno' + 'mnopnopq');
+      result = calculateSHA256(input, 0, input.length);
+      expected = hex2binary('248D6A61D20638B8E5C026930C3E6039A33CE45964FF2167' + 'F6ECEDD419DB06C1');
+      expect(result).toEqual(expected);
+    });
+  });
+  describe('calculateSHA384', function () {
+    it('should properly hash abc', function () {
+      var input, result, expected;
+      input = stringToBytes('abc');
+      result = calculateSHA384(input, 0, input.length);
+      expected = hex2binary('CB00753F45A35E8BB5A03D699AC65007272C32AB0EDED163' + '1A8B605A43FF5BED8086072BA1E7CC2358BAECA134C825A7');
+      expect(result).toEqual(expected);
+    });
+    it('should properly hash a multiblock input', function () {
+      var input, result, expected;
+      input = stringToBytes('abcdefghbcdefghicdefghijdefghijkefghijklfghijklm' + 'ghijklmnhijklmnoijklmnopjklmnopqklmnopqrlmnopqrs' + 'mnopqrstnopqrstu');
+      result = calculateSHA384(input, 0, input.length);
+      expected = hex2binary('09330C33F71147E83D192FC782CD1B4753111B173B3B05D2' + '2FA08086E3B0F712FCC7C71A557E2DB966C3E9FA91746039');
+      expect(result).toEqual(expected);
+    });
+  });
+  describe('calculateSHA512', function () {
+    it('should properly hash abc', function () {
+      var input, result, expected;
+      input = stringToBytes('abc');
+      result = calculateSHA512(input, 0, input.length);
+      expected = hex2binary('DDAF35A193617ABACC417349AE20413112E6FA4E89A97EA2' + '0A9EEEE64B55D39A2192992A274FC1A836BA3C23A3FEEBBD' + '454D4423643CE80E2A9AC94FA54CA49F');
+      expect(result).toEqual(expected);
+    });
+    it('should properly hash a multiblock input', function () {
+      var input, result, expected;
+      input = stringToBytes('abcdefghbcdefghicdefghijdefghijkefghijklfghijklm' + 'ghijklmnhijklmnoijklmnopjklmnopqklmnopqrlmnopqrs' + 'mnopqrstnopqrstu');
+      result = calculateSHA512(input, 0, input.length);
+      expected = hex2binary('8E959B75DAE313DA8CF4F72814FC143F8F7779C6EB9F7FA1' + '7299AEADB6889018501D289E4900F7E4331B99DEC4B5433A' + 'C7D329EEB6DD26545E96E55B874BE909');
+      expect(result).toEqual(expected);
+    });
+  });
+  describe('AES128', function () {
+    describe('Encryption', function () {
+      it('should be able to encrypt a block', function () {
+        var input, key, result, expected, iv, cipher;
+        input = hex2binary('00112233445566778899aabbccddeeff');
+        key = hex2binary('000102030405060708090a0b0c0d0e0f');
+        iv = hex2binary('00000000000000000000000000000000');
+        cipher = new AES128Cipher(key);
+        result = cipher.encrypt(input, iv);
+        expected = hex2binary('69c4e0d86a7b0430d8cdb78070b4c55a');
+        expect(result).toEqual(expected);
+      });
+    });
+    describe('Decryption', function () {
+      it('should be able to decrypt a block with IV in stream', function () {
+        var input, key, result, expected, cipher;
+        input = hex2binary('0000000000000000000000000000000069c4e0d86a7b0430d' + '8cdb78070b4c55a');
+        key = hex2binary('000102030405060708090a0b0c0d0e0f');
+        cipher = new AES128Cipher(key);
+        result = cipher.decryptBlock(input);
+        expected = hex2binary('00112233445566778899aabbccddeeff');
+        expect(result).toEqual(expected);
+      });
+    });
+  });
+  describe('AES256', function () {
+    describe('Encryption', function () {
+      it('should be able to encrypt a block', function () {
+        var input, key, result, expected, iv, cipher;
+        input = hex2binary('00112233445566778899aabbccddeeff');
+        key = hex2binary('000102030405060708090a0b0c0d0e0f101112131415161718' + '191a1b1c1d1e1f');
+        iv = hex2binary('00000000000000000000000000000000');
+        cipher = new AES256Cipher(key);
+        result = cipher.encrypt(input, iv);
+        expected = hex2binary('8ea2b7ca516745bfeafc49904b496089');
+        expect(result).toEqual(expected);
+      });
+    });
+    describe('Decryption', function () {
+      it('should be able to decrypt a block with specified iv', function () {
+        var input, key, result, expected, cipher, iv;
+        input = hex2binary('8ea2b7ca516745bfeafc49904b496089');
+        key = hex2binary('000102030405060708090a0b0c0d0e0f101112131415161718' + '191a1b1c1d1e1f');
+        iv = hex2binary('00000000000000000000000000000000');
+        cipher = new AES256Cipher(key);
+        result = cipher.decryptBlock(input, false, iv);
+        expected = hex2binary('00112233445566778899aabbccddeeff');
+        expect(result).toEqual(expected);
+      });
+      it('should be able to decrypt a block with IV in stream', function () {
+        var input, key, result, expected, cipher;
+        input = hex2binary('000000000000000000000000000000008ea2b7ca516745bf' + 'eafc49904b496089');
+        key = hex2binary('000102030405060708090a0b0c0d0e0f101112131415161718' + '191a1b1c1d1e1f');
+        cipher = new AES256Cipher(key);
+        result = cipher.decryptBlock(input, false);
+        expected = hex2binary('00112233445566778899aabbccddeeff');
+        expect(result).toEqual(expected);
+      });
+    });
+  });
+  describe('PDF17Algorithm', function () {
+    it('should correctly check a user key', function () {
+      var password, userValidation, userPassword, alg, result;
+      alg = new PDF17();
+      password = new Uint8Array([117, 115, 101, 114]);
+      userValidation = new Uint8Array([117, 169, 4, 32, 159, 101, 22, 220]);
+      userPassword = new Uint8Array([131, 242, 143, 160, 87, 2, 138, 134, 79, 253, 189, 173, 224, 73, 144, 241, 190, 81, 197, 15, 249, 105, 145, 151, 15, 194, 65, 3, 1, 126, 187, 221]);
+      result = alg.checkUserPassword(password, userValidation, userPassword);
+      expect(result).toEqual(true);
+    });
+    it('should correctly check an owner key', function () {
+      var password, ownerValidation, ownerPassword, alg, result, uBytes;
+      alg = new PDF17();
+      password = new Uint8Array([111, 119, 110, 101, 114]);
+      ownerValidation = new Uint8Array([243, 118, 71, 153, 128, 17, 101, 62]);
+      ownerPassword = new Uint8Array([60, 98, 137, 35, 51, 101, 200, 152, 210, 178, 226, 228, 134, 205, 163, 24, 204, 126, 177, 36, 106, 50, 36, 125, 210, 172, 171, 120, 222, 108, 139, 115]);
+      uBytes = new Uint8Array([131, 242, 143, 160, 87, 2, 138, 134, 79, 253, 189, 173, 224, 73, 144, 241, 190, 81, 197, 15, 249, 105, 145, 151, 15, 194, 65, 3, 1, 126, 187, 221, 117, 169, 4, 32, 159, 101, 22, 220, 168, 94, 215, 192, 100, 38, 188, 40]);
+      result = alg.checkOwnerPassword(password, ownerValidation, uBytes, ownerPassword);
+      expect(result).toEqual(true);
+    });
+    it('should generate a file encryption key from the user key', function () {
+      var password, userKeySalt, expected, alg, result, userEncryption;
+      alg = new PDF17();
+      password = new Uint8Array([117, 115, 101, 114]);
+      userKeySalt = new Uint8Array([168, 94, 215, 192, 100, 38, 188, 40]);
+      userEncryption = new Uint8Array([35, 150, 195, 169, 245, 51, 51, 255, 158, 158, 33, 242, 231, 75, 125, 190, 25, 126, 172, 114, 195, 244, 137, 245, 234, 165, 42, 74, 60, 38, 17, 17]);
+      result = alg.getUserKey(password, userKeySalt, userEncryption);
+      expected = new Uint8Array([63, 114, 136, 209, 87, 61, 12, 30, 249, 1, 186, 144, 254, 248, 163, 153, 151, 51, 133, 10, 80, 152, 206, 15, 72, 187, 231, 33, 224, 239, 13, 213]);
+      expect(result).toEqual(expected);
+    });
+    it('should generate a file encryption key from the owner key', function () {
+      var password, ownerKeySalt, expected, alg, result, ownerEncryption;
+      var uBytes;
+      alg = new PDF17();
+      password = new Uint8Array([111, 119, 110, 101, 114]);
+      ownerKeySalt = new Uint8Array([200, 245, 242, 12, 218, 123, 24, 120]);
+      ownerEncryption = new Uint8Array([213, 202, 14, 189, 110, 76, 70, 191, 6, 195, 10, 190, 157, 100, 144, 85, 8, 62, 123, 178, 156, 229, 50, 40, 229, 216, 54, 222, 34, 38, 106, 223]);
+      uBytes = new Uint8Array([131, 242, 143, 160, 87, 2, 138, 134, 79, 253, 189, 173, 224, 73, 144, 241, 190, 81, 197, 15, 249, 105, 145, 151, 15, 194, 65, 3, 1, 126, 187, 221, 117, 169, 4, 32, 159, 101, 22, 220, 168, 94, 215, 192, 100, 38, 188, 40]);
+      result = alg.getOwnerKey(password, ownerKeySalt, uBytes, ownerEncryption);
+      expected = new Uint8Array([63, 114, 136, 209, 87, 61, 12, 30, 249, 1, 186, 144, 254, 248, 163, 153, 151, 51, 133, 10, 80, 152, 206, 15, 72, 187, 231, 33, 224, 239, 13, 213]);
+      expect(result).toEqual(expected);
+    });
+  });
+  describe('PDF20Algorithm', function () {
+    it('should correctly check a user key', function () {
+      var password, userValidation, userPassword, alg, result;
+      alg = new PDF20();
+      password = new Uint8Array([117, 115, 101, 114]);
+      userValidation = new Uint8Array([83, 245, 146, 101, 198, 247, 34, 198]);
+      userPassword = new Uint8Array([94, 230, 205, 75, 166, 99, 250, 76, 219, 128, 17, 85, 57, 17, 33, 164, 150, 46, 103, 176, 160, 156, 187, 233, 166, 223, 163, 253, 147, 235, 95, 184]);
+      result = alg.checkUserPassword(password, userValidation, userPassword);
+      expect(result).toEqual(true);
+    });
+    it('should correctly check an owner key', function () {
+      var password, ownerValidation, ownerPassword, alg, result, uBytes;
+      alg = new PDF20();
+      password = new Uint8Array([111, 119, 110, 101, 114]);
+      ownerValidation = new Uint8Array([142, 232, 169, 208, 202, 214, 5, 185]);
+      ownerPassword = new Uint8Array([88, 232, 62, 54, 245, 26, 245, 209, 137, 123, 221, 72, 199, 49, 37, 217, 31, 74, 115, 167, 127, 158, 176, 77, 45, 163, 87, 47, 39, 90, 217, 141]);
+      uBytes = new Uint8Array([94, 230, 205, 75, 166, 99, 250, 76, 219, 128, 17, 85, 57, 17, 33, 164, 150, 46, 103, 176, 160, 156, 187, 233, 166, 223, 163, 253, 147, 235, 95, 184, 83, 245, 146, 101, 198, 247, 34, 198, 191, 11, 16, 94, 237, 216, 20, 175]);
+      result = alg.checkOwnerPassword(password, ownerValidation, uBytes, ownerPassword);
+      expect(result).toEqual(true);
+    });
+    it('should generate a file encryption key from the user key', function () {
+      var password, userKeySalt, expected, alg, result, userEncryption;
+      alg = new PDF20();
+      password = new Uint8Array([117, 115, 101, 114]);
+      userKeySalt = new Uint8Array([191, 11, 16, 94, 237, 216, 20, 175]);
+      userEncryption = new Uint8Array([121, 208, 2, 181, 230, 89, 156, 60, 253, 143, 212, 28, 84, 180, 196, 177, 173, 128, 221, 107, 46, 20, 94, 186, 135, 51, 95, 24, 20, 223, 254, 36]);
+      result = alg.getUserKey(password, userKeySalt, userEncryption);
+      expected = new Uint8Array([42, 218, 213, 39, 73, 91, 72, 79, 67, 38, 248, 133, 18, 189, 61, 34, 107, 79, 29, 56, 59, 181, 213, 118, 113, 34, 65, 210, 87, 174, 22, 239]);
+      expect(result).toEqual(expected);
+    });
+    it('should generate a file encryption key from the owner key', function () {
+      var password, ownerKeySalt, expected, alg, result, ownerEncryption;
+      var uBytes;
+      alg = new PDF20();
+      password = new Uint8Array([111, 119, 110, 101, 114]);
+      ownerKeySalt = new Uint8Array([29, 208, 185, 46, 11, 76, 135, 149]);
+      ownerEncryption = new Uint8Array([209, 73, 224, 77, 103, 155, 201, 181, 190, 68, 223, 20, 62, 90, 56, 210, 5, 240, 178, 128, 238, 124, 68, 254, 253, 244, 62, 108, 208, 135, 10, 251]);
+      uBytes = new Uint8Array([94, 230, 205, 75, 166, 99, 250, 76, 219, 128, 17, 85, 57, 17, 33, 164, 150, 46, 103, 176, 160, 156, 187, 233, 166, 223, 163, 253, 147, 235, 95, 184, 83, 245, 146, 101, 198, 247, 34, 198, 191, 11, 16, 94, 237, 216, 20, 175]);
+      result = alg.getOwnerKey(password, ownerKeySalt, uBytes, ownerEncryption);
+      expected = new Uint8Array([42, 218, 213, 39, 73, 91, 72, 79, 67, 38, 248, 133, 18, 189, 61, 34, 107, 79, 29, 56, 59, 181, 213, 118, 113, 34, 65, 210, 87, 174, 22, 239]);
+      expect(result).toEqual(expected);
+    });
   });
-  it('should pass RFC 1321 test #2', function () {
-   var input, result, expected;
-   input = stringToBytes('a');
-   result = calculateMD5(input, 0, input.length);
-   expected = hex2binary('0cc175b9c0f1b6a831c399e269772661');
-   expect(result).toEqual(expected);
-  });
-  it('should pass RFC 1321 test #3', function () {
-   var input, result, expected;
-   input = stringToBytes('abc');
-   result = calculateMD5(input, 0, input.length);
-   expected = hex2binary('900150983cd24fb0d6963f7d28e17f72');
-   expect(result).toEqual(expected);
-  });
-  it('should pass RFC 1321 test #4', function () {
-   var input, result, expected;
-   input = stringToBytes('message digest');
-   result = calculateMD5(input, 0, input.length);
-   expected = hex2binary('f96b697d7cb7938d525a2f31aaf161d0');
-   expect(result).toEqual(expected);
-  });
-  it('should pass RFC 1321 test #5', function () {
-   var input, result, expected;
-   input = stringToBytes('abcdefghijklmnopqrstuvwxyz');
-   result = calculateMD5(input, 0, input.length);
-   expected = hex2binary('c3fcd3d76192e4007dfb496cca67e13b');
-   expect(result).toEqual(expected);
-  });
-  it('should pass RFC 1321 test #6', function () {
-   var input, result, expected;
-   input = stringToBytes('ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuv' + 'wxyz0123456789');
-   result = calculateMD5(input, 0, input.length);
-   expected = hex2binary('d174ab98d277d9f5a5611c2c9f419d9f');
-   expect(result).toEqual(expected);
-  });
-  it('should pass RFC 1321 test #7', function () {
-   var input, result, expected;
-   input = stringToBytes('123456789012345678901234567890123456789012345678' + '90123456789012345678901234567890');
-   result = calculateMD5(input, 0, input.length);
-   expected = hex2binary('57edf4a22be3c955ac49da2e2107b67a');
-   expect(result).toEqual(expected);
-  });
- });
- describe('ARCFourCipher', function () {
-  it('should pass test #1', function () {
-   var key, input, result, expected, cipher;
-   key = hex2binary('0123456789abcdef');
-   input = hex2binary('0123456789abcdef');
-   cipher = new ARCFourCipher(key);
-   result = cipher.encryptBlock(input);
-   expected = hex2binary('75b7878099e0c596');
-   expect(result).toEqual(expected);
-  });
-  it('should pass test #2', function () {
-   var key, input, result, expected, cipher;
-   key = hex2binary('0123456789abcdef');
-   input = hex2binary('0000000000000000');
-   cipher = new ARCFourCipher(key);
-   result = cipher.encryptBlock(input);
-   expected = hex2binary('7494c2e7104b0879');
-   expect(result).toEqual(expected);
-  });
-  it('should pass test #3', function () {
-   var key, input, result, expected, cipher;
-   key = hex2binary('0000000000000000');
-   input = hex2binary('0000000000000000');
-   cipher = new ARCFourCipher(key);
-   result = cipher.encryptBlock(input);
-   expected = hex2binary('de188941a3375d3a');
-   expect(result).toEqual(expected);
-  });
-  it('should pass test #4', function () {
-   var key, input, result, expected, cipher;
-   key = hex2binary('ef012345');
-   input = hex2binary('00000000000000000000');
-   cipher = new ARCFourCipher(key);
-   result = cipher.encryptBlock(input);
-   expected = hex2binary('d6a141a7ec3c38dfbd61');
-   expect(result).toEqual(expected);
-  });
-  it('should pass test #5', function () {
-   var key, input, result, expected, cipher;
-   key = hex2binary('0123456789abcdef');
-   input = hex2binary('010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '10101010101010101010101010101010101010101010101010101010101010101010' + '101010101010101010101');
-   cipher = new ARCFourCipher(key);
-   result = cipher.encryptBlock(input);
-   expected = hex2binary('7595c3e6114a09780c4ad452338e1ffd9a1be9498f813d76' + '533449b6778dcad8c78a8d2ba9ac66085d0e53d59c26c2d1c490c1ebbe0ce66d1b6b' + '1b13b6b919b847c25a91447a95e75e4ef16779cde8bf0a95850e32af9689444fd377' + '108f98fdcbd4e726567500990bcc7e0ca3c4aaa304a387d20f3b8fbbcd42a1bd311d' + '7a4303dda5ab078896ae80c18b0af66dff319616eb784e495ad2ce90d7f772a81747' + 'b65f62093b1e0db9e5ba532fafec47508323e671327df9444432cb7367cec82f5d44' + 'c0d00b67d650a075cd4b70dedd77eb9b10231b6b5b741347396d62897421d43df9b4' + '2e446e358e9c11a9b2184ecbef0cd8e7a877ef968f1390ec9b3d35a5585cb009290e' + '2fcde7b5ec66d9084be44055a619d9dd7fc3166f9487f7cb272912426445998514c1' + '5d53a18c864ce3a2b7555793988126520eacf2e3066e230c91bee4dd5304f5fd0405' + 'b35bd99c73135d3d9bc335ee049ef69b3867bf2d7bd1eaa595d8bfc0066ff8d31509' + 'eb0c6caa006c807a623ef84c3d33c195d23ee320c40de0558157c822d4b8c569d849' + 'aed59d4e0fd7f379586b4b7ff684ed6a189f7486d49b9c4bad9ba24b96abf924372c' + '8a8fffb10d55354900a77a3db5f205e1b99fcd8660863a159ad4abe40fa48934163d' + 'dde542a6585540fd683cbfd8c00f12129a284deacc4cdefe58be7137541c047126c8' + 'd49e2755ab181ab7e940b0c0');
-   expect(result).toEqual(expected);
-  });
-  it('should pass test #6', function () {
-   var key, input, result, expected, cipher;
-   key = hex2binary('fb029e3031323334');
-   input = hex2binary('aaaa0300000008004500004e661a00008011be640a0001220af' + 'fffff00890089003a000080a601100001000000000000204543454a4548454346434' + '550464545494546464343414341434143414341414100002000011bd0b604');
-   cipher = new ARCFourCipher(key);
-   result = cipher.encryptBlock(input);
-   expected = hex2binary('f69c5806bd6ce84626bcbefb9474650aad1f7909b0f64d5f' + '58a503a258b7ed22eb0ea64930d3a056a55742fcce141d485f8aa836dea18df42c53' + '80805ad0c61a5d6f58f41040b24b7d1a693856ed0d4398e7aee3bf0e2a2ca8f7');
-   expect(result).toEqual(expected);
-  });
-  it('should pass test #7', function () {
-   var key, input, result, expected, cipher;
-   key = hex2binary('0123456789abcdef');
-   input = hex2binary('123456789abcdef0123456789abcdef0123456789abcdef0123' + '45678');
-   cipher = new ARCFourCipher(key);
-   result = cipher.encryptBlock(input);
-   expected = hex2binary('66a0949f8af7d6891f7f832ba833c00c892ebe30143ce287' + '40011ecf');
-   expect(result).toEqual(expected);
-  });
- });
- describe('calculateSHA256', function () {
-  it('should properly hash abc', function () {
-   var input, result, expected;
-   input = stringToBytes('abc');
-   result = calculateSHA256(input, 0, input.length);
-   expected = hex2binary('BA7816BF8F01CFEA414140DE5DAE2223B00361A396177A9C' + 'B410FF61F20015AD');
-   expect(result).toEqual(expected);
-  });
-  it('should properly hash a multiblock input', function () {
-   var input, result, expected;
-   input = stringToBytes('abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmno' + 'mnopnopq');
-   result = calculateSHA256(input, 0, input.length);
-   expected = hex2binary('248D6A61D20638B8E5C026930C3E6039A33CE45964FF2167' + 'F6ECEDD419DB06C1');
-   expect(result).toEqual(expected);
-  });
- });
- describe('calculateSHA384', function () {
-  it('should properly hash abc', function () {
-   var input, result, expected;
-   input = stringToBytes('abc');
-   result = calculateSHA384(input, 0, input.length);
-   expected = hex2binary('CB00753F45A35E8BB5A03D699AC65007272C32AB0EDED163' + '1A8B605A43FF5BED8086072BA1E7CC2358BAECA134C825A7');
-   expect(result).toEqual(expected);
-  });
-  it('should properly hash a multiblock input', function () {
-   var input, result, expected;
-   input = stringToBytes('abcdefghbcdefghicdefghijdefghijkefghijklfghijklm' + 'ghijklmnhijklmnoijklmnopjklmnopqklmnopqrlmnopqrs' + 'mnopqrstnopqrstu');
-   result = calculateSHA384(input, 0, input.length);
-   expected = hex2binary('09330C33F71147E83D192FC782CD1B4753111B173B3B05D2' + '2FA08086E3B0F712FCC7C71A557E2DB966C3E9FA91746039');
-   expect(result).toEqual(expected);
-  });
- });
- describe('calculateSHA512', function () {
-  it('should properly hash abc', function () {
-   var input, result, expected;
-   input = stringToBytes('abc');
-   result = calculateSHA512(input, 0, input.length);
-   expected = hex2binary('DDAF35A193617ABACC417349AE20413112E6FA4E89A97EA2' + '0A9EEEE64B55D39A2192992A274FC1A836BA3C23A3FEEBBD' + '454D4423643CE80E2A9AC94FA54CA49F');
-   expect(result).toEqual(expected);
-  });
-  it('should properly hash a multiblock input', function () {
-   var input, result, expected;
-   input = stringToBytes('abcdefghbcdefghicdefghijdefghijkefghijklfghijklm' + 'ghijklmnhijklmnoijklmnopjklmnopqklmnopqrlmnopqrs' + 'mnopqrstnopqrstu');
-   result = calculateSHA512(input, 0, input.length);
-   expected = hex2binary('8E959B75DAE313DA8CF4F72814FC143F8F7779C6EB9F7FA1' + '7299AEADB6889018501D289E4900F7E4331B99DEC4B5433A' + 'C7D329EEB6DD26545E96E55B874BE909');
-   expect(result).toEqual(expected);
-  });
- });
- describe('AES128', function () {
-  describe('Encryption', function () {
-   it('should be able to encrypt a block', function () {
-    var input, key, result, expected, iv, cipher;
-    input = hex2binary('00112233445566778899aabbccddeeff');
-    key = hex2binary('000102030405060708090a0b0c0d0e0f');
-    iv = hex2binary('00000000000000000000000000000000');
-    cipher = new AES128Cipher(key);
-    result = cipher.encrypt(input, iv);
-    expected = hex2binary('69c4e0d86a7b0430d8cdb78070b4c55a');
-    expect(result).toEqual(expected);
-   });
-  });
-  describe('Decryption', function () {
-   it('should be able to decrypt a block with IV in stream', function () {
-    var input, key, result, expected, cipher;
-    input = hex2binary('0000000000000000000000000000000069c4e0d86a7b0430d' + '8cdb78070b4c55a');
-    key = hex2binary('000102030405060708090a0b0c0d0e0f');
-    cipher = new AES128Cipher(key);
-    result = cipher.decryptBlock(input);
-    expected = hex2binary('00112233445566778899aabbccddeeff');
-    expect(result).toEqual(expected);
-   });
-  });
- });
- describe('AES256', function () {
-  describe('Encryption', function () {
-   it('should be able to encrypt a block', function () {
-    var input, key, result, expected, iv, cipher;
-    input = hex2binary('00112233445566778899aabbccddeeff');
-    key = hex2binary('000102030405060708090a0b0c0d0e0f101112131415161718' + '191a1b1c1d1e1f');
-    iv = hex2binary('00000000000000000000000000000000');
-    cipher = new AES256Cipher(key);
-    result = cipher.encrypt(input, iv);
-    expected = hex2binary('8ea2b7ca516745bfeafc49904b496089');
-    expect(result).toEqual(expected);
-   });
-  });
-  describe('Decryption', function () {
-   it('should be able to decrypt a block with specified iv', function () {
-    var input, key, result, expected, cipher, iv;
-    input = hex2binary('8ea2b7ca516745bfeafc49904b496089');
-    key = hex2binary('000102030405060708090a0b0c0d0e0f101112131415161718' + '191a1b1c1d1e1f');
-    iv = hex2binary('00000000000000000000000000000000');
-    cipher = new AES256Cipher(key);
-    result = cipher.decryptBlock(input, false, iv);
-    expected = hex2binary('00112233445566778899aabbccddeeff');
-    expect(result).toEqual(expected);
-   });
-   it('should be able to decrypt a block with IV in stream', function () {
-    var input, key, result, expected, cipher;
-    input = hex2binary('000000000000000000000000000000008ea2b7ca516745bf' + 'eafc49904b496089');
-    key = hex2binary('000102030405060708090a0b0c0d0e0f101112131415161718' + '191a1b1c1d1e1f');
-    cipher = new AES256Cipher(key);
-    result = cipher.decryptBlock(input, false);
-    expected = hex2binary('00112233445566778899aabbccddeeff');
-    expect(result).toEqual(expected);
-   });
-  });
- });
- describe('PDF17Algorithm', function () {
-  it('should correctly check a user key', function () {
-   var password, userValidation, userPassword, alg, result;
-   alg = new PDF17();
-   password = new Uint8Array([
-    117,
-    115,
-    101,
-    114
-   ]);
-   userValidation = new Uint8Array([
-    117,
-    169,
-    4,
-    32,
-    159,
-    101,
-    22,
-    220
-   ]);
-   userPassword = new Uint8Array([
-    131,
-    242,
-    143,
-    160,
-    87,
-    2,
-    138,
-    134,
-    79,
-    253,
-    189,
-    173,
-    224,
-    73,
-    144,
-    241,
-    190,
-    81,
-    197,
-    15,
-    249,
-    105,
-    145,
-    151,
-    15,
-    194,
-    65,
-    3,
-    1,
-    126,
-    187,
-    221
-   ]);
-   result = alg.checkUserPassword(password, userValidation, userPassword);
-   expect(result).toEqual(true);
-  });
-  it('should correctly check an owner key', function () {
-   var password, ownerValidation, ownerPassword, alg, result, uBytes;
-   alg = new PDF17();
-   password = new Uint8Array([
-    111,
-    119,
-    110,
-    101,
-    114
-   ]);
-   ownerValidation = new Uint8Array([
-    243,
-    118,
-    71,
-    153,
-    128,
-    17,
-    101,
-    62
-   ]);
-   ownerPassword = new Uint8Array([
-    60,
-    98,
-    137,
-    35,
-    51,
-    101,
-    200,
-    152,
-    210,
-    178,
-    226,
-    228,
-    134,
-    205,
-    163,
-    24,
-    204,
-    126,
-    177,
-    36,
-    106,
-    50,
-    36,
-    125,
-    210,
-    172,
-    171,
-    120,
-    222,
-    108,
-    139,
-    115
-   ]);
-   uBytes = new Uint8Array([
-    131,
-    242,
-    143,
-    160,
-    87,
-    2,
-    138,
-    134,
-    79,
-    253,
-    189,
-    173,
-    224,
-    73,
-    144,
-    241,
-    190,
-    81,
-    197,
-    15,
-    249,
-    105,
-    145,
-    151,
-    15,
-    194,
-    65,
-    3,
-    1,
-    126,
-    187,
-    221,
-    117,
-    169,
-    4,
-    32,
-    159,
-    101,
-    22,
-    220,
-    168,
-    94,
-    215,
-    192,
-    100,
-    38,
-    188,
-    40
-   ]);
-   result = alg.checkOwnerPassword(password, ownerValidation, uBytes, ownerPassword);
-   expect(result).toEqual(true);
-  });
-  it('should generate a file encryption key from the user key', function () {
-   var password, userKeySalt, expected, alg, result, userEncryption;
-   alg = new PDF17();
-   password = new Uint8Array([
-    117,
-    115,
-    101,
-    114
-   ]);
-   userKeySalt = new Uint8Array([
-    168,
-    94,
-    215,
-    192,
-    100,
-    38,
-    188,
-    40
-   ]);
-   userEncryption = new Uint8Array([
-    35,
-    150,
-    195,
-    169,
-    245,
-    51,
-    51,
-    255,
-    158,
-    158,
-    33,
-    242,
-    231,
-    75,
-    125,
-    190,
-    25,
-    126,
-    172,
-    114,
-    195,
-    244,
-    137,
-    245,
-    234,
-    165,
-    42,
-    74,
-    60,
-    38,
-    17,
-    17
-   ]);
-   result = alg.getUserKey(password, userKeySalt, userEncryption);
-   expected = new Uint8Array([
-    63,
-    114,
-    136,
-    209,
-    87,
-    61,
-    12,
-    30,
-    249,
-    1,
-    186,
-    144,
-    254,
-    248,
-    163,
-    153,
-    151,
-    51,
-    133,
-    10,
-    80,
-    152,
-    206,
-    15,
-    72,
-    187,
-    231,
-    33,
-    224,
-    239,
-    13,
-    213
-   ]);
-   expect(result).toEqual(expected);
-  });
-  it('should generate a file encryption key from the owner key', function () {
-   var password, ownerKeySalt, expected, alg, result, ownerEncryption;
-   var uBytes;
-   alg = new PDF17();
-   password = new Uint8Array([
-    111,
-    119,
-    110,
-    101,
-    114
-   ]);
-   ownerKeySalt = new Uint8Array([
-    200,
-    245,
-    242,
-    12,
-    218,
-    123,
-    24,
-    120
-   ]);
-   ownerEncryption = new Uint8Array([
-    213,
-    202,
-    14,
-    189,
-    110,
-    76,
-    70,
-    191,
-    6,
-    195,
-    10,
-    190,
-    157,
-    100,
-    144,
-    85,
-    8,
-    62,
-    123,
-    178,
-    156,
-    229,
-    50,
-    40,
-    229,
-    216,
-    54,
-    222,
-    34,
-    38,
-    106,
-    223
-   ]);
-   uBytes = new Uint8Array([
-    131,
-    242,
-    143,
-    160,
-    87,
-    2,
-    138,
-    134,
-    79,
-    253,
-    189,
-    173,
-    224,
-    73,
-    144,
-    241,
-    190,
-    81,
-    197,
-    15,
-    249,
-    105,
-    145,
-    151,
-    15,
-    194,
-    65,
-    3,
-    1,
-    126,
-    187,
-    221,
-    117,
-    169,
-    4,
-    32,
-    159,
-    101,
-    22,
-    220,
-    168,
-    94,
-    215,
-    192,
-    100,
-    38,
-    188,
-    40
-   ]);
-   result = alg.getOwnerKey(password, ownerKeySalt, uBytes, ownerEncryption);
-   expected = new Uint8Array([
-    63,
-    114,
-    136,
-    209,
-    87,
-    61,
-    12,
-    30,
-    249,
-    1,
-    186,
-    144,
-    254,
-    248,
-    163,
-    153,
-    151,
-    51,
-    133,
-    10,
-    80,
-    152,
-    206,
-    15,
-    72,
-    187,
-    231,
-    33,
-    224,
-    239,
-    13,
-    213
-   ]);
-   expect(result).toEqual(expected);
-  });
- });
- describe('PDF20Algorithm', function () {
-  it('should correctly check a user key', function () {
-   var password, userValidation, userPassword, alg, result;
-   alg = new PDF20();
-   password = new Uint8Array([
-    117,
-    115,
-    101,
-    114
-   ]);
-   userValidation = new Uint8Array([
-    83,
-    245,
-    146,
-    101,
-    198,
-    247,
-    34,
-    198
-   ]);
-   userPassword = new Uint8Array([
-    94,
-    230,
-    205,
-    75,
-    166,
-    99,
-    250,
-    76,
-    219,
-    128,
-    17,
-    85,
-    57,
-    17,
-    33,
-    164,
-    150,
-    46,
-    103,
-    176,
-    160,
-    156,
-    187,
-    233,
-    166,
-    223,
-    163,
-    253,
-    147,
-    235,
-    95,
-    184
-   ]);
-   result = alg.checkUserPassword(password, userValidation, userPassword);
-   expect(result).toEqual(true);
-  });
-  it('should correctly check an owner key', function () {
-   var password, ownerValidation, ownerPassword, alg, result, uBytes;
-   alg = new PDF20();
-   password = new Uint8Array([
-    111,
-    119,
-    110,
-    101,
-    114
-   ]);
-   ownerValidation = new Uint8Array([
-    142,
-    232,
-    169,
-    208,
-    202,
-    214,
-    5,
-    185
-   ]);
-   ownerPassword = new Uint8Array([
-    88,
-    232,
-    62,
-    54,
-    245,
-    26,
-    245,
-    209,
-    137,
-    123,
-    221,
-    72,
-    199,
-    49,
-    37,
-    217,
-    31,
-    74,
-    115,
-    167,
-    127,
-    158,
-    176,
-    77,
-    45,
-    163,
-    87,
-    47,
-    39,
-    90,
-    217,
-    141
-   ]);
-   uBytes = new Uint8Array([
-    94,
-    230,
-    205,
-    75,
-    166,
-    99,
-    250,
-    76,
-    219,
-    128,
-    17,
-    85,
-    57,
-    17,
-    33,
-    164,
-    150,
-    46,
-    103,
-    176,
-    160,
-    156,
-    187,
-    233,
-    166,
-    223,
-    163,
-    253,
-    147,
-    235,
-    95,
-    184,
-    83,
-    245,
-    146,
-    101,
-    198,
-    247,
-    34,
-    198,
-    191,
-    11,
-    16,
-    94,
-    237,
-    216,
-    20,
-    175
-   ]);
-   result = alg.checkOwnerPassword(password, ownerValidation, uBytes, ownerPassword);
-   expect(result).toEqual(true);
-  });
-  it('should generate a file encryption key from the user key', function () {
-   var password, userKeySalt, expected, alg, result, userEncryption;
-   alg = new PDF20();
-   password = new Uint8Array([
-    117,
-    115,
-    101,
-    114
-   ]);
-   userKeySalt = new Uint8Array([
-    191,
-    11,
-    16,
-    94,
-    237,
-    216,
-    20,
-    175
-   ]);
-   userEncryption = new Uint8Array([
-    121,
-    208,
-    2,
-    181,
-    230,
-    89,
-    156,
-    60,
-    253,
-    143,
-    212,
-    28,
-    84,
-    180,
-    196,
-    177,
-    173,
-    128,
-    221,
-    107,
-    46,
-    20,
-    94,
-    186,
-    135,
-    51,
-    95,
-    24,
-    20,
-    223,
-    254,
-    36
-   ]);
-   result = alg.getUserKey(password, userKeySalt, userEncryption);
-   expected = new Uint8Array([
-    42,
-    218,
-    213,
-    39,
-    73,
-    91,
-    72,
-    79,
-    67,
-    38,
-    248,
-    133,
-    18,
-    189,
-    61,
-    34,
-    107,
-    79,
-    29,
-    56,
-    59,
-    181,
-    213,
-    118,
-    113,
-    34,
-    65,
-    210,
-    87,
-    174,
-    22,
-    239
-   ]);
-   expect(result).toEqual(expected);
-  });
-  it('should generate a file encryption key from the owner key', function () {
-   var password, ownerKeySalt, expected, alg, result, ownerEncryption;
-   var uBytes;
-   alg = new PDF20();
-   password = new Uint8Array([
-    111,
-    119,
-    110,
-    101,
-    114
-   ]);
-   ownerKeySalt = new Uint8Array([
-    29,
-    208,
-    185,
-    46,
-    11,
-    76,
-    135,
-    149
-   ]);
-   ownerEncryption = new Uint8Array([
-    209,
-    73,
-    224,
-    77,
-    103,
-    155,
-    201,
-    181,
-    190,
-    68,
-    223,
-    20,
-    62,
-    90,
-    56,
-    210,
-    5,
-    240,
-    178,
-    128,
-    238,
-    124,
-    68,
-    254,
-    253,
-    244,
-    62,
-    108,
-    208,
-    135,
-    10,
-    251
-   ]);
-   uBytes = new Uint8Array([
-    94,
-    230,
-    205,
-    75,
-    166,
-    99,
-    250,
-    76,
-    219,
-    128,
-    17,
-    85,
-    57,
-    17,
-    33,
-    164,
-    150,
-    46,
-    103,
-    176,
-    160,
-    156,
-    187,
-    233,
-    166,
-    223,
-    163,
-    253,
-    147,
-    235,
-    95,
-    184,
-    83,
-    245,
-    146,
-    101,
-    198,
-    247,
-    34,
-    198,
-    191,
-    11,
-    16,
-    94,
-    237,
-    216,
-    20,
-    175
-   ]);
-   result = alg.getOwnerKey(password, ownerKeySalt, uBytes, ownerEncryption);
-   expected = new Uint8Array([
-    42,
-    218,
-    213,
-    39,
-    73,
-    91,
-    72,
-    79,
-    67,
-    38,
-    248,
-    133,
-    18,
-    189,
-    61,
-    34,
-    107,
-    79,
-    29,
-    56,
-    59,
-    181,
-    213,
-    118,
-    113,
-    34,
-    65,
-    210,
-    87,
-    174,
-    22,
-    239
-   ]);
-   expect(result).toEqual(expected);
-  });
- });
 });
 describe('CipherTransformFactory', function () {
- function buildDict(map) {
-  var dict = new Dict();
-  for (var key in map) {
-   dict.set(key, map[key]);
+  function buildDict(map) {
+    var dict = new Dict();
+    for (var key in map) {
+      dict.set(key, map[key]);
+    }
+    return dict;
   }
-  return dict;
- }
- function ensurePasswordCorrect(done, dict, fileId, password) {
-  try {
-   var factory = new CipherTransformFactory(dict, fileId, password);
-   expect('createCipherTransform' in factory).toEqual(true);
-  } catch (ex) {
-   done.fail('Password should be accepted: ' + ex);
-   return;
+  function ensurePasswordCorrect(done, dict, fileId, password) {
+    try {
+      var factory = new CipherTransformFactory(dict, fileId, password);
+      expect('createCipherTransform' in factory).toEqual(true);
+    } catch (ex) {
+      done.fail('Password should be accepted: ' + ex);
+      return;
+    }
+    done();
   }
-  done();
- }
- function ensurePasswordNeeded(done, dict, fileId, password) {
-  try {
-   new CipherTransformFactory(dict, fileId, password);
-  } catch (ex) {
-   expect(ex instanceof PasswordException).toEqual(true);
-   expect(ex.code).toEqual(PasswordResponses.NEED_PASSWORD);
-   done();
-   return;
+  function ensurePasswordNeeded(done, dict, fileId, password) {
+    try {
+      new CipherTransformFactory(dict, fileId, password);
+    } catch (ex) {
+      expect(ex instanceof PasswordException).toEqual(true);
+      expect(ex.code).toEqual(PasswordResponses.NEED_PASSWORD);
+      done();
+      return;
+    }
+    done.fail('Password should be rejected.');
   }
-  done.fail('Password should be rejected.');
- }
- function ensurePasswordIncorrect(done, dict, fileId, password) {
-  try {
-   new CipherTransformFactory(dict, fileId, password);
-  } catch (ex) {
-   expect(ex instanceof PasswordException).toEqual(true);
-   expect(ex.code).toEqual(PasswordResponses.INCORRECT_PASSWORD);
-   done();
-   return;
+  function ensurePasswordIncorrect(done, dict, fileId, password) {
+    try {
+      new CipherTransformFactory(dict, fileId, password);
+    } catch (ex) {
+      expect(ex instanceof PasswordException).toEqual(true);
+      expect(ex.code).toEqual(PasswordResponses.INCORRECT_PASSWORD);
+      done();
+      return;
+    }
+    done.fail('Password should be rejected.');
   }
-  done.fail('Password should be rejected.');
- }
- var fileId1, fileId2, dict1, dict2;
- var aes256Dict, aes256IsoDict, aes256BlankDict, aes256IsoBlankDict;
- beforeAll(function (done) {
-  fileId1 = unescape('%F6%C6%AF%17%F3rR%8DRM%9A%80%D1%EF%DF%18');
-  fileId2 = unescape('%3CL_%3AD%96%AF@%9A%9D%B3%3Cx%1Cv%AC');
-  dict1 = buildDict({
-   Filter: Name.get('Standard'),
-   V: 2,
-   Length: 128,
-   O: unescape('%80%C3%04%96%91o%20sl%3A%E6%1B%13T%91%F2%0DV%12%E3%FF%5E%B' + 'B%E9VO%D8k%9A%CA%7C%5D'),
-   U: unescape('j%0C%8D%3EY%19%00%BCjd%7D%91%BD%AA%00%18%00%00%00%00%00%00' + '%00%00%00%00%00%00%00%00%00%00'),
-   P: -1028,
-   R: 3
-  });
-  dict2 = buildDict({
-   Filter: Name.get('Standard'),
-   V: 4,
-   Length: 128,
-   O: unescape('sF%14v.y5%27%DB%97%0A5%22%B3%E1%D4%AD%BD%9B%3C%B4%A5%89u%1' + '5%B2Y%F1h%D9%E9%F4'),
-   U: unescape('%93%04%89%A9%BF%8AE%A6%88%A2%DB%C2%A0%A8gn%00%00%00%00%00%' + '00%00%00%00%00%00%00%00%00%00%00'),
-   P: -1084,
-   R: 4
-  });
-  aes256Dict = buildDict({
-   Filter: Name.get('Standard'),
-   V: 5,
-   Length: 256,
-   O: unescape('%3Cb%89%233e%C8%98%D2%B2%E2%E4%86%CD%A3%18%CC%7E%B1%24j2%2' + '4%7D%D2%AC%ABx%DEl%8Bs%F3vG%99%80%11e%3E%C8%F5%F2%0C%DA%7B' + '%18x'),
-   U: unescape('%83%F2%8F%A0W%02%8A%86O%FD%BD%AD%E0I%90%F1%BEQ%C5%0F%F9i%9' + '1%97%0F%C2A%03%01%7E%BB%DDu%A9%04%20%9Fe%16%DC%A8%5E%D7%C0' + 'd%26%BC%28'),
-   OE: unescape('%D5%CA%0E%BDnLF%BF%06%C3%0A%BE%9Dd%90U%08%3E%7B%B2%9C%E52' + '%28%E5%D86%DE%22%26j%DF'),
-   UE: unescape('%23%96%C3%A9%F533%FF%9E%9E%21%F2%E7K%7D%BE%19%7E%ACr%C3%F' + '4%89%F5%EA%A5*J%3C%26%11%11'),
-   Perms: unescape('%D8%FC%844%E5e%0DB%5D%7Ff%FD%3COMM'),
-   P: -1084,
-   R: 5
-  });
-  aes256IsoDict = buildDict({
-   Filter: Name.get('Standard'),
-   V: 5,
-   Length: 256,
-   O: unescape('X%E8%3E6%F5%1A%F5%D1%89%7B%DDH%C71%25%D9%1FJs%A7%7F%9E%B0M' + '-%A3W/%27Z%D9%8D%8E%E8%A9%D0%CA%D6%05%B9%1D%D0%B9.%0BL%87%' + '95'),
-   U: unescape('%5E%E6%CDK%A6c%FAL%DB%80%11U9%11%21%A4%96.g%B0%A0%9C%BB%E9' + '%A6%DF%A3%FD%93%EB_%B8S%F5%92e%C6%F7%22%C6%BF%0B%10%5E%ED%' + 'D8%14%AF'),
-   OE: unescape('%D1I%E0Mg%9B%C9%B5%BED%DF%14%3EZ8%D2%05%F0%B2%80%EE%7CD%F' + 'E%FD%F4%3El%D0%87%0A%FB'),
-   UE: unescape('y%D0%02%B5%E6Y%9C%3C%FD%8F%D4%1CT%B4%C4%B1%AD%80%DDk.%14%' + '5E%BA%873_%18%14%DF%FE%24'),
-   Perms: unescape('l%AD%0F%A0%EBM%86WM%3E%CB%B5%E0X%C97'),
-   P: -1084,
-   R: 6
-  });
-  aes256BlankDict = buildDict({
-   Filter: Name.get('Standard'),
-   V: 5,
-   Length: 256,
-   O: unescape('%B8p%04%C3g%26%FCW%CCN%D4%16%A1%E8%950YZ%C9%9E%B1-%97%F3%F' + 'E%03%13%19ffZn%8F%F5%EB%EC%CC5sV%10e%CEl%B5%E9G%C1'),
-   U: unescape('%83%D4zi%F1O0%961%12%CC%82%CB%CA%BF5y%FD%21%EB%E4%D1%B5%1D' + '%D6%FA%14%F3%BE%8Fqs%EF%88%DE%E2%E8%DC%F55%E4%B8%16%C8%14%' + '8De%1E'),
-   OE: unescape('%8F%19%E8%D4%27%D5%07%CA%C6%A1%11%A6a%5Bt%F4%DF%0F%84%29%' + '0F%E4%EFF7%5B%5B%11%A0%8F%17e'),
-   UE: unescape('%81%F5%5D%B0%28%81%E4%7F_%7C%8F%85b%A0%7E%10%D0%88lx%7B%7' + 'EJ%5E%912%B6d%12%27%05%F6'),
-   Perms: unescape('%86%1562%0D%AE%A2%FB%5D%3B%22%3Dq%12%B2H'),
-   P: -1084,
-   R: 5
-  });
-  aes256IsoBlankDict = buildDict({
-   Filter: Name.get('Standard'),
-   V: 5,
-   Length: 256,
-   O: unescape('%F7%DB%99U%A6M%ACk%AF%CF%D7AFw%E9%C1%91%CBDgI%23R%CF%0C%15' + 'r%D74%0D%CE%E9%91@%E4%98QF%BF%88%7Ej%DE%AD%8F%F4@%C1'),
-   U: unescape('%1A%A9%DC%918%83%93k%29%5B%117%B16%DB%E8%8E%FE%28%E5%89%D4' + '%0E%AD%12%3B%7DN_6fez%8BG%18%05YOh%7DZH%A3Z%87%17*'),
-   OE: unescape('%A4a%88%20h%1B%7F%CD%D5%CAc%D8R%83%E5%D6%1C%D2%98%07%984%' + 'BA%AF%1B%B4%7FQ%F8%1EU%7D'),
-   UE: unescape('%A0%0AZU%27%1D%27%2C%0B%FE%0E%A2L%F9b%5E%A1%B9%D6v7b%B26%' + 'A9N%99%F1%A4Deq'),
-   Perms: unescape('%03%F2i%07%0D%C3%F9%F2%28%80%B7%F5%DD%D1c%EB'),
-   P: -1084,
-   R: 6
-  });
-  done();
- });
- afterAll(function () {
-  fileId1 = fileId2 = dict1 = dict2 = null;
-  aes256Dict = aes256IsoDict = aes256BlankDict = aes256IsoBlankDict = null;
- });
- describe('#ctor', function () {
-  describe('AES256 Revision 5', function () {
-   it('should accept user password', function (done) {
-    ensurePasswordCorrect(done, aes256Dict, fileId1, 'user');
-   });
-   it('should accept owner password', function (done) {
-    ensurePasswordCorrect(done, aes256Dict, fileId1, 'owner');
-   });
-   it('should not accept blank password', function (done) {
-    ensurePasswordNeeded(done, aes256Dict, fileId1);
-   });
-   it('should not accept wrong password', function (done) {
-    ensurePasswordIncorrect(done, aes256Dict, fileId1, 'wrong');
-   });
-   it('should accept blank password', function (done) {
-    ensurePasswordCorrect(done, aes256BlankDict, fileId1);
-   });
-  });
-  describe('AES256 Revision 6', function () {
-   it('should accept user password', function (done) {
-    ensurePasswordCorrect(done, aes256IsoDict, fileId1, 'user');
-   });
-   it('should accept owner password', function (done) {
-    ensurePasswordCorrect(done, aes256IsoDict, fileId1, 'owner');
-   });
-   it('should not accept blank password', function (done) {
-    ensurePasswordNeeded(done, aes256IsoDict, fileId1);
-   });
-   it('should not accept wrong password', function (done) {
-    ensurePasswordIncorrect(done, aes256IsoDict, fileId1, 'wrong');
-   });
-   it('should accept blank password', function (done) {
-    ensurePasswordCorrect(done, aes256IsoBlankDict, fileId1);
-   });
-  });
-  it('should accept user password', function (done) {
-   ensurePasswordCorrect(done, dict1, fileId1, '123456');
-  });
-  it('should accept owner password', function (done) {
-   ensurePasswordCorrect(done, dict1, fileId1, '654321');
-  });
-  it('should not accept blank password', function (done) {
-   ensurePasswordNeeded(done, dict1, fileId1);
-  });
-  it('should not accept wrong password', function (done) {
-   ensurePasswordIncorrect(done, dict1, fileId1, 'wrong');
-  });
-  it('should accept blank password', function (done) {
-   ensurePasswordCorrect(done, dict2, fileId2);
+  var fileId1, fileId2, dict1, dict2;
+  var aes256Dict, aes256IsoDict, aes256BlankDict, aes256IsoBlankDict;
+  beforeAll(function (done) {
+    fileId1 = unescape('%F6%C6%AF%17%F3rR%8DRM%9A%80%D1%EF%DF%18');
+    fileId2 = unescape('%3CL_%3AD%96%AF@%9A%9D%B3%3Cx%1Cv%AC');
+    dict1 = buildDict({
+      Filter: Name.get('Standard'),
+      V: 2,
+      Length: 128,
+      O: unescape('%80%C3%04%96%91o%20sl%3A%E6%1B%13T%91%F2%0DV%12%E3%FF%5E%B' + 'B%E9VO%D8k%9A%CA%7C%5D'),
+      U: unescape('j%0C%8D%3EY%19%00%BCjd%7D%91%BD%AA%00%18%00%00%00%00%00%00' + '%00%00%00%00%00%00%00%00%00%00'),
+      P: -1028,
+      R: 3
+    });
+    dict2 = buildDict({
+      Filter: Name.get('Standard'),
+      V: 4,
+      Length: 128,
+      O: unescape('sF%14v.y5%27%DB%97%0A5%22%B3%E1%D4%AD%BD%9B%3C%B4%A5%89u%1' + '5%B2Y%F1h%D9%E9%F4'),
+      U: unescape('%93%04%89%A9%BF%8AE%A6%88%A2%DB%C2%A0%A8gn%00%00%00%00%00%' + '00%00%00%00%00%00%00%00%00%00%00'),
+      P: -1084,
+      R: 4
+    });
+    aes256Dict = buildDict({
+      Filter: Name.get('Standard'),
+      V: 5,
+      Length: 256,
+      O: unescape('%3Cb%89%233e%C8%98%D2%B2%E2%E4%86%CD%A3%18%CC%7E%B1%24j2%2' + '4%7D%D2%AC%ABx%DEl%8Bs%F3vG%99%80%11e%3E%C8%F5%F2%0C%DA%7B' + '%18x'),
+      U: unescape('%83%F2%8F%A0W%02%8A%86O%FD%BD%AD%E0I%90%F1%BEQ%C5%0F%F9i%9' + '1%97%0F%C2A%03%01%7E%BB%DDu%A9%04%20%9Fe%16%DC%A8%5E%D7%C0' + 'd%26%BC%28'),
+      OE: unescape('%D5%CA%0E%BDnLF%BF%06%C3%0A%BE%9Dd%90U%08%3E%7B%B2%9C%E52' + '%28%E5%D86%DE%22%26j%DF'),
+      UE: unescape('%23%96%C3%A9%F533%FF%9E%9E%21%F2%E7K%7D%BE%19%7E%ACr%C3%F' + '4%89%F5%EA%A5*J%3C%26%11%11'),
+      Perms: unescape('%D8%FC%844%E5e%0DB%5D%7Ff%FD%3COMM'),
+      P: -1084,
+      R: 5
+    });
+    aes256IsoDict = buildDict({
+      Filter: Name.get('Standard'),
+      V: 5,
+      Length: 256,
+      O: unescape('X%E8%3E6%F5%1A%F5%D1%89%7B%DDH%C71%25%D9%1FJs%A7%7F%9E%B0M' + '-%A3W/%27Z%D9%8D%8E%E8%A9%D0%CA%D6%05%B9%1D%D0%B9.%0BL%87%' + '95'),
+      U: unescape('%5E%E6%CDK%A6c%FAL%DB%80%11U9%11%21%A4%96.g%B0%A0%9C%BB%E9' + '%A6%DF%A3%FD%93%EB_%B8S%F5%92e%C6%F7%22%C6%BF%0B%10%5E%ED%' + 'D8%14%AF'),
+      OE: unescape('%D1I%E0Mg%9B%C9%B5%BED%DF%14%3EZ8%D2%05%F0%B2%80%EE%7CD%F' + 'E%FD%F4%3El%D0%87%0A%FB'),
+      UE: unescape('y%D0%02%B5%E6Y%9C%3C%FD%8F%D4%1CT%B4%C4%B1%AD%80%DDk.%14%' + '5E%BA%873_%18%14%DF%FE%24'),
+      Perms: unescape('l%AD%0F%A0%EBM%86WM%3E%CB%B5%E0X%C97'),
+      P: -1084,
+      R: 6
+    });
+    aes256BlankDict = buildDict({
+      Filter: Name.get('Standard'),
+      V: 5,
+      Length: 256,
+      O: unescape('%B8p%04%C3g%26%FCW%CCN%D4%16%A1%E8%950YZ%C9%9E%B1-%97%F3%F' + 'E%03%13%19ffZn%8F%F5%EB%EC%CC5sV%10e%CEl%B5%E9G%C1'),
+      U: unescape('%83%D4zi%F1O0%961%12%CC%82%CB%CA%BF5y%FD%21%EB%E4%D1%B5%1D' + '%D6%FA%14%F3%BE%8Fqs%EF%88%DE%E2%E8%DC%F55%E4%B8%16%C8%14%' + '8De%1E'),
+      OE: unescape('%8F%19%E8%D4%27%D5%07%CA%C6%A1%11%A6a%5Bt%F4%DF%0F%84%29%' + '0F%E4%EFF7%5B%5B%11%A0%8F%17e'),
+      UE: unescape('%81%F5%5D%B0%28%81%E4%7F_%7C%8F%85b%A0%7E%10%D0%88lx%7B%7' + 'EJ%5E%912%B6d%12%27%05%F6'),
+      Perms: unescape('%86%1562%0D%AE%A2%FB%5D%3B%22%3Dq%12%B2H'),
+      P: -1084,
+      R: 5
+    });
+    aes256IsoBlankDict = buildDict({
+      Filter: Name.get('Standard'),
+      V: 5,
+      Length: 256,
+      O: unescape('%F7%DB%99U%A6M%ACk%AF%CF%D7AFw%E9%C1%91%CBDgI%23R%CF%0C%15' + 'r%D74%0D%CE%E9%91@%E4%98QF%BF%88%7Ej%DE%AD%8F%F4@%C1'),
+      U: unescape('%1A%A9%DC%918%83%93k%29%5B%117%B16%DB%E8%8E%FE%28%E5%89%D4' + '%0E%AD%12%3B%7DN_6fez%8BG%18%05YOh%7DZH%A3Z%87%17*'),
+      OE: unescape('%A4a%88%20h%1B%7F%CD%D5%CAc%D8R%83%E5%D6%1C%D2%98%07%984%' + 'BA%AF%1B%B4%7FQ%F8%1EU%7D'),
+      UE: unescape('%A0%0AZU%27%1D%27%2C%0B%FE%0E%A2L%F9b%5E%A1%B9%D6v7b%B26%' + 'A9N%99%F1%A4Deq'),
+      Perms: unescape('%03%F2i%07%0D%C3%F9%F2%28%80%B7%F5%DD%D1c%EB'),
+      P: -1084,
+      R: 6
+    });
+    done();
+  });
+  afterAll(function () {
+    fileId1 = fileId2 = dict1 = dict2 = null;
+    aes256Dict = aes256IsoDict = aes256BlankDict = aes256IsoBlankDict = null;
+  });
+  describe('#ctor', function () {
+    describe('AES256 Revision 5', function () {
+      it('should accept user password', function (done) {
+        ensurePasswordCorrect(done, aes256Dict, fileId1, 'user');
+      });
+      it('should accept owner password', function (done) {
+        ensurePasswordCorrect(done, aes256Dict, fileId1, 'owner');
+      });
+      it('should not accept blank password', function (done) {
+        ensurePasswordNeeded(done, aes256Dict, fileId1);
+      });
+      it('should not accept wrong password', function (done) {
+        ensurePasswordIncorrect(done, aes256Dict, fileId1, 'wrong');
+      });
+      it('should accept blank password', function (done) {
+        ensurePasswordCorrect(done, aes256BlankDict, fileId1);
+      });
+    });
+    describe('AES256 Revision 6', function () {
+      it('should accept user password', function (done) {
+        ensurePasswordCorrect(done, aes256IsoDict, fileId1, 'user');
+      });
+      it('should accept owner password', function (done) {
+        ensurePasswordCorrect(done, aes256IsoDict, fileId1, 'owner');
+      });
+      it('should not accept blank password', function (done) {
+        ensurePasswordNeeded(done, aes256IsoDict, fileId1);
+      });
+      it('should not accept wrong password', function (done) {
+        ensurePasswordIncorrect(done, aes256IsoDict, fileId1, 'wrong');
+      });
+      it('should accept blank password', function (done) {
+        ensurePasswordCorrect(done, aes256IsoBlankDict, fileId1);
+      });
+    });
+    it('should accept user password', function (done) {
+      ensurePasswordCorrect(done, dict1, fileId1, '123456');
+    });
+    it('should accept owner password', function (done) {
+      ensurePasswordCorrect(done, dict1, fileId1, '654321');
+    });
+    it('should not accept blank password', function (done) {
+      ensurePasswordNeeded(done, dict1, fileId1);
+    });
+    it('should not accept wrong password', function (done) {
+      ensurePasswordIncorrect(done, dict1, fileId1, 'wrong');
+    });
+    it('should accept blank password', function (done) {
+      ensurePasswordCorrect(done, dict2, fileId2);
+    });
   });
- });
 });

+ 14 - 12
lib/test/unit/document_spec.js

@@ -13,20 +13,22 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreDocument = require('../../core/document.js');
 var Page = coreDocument.Page;
 describe('document', function () {
- describe('Page', function () {
-  it('should create correct objId using the idFactory', function () {
-   var page1 = new Page({}, null, 0, null, null, null, null);
-   var page2 = new Page({}, null, 1, null, null, null, null);
-   var idFactory1 = page1.idFactory, idFactory2 = page2.idFactory;
-   expect(idFactory1.createObjId()).toEqual('p0_1');
-   expect(idFactory1.createObjId()).toEqual('p0_2');
-   expect(idFactory2.createObjId()).toEqual('p1_1');
-   expect(idFactory2.createObjId()).toEqual('p1_2');
-   expect(idFactory1.createObjId()).toEqual('p0_3');
-   expect(idFactory1.createObjId()).toEqual('p0_4');
+  describe('Page', function () {
+    it('should create correct objId using the idFactory', function () {
+      var page1 = new Page({}, null, 0, null, null, null, null);
+      var page2 = new Page({}, null, 1, null, null, null, null);
+      var idFactory1 = page1.idFactory,
+          idFactory2 = page2.idFactory;
+      expect(idFactory1.createObjId()).toEqual('p0_1');
+      expect(idFactory1.createObjId()).toEqual('p0_2');
+      expect(idFactory2.createObjId()).toEqual('p1_1');
+      expect(idFactory2.createObjId()).toEqual('p1_2');
+      expect(idFactory1.createObjId()).toEqual('p0_3');
+      expect(idFactory1.createObjId()).toEqual('p0_4');
+    });
   });
- });
 });

+ 38 - 44
lib/test/unit/dom_utils_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var displayDOMUtils = require('../../display/dom_utils.js');
 var displayGlobal = require('../../display/global.js');
 var PDFJS = displayGlobal.PDFJS;
@@ -20,50 +21,43 @@ var getFilenameFromUrl = displayDOMUtils.getFilenameFromUrl;
 var LinkTarget = displayDOMUtils.LinkTarget;
 var isExternalLinkTargetSet = displayDOMUtils.isExternalLinkTargetSet;
 describe('dom_utils', function () {
- describe('getFilenameFromUrl', function () {
-  it('should get the filename from an absolute URL', function () {
-   var url = 'http://server.org/filename.pdf';
-   var result = getFilenameFromUrl(url);
-   var expected = 'filename.pdf';
-   expect(result).toEqual(expected);
+  describe('getFilenameFromUrl', function () {
+    it('should get the filename from an absolute URL', function () {
+      var url = 'http://server.org/filename.pdf';
+      var result = getFilenameFromUrl(url);
+      var expected = 'filename.pdf';
+      expect(result).toEqual(expected);
+    });
+    it('should get the filename from a relative URL', function () {
+      var url = '../../filename.pdf';
+      var result = getFilenameFromUrl(url);
+      var expected = 'filename.pdf';
+      expect(result).toEqual(expected);
+    });
   });
-  it('should get the filename from a relative URL', function () {
-   var url = '../../filename.pdf';
-   var result = getFilenameFromUrl(url);
-   var expected = 'filename.pdf';
-   expect(result).toEqual(expected);
+  describe('isExternalLinkTargetSet', function () {
+    var savedExternalLinkTarget;
+    beforeAll(function (done) {
+      savedExternalLinkTarget = PDFJS.externalLinkTarget;
+      done();
+    });
+    afterAll(function () {
+      PDFJS.externalLinkTarget = savedExternalLinkTarget;
+    });
+    it('handles the predefined LinkTargets', function () {
+      for (var key in LinkTarget) {
+        var linkTarget = LinkTarget[key];
+        PDFJS.externalLinkTarget = linkTarget;
+        expect(isExternalLinkTargetSet()).toEqual(!!linkTarget);
+      }
+    });
+    it('handles incorrect LinkTargets', function () {
+      var targets = [true, '', false, -1, '_blank', null];
+      for (var i = 0, ii = targets.length; i < ii; i++) {
+        var linkTarget = targets[i];
+        PDFJS.externalLinkTarget = linkTarget;
+        expect(isExternalLinkTargetSet()).toEqual(false);
+      }
+    });
   });
- });
- describe('isExternalLinkTargetSet', function () {
-  var savedExternalLinkTarget;
-  beforeAll(function (done) {
-   savedExternalLinkTarget = PDFJS.externalLinkTarget;
-   done();
-  });
-  afterAll(function () {
-   PDFJS.externalLinkTarget = savedExternalLinkTarget;
-  });
-  it('handles the predefined LinkTargets', function () {
-   for (var key in LinkTarget) {
-    var linkTarget = LinkTarget[key];
-    PDFJS.externalLinkTarget = linkTarget;
-    expect(isExternalLinkTargetSet()).toEqual(!!linkTarget);
-   }
-  });
-  it('handles incorrect LinkTargets', function () {
-   var targets = [
-    true,
-    '',
-    false,
-    -1,
-    '_blank',
-    null
-   ];
-   for (var i = 0, ii = targets.length; i < ii; i++) {
-    var linkTarget = targets[i];
-    PDFJS.externalLinkTarget = linkTarget;
-    expect(isExternalLinkTargetSet()).toEqual(false);
-   }
-  });
- });
 });

+ 247 - 250
lib/test/unit/evaluator_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreEvaluator = require('../../core/evaluator.js');
 var corePrimitives = require('../../core/primitives.js');
 var coreStream = require('../../core/stream.js');
@@ -27,261 +28,257 @@ var StringStream = coreStream.StringStream;
 var WorkerTask = coreWorker.WorkerTask;
 var OPS = sharedUtil.OPS;
 describe('evaluator', function () {
- function XrefMock(queue) {
-  this.queue = queue || [];
- }
- XrefMock.prototype = {
-  fetchIfRef: function () {
-   return this.queue.shift();
+  function XrefMock(queue) {
+    this.queue = queue || [];
   }
- };
- function HandlerMock() {
-  this.inputs = [];
- }
- HandlerMock.prototype = {
-  send: function (name, data) {
-   this.inputs.push({
-    name: name,
-    data: data
-   });
+  XrefMock.prototype = {
+    fetchIfRef: function () {
+      return this.queue.shift();
+    }
+  };
+  function HandlerMock() {
+    this.inputs = [];
   }
- };
- function ResourcesMock() {
- }
- ResourcesMock.prototype = {
-  get: function (name) {
-   return this[name];
+  HandlerMock.prototype = {
+    send: function (name, data) {
+      this.inputs.push({
+        name: name,
+        data: data
+      });
+    }
+  };
+  function ResourcesMock() {}
+  ResourcesMock.prototype = {
+    get: function (name) {
+      return this[name];
+    }
+  };
+  function PdfManagerMock() {}
+  function runOperatorListCheck(evaluator, stream, resources, callback) {
+    var result = new OperatorList();
+    var task = new WorkerTask('OperatorListCheck');
+    evaluator.getOperatorList(stream, task, resources, result).then(function () {
+      callback(result);
+    });
   }
- };
- function PdfManagerMock() {
- }
- function runOperatorListCheck(evaluator, stream, resources, callback) {
-  var result = new OperatorList();
-  var task = new WorkerTask('OperatorListCheck');
-  evaluator.getOperatorList(stream, task, resources, result).then(function () {
-   callback(result);
-  });
- }
- describe('splitCombinedOperations', function () {
-  it('should reject unknown operations', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('fTT');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(!!result.fnArray && !!result.argsArray).toEqual(true);
-    expect(result.fnArray.length).toEqual(1);
-    expect(result.fnArray[0]).toEqual(OPS.fill);
-    expect(result.argsArray[0]).toEqual(null);
-    done();
-   });
-  });
-  it('should handle one operations', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('Q');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(!!result.fnArray && !!result.argsArray).toEqual(true);
-    expect(result.fnArray.length).toEqual(1);
-    expect(result.fnArray[0]).toEqual(OPS.restore);
-    done();
-   });
-  });
-  it('should handle two glued operations', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var resources = new ResourcesMock();
-   resources.Res1 = {};
-   var stream = new StringStream('/Res1 DoQ');
-   runOperatorListCheck(evaluator, stream, resources, function (result) {
-    expect(!!result.fnArray && !!result.argsArray).toEqual(true);
-    expect(result.fnArray.length).toEqual(2);
-    expect(result.fnArray[0]).toEqual(OPS.paintXObject);
-    expect(result.fnArray[1]).toEqual(OPS.restore);
-    done();
-   });
-  });
-  it('should handle tree glued operations', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('fff');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(!!result.fnArray && !!result.argsArray).toEqual(true);
-    expect(result.fnArray.length).toEqual(3);
-    expect(result.fnArray[0]).toEqual(OPS.fill);
-    expect(result.fnArray[1]).toEqual(OPS.fill);
-    expect(result.fnArray[2]).toEqual(OPS.fill);
-    done();
-   });
-  });
-  it('should handle three glued operations #2', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var resources = new ResourcesMock();
-   resources.Res1 = {};
-   var stream = new StringStream('B*Bf*');
-   runOperatorListCheck(evaluator, stream, resources, function (result) {
-    expect(!!result.fnArray && !!result.argsArray).toEqual(true);
-    expect(result.fnArray.length).toEqual(3);
-    expect(result.fnArray[0]).toEqual(OPS.eoFillStroke);
-    expect(result.fnArray[1]).toEqual(OPS.fillStroke);
-    expect(result.fnArray[2]).toEqual(OPS.eoFill);
-    done();
-   });
-  });
-  it('should handle glued operations and operands', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('f5 Ts');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(!!result.fnArray && !!result.argsArray).toEqual(true);
-    expect(result.fnArray.length).toEqual(2);
-    expect(result.fnArray[0]).toEqual(OPS.fill);
-    expect(result.fnArray[1]).toEqual(OPS.setTextRise);
-    expect(result.argsArray.length).toEqual(2);
-    expect(result.argsArray[1].length).toEqual(1);
-    expect(result.argsArray[1][0]).toEqual(5);
-    done();
-   });
-  });
-  it('should handle glued operations and literals', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('trueifalserinulln');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(!!result.fnArray && !!result.argsArray).toEqual(true);
-    expect(result.fnArray.length).toEqual(3);
-    expect(result.fnArray[0]).toEqual(OPS.setFlatness);
-    expect(result.fnArray[1]).toEqual(OPS.setRenderingIntent);
-    expect(result.fnArray[2]).toEqual(OPS.endPath);
-    expect(result.argsArray.length).toEqual(3);
-    expect(result.argsArray[0].length).toEqual(1);
-    expect(result.argsArray[0][0]).toEqual(true);
-    expect(result.argsArray[1].length).toEqual(1);
-    expect(result.argsArray[1][0]).toEqual(false);
-    expect(result.argsArray[2]).toEqual(null);
-    done();
-   });
+  describe('splitCombinedOperations', function () {
+    it('should reject unknown operations', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('fTT');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(!!result.fnArray && !!result.argsArray).toEqual(true);
+        expect(result.fnArray.length).toEqual(1);
+        expect(result.fnArray[0]).toEqual(OPS.fill);
+        expect(result.argsArray[0]).toEqual(null);
+        done();
+      });
+    });
+    it('should handle one operations', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('Q');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(!!result.fnArray && !!result.argsArray).toEqual(true);
+        expect(result.fnArray.length).toEqual(1);
+        expect(result.fnArray[0]).toEqual(OPS.restore);
+        done();
+      });
+    });
+    it('should handle two glued operations', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var resources = new ResourcesMock();
+      resources.Res1 = {};
+      var stream = new StringStream('/Res1 DoQ');
+      runOperatorListCheck(evaluator, stream, resources, function (result) {
+        expect(!!result.fnArray && !!result.argsArray).toEqual(true);
+        expect(result.fnArray.length).toEqual(2);
+        expect(result.fnArray[0]).toEqual(OPS.paintXObject);
+        expect(result.fnArray[1]).toEqual(OPS.restore);
+        done();
+      });
+    });
+    it('should handle tree glued operations', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('fff');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(!!result.fnArray && !!result.argsArray).toEqual(true);
+        expect(result.fnArray.length).toEqual(3);
+        expect(result.fnArray[0]).toEqual(OPS.fill);
+        expect(result.fnArray[1]).toEqual(OPS.fill);
+        expect(result.fnArray[2]).toEqual(OPS.fill);
+        done();
+      });
+    });
+    it('should handle three glued operations #2', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var resources = new ResourcesMock();
+      resources.Res1 = {};
+      var stream = new StringStream('B*Bf*');
+      runOperatorListCheck(evaluator, stream, resources, function (result) {
+        expect(!!result.fnArray && !!result.argsArray).toEqual(true);
+        expect(result.fnArray.length).toEqual(3);
+        expect(result.fnArray[0]).toEqual(OPS.eoFillStroke);
+        expect(result.fnArray[1]).toEqual(OPS.fillStroke);
+        expect(result.fnArray[2]).toEqual(OPS.eoFill);
+        done();
+      });
+    });
+    it('should handle glued operations and operands', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('f5 Ts');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(!!result.fnArray && !!result.argsArray).toEqual(true);
+        expect(result.fnArray.length).toEqual(2);
+        expect(result.fnArray[0]).toEqual(OPS.fill);
+        expect(result.fnArray[1]).toEqual(OPS.setTextRise);
+        expect(result.argsArray.length).toEqual(2);
+        expect(result.argsArray[1].length).toEqual(1);
+        expect(result.argsArray[1][0]).toEqual(5);
+        done();
+      });
+    });
+    it('should handle glued operations and literals', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('trueifalserinulln');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(!!result.fnArray && !!result.argsArray).toEqual(true);
+        expect(result.fnArray.length).toEqual(3);
+        expect(result.fnArray[0]).toEqual(OPS.setFlatness);
+        expect(result.fnArray[1]).toEqual(OPS.setRenderingIntent);
+        expect(result.fnArray[2]).toEqual(OPS.endPath);
+        expect(result.argsArray.length).toEqual(3);
+        expect(result.argsArray[0].length).toEqual(1);
+        expect(result.argsArray[0][0]).toEqual(true);
+        expect(result.argsArray[1].length).toEqual(1);
+        expect(result.argsArray[1][0]).toEqual(false);
+        expect(result.argsArray[2]).toEqual(null);
+        done();
+      });
+    });
   });
- });
- describe('validateNumberOfArgs', function () {
-  it('should execute if correct number of arguments', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('5 1 d0');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(result.argsArray[0][0]).toEqual(5);
-    expect(result.argsArray[0][1]).toEqual(1);
-    expect(result.fnArray[0]).toEqual(OPS.setCharWidth);
-    done();
-   });
+  describe('validateNumberOfArgs', function () {
+    it('should execute if correct number of arguments', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('5 1 d0');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(result.argsArray[0][0]).toEqual(5);
+        expect(result.argsArray[0][1]).toEqual(1);
+        expect(result.fnArray[0]).toEqual(OPS.setCharWidth);
+        done();
+      });
+    });
+    it('should execute if too many arguments', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('5 1 4 d0');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(result.argsArray[0][0]).toEqual(1);
+        expect(result.argsArray[0][1]).toEqual(4);
+        expect(result.fnArray[0]).toEqual(OPS.setCharWidth);
+        done();
+      });
+    });
+    it('should execute if nested commands', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('/F2 /GS2 gs 5.711 Tf');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(result.fnArray.length).toEqual(3);
+        expect(result.fnArray[0]).toEqual(OPS.setGState);
+        expect(result.fnArray[1]).toEqual(OPS.dependency);
+        expect(result.fnArray[2]).toEqual(OPS.setFont);
+        expect(result.argsArray.length).toEqual(3);
+        expect(result.argsArray[0].length).toEqual(1);
+        expect(result.argsArray[1].length).toEqual(1);
+        expect(result.argsArray[2].length).toEqual(2);
+        done();
+      });
+    });
+    it('should skip if too few arguments', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('5 d0');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(result.argsArray).toEqual([]);
+        expect(result.fnArray).toEqual([]);
+        done();
+      });
+    });
+    it('should close opened saves', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('qq');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(!!result.fnArray && !!result.argsArray).toEqual(true);
+        expect(result.fnArray.length).toEqual(4);
+        expect(result.fnArray[0]).toEqual(OPS.save);
+        expect(result.fnArray[1]).toEqual(OPS.save);
+        expect(result.fnArray[2]).toEqual(OPS.restore);
+        expect(result.fnArray[3]).toEqual(OPS.restore);
+        done();
+      });
+    });
+    it('should skip paintXObject if name is missing', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('/ Do');
+      runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
+        expect(result.argsArray).toEqual([]);
+        expect(result.fnArray).toEqual([]);
+        done();
+      });
+    });
+    it('should skip paintXObject if subtype is PS', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var xobjStreamDict = new Dict();
+      xobjStreamDict.set('Subtype', Name.get('PS'));
+      var xobjStream = new Stream([], 0, 0, xobjStreamDict);
+      var xobjs = new Dict();
+      xobjs.set('Res1', xobjStream);
+      var resources = new Dict();
+      resources.set('XObject', xobjs);
+      var stream = new StringStream('/Res1 Do');
+      runOperatorListCheck(evaluator, stream, resources, function (result) {
+        expect(result.argsArray).toEqual([]);
+        expect(result.fnArray).toEqual([]);
+        done();
+      });
+    });
   });
-  it('should execute if too many arguments', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('5 1 4 d0');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(result.argsArray[0][0]).toEqual(1);
-    expect(result.argsArray[0][1]).toEqual(4);
-    expect(result.fnArray[0]).toEqual(OPS.setCharWidth);
-    done();
-   });
+  describe('thread control', function () {
+    it('should abort operator list parsing', function (done) {
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('qqQQ');
+      var resources = new ResourcesMock();
+      var result = new OperatorList();
+      var task = new WorkerTask('OperatorListAbort');
+      task.terminate();
+      evaluator.getOperatorList(stream, task, resources, result).catch(function () {
+        expect(!!result.fnArray && !!result.argsArray).toEqual(true);
+        expect(result.fnArray.length).toEqual(0);
+        done();
+      });
+    });
+    it('should abort text parsing parsing', function (done) {
+      var resources = new ResourcesMock();
+      var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
+      var stream = new StringStream('qqQQ');
+      var task = new WorkerTask('TextContentAbort');
+      task.terminate();
+      evaluator.getTextContent(stream, task, resources).catch(function () {
+        expect(true).toEqual(true);
+        done();
+      });
+    });
   });
-  it('should execute if nested commands', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('/F2 /GS2 gs 5.711 Tf');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(result.fnArray.length).toEqual(3);
-    expect(result.fnArray[0]).toEqual(OPS.setGState);
-    expect(result.fnArray[1]).toEqual(OPS.dependency);
-    expect(result.fnArray[2]).toEqual(OPS.setFont);
-    expect(result.argsArray.length).toEqual(3);
-    expect(result.argsArray[0].length).toEqual(1);
-    expect(result.argsArray[1].length).toEqual(1);
-    expect(result.argsArray[2].length).toEqual(2);
-    done();
-   });
-  });
-  it('should skip if too few arguments', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('5 d0');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(result.argsArray).toEqual([]);
-    expect(result.fnArray).toEqual([]);
-    done();
-   });
-  });
-  it('should close opened saves', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('qq');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(!!result.fnArray && !!result.argsArray).toEqual(true);
-    expect(result.fnArray.length).toEqual(4);
-    expect(result.fnArray[0]).toEqual(OPS.save);
-    expect(result.fnArray[1]).toEqual(OPS.save);
-    expect(result.fnArray[2]).toEqual(OPS.restore);
-    expect(result.fnArray[3]).toEqual(OPS.restore);
-    done();
-   });
-  });
-  it('should skip paintXObject if name is missing', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('/ Do');
-   runOperatorListCheck(evaluator, stream, new ResourcesMock(), function (result) {
-    expect(result.argsArray).toEqual([]);
-    expect(result.fnArray).toEqual([]);
-    done();
-   });
-  });
-  it('should skip paintXObject if subtype is PS', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var xobjStreamDict = new Dict();
-   xobjStreamDict.set('Subtype', Name.get('PS'));
-   var xobjStream = new Stream([], 0, 0, xobjStreamDict);
-   var xobjs = new Dict();
-   xobjs.set('Res1', xobjStream);
-   var resources = new Dict();
-   resources.set('XObject', xobjs);
-   var stream = new StringStream('/Res1 Do');
-   runOperatorListCheck(evaluator, stream, resources, function (result) {
-    expect(result.argsArray).toEqual([]);
-    expect(result.fnArray).toEqual([]);
-    done();
-   });
-  });
- });
- describe('thread control', function () {
-  it('should abort operator list parsing', function (done) {
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('qqQQ');
-   var resources = new ResourcesMock();
-   var result = new OperatorList();
-   var task = new WorkerTask('OperatorListAbort');
-   task.terminate();
-   evaluator.getOperatorList(stream, task, resources, result).catch(function () {
-    expect(!!result.fnArray && !!result.argsArray).toEqual(true);
-    expect(result.fnArray.length).toEqual(0);
-    done();
-   });
-  });
-  it('should abort text parsing parsing', function (done) {
-   var resources = new ResourcesMock();
-   var evaluator = new PartialEvaluator(new PdfManagerMock(), new XrefMock(), new HandlerMock(), 'prefix');
-   var stream = new StringStream('qqQQ');
-   var task = new WorkerTask('TextContentAbort');
-   task.terminate();
-   evaluator.getTextContent(stream, task, resources).catch(function () {
-    expect(true).toEqual(true);
-    done();
-   });
-  });
- });
- describe('operator list', function () {
-  function MessageHandlerMock() {
-  }
-  MessageHandlerMock.prototype = {
-   send: function () {
-   }
-  };
-  it('should get correct total length after flushing', function () {
-   var operatorList = new OperatorList(null, new MessageHandlerMock());
-   operatorList.addOp(OPS.save, null);
-   operatorList.addOp(OPS.restore, null);
-   expect(operatorList.totalLength).toEqual(2);
-   expect(operatorList.length).toEqual(2);
-   operatorList.flush();
-   expect(operatorList.totalLength).toEqual(2);
-   expect(operatorList.length).toEqual(0);
+  describe('operator list', function () {
+    function MessageHandlerMock() {}
+    MessageHandlerMock.prototype = {
+      send: function () {}
+    };
+    it('should get correct total length after flushing', function () {
+      var operatorList = new OperatorList(null, new MessageHandlerMock());
+      operatorList.addOp(OPS.save, null);
+      operatorList.addOp(OPS.restore, null);
+      expect(operatorList.totalLength).toEqual(2);
+      expect(operatorList.length).toEqual(2);
+      operatorList.flush();
+      expect(operatorList.totalLength).toEqual(2);
+      expect(operatorList.length).toEqual(0);
+    });
   });
- });
 });

+ 48 - 46
lib/test/unit/fonts_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreFonts = require('../../core/fonts.js');
 var sharedUtil = require('../../shared/util.js');
 var ProblematicCharRanges = coreFonts.ProblematicCharRanges;
@@ -20,57 +21,58 @@ var PRIVATE_USE_OFFSET_START = coreFonts.PRIVATE_USE_OFFSET_START;
 var PRIVATE_USE_OFFSET_END = coreFonts.PRIVATE_USE_OFFSET_END;
 var isInt = sharedUtil.isInt;
 var checkProblematicCharRanges = function checkProblematicCharRanges() {
- function printRange(limits) {
-  return '[' + limits.lower.toString('16').toUpperCase() + ', ' + limits.upper.toString('16').toUpperCase() + ')';
- }
- var numRanges = ProblematicCharRanges.length;
- if (numRanges % 2 !== 0) {
-  throw new Error('Char ranges must contain an even number of elements.');
- }
- var prevLimits, numChars = 0;
- for (var i = 0; i < numRanges; i += 2) {
-  var limits = {
-   lower: ProblematicCharRanges[i],
-   upper: ProblematicCharRanges[i + 1]
-  };
-  if (!isInt(limits.lower) || !isInt(limits.upper)) {
-   throw new Error('Range endpoints must be integers: ' + printRange(limits));
+  function printRange(limits) {
+    return '[' + limits.lower.toString('16').toUpperCase() + ', ' + limits.upper.toString('16').toUpperCase() + ')';
   }
-  if (limits.lower < 0 || limits.upper < 0) {
-   throw new Error('Range endpoints must be non-negative: ' + printRange(limits));
+  var numRanges = ProblematicCharRanges.length;
+  if (numRanges % 2 !== 0) {
+    throw new Error('Char ranges must contain an even number of elements.');
   }
-  var range = limits.upper - limits.lower;
-  if (range < 1) {
-   throw new Error('Range must contain at least one element: ' + printRange(limits));
+  var prevLimits,
+      numChars = 0;
+  for (var i = 0; i < numRanges; i += 2) {
+    var limits = {
+      lower: ProblematicCharRanges[i],
+      upper: ProblematicCharRanges[i + 1]
+    };
+    if (!isInt(limits.lower) || !isInt(limits.upper)) {
+      throw new Error('Range endpoints must be integers: ' + printRange(limits));
+    }
+    if (limits.lower < 0 || limits.upper < 0) {
+      throw new Error('Range endpoints must be non-negative: ' + printRange(limits));
+    }
+    var range = limits.upper - limits.lower;
+    if (range < 1) {
+      throw new Error('Range must contain at least one element: ' + printRange(limits));
+    }
+    if (prevLimits) {
+      if (limits.lower < prevLimits.lower) {
+        throw new Error('Ranges must be sorted in ascending order: ' + printRange(limits) + ', ' + printRange(prevLimits));
+      }
+      if (limits.lower < prevLimits.upper) {
+        throw new Error('Ranges must not overlap: ' + printRange(limits) + ', ' + printRange(prevLimits));
+      }
+    }
+    prevLimits = {
+      lower: limits.lower,
+      upper: limits.upper
+    };
+    numChars += range;
   }
-  if (prevLimits) {
-   if (limits.lower < prevLimits.lower) {
-    throw new Error('Ranges must be sorted in ascending order: ' + printRange(limits) + ', ' + printRange(prevLimits));
-   }
-   if (limits.lower < prevLimits.upper) {
-    throw new Error('Ranges must not overlap: ' + printRange(limits) + ', ' + printRange(prevLimits));
-   }
+  var puaLength = PRIVATE_USE_OFFSET_END + 1 - PRIVATE_USE_OFFSET_START;
+  if (numChars > puaLength) {
+    throw new Error('Total number of chars must not exceed the PUA length.');
   }
-  prevLimits = {
-   lower: limits.lower,
-   upper: limits.upper
+  return {
+    numChars: numChars,
+    puaLength: puaLength,
+    percentage: 100 * (numChars / puaLength)
   };
-  numChars += range;
- }
- var puaLength = PRIVATE_USE_OFFSET_END + 1 - PRIVATE_USE_OFFSET_START;
- if (numChars > puaLength) {
-  throw new Error('Total number of chars must not exceed the PUA length.');
- }
- return {
-  numChars: numChars,
-  puaLength: puaLength,
-  percentage: 100 * (numChars / puaLength)
- };
 };
 describe('Fonts', function () {
- it('checkProblematicCharRanges', function () {
-  var EXPECTED_PERCENTAGE = 45;
-  var result = checkProblematicCharRanges();
-  expect(result.percentage).toBeLessThan(EXPECTED_PERCENTAGE);
- });
+  it('checkProblematicCharRanges', function () {
+    var EXPECTED_PERCENTAGE = 45;
+    var result = checkProblematicCharRanges();
+    expect(result.percentage).toBeLessThan(EXPECTED_PERCENTAGE);
+  });
 });

+ 596 - 1082
lib/test/unit/function_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreFunction = require('../../core/function.js');
 var corePsParser = require('../../core/ps_parser.js');
 var coreStream = require('../../core/stream.js');
@@ -24,1091 +25,604 @@ var PostScriptLexer = corePsParser.PostScriptLexer;
 var StringStream = coreStream.StringStream;
 var isArray = sharedUtil.isArray;
 describe('function', function () {
- beforeEach(function () {
-  jasmine.addMatchers({
-   toMatchArray: function (util, customEqualityTesters) {
-    return {
-     compare: function (actual, expected) {
-      var result = {};
-      if (actual.length !== expected.length) {
-       result.pass = false;
-       result.message = 'Array length: ' + actual.length + ', expected: ' + expected.length;
-       return result;
+  beforeEach(function () {
+    jasmine.addMatchers({
+      toMatchArray: function (util, customEqualityTesters) {
+        return {
+          compare: function (actual, expected) {
+            var result = {};
+            if (actual.length !== expected.length) {
+              result.pass = false;
+              result.message = 'Array length: ' + actual.length + ', expected: ' + expected.length;
+              return result;
+            }
+            result.pass = true;
+            for (var i = 0; i < expected.length; i++) {
+              var a = actual[i],
+                  b = expected[i];
+              if (isArray(b)) {
+                if (a.length !== b.length) {
+                  result.pass = false;
+                  break;
+                }
+                for (var j = 0; j < a.length; j++) {
+                  var suba = a[j],
+                      subb = b[j];
+                  if (suba !== subb) {
+                    result.pass = false;
+                    break;
+                  }
+                }
+              } else {
+                if (a !== b) {
+                  result.pass = false;
+                  break;
+                }
+              }
+            }
+            return result;
+          }
+        };
       }
-      result.pass = true;
-      for (var i = 0; i < expected.length; i++) {
-       var a = actual[i], b = expected[i];
-       if (isArray(b)) {
-        if (a.length !== b.length) {
-         result.pass = false;
-         break;
-        }
-        for (var j = 0; j < a.length; j++) {
-         var suba = a[j], subb = b[j];
-         if (suba !== subb) {
-          result.pass = false;
-          break;
-         }
-        }
-       } else {
-        if (a !== b) {
-         result.pass = false;
-         break;
+    });
+  });
+  describe('PostScriptParser', function () {
+    function parse(program) {
+      var stream = new StringStream(program);
+      var parser = new PostScriptParser(new PostScriptLexer(stream));
+      return parser.parse();
+    }
+    it('parses empty programs', function () {
+      var output = parse('{}');
+      expect(output.length).toEqual(0);
+    });
+    it('parses positive numbers', function () {
+      var number = 999;
+      var program = parse('{ ' + number + ' }');
+      var expectedProgram = [number];
+      expect(program).toMatchArray(expectedProgram);
+    });
+    it('parses negative numbers', function () {
+      var number = -999;
+      var program = parse('{ ' + number + ' }');
+      var expectedProgram = [number];
+      expect(program).toMatchArray(expectedProgram);
+    });
+    it('parses negative floats', function () {
+      var number = 3.3;
+      var program = parse('{ ' + number + ' }');
+      var expectedProgram = [number];
+      expect(program).toMatchArray(expectedProgram);
+    });
+    it('parses operators', function () {
+      var program = parse('{ sub }');
+      var expectedProgram = ['sub'];
+      expect(program).toMatchArray(expectedProgram);
+    });
+    it('parses if statements', function () {
+      var program = parse('{ { 99 } if }');
+      var expectedProgram = [3, 'jz', 99];
+      expect(program).toMatchArray(expectedProgram);
+    });
+    it('parses ifelse statements', function () {
+      var program = parse('{ { 99 } { 44 } ifelse }');
+      var expectedProgram = [5, 'jz', 99, 6, 'j', 44];
+      expect(program).toMatchArray(expectedProgram);
+    });
+    it('handles missing brackets', function () {
+      expect(function () {
+        parse('{');
+      }).toThrow(new Error('Unexpected symbol: found undefined expected 1.'));
+    });
+    it('handles junk after the end', function () {
+      var number = 3.3;
+      var program = parse('{ ' + number + ' }#');
+      var expectedProgram = [number];
+      expect(program).toMatchArray(expectedProgram);
+    });
+  });
+  describe('PostScriptEvaluator', function () {
+    function evaluate(program) {
+      var stream = new StringStream(program);
+      var parser = new PostScriptParser(new PostScriptLexer(stream));
+      var code = parser.parse();
+      var evaluator = new PostScriptEvaluator(code);
+      var output = evaluator.execute();
+      return output;
+    }
+    it('pushes stack', function () {
+      var stack = evaluate('{ 99 }');
+      var expectedStack = [99];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles if with true', function () {
+      var stack = evaluate('{ 1 {99} if }');
+      var expectedStack = [99];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles if with false', function () {
+      var stack = evaluate('{ 0 {99} if }');
+      var expectedStack = [];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles ifelse with true', function () {
+      var stack = evaluate('{ 1 {99} {77} ifelse }');
+      var expectedStack = [99];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles ifelse with false', function () {
+      var stack = evaluate('{ 0 {99} {77} ifelse }');
+      var expectedStack = [77];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles nested if', function () {
+      var stack = evaluate('{ 1 {1 {77} if} if }');
+      var expectedStack = [77];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('abs', function () {
+      var stack = evaluate('{ -2 abs }');
+      var expectedStack = [2];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('adds', function () {
+      var stack = evaluate('{ 1 2 add }');
+      var expectedStack = [3];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('boolean and', function () {
+      var stack = evaluate('{ true false and }');
+      var expectedStack = [false];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('bitwise and', function () {
+      var stack = evaluate('{ 254 1 and }');
+      var expectedStack = [254 & 1];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates the inverse tangent of a number', function () {
+      var stack = evaluate('{ 90 atan }');
+      var expectedStack = [Math.atan(90)];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles bitshifting ', function () {
+      var stack = evaluate('{ 50 2 bitshift }');
+      var expectedStack = [200];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates the ceiling value', function () {
+      var stack = evaluate('{ 9.9 ceiling }');
+      var expectedStack = [10];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('copies', function () {
+      var stack = evaluate('{ 99 98 2 copy }');
+      var expectedStack = [99, 98, 99, 98];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates the cosine of a number', function () {
+      var stack = evaluate('{ 90 cos }');
+      var expectedStack = [Math.cos(90)];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('converts to int', function () {
+      var stack = evaluate('{ 9.9 cvi }');
+      var expectedStack = [9];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('converts negatives to int', function () {
+      var stack = evaluate('{ -9.9 cvi }');
+      var expectedStack = [-9];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('converts to real', function () {
+      var stack = evaluate('{ 55.34 cvr }');
+      var expectedStack = [55.34];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('divides', function () {
+      var stack = evaluate('{ 6 5 div }');
+      var expectedStack = [1.2];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('maps division by zero to infinity', function () {
+      var stack = evaluate('{ 6 0 div }');
+      var expectedStack = [Infinity];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('duplicates', function () {
+      var stack = evaluate('{ 99 dup }');
+      var expectedStack = [99, 99];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('accepts an equality', function () {
+      var stack = evaluate('{ 9 9 eq }');
+      var expectedStack = [true];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('rejects an inequality', function () {
+      var stack = evaluate('{ 9 8 eq }');
+      var expectedStack = [false];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('exchanges', function () {
+      var stack = evaluate('{ 44 99 exch }');
+      var expectedStack = [99, 44];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles exponentiation', function () {
+      var stack = evaluate('{ 10 2 exp }');
+      var expectedStack = [100];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('pushes false onto the stack', function () {
+      var stack = evaluate('{ false }');
+      var expectedStack = [false];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates the floor value', function () {
+      var stack = evaluate('{ 9.9 floor }');
+      var expectedStack = [9];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles greater than or equal to', function () {
+      var stack = evaluate('{ 10 9 ge }');
+      var expectedStack = [true];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('rejects less than for greater than or equal to', function () {
+      var stack = evaluate('{ 8 9 ge }');
+      var expectedStack = [false];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles greater than', function () {
+      var stack = evaluate('{ 10 9 gt }');
+      var expectedStack = [true];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('rejects less than or equal for greater than', function () {
+      var stack = evaluate('{ 9 9 gt }');
+      var expectedStack = [false];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('divides to integer', function () {
+      var stack = evaluate('{ 2 3 idiv }');
+      var expectedStack = [0];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('divides to negative integer', function () {
+      var stack = evaluate('{ -2 3 idiv }');
+      var expectedStack = [0];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('duplicates index', function () {
+      var stack = evaluate('{ 4 3 2 1 2 index }');
+      var expectedStack = [4, 3, 2, 1, 3];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles less than or equal to', function () {
+      var stack = evaluate('{ 9 10 le }');
+      var expectedStack = [true];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('rejects greater than for less than or equal to', function () {
+      var stack = evaluate('{ 10 9 le }');
+      var expectedStack = [false];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates the natural logarithm', function () {
+      var stack = evaluate('{ 10 ln }');
+      var expectedStack = [Math.log(10)];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates the base 10 logarithm', function () {
+      var stack = evaluate('{ 100 log }');
+      var expectedStack = [2];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('handles less than', function () {
+      var stack = evaluate('{ 9 10 lt }');
+      var expectedStack = [true];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('rejects greater than or equal to for less than', function () {
+      var stack = evaluate('{ 10 9 lt }');
+      var expectedStack = [false];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('performs the modulo operation', function () {
+      var stack = evaluate('{ 4 3 mod }');
+      var expectedStack = [1];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('multiplies two numbers (positive result)', function () {
+      var stack = evaluate('{ 9 8 mul }');
+      var expectedStack = [72];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('multiplies two numbers (negative result)', function () {
+      var stack = evaluate('{ 9 -8 mul }');
+      var expectedStack = [-72];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('accepts an inequality', function () {
+      var stack = evaluate('{ 9 8 ne }');
+      var expectedStack = [true];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('rejects an equality', function () {
+      var stack = evaluate('{ 9 9 ne }');
+      var expectedStack = [false];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('negates', function () {
+      var stack = evaluate('{ 4.5 neg }');
+      var expectedStack = [-4.5];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('boolean not', function () {
+      var stack = evaluate('{ true not }');
+      var expectedStack = [false];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('bitwise not', function () {
+      var stack = evaluate('{ 12 not }');
+      var expectedStack = [-13];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('boolean or', function () {
+      var stack = evaluate('{ true false or }');
+      var expectedStack = [true];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('bitwise or', function () {
+      var stack = evaluate('{ 254 1 or }');
+      var expectedStack = [254 | 1];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('pops stack', function () {
+      var stack = evaluate('{ 1 2 pop }');
+      var expectedStack = [1];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('rolls stack right', function () {
+      var stack = evaluate('{ 1 3 2 2 4 1 roll }');
+      var expectedStack = [2, 1, 3, 2];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('rolls stack left', function () {
+      var stack = evaluate('{ 1 3 2 2 4 -1 roll }');
+      var expectedStack = [3, 2, 2, 1];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('rounds a number', function () {
+      var stack = evaluate('{ 9.52 round }');
+      var expectedStack = [10];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates the sine of a number', function () {
+      var stack = evaluate('{ 90 sin }');
+      var expectedStack = [Math.sin(90)];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates a square root (integer)', function () {
+      var stack = evaluate('{ 100 sqrt }');
+      var expectedStack = [10];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates a square root (float)', function () {
+      var stack = evaluate('{ 99 sqrt }');
+      var expectedStack = [Math.sqrt(99)];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('subtracts (positive result)', function () {
+      var stack = evaluate('{ 6 4 sub }');
+      var expectedStack = [2];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('subtracts (negative result)', function () {
+      var stack = evaluate('{ 4 6 sub }');
+      var expectedStack = [-2];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('pushes true onto the stack', function () {
+      var stack = evaluate('{ true }');
+      var expectedStack = [true];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('truncates a number', function () {
+      var stack = evaluate('{ 35.004 truncate }');
+      var expectedStack = [35];
+      expect(stack).toMatchArray(expectedStack);
+    });
+    it('calculates an exclusive or value', function () {
+      var stack = evaluate('{ 3 9 xor }');
+      var expectedStack = [10];
+      expect(stack).toMatchArray(expectedStack);
+    });
+  });
+  describe('PostScriptCompiler', function () {
+    function check(code, domain, range, samples) {
+      var compiler = new PostScriptCompiler();
+      var compiledCode = compiler.compile(code, domain, range);
+      if (samples === null) {
+        expect(compiledCode).toBeNull();
+      } else {
+        expect(compiledCode).not.toBeNull();
+        var fn = new Function('src', 'srcOffset', 'dest', 'destOffset', compiledCode);
+        for (var i = 0; i < samples.length; i++) {
+          var out = new Float32Array(samples[i].output.length);
+          fn(samples[i].input, 0, out, 0);
+          expect(Array.prototype.slice.call(out, 0)).toMatchArray(samples[i].output);
         }
-       }
       }
-      return result;
-     }
-    };
-   }
-  });
- });
- describe('PostScriptParser', function () {
-  function parse(program) {
-   var stream = new StringStream(program);
-   var parser = new PostScriptParser(new PostScriptLexer(stream));
-   return parser.parse();
-  }
-  it('parses empty programs', function () {
-   var output = parse('{}');
-   expect(output.length).toEqual(0);
-  });
-  it('parses positive numbers', function () {
-   var number = 999;
-   var program = parse('{ ' + number + ' }');
-   var expectedProgram = [number];
-   expect(program).toMatchArray(expectedProgram);
-  });
-  it('parses negative numbers', function () {
-   var number = -999;
-   var program = parse('{ ' + number + ' }');
-   var expectedProgram = [number];
-   expect(program).toMatchArray(expectedProgram);
-  });
-  it('parses negative floats', function () {
-   var number = 3.3;
-   var program = parse('{ ' + number + ' }');
-   var expectedProgram = [number];
-   expect(program).toMatchArray(expectedProgram);
-  });
-  it('parses operators', function () {
-   var program = parse('{ sub }');
-   var expectedProgram = ['sub'];
-   expect(program).toMatchArray(expectedProgram);
-  });
-  it('parses if statements', function () {
-   var program = parse('{ { 99 } if }');
-   var expectedProgram = [
-    3,
-    'jz',
-    99
-   ];
-   expect(program).toMatchArray(expectedProgram);
-  });
-  it('parses ifelse statements', function () {
-   var program = parse('{ { 99 } { 44 } ifelse }');
-   var expectedProgram = [
-    5,
-    'jz',
-    99,
-    6,
-    'j',
-    44
-   ];
-   expect(program).toMatchArray(expectedProgram);
-  });
-  it('handles missing brackets', function () {
-   expect(function () {
-    parse('{');
-   }).toThrow(new Error('Unexpected symbol: found undefined expected 1.'));
-  });
-  it('handles junk after the end', function () {
-   var number = 3.3;
-   var program = parse('{ ' + number + ' }#');
-   var expectedProgram = [number];
-   expect(program).toMatchArray(expectedProgram);
-  });
- });
- describe('PostScriptEvaluator', function () {
-  function evaluate(program) {
-   var stream = new StringStream(program);
-   var parser = new PostScriptParser(new PostScriptLexer(stream));
-   var code = parser.parse();
-   var evaluator = new PostScriptEvaluator(code);
-   var output = evaluator.execute();
-   return output;
-  }
-  it('pushes stack', function () {
-   var stack = evaluate('{ 99 }');
-   var expectedStack = [99];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles if with true', function () {
-   var stack = evaluate('{ 1 {99} if }');
-   var expectedStack = [99];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles if with false', function () {
-   var stack = evaluate('{ 0 {99} if }');
-   var expectedStack = [];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles ifelse with true', function () {
-   var stack = evaluate('{ 1 {99} {77} ifelse }');
-   var expectedStack = [99];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles ifelse with false', function () {
-   var stack = evaluate('{ 0 {99} {77} ifelse }');
-   var expectedStack = [77];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles nested if', function () {
-   var stack = evaluate('{ 1 {1 {77} if} if }');
-   var expectedStack = [77];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('abs', function () {
-   var stack = evaluate('{ -2 abs }');
-   var expectedStack = [2];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('adds', function () {
-   var stack = evaluate('{ 1 2 add }');
-   var expectedStack = [3];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('boolean and', function () {
-   var stack = evaluate('{ true false and }');
-   var expectedStack = [false];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('bitwise and', function () {
-   var stack = evaluate('{ 254 1 and }');
-   var expectedStack = [254 & 1];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates the inverse tangent of a number', function () {
-   var stack = evaluate('{ 90 atan }');
-   var expectedStack = [Math.atan(90)];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles bitshifting ', function () {
-   var stack = evaluate('{ 50 2 bitshift }');
-   var expectedStack = [200];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates the ceiling value', function () {
-   var stack = evaluate('{ 9.9 ceiling }');
-   var expectedStack = [10];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('copies', function () {
-   var stack = evaluate('{ 99 98 2 copy }');
-   var expectedStack = [
-    99,
-    98,
-    99,
-    98
-   ];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates the cosine of a number', function () {
-   var stack = evaluate('{ 90 cos }');
-   var expectedStack = [Math.cos(90)];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('converts to int', function () {
-   var stack = evaluate('{ 9.9 cvi }');
-   var expectedStack = [9];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('converts negatives to int', function () {
-   var stack = evaluate('{ -9.9 cvi }');
-   var expectedStack = [-9];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('converts to real', function () {
-   var stack = evaluate('{ 55.34 cvr }');
-   var expectedStack = [55.34];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('divides', function () {
-   var stack = evaluate('{ 6 5 div }');
-   var expectedStack = [1.2];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('maps division by zero to infinity', function () {
-   var stack = evaluate('{ 6 0 div }');
-   var expectedStack = [Infinity];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('duplicates', function () {
-   var stack = evaluate('{ 99 dup }');
-   var expectedStack = [
-    99,
-    99
-   ];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('accepts an equality', function () {
-   var stack = evaluate('{ 9 9 eq }');
-   var expectedStack = [true];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('rejects an inequality', function () {
-   var stack = evaluate('{ 9 8 eq }');
-   var expectedStack = [false];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('exchanges', function () {
-   var stack = evaluate('{ 44 99 exch }');
-   var expectedStack = [
-    99,
-    44
-   ];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles exponentiation', function () {
-   var stack = evaluate('{ 10 2 exp }');
-   var expectedStack = [100];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('pushes false onto the stack', function () {
-   var stack = evaluate('{ false }');
-   var expectedStack = [false];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates the floor value', function () {
-   var stack = evaluate('{ 9.9 floor }');
-   var expectedStack = [9];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles greater than or equal to', function () {
-   var stack = evaluate('{ 10 9 ge }');
-   var expectedStack = [true];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('rejects less than for greater than or equal to', function () {
-   var stack = evaluate('{ 8 9 ge }');
-   var expectedStack = [false];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles greater than', function () {
-   var stack = evaluate('{ 10 9 gt }');
-   var expectedStack = [true];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('rejects less than or equal for greater than', function () {
-   var stack = evaluate('{ 9 9 gt }');
-   var expectedStack = [false];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('divides to integer', function () {
-   var stack = evaluate('{ 2 3 idiv }');
-   var expectedStack = [0];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('divides to negative integer', function () {
-   var stack = evaluate('{ -2 3 idiv }');
-   var expectedStack = [0];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('duplicates index', function () {
-   var stack = evaluate('{ 4 3 2 1 2 index }');
-   var expectedStack = [
-    4,
-    3,
-    2,
-    1,
-    3
-   ];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles less than or equal to', function () {
-   var stack = evaluate('{ 9 10 le }');
-   var expectedStack = [true];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('rejects greater than for less than or equal to', function () {
-   var stack = evaluate('{ 10 9 le }');
-   var expectedStack = [false];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates the natural logarithm', function () {
-   var stack = evaluate('{ 10 ln }');
-   var expectedStack = [Math.log(10)];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates the base 10 logarithm', function () {
-   var stack = evaluate('{ 100 log }');
-   var expectedStack = [2];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('handles less than', function () {
-   var stack = evaluate('{ 9 10 lt }');
-   var expectedStack = [true];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('rejects greater than or equal to for less than', function () {
-   var stack = evaluate('{ 10 9 lt }');
-   var expectedStack = [false];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('performs the modulo operation', function () {
-   var stack = evaluate('{ 4 3 mod }');
-   var expectedStack = [1];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('multiplies two numbers (positive result)', function () {
-   var stack = evaluate('{ 9 8 mul }');
-   var expectedStack = [72];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('multiplies two numbers (negative result)', function () {
-   var stack = evaluate('{ 9 -8 mul }');
-   var expectedStack = [-72];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('accepts an inequality', function () {
-   var stack = evaluate('{ 9 8 ne }');
-   var expectedStack = [true];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('rejects an equality', function () {
-   var stack = evaluate('{ 9 9 ne }');
-   var expectedStack = [false];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('negates', function () {
-   var stack = evaluate('{ 4.5 neg }');
-   var expectedStack = [-4.5];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('boolean not', function () {
-   var stack = evaluate('{ true not }');
-   var expectedStack = [false];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('bitwise not', function () {
-   var stack = evaluate('{ 12 not }');
-   var expectedStack = [-13];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('boolean or', function () {
-   var stack = evaluate('{ true false or }');
-   var expectedStack = [true];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('bitwise or', function () {
-   var stack = evaluate('{ 254 1 or }');
-   var expectedStack = [254 | 1];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('pops stack', function () {
-   var stack = evaluate('{ 1 2 pop }');
-   var expectedStack = [1];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('rolls stack right', function () {
-   var stack = evaluate('{ 1 3 2 2 4 1 roll }');
-   var expectedStack = [
-    2,
-    1,
-    3,
-    2
-   ];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('rolls stack left', function () {
-   var stack = evaluate('{ 1 3 2 2 4 -1 roll }');
-   var expectedStack = [
-    3,
-    2,
-    2,
-    1
-   ];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('rounds a number', function () {
-   var stack = evaluate('{ 9.52 round }');
-   var expectedStack = [10];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates the sine of a number', function () {
-   var stack = evaluate('{ 90 sin }');
-   var expectedStack = [Math.sin(90)];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates a square root (integer)', function () {
-   var stack = evaluate('{ 100 sqrt }');
-   var expectedStack = [10];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates a square root (float)', function () {
-   var stack = evaluate('{ 99 sqrt }');
-   var expectedStack = [Math.sqrt(99)];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('subtracts (positive result)', function () {
-   var stack = evaluate('{ 6 4 sub }');
-   var expectedStack = [2];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('subtracts (negative result)', function () {
-   var stack = evaluate('{ 4 6 sub }');
-   var expectedStack = [-2];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('pushes true onto the stack', function () {
-   var stack = evaluate('{ true }');
-   var expectedStack = [true];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('truncates a number', function () {
-   var stack = evaluate('{ 35.004 truncate }');
-   var expectedStack = [35];
-   expect(stack).toMatchArray(expectedStack);
-  });
-  it('calculates an exclusive or value', function () {
-   var stack = evaluate('{ 3 9 xor }');
-   var expectedStack = [10];
-   expect(stack).toMatchArray(expectedStack);
-  });
- });
- describe('PostScriptCompiler', function () {
-  function check(code, domain, range, samples) {
-   var compiler = new PostScriptCompiler();
-   var compiledCode = compiler.compile(code, domain, range);
-   if (samples === null) {
-    expect(compiledCode).toBeNull();
-   } else {
-    expect(compiledCode).not.toBeNull();
-    var fn = new Function('src', 'srcOffset', 'dest', 'destOffset', compiledCode);
-    for (var i = 0; i < samples.length; i++) {
-     var out = new Float32Array(samples[i].output.length);
-     fn(samples[i].input, 0, out, 0);
-     expect(Array.prototype.slice.call(out, 0)).toMatchArray(samples[i].output);
     }
-   }
-  }
-  it('check compiled add', function () {
-   check([
-    0.25,
-    0.5,
-    'add'
-   ], [], [
-    0,
-    1
-   ], [{
-     input: [],
-     output: [0.75]
-    }]);
-   check([
-    0,
-    'add'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.25]
-    }]);
-   check([
-    0.5,
-    'add'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.75]
-    }]);
-   check([
-    0,
-    'exch',
-    'add'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.25]
-    }]);
-   check([
-    0.5,
-    'exch',
-    'add'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.75]
-    }]);
-   check(['add'], [
-    0,
-    1,
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [
-      0.25,
-      0.5
-     ],
-     output: [0.75]
-    }]);
-   check(['add'], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], null);
-  });
-  it('check compiled sub', function () {
-   check([
-    0.5,
-    0.25,
-    'sub'
-   ], [], [
-    0,
-    1
-   ], [{
-     input: [],
-     output: [0.25]
-    }]);
-   check([
-    0,
-    'sub'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.25]
-    }]);
-   check([
-    0.5,
-    'sub'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.75],
-     output: [0.25]
-    }]);
-   check([
-    0,
-    'exch',
-    'sub'
-   ], [
-    0,
-    1
-   ], [
-    -1,
-    1
-   ], [{
-     input: [0.25],
-     output: [-0.25]
-    }]);
-   check([
-    0.75,
-    'exch',
-    'sub'
-   ], [
-    0,
-    1
-   ], [
-    -1,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.5]
-    }]);
-   check(['sub'], [
-    0,
-    1,
-    0,
-    1
-   ], [
-    -1,
-    1
-   ], [{
-     input: [
-      0.25,
-      0.5
-     ],
-     output: [-0.25]
-    }]);
-   check(['sub'], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], null);
-   check([
-    1,
-    'dup',
-    3,
-    2,
-    'roll',
-    'sub',
-    'sub'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.75],
-     output: [0.75]
-    }]);
-  });
-  it('check compiled mul', function () {
-   check([
-    0.25,
-    0.5,
-    'mul'
-   ], [], [
-    0,
-    1
-   ], [{
-     input: [],
-     output: [0.125]
-    }]);
-   check([
-    0,
-    'mul'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0]
-    }]);
-   check([
-    0.5,
-    'mul'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.125]
-    }]);
-   check([
-    1,
-    'mul'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.25]
-    }]);
-   check([
-    0,
-    'exch',
-    'mul'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0]
-    }]);
-   check([
-    0.5,
-    'exch',
-    'mul'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.125]
-    }]);
-   check([
-    1,
-    'exch',
-    'mul'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.25],
-     output: [0.25]
-    }]);
-   check(['mul'], [
-    0,
-    1,
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [
-      0.25,
-      0.5
-     ],
-     output: [0.125]
-    }]);
-   check(['mul'], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], null);
-  });
-  it('check compiled max', function () {
-   check([
-    'dup',
-    0.75,
-    'gt',
-    7,
-    'jz',
-    'pop',
-    0.75
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.5],
-     output: [0.5]
-    }]);
-   check([
-    'dup',
-    0.75,
-    'gt',
-    7,
-    'jz',
-    'pop',
-    0.75
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [1],
-     output: [0.75]
-    }]);
-   check([
-    'dup',
-    0.75,
-    'gt',
-    5,
-    'jz',
-    'pop',
-    0.75
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], null);
-  });
-  it('check pop/roll/index', function () {
-   check([
-    1,
-    'pop'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0.5],
-     output: [0.5]
-    }]);
-   check([
-    1,
-    3,
-    -1,
-    'roll'
-   ], [
-    0,
-    1,
-    0,
-    1
-   ], [
-    0,
-    1,
-    0,
-    1,
-    0,
-    1
-   ], [{
-     input: [
-      0.25,
-      0.5
-     ],
-     output: [
-      0.5,
-      1,
-      0.25
-     ]
-    }]);
-   check([
-    1,
-    3,
-    1,
-    'roll'
-   ], [
-    0,
-    1,
-    0,
-    1
-   ], [
-    0,
-    1,
-    0,
-    1,
-    0,
-    1
-   ], [{
-     input: [
-      0.25,
-      0.5
-     ],
-     output: [
-      1,
-      0.25,
-      0.5
-     ]
-    }]);
-   check([
-    1,
-    3,
-    1.5,
-    'roll'
-   ], [
-    0,
-    1,
-    0,
-    1
-   ], [
-    0,
-    1,
-    0,
-    1,
-    0,
-    1
-   ], null);
-   check([
-    1,
-    1,
-    'index'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1,
-    0,
-    1,
-    0,
-    1
-   ], [{
-     input: [0.5],
-     output: [
-      0.5,
-      1,
-      0.5
-     ]
-    }]);
-   check([
-    1,
-    3,
-    'index',
-    'pop'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], null);
-   check([
-    1,
-    0.5,
-    'index',
-    'pop'
-   ], [
-    0,
-    1
-   ], [
-    0,
-    1
-   ], null);
-  });
-  it('check input boundaries', function () {
-   check([], [
-    0,
-    0.5
-   ], [
-    0,
-    1
-   ], [{
-     input: [1],
-     output: [0.5]
-    }]);
-   check([], [
-    0.5,
-    1
-   ], [
-    0,
-    1
-   ], [{
-     input: [0],
-     output: [0.5]
-    }]);
-   check(['dup'], [
-    0.5,
-    0.75
-   ], [
-    0,
-    1,
-    0,
-    1
-   ], [{
-     input: [0],
-     output: [
-      0.5,
-      0.5
-     ]
-    }]);
-   check([], [
-    100,
-    1001
-   ], [
-    0,
-    10000
-   ], [{
-     input: [1000],
-     output: [1000]
-    }]);
-  });
-  it('check output boundaries', function () {
-   check([], [
-    0,
-    1
-   ], [
-    0,
-    0.5
-   ], [{
-     input: [1],
-     output: [0.5]
-    }]);
-   check([], [
-    0,
-    1
-   ], [
-    0.5,
-    1
-   ], [{
-     input: [0],
-     output: [0.5]
-    }]);
-   check(['dup'], [
-    0,
-    1
-   ], [
-    0.5,
-    1,
-    0.75,
-    1
-   ], [{
-     input: [0],
-     output: [
-      0.5,
-      0.75
-     ]
-    }]);
-   check([], [
-    0,
-    10000
-   ], [
-    100,
-    1001
-   ], [{
-     input: [1000],
-     output: [1000]
-    }]);
-  });
-  it('compile optimized', function () {
-   var compiler = new PostScriptCompiler();
-   var code = [
-    0,
-    'add',
-    1,
-    1,
-    3,
-    -1,
-    'roll',
-    'sub',
-    'sub',
-    1,
-    'mul'
-   ];
-   var compiledCode = compiler.compile(code, [
-    0,
-    1
-   ], [
-    0,
-    1
-   ]);
-   expect(compiledCode).toEqual('dest[destOffset + 0] = Math.max(0, Math.min(1, src[srcOffset + 0]));');
+    it('check compiled add', function () {
+      check([0.25, 0.5, 'add'], [], [0, 1], [{
+        input: [],
+        output: [0.75]
+      }]);
+      check([0, 'add'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0.25]
+      }]);
+      check([0.5, 'add'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0.75]
+      }]);
+      check([0, 'exch', 'add'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0.25]
+      }]);
+      check([0.5, 'exch', 'add'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0.75]
+      }]);
+      check(['add'], [0, 1, 0, 1], [0, 1], [{
+        input: [0.25, 0.5],
+        output: [0.75]
+      }]);
+      check(['add'], [0, 1], [0, 1], null);
+    });
+    it('check compiled sub', function () {
+      check([0.5, 0.25, 'sub'], [], [0, 1], [{
+        input: [],
+        output: [0.25]
+      }]);
+      check([0, 'sub'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0.25]
+      }]);
+      check([0.5, 'sub'], [0, 1], [0, 1], [{
+        input: [0.75],
+        output: [0.25]
+      }]);
+      check([0, 'exch', 'sub'], [0, 1], [-1, 1], [{
+        input: [0.25],
+        output: [-0.25]
+      }]);
+      check([0.75, 'exch', 'sub'], [0, 1], [-1, 1], [{
+        input: [0.25],
+        output: [0.5]
+      }]);
+      check(['sub'], [0, 1, 0, 1], [-1, 1], [{
+        input: [0.25, 0.5],
+        output: [-0.25]
+      }]);
+      check(['sub'], [0, 1], [0, 1], null);
+      check([1, 'dup', 3, 2, 'roll', 'sub', 'sub'], [0, 1], [0, 1], [{
+        input: [0.75],
+        output: [0.75]
+      }]);
+    });
+    it('check compiled mul', function () {
+      check([0.25, 0.5, 'mul'], [], [0, 1], [{
+        input: [],
+        output: [0.125]
+      }]);
+      check([0, 'mul'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0]
+      }]);
+      check([0.5, 'mul'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0.125]
+      }]);
+      check([1, 'mul'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0.25]
+      }]);
+      check([0, 'exch', 'mul'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0]
+      }]);
+      check([0.5, 'exch', 'mul'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0.125]
+      }]);
+      check([1, 'exch', 'mul'], [0, 1], [0, 1], [{
+        input: [0.25],
+        output: [0.25]
+      }]);
+      check(['mul'], [0, 1, 0, 1], [0, 1], [{
+        input: [0.25, 0.5],
+        output: [0.125]
+      }]);
+      check(['mul'], [0, 1], [0, 1], null);
+    });
+    it('check compiled max', function () {
+      check(['dup', 0.75, 'gt', 7, 'jz', 'pop', 0.75], [0, 1], [0, 1], [{
+        input: [0.5],
+        output: [0.5]
+      }]);
+      check(['dup', 0.75, 'gt', 7, 'jz', 'pop', 0.75], [0, 1], [0, 1], [{
+        input: [1],
+        output: [0.75]
+      }]);
+      check(['dup', 0.75, 'gt', 5, 'jz', 'pop', 0.75], [0, 1], [0, 1], null);
+    });
+    it('check pop/roll/index', function () {
+      check([1, 'pop'], [0, 1], [0, 1], [{
+        input: [0.5],
+        output: [0.5]
+      }]);
+      check([1, 3, -1, 'roll'], [0, 1, 0, 1], [0, 1, 0, 1, 0, 1], [{
+        input: [0.25, 0.5],
+        output: [0.5, 1, 0.25]
+      }]);
+      check([1, 3, 1, 'roll'], [0, 1, 0, 1], [0, 1, 0, 1, 0, 1], [{
+        input: [0.25, 0.5],
+        output: [1, 0.25, 0.5]
+      }]);
+      check([1, 3, 1.5, 'roll'], [0, 1, 0, 1], [0, 1, 0, 1, 0, 1], null);
+      check([1, 1, 'index'], [0, 1], [0, 1, 0, 1, 0, 1], [{
+        input: [0.5],
+        output: [0.5, 1, 0.5]
+      }]);
+      check([1, 3, 'index', 'pop'], [0, 1], [0, 1], null);
+      check([1, 0.5, 'index', 'pop'], [0, 1], [0, 1], null);
+    });
+    it('check input boundaries', function () {
+      check([], [0, 0.5], [0, 1], [{
+        input: [1],
+        output: [0.5]
+      }]);
+      check([], [0.5, 1], [0, 1], [{
+        input: [0],
+        output: [0.5]
+      }]);
+      check(['dup'], [0.5, 0.75], [0, 1, 0, 1], [{
+        input: [0],
+        output: [0.5, 0.5]
+      }]);
+      check([], [100, 1001], [0, 10000], [{
+        input: [1000],
+        output: [1000]
+      }]);
+    });
+    it('check output boundaries', function () {
+      check([], [0, 1], [0, 0.5], [{
+        input: [1],
+        output: [0.5]
+      }]);
+      check([], [0, 1], [0.5, 1], [{
+        input: [0],
+        output: [0.5]
+      }]);
+      check(['dup'], [0, 1], [0.5, 1, 0.75, 1], [{
+        input: [0],
+        output: [0.5, 0.75]
+      }]);
+      check([], [0, 10000], [100, 1001], [{
+        input: [1000],
+        output: [1000]
+      }]);
+    });
+    it('compile optimized', function () {
+      var compiler = new PostScriptCompiler();
+      var code = [0, 'add', 1, 1, 3, -1, 'roll', 'sub', 'sub', 1, 'mul'];
+      var compiledCode = compiler.compile(code, [0, 1], [0, 1]);
+      expect(compiledCode).toEqual('dest[destOffset + 0] = Math.max(0, Math.min(1, src[srcOffset + 0]));');
+    });
   });
- });
 });

+ 80 - 102
lib/test/unit/jasmine-boot.js

@@ -13,112 +13,90 @@
  * limitations under the License.
  */
 'use strict';
+
 function initializePDFJS(callback) {
- Promise.all([
-  'pdfjs/display/global',
-  'pdfjs-test/unit/annotation_spec',
-  'pdfjs-test/unit/api_spec',
-  'pdfjs-test/unit/bidi_spec',
-  'pdfjs-test/unit/cff_parser_spec',
-  'pdfjs-test/unit/cmap_spec',
-  'pdfjs-test/unit/crypto_spec',
-  'pdfjs-test/unit/document_spec',
-  'pdfjs-test/unit/dom_utils_spec',
-  'pdfjs-test/unit/evaluator_spec',
-  'pdfjs-test/unit/fonts_spec',
-  'pdfjs-test/unit/function_spec',
-  'pdfjs-test/unit/metadata_spec',
-  'pdfjs-test/unit/murmurhash3_spec',
-  'pdfjs-test/unit/network_spec',
-  'pdfjs-test/unit/parser_spec',
-  'pdfjs-test/unit/primitives_spec',
-  'pdfjs-test/unit/stream_spec',
-  'pdfjs-test/unit/type1_parser_spec',
-  'pdfjs-test/unit/ui_utils_spec',
-  'pdfjs-test/unit/unicode_spec',
-  'pdfjs-test/unit/util_spec'
- ].map(function (moduleName) {
-  return SystemJS.import(moduleName);
- })).then(function (modules) {
-  var displayGlobal = modules[0];
-  displayGlobal.PDFJS.workerSrc = '../../src/worker_loader.js';
-  callback();
- });
+  Promise.all(['pdfjs/display/global', 'pdfjs-test/unit/annotation_spec', 'pdfjs-test/unit/api_spec', 'pdfjs-test/unit/bidi_spec', 'pdfjs-test/unit/cff_parser_spec', 'pdfjs-test/unit/cmap_spec', 'pdfjs-test/unit/crypto_spec', 'pdfjs-test/unit/document_spec', 'pdfjs-test/unit/dom_utils_spec', 'pdfjs-test/unit/evaluator_spec', 'pdfjs-test/unit/fonts_spec', 'pdfjs-test/unit/function_spec', 'pdfjs-test/unit/metadata_spec', 'pdfjs-test/unit/murmurhash3_spec', 'pdfjs-test/unit/network_spec', 'pdfjs-test/unit/parser_spec', 'pdfjs-test/unit/primitives_spec', 'pdfjs-test/unit/stream_spec', 'pdfjs-test/unit/type1_parser_spec', 'pdfjs-test/unit/ui_utils_spec', 'pdfjs-test/unit/unicode_spec', 'pdfjs-test/unit/util_spec'].map(function (moduleName) {
+    return SystemJS.import(moduleName);
+  })).then(function (modules) {
+    var displayGlobal = modules[0];
+    displayGlobal.PDFJS.workerSrc = '../../src/worker_loader.js';
+    callback();
+  });
 }
 (function () {
- window.jasmine = jasmineRequire.core(jasmineRequire);
- jasmineRequire.html(jasmine);
- var env = jasmine.getEnv();
- var jasmineInterface = jasmineRequire.interface(jasmine, env);
- extend(window, jasmineInterface);
- var queryString = new jasmine.QueryString({
-  getWindowLocation: function () {
-   return window.location;
-  }
- });
- var catchingExceptions = queryString.getParam('catch');
- env.catchExceptions(typeof catchingExceptions === 'undefined' ? true : catchingExceptions);
- var throwingExpectationFailures = queryString.getParam('throwFailures');
- env.throwOnExpectationFailure(throwingExpectationFailures);
- var random = queryString.getParam('random');
- env.randomizeTests(random);
- var seed = queryString.getParam('seed');
- if (seed) {
-  env.seed(seed);
- }
- var htmlReporter = new jasmine.HtmlReporter({
-  env: env,
-  onRaiseExceptionsClick: function () {
-   queryString.navigateWithNewParam('catch', !env.catchingExceptions());
-  },
-  onThrowExpectationsClick: function () {
-   queryString.navigateWithNewParam('throwFailures', !env.throwingExpectationFailures());
-  },
-  onRandomClick: function () {
-   queryString.navigateWithNewParam('random', !env.randomTests());
-  },
-  addToExistingQueryString: function (key, value) {
-   return queryString.fullStringWithNewParam(key, value);
-  },
-  getContainer: function () {
-   return document.body;
-  },
-  createElement: function () {
-   return document.createElement.apply(document, arguments);
-  },
-  createTextNode: function () {
-   return document.createTextNode.apply(document, arguments);
-  },
-  timer: new jasmine.Timer()
- });
- env.addReporter(htmlReporter);
- if (queryString.getParam('browser')) {
-  var testReporter = new TestReporter(queryString.getParam('browser'), queryString.getParam('path'));
-  env.addReporter(testReporter);
- }
- var specFilter = new jasmine.HtmlSpecFilter({
-  filterString: function () {
-   return queryString.getParam('spec');
+  window.jasmine = jasmineRequire.core(jasmineRequire);
+  jasmineRequire.html(jasmine);
+  var env = jasmine.getEnv();
+  var jasmineInterface = jasmineRequire.interface(jasmine, env);
+  extend(window, jasmineInterface);
+  var queryString = new jasmine.QueryString({
+    getWindowLocation: function () {
+      return window.location;
+    }
+  });
+  var catchingExceptions = queryString.getParam('catch');
+  env.catchExceptions(typeof catchingExceptions === 'undefined' ? true : catchingExceptions);
+  var throwingExpectationFailures = queryString.getParam('throwFailures');
+  env.throwOnExpectationFailure(throwingExpectationFailures);
+  var random = queryString.getParam('random');
+  env.randomizeTests(random);
+  var seed = queryString.getParam('seed');
+  if (seed) {
+    env.seed(seed);
   }
- });
- env.specFilter = function (spec) {
-  return specFilter.matches(spec.getFullName());
- };
- jasmine.DEFAULT_TIMEOUT_INTERVAL = 30000;
- var currentWindowOnload = window.onload;
- window.onload = function () {
-  if (currentWindowOnload) {
-   currentWindowOnload();
+  var htmlReporter = new jasmine.HtmlReporter({
+    env: env,
+    onRaiseExceptionsClick: function () {
+      queryString.navigateWithNewParam('catch', !env.catchingExceptions());
+    },
+    onThrowExpectationsClick: function () {
+      queryString.navigateWithNewParam('throwFailures', !env.throwingExpectationFailures());
+    },
+    onRandomClick: function () {
+      queryString.navigateWithNewParam('random', !env.randomTests());
+    },
+    addToExistingQueryString: function (key, value) {
+      return queryString.fullStringWithNewParam(key, value);
+    },
+    getContainer: function () {
+      return document.body;
+    },
+    createElement: function () {
+      return document.createElement.apply(document, arguments);
+    },
+    createTextNode: function () {
+      return document.createTextNode.apply(document, arguments);
+    },
+    timer: new jasmine.Timer()
+  });
+  env.addReporter(htmlReporter);
+  if (queryString.getParam('browser')) {
+    var testReporter = new TestReporter(queryString.getParam('browser'), queryString.getParam('path'));
+    env.addReporter(testReporter);
   }
-  initializePDFJS(function () {
-   htmlReporter.initialize();
-   env.execute();
+  var specFilter = new jasmine.HtmlSpecFilter({
+    filterString: function () {
+      return queryString.getParam('spec');
+    }
   });
- };
- function extend(destination, source) {
-  for (var property in source) {
-   destination[property] = source[property];
+  env.specFilter = function (spec) {
+    return specFilter.matches(spec.getFullName());
+  };
+  jasmine.DEFAULT_TIMEOUT_INTERVAL = 30000;
+  var currentWindowOnload = window.onload;
+  window.onload = function () {
+    if (currentWindowOnload) {
+      currentWindowOnload();
+    }
+    initializePDFJS(function () {
+      htmlReporter.initialize();
+      env.execute();
+    });
+  };
+  function extend(destination, source) {
+    for (var property in source) {
+      destination[property] = source[property];
+    }
+    return destination;
   }
-  return destination;
- }
-}());
+})();

+ 7 - 6
lib/test/unit/metadata_spec.js

@@ -13,14 +13,15 @@
  * limitations under the License.
  */
 'use strict';
+
 var displayMetadata = require('../../display/metadata.js');
 var Metadata = displayMetadata.Metadata;
 describe('metadata', function () {
- describe('incorrect_xmp', function () {
-  it('should fix the incorrect XMP data', function () {
-   var invalidXMP = '<x:xmpmeta xmlns:x=\'adobe:ns:meta/\'>' + '<rdf:RDF xmlns:rdf=\'http://www.w3.org/1999/02/22-rdf-syntax-ns#\'>' + '<rdf:Description xmlns:dc=\'http://purl.org/dc/elements/1.1/\'>' + '<dc:title>\\376\\377\\000P\\000D\\000F\\000&</dc:title>' + '</rdf:Description></rdf:RDF></x:xmpmeta>';
-   var meta = new Metadata(invalidXMP);
-   expect(meta.get('dc:title')).toEqual('PDF&');
+  describe('incorrect_xmp', function () {
+    it('should fix the incorrect XMP data', function () {
+      var invalidXMP = '<x:xmpmeta xmlns:x=\'adobe:ns:meta/\'>' + '<rdf:RDF xmlns:rdf=\'http://www.w3.org/1999/02/22-rdf-syntax-ns#\'>' + '<rdf:Description xmlns:dc=\'http://purl.org/dc/elements/1.1/\'>' + '<dc:title>\\376\\377\\000P\\000D\\000F\\000&</dc:title>' + '</rdf:Description></rdf:RDF></x:xmpmeta>';
+      var meta = new Metadata(invalidXMP);
+      expect(meta.get('dc:title')).toEqual('PDF&');
+    });
   });
- });
 });

+ 45 - 49
lib/test/unit/murmurhash3_spec.js

@@ -13,56 +13,52 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreMurmurHash3 = require('../../core/murmurhash3.js');
 var MurmurHash3_64 = coreMurmurHash3.MurmurHash3_64;
 describe('MurmurHash3_64', function () {
- it('instantiates without seed', function () {
-  var hash = new MurmurHash3_64();
-  expect(hash).toEqual(jasmine.any(MurmurHash3_64));
- });
- it('instantiates with seed', function () {
-  var hash = new MurmurHash3_64(1);
-  expect(hash).toEqual(jasmine.any(MurmurHash3_64));
- });
- var hexDigestExpected = 'f61cfdbfdae0f65e';
- var sourceText = 'test';
- var sourceCharCodes = [
-  116,
-  101,
-  115,
-  116
- ];
- it('correctly generates a hash from a string', function () {
-  var hash = new MurmurHash3_64();
-  hash.update(sourceText);
-  expect(hash.hexdigest()).toEqual(hexDigestExpected);
- });
- it('correctly generates a hash from a Uint8Array', function () {
-  var hash = new MurmurHash3_64();
-  hash.update(new Uint8Array(sourceCharCodes));
-  expect(hash.hexdigest()).toEqual(hexDigestExpected);
- });
- it('correctly generates a hash from a Uint32Array', function () {
-  var hash = new MurmurHash3_64();
-  hash.update(new Uint32Array(sourceCharCodes));
-  expect(hash.hexdigest()).toEqual(hexDigestExpected);
- });
- it('changes the hash after update without seed', function () {
-  var hash = new MurmurHash3_64();
-  var hexdigest1, hexdigest2;
-  hash.update(sourceText);
-  hexdigest1 = hash.hexdigest();
-  hash.update(sourceText);
-  hexdigest2 = hash.hexdigest();
-  expect(hexdigest1).not.toEqual(hexdigest2);
- });
- it('changes the hash after update with seed', function () {
-  var hash = new MurmurHash3_64(1);
-  var hexdigest1, hexdigest2;
-  hash.update(sourceText);
-  hexdigest1 = hash.hexdigest();
-  hash.update(sourceText);
-  hexdigest2 = hash.hexdigest();
-  expect(hexdigest1).not.toEqual(hexdigest2);
- });
+  it('instantiates without seed', function () {
+    var hash = new MurmurHash3_64();
+    expect(hash).toEqual(jasmine.any(MurmurHash3_64));
+  });
+  it('instantiates with seed', function () {
+    var hash = new MurmurHash3_64(1);
+    expect(hash).toEqual(jasmine.any(MurmurHash3_64));
+  });
+  var hexDigestExpected = 'f61cfdbfdae0f65e';
+  var sourceText = 'test';
+  var sourceCharCodes = [116, 101, 115, 116];
+  it('correctly generates a hash from a string', function () {
+    var hash = new MurmurHash3_64();
+    hash.update(sourceText);
+    expect(hash.hexdigest()).toEqual(hexDigestExpected);
+  });
+  it('correctly generates a hash from a Uint8Array', function () {
+    var hash = new MurmurHash3_64();
+    hash.update(new Uint8Array(sourceCharCodes));
+    expect(hash.hexdigest()).toEqual(hexDigestExpected);
+  });
+  it('correctly generates a hash from a Uint32Array', function () {
+    var hash = new MurmurHash3_64();
+    hash.update(new Uint32Array(sourceCharCodes));
+    expect(hash.hexdigest()).toEqual(hexDigestExpected);
+  });
+  it('changes the hash after update without seed', function () {
+    var hash = new MurmurHash3_64();
+    var hexdigest1, hexdigest2;
+    hash.update(sourceText);
+    hexdigest1 = hash.hexdigest();
+    hash.update(sourceText);
+    hexdigest2 = hash.hexdigest();
+    expect(hexdigest1).not.toEqual(hexdigest2);
+  });
+  it('changes the hash after update with seed', function () {
+    var hash = new MurmurHash3_64(1);
+    var hexdigest1, hexdigest2;
+    hash.update(sourceText);
+    hexdigest1 = hash.hexdigest();
+    hash.update(sourceText);
+    hexdigest2 = hash.hexdigest();
+    expect(hexdigest1).not.toEqual(hexdigest2);
+  });
 });

+ 129 - 135
lib/test/unit/network_spec.js

@@ -13,146 +13,140 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreNetwork = require('../../core/network.js');
 var PDFNetworkStream = coreNetwork.PDFNetworkStream;
 describe('network', function () {
- var pdf1 = new URL('../pdfs/tracemonkey.pdf', window.location).href;
- var pdf1Length = 1016315;
- var pdf2 = new URL('../pdfs/pdf.pdf', window.location).href;
- var pdf2Length = 32472771;
- it('read without stream and range', function (done) {
-  var stream = new PDFNetworkStream({
-   source: {
-    url: pdf1,
-    rangeChunkSize: 65536,
-    disableStream: true
-   },
-   disableRange: true
+  var pdf1 = new URL('../pdfs/tracemonkey.pdf', window.location).href;
+  var pdf1Length = 1016315;
+  var pdf2 = new URL('../pdfs/pdf.pdf', window.location).href;
+  var pdf2Length = 32472771;
+  it('read without stream and range', function (done) {
+    var stream = new PDFNetworkStream({
+      source: {
+        url: pdf1,
+        rangeChunkSize: 65536,
+        disableStream: true
+      },
+      disableRange: true
+    });
+    var fullReader = stream.getFullReader();
+    var isStreamingSupported, isRangeSupported;
+    var promise = fullReader.headersReady.then(function () {
+      isStreamingSupported = fullReader.isStreamingSupported;
+      isRangeSupported = fullReader.isRangeSupported;
+    });
+    var len = 0,
+        count = 0;
+    var read = function () {
+      return fullReader.read().then(function (result) {
+        if (result.done) {
+          return;
+        }
+        count++;
+        len += result.value.byteLength;
+        return read();
+      });
+    };
+    var readPromise = Promise.all([read(), promise]);
+    readPromise.then(function (page) {
+      expect(len).toEqual(pdf1Length);
+      expect(count).toEqual(1);
+      expect(isStreamingSupported).toEqual(false);
+      expect(isRangeSupported).toEqual(false);
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
-  var fullReader = stream.getFullReader();
-  var isStreamingSupported, isRangeSupported;
-  var promise = fullReader.headersReady.then(function () {
-   isStreamingSupported = fullReader.isStreamingSupported;
-   isRangeSupported = fullReader.isRangeSupported;
-  });
-  var len = 0, count = 0;
-  var read = function () {
-   return fullReader.read().then(function (result) {
-    if (result.done) {
-     return;
-    }
-    count++;
-    len += result.value.byteLength;
-    return read();
-   });
-  };
-  var readPromise = Promise.all([
-   read(),
-   promise
-  ]);
-  readPromise.then(function (page) {
-   expect(len).toEqual(pdf1Length);
-   expect(count).toEqual(1);
-   expect(isStreamingSupported).toEqual(false);
-   expect(isRangeSupported).toEqual(false);
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
-  });
- });
- it('read with streaming', function (done) {
-  var userAgent = window.navigator.userAgent;
-  var m = /Mozilla\/5.0.*?rv:(\d+).*? Gecko/.exec(userAgent);
-  if (!m || m[1] < 9) {
-   expect(true).toEqual(true);
-   done();
-   return;
-  }
-  var stream = new PDFNetworkStream({
-   source: {
-    url: pdf2,
-    rangeChunkSize: 65536,
-    disableStream: false
-   },
-   disableRange: false
-  });
-  var fullReader = stream.getFullReader();
-  var isStreamingSupported, isRangeSupported;
-  var promise = fullReader.headersReady.then(function () {
-   isStreamingSupported = fullReader.isStreamingSupported;
-   isRangeSupported = fullReader.isRangeSupported;
-  });
-  var len = 0, count = 0;
-  var read = function () {
-   return fullReader.read().then(function (result) {
-    if (result.done) {
-     return;
+  it('read with streaming', function (done) {
+    var userAgent = window.navigator.userAgent;
+    var m = /Mozilla\/5.0.*?rv:(\d+).*? Gecko/.exec(userAgent);
+    if (!m || m[1] < 9) {
+      expect(true).toEqual(true);
+      done();
+      return;
     }
-    count++;
-    len += result.value.byteLength;
-    return read();
-   });
-  };
-  var readPromise = Promise.all([
-   read(),
-   promise
-  ]);
-  readPromise.then(function () {
-   expect(len).toEqual(pdf2Length);
-   expect(count).toBeGreaterThan(1);
-   expect(isStreamingSupported).toEqual(true);
-   expect(isRangeSupported).toEqual(true);
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
-  });
- });
- it('read custom ranges', function (done) {
-  var rangeSize = 32768;
-  var stream = new PDFNetworkStream({
-   source: {
-    url: pdf1,
-    length: pdf1Length,
-    rangeChunkSize: rangeSize,
-    disableStream: true
-   },
-   disableRange: false
+    var stream = new PDFNetworkStream({
+      source: {
+        url: pdf2,
+        rangeChunkSize: 65536,
+        disableStream: false
+      },
+      disableRange: false
+    });
+    var fullReader = stream.getFullReader();
+    var isStreamingSupported, isRangeSupported;
+    var promise = fullReader.headersReady.then(function () {
+      isStreamingSupported = fullReader.isStreamingSupported;
+      isRangeSupported = fullReader.isRangeSupported;
+    });
+    var len = 0,
+        count = 0;
+    var read = function () {
+      return fullReader.read().then(function (result) {
+        if (result.done) {
+          return;
+        }
+        count++;
+        len += result.value.byteLength;
+        return read();
+      });
+    };
+    var readPromise = Promise.all([read(), promise]);
+    readPromise.then(function () {
+      expect(len).toEqual(pdf2Length);
+      expect(count).toBeGreaterThan(1);
+      expect(isStreamingSupported).toEqual(true);
+      expect(isRangeSupported).toEqual(true);
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
-  var fullReader = stream.getFullReader();
-  var isStreamingSupported, isRangeSupported, fullReaderCancelled;
-  var promise = fullReader.headersReady.then(function () {
-   isStreamingSupported = fullReader.isStreamingSupported;
-   isRangeSupported = fullReader.isRangeSupported;
-   fullReader.cancel('Don\'t need full reader');
-   fullReaderCancelled = true;
-  });
-  var tailSize = pdf1Length % rangeSize || rangeSize;
-  var range1Reader = stream.getRangeReader(pdf1Length - tailSize - rangeSize, pdf1Length - tailSize);
-  var range2Reader = stream.getRangeReader(pdf1Length - tailSize, pdf1Length);
-  var result1 = { value: 0 }, result2 = { value: 0 };
-  var read = function (reader, lenResult) {
-   return reader.read().then(function (result) {
-    if (result.done) {
-     return;
-    }
-    lenResult.value += result.value.byteLength;
-    return read(reader, lenResult);
-   });
-  };
-  var readPromises = Promise.all([
-   read(range1Reader, result1),
-   read(range2Reader, result2),
-   promise
-  ]);
-  readPromises.then(function () {
-   expect(result1.value).toEqual(rangeSize);
-   expect(result2.value).toEqual(tailSize);
-   expect(isStreamingSupported).toEqual(false);
-   expect(isRangeSupported).toEqual(true);
-   expect(fullReaderCancelled).toEqual(true);
-   done();
-  }).catch(function (reason) {
-   done.fail(reason);
+  it('read custom ranges', function (done) {
+    var rangeSize = 32768;
+    var stream = new PDFNetworkStream({
+      source: {
+        url: pdf1,
+        length: pdf1Length,
+        rangeChunkSize: rangeSize,
+        disableStream: true
+      },
+      disableRange: false
+    });
+    var fullReader = stream.getFullReader();
+    var isStreamingSupported, isRangeSupported, fullReaderCancelled;
+    var promise = fullReader.headersReady.then(function () {
+      isStreamingSupported = fullReader.isStreamingSupported;
+      isRangeSupported = fullReader.isRangeSupported;
+      fullReader.cancel('Don\'t need full reader');
+      fullReaderCancelled = true;
+    });
+    var tailSize = pdf1Length % rangeSize || rangeSize;
+    var range1Reader = stream.getRangeReader(pdf1Length - tailSize - rangeSize, pdf1Length - tailSize);
+    var range2Reader = stream.getRangeReader(pdf1Length - tailSize, pdf1Length);
+    var result1 = { value: 0 },
+        result2 = { value: 0 };
+    var read = function (reader, lenResult) {
+      return reader.read().then(function (result) {
+        if (result.done) {
+          return;
+        }
+        lenResult.value += result.value.byteLength;
+        return read(reader, lenResult);
+      });
+    };
+    var readPromises = Promise.all([read(range1Reader, result1), read(range2Reader, result2), promise]);
+    readPromises.then(function () {
+      expect(result1.value).toEqual(rangeSize);
+      expect(result2.value).toEqual(tailSize);
+      expect(isStreamingSupported).toEqual(false);
+      expect(isRangeSupported).toEqual(true);
+      expect(fullReaderCancelled).toEqual(true);
+      done();
+    }).catch(function (reason) {
+      done.fail(reason);
+    });
   });
- });
 });

+ 111 - 134
lib/test/unit/parser_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreParser = require('../../core/parser.js');
 var corePrimitives = require('../../core/primitives.js');
 var coreStream = require('../../core/stream.js');
@@ -21,140 +22,116 @@ var Linearization = coreParser.Linearization;
 var Name = corePrimitives.Name;
 var StringStream = coreStream.StringStream;
 describe('parser', function () {
- describe('Lexer', function () {
-  it('should stop parsing numbers at the end of stream', function () {
-   var input = new StringStream('11.234');
-   var lexer = new Lexer(input);
-   var result = lexer.getNumber();
-   expect(result).toEqual(11.234);
+  describe('Lexer', function () {
+    it('should stop parsing numbers at the end of stream', function () {
+      var input = new StringStream('11.234');
+      var lexer = new Lexer(input);
+      var result = lexer.getNumber();
+      expect(result).toEqual(11.234);
+    });
+    it('should parse PostScript numbers', function () {
+      var numbers = ['-.002', '34.5', '-3.62', '123.6e10', '1E-5', '-1.', '0.0', '123', '-98', '43445', '0', '+17'];
+      for (var i = 0, ii = numbers.length; i < ii; i++) {
+        var num = numbers[i];
+        var input = new StringStream(num);
+        var lexer = new Lexer(input);
+        var result = lexer.getNumber();
+        expect(result).toEqual(parseFloat(num));
+      }
+    });
+    it('should ignore double negative before number', function () {
+      var input = new StringStream('--205.88');
+      var lexer = new Lexer(input);
+      var result = lexer.getNumber();
+      expect(result).toEqual(-205.88);
+    });
+    it('should handle glued numbers and operators', function () {
+      var input = new StringStream('123ET');
+      var lexer = new Lexer(input);
+      var value = lexer.getNumber();
+      expect(value).toEqual(123);
+      expect(lexer.currentChar).toEqual(0x45);
+    });
+    it('should stop parsing strings at the end of stream', function () {
+      var input = new StringStream('(1$4)');
+      input.getByte = function (super_getByte) {
+        var ch = super_getByte.call(input);
+        return ch === 0x24 ? -1 : ch;
+      }.bind(input, input.getByte);
+      var lexer = new Lexer(input);
+      var result = lexer.getString();
+      expect(result).toEqual('1');
+    });
+    it('should not throw exception on bad input', function () {
+      var input = new StringStream('<7 0 2 15 5 2 2 2 4 3 2 4>');
+      var lexer = new Lexer(input);
+      var result = lexer.getHexString();
+      expect(result).toEqual('p!U"$2');
+    });
+    it('should ignore escaped CR and LF', function () {
+      var input = new StringStream('(\\101\\\r\n\\102\\\r\\103\\\n\\104)');
+      var lexer = new Lexer(input);
+      var result = lexer.getString();
+      expect(result).toEqual('ABCD');
+    });
+    it('should handle Names with invalid usage of NUMBER SIGN (#)', function () {
+      var inputNames = ['/# 680 0 R', '/#AQwerty', '/#A<</B'];
+      var expectedNames = ['#', '#AQwerty', '#A'];
+      for (var i = 0, ii = inputNames.length; i < ii; i++) {
+        var input = new StringStream(inputNames[i]);
+        var lexer = new Lexer(input);
+        var result = lexer.getName();
+        expect(result).toEqual(Name.get(expectedNames[i]));
+      }
+    });
   });
-  it('should parse PostScript numbers', function () {
-   var numbers = [
-    '-.002',
-    '34.5',
-    '-3.62',
-    '123.6e10',
-    '1E-5',
-    '-1.',
-    '0.0',
-    '123',
-    '-98',
-    '43445',
-    '0',
-    '+17'
-   ];
-   for (var i = 0, ii = numbers.length; i < ii; i++) {
-    var num = numbers[i];
-    var input = new StringStream(num);
-    var lexer = new Lexer(input);
-    var result = lexer.getNumber();
-    expect(result).toEqual(parseFloat(num));
-   }
+  describe('Linearization', function () {
+    it('should not find a linearization dictionary', function () {
+      var stream1 = new StringStream('3 0 obj\n' + '<<\n' + '/Length 4622\n' + '/Filter /FlateDecode\n' + '>>\n' + 'endobj');
+      expect(Linearization.create(stream1)).toEqual(null);
+      var stream2 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 0\n' + '>>\n' + 'endobj');
+      expect(Linearization.create(stream2)).toEqual(null);
+    });
+    it('should accept a valid linearization dictionary', function () {
+      var stream = new StringStream('131 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 863 ]\n' + '/L 90\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
+      var expectedLinearizationDict = {
+        length: 90,
+        hints: [1388, 863],
+        objectNumberFirst: 133,
+        endFirst: 43573,
+        numPages: 18,
+        mainXRefEntriesOffset: 193883,
+        pageFirst: 0
+      };
+      expect(Linearization.create(stream)).toEqual(expectedLinearizationDict);
+    });
+    it('should reject a linearization dictionary with invalid ' + 'integer parameters', function () {
+      var stream1 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 863 ]\n' + '/L 196622\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
+      expect(function () {
+        return Linearization.create(stream1);
+      }).toThrow(new Error('The "L" parameter in the linearization ' + 'dictionary does not equal the stream length.'));
+      var stream2 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 863 ]\n' + '/L 84\n' + '/E 0\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
+      expect(function () {
+        return Linearization.create(stream2);
+      }).toThrow(new Error('The "E" parameter in the linearization ' + 'dictionary is invalid.'));
+      var stream3 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O /abc\n' + '/H [ 1388 863 ]\n' + '/L 89\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
+      expect(function () {
+        return Linearization.create(stream3);
+      }).toThrow(new Error('The "O" parameter in the linearization ' + 'dictionary is invalid.'));
+    });
+    it('should reject a linearization dictionary with invalid hint parameters', function () {
+      var stream1 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H 1388\n' + '/L 80\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
+      expect(function () {
+        return Linearization.create(stream1);
+      }).toThrow(new Error('Hint array in the linearization dictionary ' + 'is invalid.'));
+      var stream2 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 ]\n' + '/L 84\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
+      expect(function () {
+        return Linearization.create(stream2);
+      }).toThrow(new Error('Hint array in the linearization dictionary ' + 'is invalid.'));
+      var stream3 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 863 0 234]\n' + '/L 93\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
+      expect(function () {
+        return Linearization.create(stream3);
+      }).toThrow(new Error('Hint (2) in the linearization dictionary ' + 'is invalid.'));
+    });
   });
-  it('should ignore double negative before number', function () {
-   var input = new StringStream('--205.88');
-   var lexer = new Lexer(input);
-   var result = lexer.getNumber();
-   expect(result).toEqual(-205.88);
-  });
-  it('should handle glued numbers and operators', function () {
-   var input = new StringStream('123ET');
-   var lexer = new Lexer(input);
-   var value = lexer.getNumber();
-   expect(value).toEqual(123);
-   expect(lexer.currentChar).toEqual(0x45);
-  });
-  it('should stop parsing strings at the end of stream', function () {
-   var input = new StringStream('(1$4)');
-   input.getByte = function (super_getByte) {
-    var ch = super_getByte.call(input);
-    return ch === 0x24 ? -1 : ch;
-   }.bind(input, input.getByte);
-   var lexer = new Lexer(input);
-   var result = lexer.getString();
-   expect(result).toEqual('1');
-  });
-  it('should not throw exception on bad input', function () {
-   var input = new StringStream('<7 0 2 15 5 2 2 2 4 3 2 4>');
-   var lexer = new Lexer(input);
-   var result = lexer.getHexString();
-   expect(result).toEqual('p!U"$2');
-  });
-  it('should ignore escaped CR and LF', function () {
-   var input = new StringStream('(\\101\\\r\n\\102\\\r\\103\\\n\\104)');
-   var lexer = new Lexer(input);
-   var result = lexer.getString();
-   expect(result).toEqual('ABCD');
-  });
-  it('should handle Names with invalid usage of NUMBER SIGN (#)', function () {
-   var inputNames = [
-    '/# 680 0 R',
-    '/#AQwerty',
-    '/#A<</B'
-   ];
-   var expectedNames = [
-    '#',
-    '#AQwerty',
-    '#A'
-   ];
-   for (var i = 0, ii = inputNames.length; i < ii; i++) {
-    var input = new StringStream(inputNames[i]);
-    var lexer = new Lexer(input);
-    var result = lexer.getName();
-    expect(result).toEqual(Name.get(expectedNames[i]));
-   }
-  });
- });
- describe('Linearization', function () {
-  it('should not find a linearization dictionary', function () {
-   var stream1 = new StringStream('3 0 obj\n' + '<<\n' + '/Length 4622\n' + '/Filter /FlateDecode\n' + '>>\n' + 'endobj');
-   expect(Linearization.create(stream1)).toEqual(null);
-   var stream2 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 0\n' + '>>\n' + 'endobj');
-   expect(Linearization.create(stream2)).toEqual(null);
-  });
-  it('should accept a valid linearization dictionary', function () {
-   var stream = new StringStream('131 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 863 ]\n' + '/L 90\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
-   var expectedLinearizationDict = {
-    length: 90,
-    hints: [
-     1388,
-     863
-    ],
-    objectNumberFirst: 133,
-    endFirst: 43573,
-    numPages: 18,
-    mainXRefEntriesOffset: 193883,
-    pageFirst: 0
-   };
-   expect(Linearization.create(stream)).toEqual(expectedLinearizationDict);
-  });
-  it('should reject a linearization dictionary with invalid ' + 'integer parameters', function () {
-   var stream1 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 863 ]\n' + '/L 196622\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
-   expect(function () {
-    return Linearization.create(stream1);
-   }).toThrow(new Error('The "L" parameter in the linearization ' + 'dictionary does not equal the stream length.'));
-   var stream2 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 863 ]\n' + '/L 84\n' + '/E 0\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
-   expect(function () {
-    return Linearization.create(stream2);
-   }).toThrow(new Error('The "E" parameter in the linearization ' + 'dictionary is invalid.'));
-   var stream3 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O /abc\n' + '/H [ 1388 863 ]\n' + '/L 89\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
-   expect(function () {
-    return Linearization.create(stream3);
-   }).toThrow(new Error('The "O" parameter in the linearization ' + 'dictionary is invalid.'));
-  });
-  it('should reject a linearization dictionary with invalid hint parameters', function () {
-   var stream1 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H 1388\n' + '/L 80\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
-   expect(function () {
-    return Linearization.create(stream1);
-   }).toThrow(new Error('Hint array in the linearization dictionary ' + 'is invalid.'));
-   var stream2 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 ]\n' + '/L 84\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
-   expect(function () {
-    return Linearization.create(stream2);
-   }).toThrow(new Error('Hint array in the linearization dictionary ' + 'is invalid.'));
-   var stream3 = new StringStream('1 0 obj\n' + '<<\n' + '/Linearized 1\n' + '/O 133\n' + '/H [ 1388 863 0 234]\n' + '/L 93\n' + '/E 43573\n' + '/N 18\n' + '/T 193883\n' + '>>\n' + 'endobj');
-   expect(function () {
-    return Linearization.create(stream3);
-   }).toThrow(new Error('Hint (2) in the linearization dictionary ' + 'is invalid.'));
-  });
- });
 });

+ 292 - 335
lib/test/unit/primitives_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var corePrimitives = require('../../core/primitives.js');
 var Name = corePrimitives.Name;
 var Dict = corePrimitives.Dict;
@@ -25,343 +26,299 @@ var isDict = corePrimitives.isDict;
 var isRef = corePrimitives.isRef;
 var isRefsEqual = corePrimitives.isRefsEqual;
 describe('primitives', function () {
- function XRefMock(array) {
-  this.map = Object.create(null);
-  for (var elem in array) {
-   var obj = array[elem];
-   var ref = obj.ref, data = obj.data;
-   this.map[ref.toString()] = data;
-  }
- }
- XRefMock.prototype = {
-  fetch: function (ref) {
-   return this.map[ref.toString()];
-  },
-  fetchIfRef: function (obj) {
-   if (!isRef(obj)) {
-    return obj;
-   }
-   return this.fetch(obj);
-  },
-  fetchAsync: function (ref) {
-   return Promise.resolve(this.fetch(ref));
-  },
-  fetchIfRefAsync: function (obj) {
-   return Promise.resolve(this.fetchIfRef(obj));
+  function XRefMock(array) {
+    this.map = Object.create(null);
+    for (var elem in array) {
+      var obj = array[elem];
+      var ref = obj.ref,
+          data = obj.data;
+      this.map[ref.toString()] = data;
+    }
   }
- };
- describe('Name', function () {
-  it('should retain the given name', function () {
-   var givenName = 'Font';
-   var name = Name.get(givenName);
-   expect(name.name).toEqual(givenName);
-  });
-  it('should create only one object for a name and cache it', function () {
-   var firstFont = Name.get('Font');
-   var secondFont = Name.get('Font');
-   var firstSubtype = Name.get('Subtype');
-   var secondSubtype = Name.get('Subtype');
-   expect(firstFont).toBe(secondFont);
-   expect(firstSubtype).toBe(secondSubtype);
-   expect(firstFont).not.toBe(firstSubtype);
-  });
- });
- describe('Cmd', function () {
-  it('should retain the given cmd name', function () {
-   var givenCmd = 'BT';
-   var cmd = Cmd.get(givenCmd);
-   expect(cmd.cmd).toEqual(givenCmd);
-  });
-  it('should create only one object for a command and cache it', function () {
-   var firstBT = Cmd.get('BT');
-   var secondBT = Cmd.get('BT');
-   var firstET = Cmd.get('ET');
-   var secondET = Cmd.get('ET');
-   expect(firstBT).toBe(secondBT);
-   expect(firstET).toBe(secondET);
-   expect(firstBT).not.toBe(firstET);
-  });
- });
- describe('Dict', function () {
-  var checkInvalidHasValues = function (dict) {
-   expect(dict.has()).toBeFalsy();
-   expect(dict.has('Prev')).toBeFalsy();
-  };
-  var checkInvalidKeyValues = function (dict) {
-   expect(dict.get()).toBeUndefined();
-   expect(dict.get('Prev')).toBeUndefined();
-   expect(dict.get('Decode', 'D')).toBeUndefined();
-   expect(dict.get('FontFile', 'FontFile2', 'FontFile3')).toBeNull();
-  };
-  var emptyDict, dictWithSizeKey, dictWithManyKeys;
-  var storedSize = 42;
-  var testFontFile = 'file1';
-  var testFontFile2 = 'file2';
-  var testFontFile3 = 'file3';
-  beforeAll(function (done) {
-   emptyDict = new Dict();
-   dictWithSizeKey = new Dict();
-   dictWithSizeKey.set('Size', storedSize);
-   dictWithManyKeys = new Dict();
-   dictWithManyKeys.set('FontFile', testFontFile);
-   dictWithManyKeys.set('FontFile2', testFontFile2);
-   dictWithManyKeys.set('FontFile3', testFontFile3);
-   done();
-  });
-  afterAll(function () {
-   emptyDict = dictWithSizeKey = dictWithManyKeys = null;
-  });
-  it('should return invalid values for unknown keys', function () {
-   checkInvalidHasValues(emptyDict);
-   checkInvalidKeyValues(emptyDict);
-  });
-  it('should return correct value for stored Size key', function () {
-   expect(dictWithSizeKey.has('Size')).toBeTruthy();
-   expect(dictWithSizeKey.get('Size')).toEqual(storedSize);
-   expect(dictWithSizeKey.get('Prev', 'Size')).toEqual(storedSize);
-   expect(dictWithSizeKey.get('Prev', 'Root', 'Size')).toEqual(storedSize);
-  });
-  it('should return invalid values for unknown keys when Size key is stored', function () {
-   checkInvalidHasValues(dictWithSizeKey);
-   checkInvalidKeyValues(dictWithSizeKey);
-  });
-  it('should return correct value for stored Size key with undefined value', function () {
-   var dict = new Dict();
-   dict.set('Size');
-   expect(dict.has('Size')).toBeTruthy();
-   checkInvalidKeyValues(dict);
-  });
-  it('should return correct values for multiple stored keys', function () {
-   expect(dictWithManyKeys.has('FontFile')).toBeTruthy();
-   expect(dictWithManyKeys.has('FontFile2')).toBeTruthy();
-   expect(dictWithManyKeys.has('FontFile3')).toBeTruthy();
-   expect(dictWithManyKeys.get('FontFile3')).toEqual(testFontFile3);
-   expect(dictWithManyKeys.get('FontFile2', 'FontFile3')).toEqual(testFontFile2);
-   expect(dictWithManyKeys.get('FontFile', 'FontFile2', 'FontFile3')).toEqual(testFontFile);
-  });
-  it('should asynchronously fetch unknown keys', function (done) {
-   var keyPromises = [
-    dictWithManyKeys.getAsync('Size'),
-    dictWithSizeKey.getAsync('FontFile', 'FontFile2', 'FontFile3')
-   ];
-   Promise.all(keyPromises).then(function (values) {
-    expect(values[0]).toBeUndefined();
-    expect(values[1]).toBeNull();
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('should asynchronously fetch correct values for multiple stored keys', function (done) {
-   var keyPromises = [
-    dictWithManyKeys.getAsync('FontFile3'),
-    dictWithManyKeys.getAsync('FontFile2', 'FontFile3'),
-    dictWithManyKeys.getAsync('FontFile', 'FontFile2', 'FontFile3')
-   ];
-   Promise.all(keyPromises).then(function (values) {
-    expect(values[0]).toEqual(testFontFile3);
-    expect(values[1]).toEqual(testFontFile2);
-    expect(values[2]).toEqual(testFontFile);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('should callback for each stored key', function () {
-   var callbackSpy = jasmine.createSpy('spy on callback in dictionary');
-   dictWithManyKeys.forEach(callbackSpy);
-   expect(callbackSpy).toHaveBeenCalled();
-   var callbackSpyCalls = callbackSpy.calls;
-   expect(callbackSpyCalls.argsFor(0)).toEqual([
-    'FontFile',
-    testFontFile
-   ]);
-   expect(callbackSpyCalls.argsFor(1)).toEqual([
-    'FontFile2',
-    testFontFile2
-   ]);
-   expect(callbackSpyCalls.argsFor(2)).toEqual([
-    'FontFile3',
-    testFontFile3
-   ]);
-   expect(callbackSpyCalls.count()).toEqual(3);
-  });
-  it('should handle keys pointing to indirect objects, both sync and async', function (done) {
-   var fontRef = new Ref(1, 0);
-   var xref = new XRefMock([{
-     ref: fontRef,
-     data: testFontFile
-    }]);
-   var fontDict = new Dict(xref);
-   fontDict.set('FontFile', fontRef);
-   expect(fontDict.getRaw('FontFile')).toEqual(fontRef);
-   expect(fontDict.get('FontFile', 'FontFile2', 'FontFile3')).toEqual(testFontFile);
-   fontDict.getAsync('FontFile', 'FontFile2', 'FontFile3').then(function (value) {
-    expect(value).toEqual(testFontFile);
-    done();
-   }).catch(function (reason) {
-    done.fail(reason);
-   });
-  });
-  it('should handle arrays containing indirect objects', function () {
-   var minCoordRef = new Ref(1, 0), maxCoordRef = new Ref(2, 0);
-   var minCoord = 0, maxCoord = 1;
-   var xref = new XRefMock([
-    {
-     ref: minCoordRef,
-     data: minCoord
+  XRefMock.prototype = {
+    fetch: function (ref) {
+      return this.map[ref.toString()];
+    },
+    fetchIfRef: function (obj) {
+      if (!isRef(obj)) {
+        return obj;
+      }
+      return this.fetch(obj);
+    },
+    fetchAsync: function (ref) {
+      return Promise.resolve(this.fetch(ref));
     },
-    {
-     ref: maxCoordRef,
-     data: maxCoord
+    fetchIfRefAsync: function (obj) {
+      return Promise.resolve(this.fetchIfRef(obj));
     }
-   ]);
-   var xObjectDict = new Dict(xref);
-   xObjectDict.set('BBox', [
-    minCoord,
-    maxCoord,
-    minCoordRef,
-    maxCoordRef
-   ]);
-   expect(xObjectDict.get('BBox')).toEqual([
-    minCoord,
-    maxCoord,
-    minCoordRef,
-    maxCoordRef
-   ]);
-   expect(xObjectDict.getArray('BBox')).toEqual([
-    minCoord,
-    maxCoord,
-    minCoord,
-    maxCoord
-   ]);
-  });
-  it('should get all key names', function () {
-   var expectedKeys = [
-    'FontFile',
-    'FontFile2',
-    'FontFile3'
-   ];
-   var keys = dictWithManyKeys.getKeys();
-   expect(keys.sort()).toEqual(expectedKeys);
-  });
-  it('should create only one object for Dict.empty', function () {
-   var firstDictEmpty = Dict.empty;
-   var secondDictEmpty = Dict.empty;
-   expect(firstDictEmpty).toBe(secondDictEmpty);
-   expect(firstDictEmpty).not.toBe(emptyDict);
-  });
-  it('should correctly merge dictionaries', function () {
-   var expectedKeys = [
-    'FontFile',
-    'FontFile2',
-    'FontFile3',
-    'Size'
-   ];
-   var fontFileDict = new Dict();
-   fontFileDict.set('FontFile', 'Type1 font file');
-   var mergedDict = Dict.merge(null, [
-    dictWithManyKeys,
-    dictWithSizeKey,
-    fontFileDict
-   ]);
-   var mergedKeys = mergedDict.getKeys();
-   expect(mergedKeys.sort()).toEqual(expectedKeys);
-   expect(mergedDict.get('FontFile')).toEqual(testFontFile);
-  });
- });
- describe('Ref', function () {
-  it('should retain the stored values', function () {
-   var storedNum = 4;
-   var storedGen = 2;
-   var ref = new Ref(storedNum, storedGen);
-   expect(ref.num).toEqual(storedNum);
-   expect(ref.gen).toEqual(storedGen);
-  });
- });
- describe('RefSet', function () {
-  it('should have a stored value', function () {
-   var ref = new Ref(4, 2);
-   var refset = new RefSet();
-   refset.put(ref);
-   expect(refset.has(ref)).toBeTruthy();
-  });
-  it('should not have an unknown value', function () {
-   var ref = new Ref(4, 2);
-   var refset = new RefSet();
-   expect(refset.has(ref)).toBeFalsy();
-   refset.put(ref);
-   var anotherRef = new Ref(2, 4);
-   expect(refset.has(anotherRef)).toBeFalsy();
-  });
- });
- describe('isName', function () {
-  it('handles non-names', function () {
-   var nonName = {};
-   expect(isName(nonName)).toEqual(false);
-  });
-  it('handles names', function () {
-   var name = Name.get('Font');
-   expect(isName(name)).toEqual(true);
-  });
-  it('handles names with name check', function () {
-   var name = Name.get('Font');
-   expect(isName(name, 'Font')).toEqual(true);
-   expect(isName(name, 'Subtype')).toEqual(false);
-  });
- });
- describe('isCmd', function () {
-  it('handles non-commands', function () {
-   var nonCmd = {};
-   expect(isCmd(nonCmd)).toEqual(false);
-  });
-  it('handles commands', function () {
-   var cmd = Cmd.get('BT');
-   expect(isCmd(cmd)).toEqual(true);
-  });
-  it('handles commands with cmd check', function () {
-   var cmd = Cmd.get('BT');
-   expect(isCmd(cmd, 'BT')).toEqual(true);
-   expect(isCmd(cmd, 'ET')).toEqual(false);
-  });
- });
- describe('isDict', function () {
-  it('handles non-dictionaries', function () {
-   var nonDict = {};
-   expect(isDict(nonDict)).toEqual(false);
-  });
-  it('handles empty dictionaries with type check', function () {
-   var dict = Dict.empty;
-   expect(isDict(dict)).toEqual(true);
-   expect(isDict(dict, 'Page')).toEqual(false);
-  });
-  it('handles dictionaries with type check', function () {
-   var dict = new Dict();
-   dict.set('Type', Name.get('Page'));
-   expect(isDict(dict, 'Page')).toEqual(true);
-   expect(isDict(dict, 'Contents')).toEqual(false);
-  });
- });
- describe('isRef', function () {
-  it('handles non-refs', function () {
-   var nonRef = {};
-   expect(isRef(nonRef)).toEqual(false);
-  });
-  it('handles refs', function () {
-   var ref = new Ref(1, 0);
-   expect(isRef(ref)).toEqual(true);
-  });
- });
- describe('isRefsEqual', function () {
-  it('should handle different Refs pointing to the same object', function () {
-   var ref1 = new Ref(1, 0);
-   var ref2 = new Ref(1, 0);
-   expect(isRefsEqual(ref1, ref2)).toEqual(true);
-  });
-  it('should handle Refs pointing to different objects', function () {
-   var ref1 = new Ref(1, 0);
-   var ref2 = new Ref(2, 0);
-   expect(isRefsEqual(ref1, ref2)).toEqual(false);
+  };
+  describe('Name', function () {
+    it('should retain the given name', function () {
+      var givenName = 'Font';
+      var name = Name.get(givenName);
+      expect(name.name).toEqual(givenName);
+    });
+    it('should create only one object for a name and cache it', function () {
+      var firstFont = Name.get('Font');
+      var secondFont = Name.get('Font');
+      var firstSubtype = Name.get('Subtype');
+      var secondSubtype = Name.get('Subtype');
+      expect(firstFont).toBe(secondFont);
+      expect(firstSubtype).toBe(secondSubtype);
+      expect(firstFont).not.toBe(firstSubtype);
+    });
+  });
+  describe('Cmd', function () {
+    it('should retain the given cmd name', function () {
+      var givenCmd = 'BT';
+      var cmd = Cmd.get(givenCmd);
+      expect(cmd.cmd).toEqual(givenCmd);
+    });
+    it('should create only one object for a command and cache it', function () {
+      var firstBT = Cmd.get('BT');
+      var secondBT = Cmd.get('BT');
+      var firstET = Cmd.get('ET');
+      var secondET = Cmd.get('ET');
+      expect(firstBT).toBe(secondBT);
+      expect(firstET).toBe(secondET);
+      expect(firstBT).not.toBe(firstET);
+    });
+  });
+  describe('Dict', function () {
+    var checkInvalidHasValues = function (dict) {
+      expect(dict.has()).toBeFalsy();
+      expect(dict.has('Prev')).toBeFalsy();
+    };
+    var checkInvalidKeyValues = function (dict) {
+      expect(dict.get()).toBeUndefined();
+      expect(dict.get('Prev')).toBeUndefined();
+      expect(dict.get('Decode', 'D')).toBeUndefined();
+      expect(dict.get('FontFile', 'FontFile2', 'FontFile3')).toBeNull();
+    };
+    var emptyDict, dictWithSizeKey, dictWithManyKeys;
+    var storedSize = 42;
+    var testFontFile = 'file1';
+    var testFontFile2 = 'file2';
+    var testFontFile3 = 'file3';
+    beforeAll(function (done) {
+      emptyDict = new Dict();
+      dictWithSizeKey = new Dict();
+      dictWithSizeKey.set('Size', storedSize);
+      dictWithManyKeys = new Dict();
+      dictWithManyKeys.set('FontFile', testFontFile);
+      dictWithManyKeys.set('FontFile2', testFontFile2);
+      dictWithManyKeys.set('FontFile3', testFontFile3);
+      done();
+    });
+    afterAll(function () {
+      emptyDict = dictWithSizeKey = dictWithManyKeys = null;
+    });
+    it('should return invalid values for unknown keys', function () {
+      checkInvalidHasValues(emptyDict);
+      checkInvalidKeyValues(emptyDict);
+    });
+    it('should return correct value for stored Size key', function () {
+      expect(dictWithSizeKey.has('Size')).toBeTruthy();
+      expect(dictWithSizeKey.get('Size')).toEqual(storedSize);
+      expect(dictWithSizeKey.get('Prev', 'Size')).toEqual(storedSize);
+      expect(dictWithSizeKey.get('Prev', 'Root', 'Size')).toEqual(storedSize);
+    });
+    it('should return invalid values for unknown keys when Size key is stored', function () {
+      checkInvalidHasValues(dictWithSizeKey);
+      checkInvalidKeyValues(dictWithSizeKey);
+    });
+    it('should return correct value for stored Size key with undefined value', function () {
+      var dict = new Dict();
+      dict.set('Size');
+      expect(dict.has('Size')).toBeTruthy();
+      checkInvalidKeyValues(dict);
+    });
+    it('should return correct values for multiple stored keys', function () {
+      expect(dictWithManyKeys.has('FontFile')).toBeTruthy();
+      expect(dictWithManyKeys.has('FontFile2')).toBeTruthy();
+      expect(dictWithManyKeys.has('FontFile3')).toBeTruthy();
+      expect(dictWithManyKeys.get('FontFile3')).toEqual(testFontFile3);
+      expect(dictWithManyKeys.get('FontFile2', 'FontFile3')).toEqual(testFontFile2);
+      expect(dictWithManyKeys.get('FontFile', 'FontFile2', 'FontFile3')).toEqual(testFontFile);
+    });
+    it('should asynchronously fetch unknown keys', function (done) {
+      var keyPromises = [dictWithManyKeys.getAsync('Size'), dictWithSizeKey.getAsync('FontFile', 'FontFile2', 'FontFile3')];
+      Promise.all(keyPromises).then(function (values) {
+        expect(values[0]).toBeUndefined();
+        expect(values[1]).toBeNull();
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('should asynchronously fetch correct values for multiple stored keys', function (done) {
+      var keyPromises = [dictWithManyKeys.getAsync('FontFile3'), dictWithManyKeys.getAsync('FontFile2', 'FontFile3'), dictWithManyKeys.getAsync('FontFile', 'FontFile2', 'FontFile3')];
+      Promise.all(keyPromises).then(function (values) {
+        expect(values[0]).toEqual(testFontFile3);
+        expect(values[1]).toEqual(testFontFile2);
+        expect(values[2]).toEqual(testFontFile);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('should callback for each stored key', function () {
+      var callbackSpy = jasmine.createSpy('spy on callback in dictionary');
+      dictWithManyKeys.forEach(callbackSpy);
+      expect(callbackSpy).toHaveBeenCalled();
+      var callbackSpyCalls = callbackSpy.calls;
+      expect(callbackSpyCalls.argsFor(0)).toEqual(['FontFile', testFontFile]);
+      expect(callbackSpyCalls.argsFor(1)).toEqual(['FontFile2', testFontFile2]);
+      expect(callbackSpyCalls.argsFor(2)).toEqual(['FontFile3', testFontFile3]);
+      expect(callbackSpyCalls.count()).toEqual(3);
+    });
+    it('should handle keys pointing to indirect objects, both sync and async', function (done) {
+      var fontRef = new Ref(1, 0);
+      var xref = new XRefMock([{
+        ref: fontRef,
+        data: testFontFile
+      }]);
+      var fontDict = new Dict(xref);
+      fontDict.set('FontFile', fontRef);
+      expect(fontDict.getRaw('FontFile')).toEqual(fontRef);
+      expect(fontDict.get('FontFile', 'FontFile2', 'FontFile3')).toEqual(testFontFile);
+      fontDict.getAsync('FontFile', 'FontFile2', 'FontFile3').then(function (value) {
+        expect(value).toEqual(testFontFile);
+        done();
+      }).catch(function (reason) {
+        done.fail(reason);
+      });
+    });
+    it('should handle arrays containing indirect objects', function () {
+      var minCoordRef = new Ref(1, 0),
+          maxCoordRef = new Ref(2, 0);
+      var minCoord = 0,
+          maxCoord = 1;
+      var xref = new XRefMock([{
+        ref: minCoordRef,
+        data: minCoord
+      }, {
+        ref: maxCoordRef,
+        data: maxCoord
+      }]);
+      var xObjectDict = new Dict(xref);
+      xObjectDict.set('BBox', [minCoord, maxCoord, minCoordRef, maxCoordRef]);
+      expect(xObjectDict.get('BBox')).toEqual([minCoord, maxCoord, minCoordRef, maxCoordRef]);
+      expect(xObjectDict.getArray('BBox')).toEqual([minCoord, maxCoord, minCoord, maxCoord]);
+    });
+    it('should get all key names', function () {
+      var expectedKeys = ['FontFile', 'FontFile2', 'FontFile3'];
+      var keys = dictWithManyKeys.getKeys();
+      expect(keys.sort()).toEqual(expectedKeys);
+    });
+    it('should create only one object for Dict.empty', function () {
+      var firstDictEmpty = Dict.empty;
+      var secondDictEmpty = Dict.empty;
+      expect(firstDictEmpty).toBe(secondDictEmpty);
+      expect(firstDictEmpty).not.toBe(emptyDict);
+    });
+    it('should correctly merge dictionaries', function () {
+      var expectedKeys = ['FontFile', 'FontFile2', 'FontFile3', 'Size'];
+      var fontFileDict = new Dict();
+      fontFileDict.set('FontFile', 'Type1 font file');
+      var mergedDict = Dict.merge(null, [dictWithManyKeys, dictWithSizeKey, fontFileDict]);
+      var mergedKeys = mergedDict.getKeys();
+      expect(mergedKeys.sort()).toEqual(expectedKeys);
+      expect(mergedDict.get('FontFile')).toEqual(testFontFile);
+    });
+  });
+  describe('Ref', function () {
+    it('should retain the stored values', function () {
+      var storedNum = 4;
+      var storedGen = 2;
+      var ref = new Ref(storedNum, storedGen);
+      expect(ref.num).toEqual(storedNum);
+      expect(ref.gen).toEqual(storedGen);
+    });
+  });
+  describe('RefSet', function () {
+    it('should have a stored value', function () {
+      var ref = new Ref(4, 2);
+      var refset = new RefSet();
+      refset.put(ref);
+      expect(refset.has(ref)).toBeTruthy();
+    });
+    it('should not have an unknown value', function () {
+      var ref = new Ref(4, 2);
+      var refset = new RefSet();
+      expect(refset.has(ref)).toBeFalsy();
+      refset.put(ref);
+      var anotherRef = new Ref(2, 4);
+      expect(refset.has(anotherRef)).toBeFalsy();
+    });
+  });
+  describe('isName', function () {
+    it('handles non-names', function () {
+      var nonName = {};
+      expect(isName(nonName)).toEqual(false);
+    });
+    it('handles names', function () {
+      var name = Name.get('Font');
+      expect(isName(name)).toEqual(true);
+    });
+    it('handles names with name check', function () {
+      var name = Name.get('Font');
+      expect(isName(name, 'Font')).toEqual(true);
+      expect(isName(name, 'Subtype')).toEqual(false);
+    });
+  });
+  describe('isCmd', function () {
+    it('handles non-commands', function () {
+      var nonCmd = {};
+      expect(isCmd(nonCmd)).toEqual(false);
+    });
+    it('handles commands', function () {
+      var cmd = Cmd.get('BT');
+      expect(isCmd(cmd)).toEqual(true);
+    });
+    it('handles commands with cmd check', function () {
+      var cmd = Cmd.get('BT');
+      expect(isCmd(cmd, 'BT')).toEqual(true);
+      expect(isCmd(cmd, 'ET')).toEqual(false);
+    });
+  });
+  describe('isDict', function () {
+    it('handles non-dictionaries', function () {
+      var nonDict = {};
+      expect(isDict(nonDict)).toEqual(false);
+    });
+    it('handles empty dictionaries with type check', function () {
+      var dict = Dict.empty;
+      expect(isDict(dict)).toEqual(true);
+      expect(isDict(dict, 'Page')).toEqual(false);
+    });
+    it('handles dictionaries with type check', function () {
+      var dict = new Dict();
+      dict.set('Type', Name.get('Page'));
+      expect(isDict(dict, 'Page')).toEqual(true);
+      expect(isDict(dict, 'Contents')).toEqual(false);
+    });
+  });
+  describe('isRef', function () {
+    it('handles non-refs', function () {
+      var nonRef = {};
+      expect(isRef(nonRef)).toEqual(false);
+    });
+    it('handles refs', function () {
+      var ref = new Ref(1, 0);
+      expect(isRef(ref)).toEqual(true);
+    });
+  });
+  describe('isRefsEqual', function () {
+    it('should handle different Refs pointing to the same object', function () {
+      var ref1 = new Ref(1, 0);
+      var ref2 = new Ref(1, 0);
+      expect(isRefsEqual(ref1, ref2)).toEqual(true);
+    });
+    it('should handle Refs pointing to different objects', function () {
+      var ref1 = new Ref(1, 0);
+      var ref2 = new Ref(2, 0);
+      expect(isRefsEqual(ref1, ref2)).toEqual(false);
+    });
   });
- });
 });

+ 37 - 52
lib/test/unit/stream_spec.js

@@ -13,65 +13,50 @@
  * limitations under the License.
  */
 'use strict';
+
 var corePrimitives = require('../../core/primitives.js');
 var coreStream = require('../../core/stream.js');
 var Dict = corePrimitives.Dict;
 var Stream = coreStream.Stream;
 var PredictorStream = coreStream.PredictorStream;
 describe('stream', function () {
- beforeEach(function () {
-  jasmine.addMatchers({
-   toMatchTypedArray: function (util, customEqualityTesters) {
-    return {
-     compare: function (actual, expected) {
-      var result = {};
-      if (actual.length !== expected.length) {
-       result.pass = false;
-       result.message = 'Array length: ' + actual.length + ', expected: ' + expected.length;
-       return result;
+  beforeEach(function () {
+    jasmine.addMatchers({
+      toMatchTypedArray: function (util, customEqualityTesters) {
+        return {
+          compare: function (actual, expected) {
+            var result = {};
+            if (actual.length !== expected.length) {
+              result.pass = false;
+              result.message = 'Array length: ' + actual.length + ', expected: ' + expected.length;
+              return result;
+            }
+            result.pass = true;
+            for (var i = 0, ii = expected.length; i < ii; i++) {
+              var a = actual[i],
+                  b = expected[i];
+              if (a !== b) {
+                result.pass = false;
+                break;
+              }
+            }
+            return result;
+          }
+        };
       }
-      result.pass = true;
-      for (var i = 0, ii = expected.length; i < ii; i++) {
-       var a = actual[i], b = expected[i];
-       if (a !== b) {
-        result.pass = false;
-        break;
-       }
-      }
-      return result;
-     }
-    };
-   }
+    });
   });
- });
- describe('PredictorStream', function () {
-  it('should decode simple predictor data', function () {
-   var dict = new Dict();
-   dict.set('Predictor', 12);
-   dict.set('Colors', 1);
-   dict.set('BitsPerComponent', 8);
-   dict.set('Columns', 2);
-   var input = new Stream(new Uint8Array([
-    2,
-    100,
-    3,
-    2,
-    1,
-    255,
-    2,
-    1,
-    255
-   ]), 0, 9, dict);
-   var predictor = new PredictorStream(input, 9, dict);
-   var result = predictor.getBytes(6);
-   expect(result).toMatchTypedArray(new Uint8Array([
-    100,
-    3,
-    101,
-    2,
-    102,
-    1
-   ]));
+  describe('PredictorStream', function () {
+    it('should decode simple predictor data', function () {
+      var dict = new Dict();
+      dict.set('Predictor', 12);
+      dict.set('Colors', 1);
+      dict.set('BitsPerComponent', 8);
+      dict.set('Columns', 2);
+      var input = new Stream(new Uint8Array([2, 100, 3, 2, 1, 255, 2, 1, 255]), 0, 9, dict);
+      var predictor = new PredictorStream(input, 9, dict);
+      var result = predictor.getBytes(6);
+      expect(result).toMatchTypedArray(new Uint8Array([100, 3, 101, 2, 102, 1]));
+    });
   });
- });
 });

+ 27 - 26
lib/test/unit/test_utils.js

@@ -13,35 +13,36 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../../shared/util.js');
 var CMapCompressionType = sharedUtil.CMapCompressionType;
 var NodeCMapReaderFactory = function NodeCMapReaderFactoryClosure() {
- function NodeCMapReaderFactory(params) {
-  this.baseUrl = params.baseUrl || null;
-  this.isCompressed = params.isCompressed || false;
- }
- NodeCMapReaderFactory.prototype = {
-  fetch: function (params) {
-   var name = params.name;
-   if (!name) {
-    return Promise.reject(new Error('CMap name must be specified.'));
-   }
-   return new Promise(function (resolve, reject) {
-    var url = this.baseUrl + name + (this.isCompressed ? '.bcmap' : '');
-    var fs = require('fs');
-    fs.readFile(url, function (error, data) {
-     if (error || !data) {
-      reject(new Error('Unable to load ' + (this.isCompressed ? 'binary ' : '') + 'CMap at: ' + url));
-      return;
-     }
-     resolve({
-      cMapData: new Uint8Array(data),
-      compressionType: this.isCompressed ? CMapCompressionType.BINARY : CMapCompressionType.NONE
-     });
-    }.bind(this));
-   }.bind(this));
+  function NodeCMapReaderFactory(params) {
+    this.baseUrl = params.baseUrl || null;
+    this.isCompressed = params.isCompressed || false;
   }
- };
- return NodeCMapReaderFactory;
+  NodeCMapReaderFactory.prototype = {
+    fetch: function (params) {
+      var name = params.name;
+      if (!name) {
+        return Promise.reject(new Error('CMap name must be specified.'));
+      }
+      return new Promise(function (resolve, reject) {
+        var url = this.baseUrl + name + (this.isCompressed ? '.bcmap' : '');
+        var fs = require('fs');
+        fs.readFile(url, function (error, data) {
+          if (error || !data) {
+            reject(new Error('Unable to load ' + (this.isCompressed ? 'binary ' : '') + 'CMap at: ' + url));
+            return;
+          }
+          resolve({
+            cMapData: new Uint8Array(data),
+            compressionType: this.isCompressed ? CMapCompressionType.BINARY : CMapCompressionType.NONE
+          });
+        }.bind(this));
+      }.bind(this));
+    }
+  };
+  return NodeCMapReaderFactory;
 }();
 exports.NodeCMapReaderFactory = NodeCMapReaderFactory;

+ 56 - 58
lib/test/unit/testreporter.js

@@ -13,67 +13,65 @@
  * limitations under the License.
  */
 'use strict';
+
 var TestReporter = function (browser, appPath) {
- function send(action, json, cb) {
-  var r = new XMLHttpRequest();
-  r.open('POST', action, true);
-  r.setRequestHeader('Content-Type', 'application/json');
-  r.onreadystatechange = function sendTaskResultOnreadystatechange(e) {
-   if (r.readyState === 4) {
-    if (r.status !== 200) {
-     send(action, json, cb);
+  function send(action, json, cb) {
+    var r = new XMLHttpRequest();
+    r.open('POST', action, true);
+    r.setRequestHeader('Content-Type', 'application/json');
+    r.onreadystatechange = function sendTaskResultOnreadystatechange(e) {
+      if (r.readyState === 4) {
+        if (r.status !== 200) {
+          send(action, json, cb);
+        } else {
+          if (cb) {
+            cb();
+          }
+        }
+      }
+    };
+    json['browser'] = browser;
+    r.send(JSON.stringify(json));
+  }
+  function sendInfo(message) {
+    send('/info', { message: message });
+  }
+  function sendResult(status, description, error) {
+    var message = {
+      status: status,
+      description: description
+    };
+    if (typeof error !== 'undefined') {
+      message['error'] = error;
+    }
+    send('/submit_task_results', message);
+  }
+  function sendQuitRequest() {
+    send('/tellMeToQuit?path=' + escape(appPath), {});
+  }
+  this.now = function () {
+    return new Date().getTime();
+  };
+  this.jasmineStarted = function (suiteInfo) {
+    this.runnerStartTime = this.now();
+    sendInfo('Started unit tests for ' + browser + '.');
+  };
+  this.suiteStarted = function (result) {};
+  this.specStarted = function (result) {};
+  this.specDone = function (result) {
+    if (result.failedExpectations.length === 0) {
+      sendResult('TEST-PASSED', result.description);
     } else {
-     if (cb) {
-      cb();
-     }
+      var failedMessages = '';
+      var items = result.failedExpectations;
+      for (var i = 0, ii = items.length; i < ii; i++) {
+        failedMessages += items[i].message + ' ';
+      }
+      sendResult('TEST-UNEXPECTED-FAIL', result.description, failedMessages);
     }
-   }
   };
-  json['browser'] = browser;
-  r.send(JSON.stringify(json));
- }
- function sendInfo(message) {
-  send('/info', { message: message });
- }
- function sendResult(status, description, error) {
-  var message = {
-   status: status,
-   description: description
+  this.suiteDone = function (result) {};
+  this.jasmineDone = function () {
+    setTimeout(sendQuitRequest, 500);
   };
-  if (typeof error !== 'undefined') {
-   message['error'] = error;
-  }
-  send('/submit_task_results', message);
- }
- function sendQuitRequest() {
-  send('/tellMeToQuit?path=' + escape(appPath), {});
- }
- this.now = function () {
-  return new Date().getTime();
- };
- this.jasmineStarted = function (suiteInfo) {
-  this.runnerStartTime = this.now();
-  sendInfo('Started unit tests for ' + browser + '.');
- };
- this.suiteStarted = function (result) {
- };
- this.specStarted = function (result) {
- };
- this.specDone = function (result) {
-  if (result.failedExpectations.length === 0) {
-   sendResult('TEST-PASSED', result.description);
-  } else {
-   var failedMessages = '';
-   var items = result.failedExpectations;
-   for (var i = 0, ii = items.length; i < ii; i++) {
-    failedMessages += items[i].message + ' ';
-   }
-   sendResult('TEST-UNEXPECTED-FAIL', result.description, failedMessages);
-  }
- };
- this.suiteDone = function (result) {
- };
- this.jasmineDone = function () {
-  setTimeout(sendQuitRequest, 500);
- };
 };

+ 72 - 84
lib/test/unit/type1_parser_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreFonts = require('../../core/fonts.js');
 var coreStream = require('../../core/stream.js');
 var coreType1Parser = require('../../core/type1_parser.js');
@@ -20,88 +21,75 @@ var SEAC_ANALYSIS_ENABLED = coreFonts.SEAC_ANALYSIS_ENABLED;
 var StringStream = coreStream.StringStream;
 var Type1Parser = coreType1Parser.Type1Parser;
 describe('Type1Parser', function () {
- it('splits tokens', function () {
-  var stream = new StringStream('/BlueValues[-17 0]noaccess def');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  expect(parser.getToken()).toEqual('/');
-  expect(parser.getToken()).toEqual('BlueValues');
-  expect(parser.getToken()).toEqual('[');
-  expect(parser.getToken()).toEqual('-17');
-  expect(parser.getToken()).toEqual('0');
-  expect(parser.getToken()).toEqual(']');
-  expect(parser.getToken()).toEqual('noaccess');
-  expect(parser.getToken()).toEqual('def');
-  expect(parser.getToken()).toEqual(null);
- });
- it('handles glued tokens', function () {
-  var stream = new StringStream('dup/CharStrings');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  expect(parser.getToken()).toEqual('dup');
-  expect(parser.getToken()).toEqual('/');
-  expect(parser.getToken()).toEqual('CharStrings');
- });
- it('ignores whitespace', function () {
-  var stream = new StringStream('\nab   c\t');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  expect(parser.getToken()).toEqual('ab');
-  expect(parser.getToken()).toEqual('c');
- });
- it('parses numbers', function () {
-  var stream = new StringStream('123');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  expect(parser.readNumber()).toEqual(123);
- });
- it('parses booleans', function () {
-  var stream = new StringStream('true false');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  expect(parser.readBoolean()).toEqual(1);
-  expect(parser.readBoolean()).toEqual(0);
- });
- it('parses number arrays', function () {
-  var stream = new StringStream('[1 2]');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  expect(parser.readNumberArray()).toEqual([
-   1,
-   2
-  ]);
-  stream = new StringStream('[ 1 2 ]');
-  parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  expect(parser.readNumberArray()).toEqual([
-   1,
-   2
-  ]);
- });
- it('skips comments', function () {
-  var stream = new StringStream('%!PS-AdobeFont-1.0: CMSY10 003.002\n' + '%%Title: CMSY10\n' + '%Version: 003.002\n' + 'FontDirectory');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  expect(parser.getToken()).toEqual('FontDirectory');
- });
- it('parses font program', function () {
-  var stream = new StringStream('/ExpansionFactor  99\n' + '/Subrs 1 array\n' + 'dup 0 1 RD x noaccess put\n' + 'end\n' + '/CharStrings 46 dict dup begin\n' + '/.notdef 1 RD x ND\n' + 'end');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  var program = parser.extractFontProgram();
-  expect(program.charstrings.length).toEqual(1);
-  expect(program.properties.privateData.ExpansionFactor).toEqual(99);
- });
- it('parses font header font matrix', function () {
-  var stream = new StringStream('/FontMatrix [0.001 0 0 0.001 0 0 ]readonly def\n');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  var props = {};
-  parser.extractFontHeader(props);
-  expect(props.fontMatrix).toEqual([
-   0.001,
-   0,
-   0,
-   0.001,
-   0,
-   0
-  ]);
- });
- it('parses font header encoding', function () {
-  var stream = new StringStream('/Encoding 256 array\n' + '0 1 255 {1 index exch /.notdef put} for\n' + 'dup 33 /arrowright put\n' + 'readonly def\n');
-  var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
-  var props = { overridableEncoding: true };
-  parser.extractFontHeader(props);
-  expect(props.builtInEncoding[33]).toEqual('arrowright');
- });
+  it('splits tokens', function () {
+    var stream = new StringStream('/BlueValues[-17 0]noaccess def');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    expect(parser.getToken()).toEqual('/');
+    expect(parser.getToken()).toEqual('BlueValues');
+    expect(parser.getToken()).toEqual('[');
+    expect(parser.getToken()).toEqual('-17');
+    expect(parser.getToken()).toEqual('0');
+    expect(parser.getToken()).toEqual(']');
+    expect(parser.getToken()).toEqual('noaccess');
+    expect(parser.getToken()).toEqual('def');
+    expect(parser.getToken()).toEqual(null);
+  });
+  it('handles glued tokens', function () {
+    var stream = new StringStream('dup/CharStrings');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    expect(parser.getToken()).toEqual('dup');
+    expect(parser.getToken()).toEqual('/');
+    expect(parser.getToken()).toEqual('CharStrings');
+  });
+  it('ignores whitespace', function () {
+    var stream = new StringStream('\nab   c\t');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    expect(parser.getToken()).toEqual('ab');
+    expect(parser.getToken()).toEqual('c');
+  });
+  it('parses numbers', function () {
+    var stream = new StringStream('123');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    expect(parser.readNumber()).toEqual(123);
+  });
+  it('parses booleans', function () {
+    var stream = new StringStream('true false');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    expect(parser.readBoolean()).toEqual(1);
+    expect(parser.readBoolean()).toEqual(0);
+  });
+  it('parses number arrays', function () {
+    var stream = new StringStream('[1 2]');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    expect(parser.readNumberArray()).toEqual([1, 2]);
+    stream = new StringStream('[ 1 2 ]');
+    parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    expect(parser.readNumberArray()).toEqual([1, 2]);
+  });
+  it('skips comments', function () {
+    var stream = new StringStream('%!PS-AdobeFont-1.0: CMSY10 003.002\n' + '%%Title: CMSY10\n' + '%Version: 003.002\n' + 'FontDirectory');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    expect(parser.getToken()).toEqual('FontDirectory');
+  });
+  it('parses font program', function () {
+    var stream = new StringStream('/ExpansionFactor  99\n' + '/Subrs 1 array\n' + 'dup 0 1 RD x noaccess put\n' + 'end\n' + '/CharStrings 46 dict dup begin\n' + '/.notdef 1 RD x ND\n' + 'end');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    var program = parser.extractFontProgram();
+    expect(program.charstrings.length).toEqual(1);
+    expect(program.properties.privateData.ExpansionFactor).toEqual(99);
+  });
+  it('parses font header font matrix', function () {
+    var stream = new StringStream('/FontMatrix [0.001 0 0 0.001 0 0 ]readonly def\n');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    var props = {};
+    parser.extractFontHeader(props);
+    expect(props.fontMatrix).toEqual([0.001, 0, 0, 0.001, 0, 0]);
+  });
+  it('parses font header encoding', function () {
+    var stream = new StringStream('/Encoding 256 array\n' + '0 1 255 {1 index exch /.notdef put} for\n' + 'dup 33 /arrowright put\n' + 'readonly def\n');
+    var parser = new Type1Parser(stream, false, SEAC_ANALYSIS_ENABLED);
+    var props = { overridableEncoding: true };
+    parser.extractFontHeader(props);
+    expect(props.builtInEncoding[33]).toEqual('arrowright');
+  });
 });

+ 110 - 137
lib/test/unit/ui_utils_spec.js

@@ -13,147 +13,120 @@
  * limitations under the License.
  */
 'use strict';
+
 var webUiUtils = require('../../web/ui_utils.js');
 var binarySearchFirstItem = webUiUtils.binarySearchFirstItem;
 var EventBus = webUiUtils.EventBus;
 describe('ui_utils', function () {
- describe('binary search', function () {
-  function isTrue(boolean) {
-   return boolean;
-  }
-  function isGreater3(number) {
-   return number > 3;
-  }
-  it('empty array', function () {
-   expect(binarySearchFirstItem([], isTrue)).toEqual(0);
+  describe('binary search', function () {
+    function isTrue(boolean) {
+      return boolean;
+    }
+    function isGreater3(number) {
+      return number > 3;
+    }
+    it('empty array', function () {
+      expect(binarySearchFirstItem([], isTrue)).toEqual(0);
+    });
+    it('single boolean entry', function () {
+      expect(binarySearchFirstItem([false], isTrue)).toEqual(1);
+      expect(binarySearchFirstItem([true], isTrue)).toEqual(0);
+    });
+    it('three boolean entries', function () {
+      expect(binarySearchFirstItem([true, true, true], isTrue)).toEqual(0);
+      expect(binarySearchFirstItem([false, true, true], isTrue)).toEqual(1);
+      expect(binarySearchFirstItem([false, false, true], isTrue)).toEqual(2);
+      expect(binarySearchFirstItem([false, false, false], isTrue)).toEqual(3);
+    });
+    it('three numeric entries', function () {
+      expect(binarySearchFirstItem([0, 1, 2], isGreater3)).toEqual(3);
+      expect(binarySearchFirstItem([2, 3, 4], isGreater3)).toEqual(2);
+      expect(binarySearchFirstItem([4, 5, 6], isGreater3)).toEqual(0);
+    });
   });
-  it('single boolean entry', function () {
-   expect(binarySearchFirstItem([false], isTrue)).toEqual(1);
-   expect(binarySearchFirstItem([true], isTrue)).toEqual(0);
+  describe('EventBus', function () {
+    it('dispatch event', function () {
+      var eventBus = new EventBus();
+      var count = 0;
+      eventBus.on('test', function () {
+        count++;
+      });
+      eventBus.dispatch('test');
+      expect(count).toEqual(1);
+    });
+    it('dispatch different event', function () {
+      var eventBus = new EventBus();
+      var count = 0;
+      eventBus.on('test', function () {
+        count++;
+      });
+      eventBus.dispatch('nottest');
+      expect(count).toEqual(0);
+    });
+    it('dispatch event multiple times', function () {
+      var eventBus = new EventBus();
+      var count = 0;
+      eventBus.dispatch('test');
+      eventBus.on('test', function () {
+        count++;
+      });
+      eventBus.dispatch('test');
+      eventBus.dispatch('test');
+      expect(count).toEqual(2);
+    });
+    it('dispatch event to multiple handlers', function () {
+      var eventBus = new EventBus();
+      var count = 0;
+      eventBus.on('test', function () {
+        count++;
+      });
+      eventBus.on('test', function () {
+        count++;
+      });
+      eventBus.dispatch('test');
+      expect(count).toEqual(2);
+    });
+    it('dispatch to detached', function () {
+      var eventBus = new EventBus();
+      var count = 0;
+      var listener = function () {
+        count++;
+      };
+      eventBus.on('test', listener);
+      eventBus.dispatch('test');
+      eventBus.off('test', listener);
+      eventBus.dispatch('test');
+      expect(count).toEqual(1);
+    });
+    it('dispatch to wrong detached', function () {
+      var eventBus = new EventBus();
+      var count = 0;
+      eventBus.on('test', function () {
+        count++;
+      });
+      eventBus.dispatch('test');
+      eventBus.off('test', function () {
+        count++;
+      });
+      eventBus.dispatch('test');
+      expect(count).toEqual(2);
+    });
+    it('dispatch to detached during handling', function () {
+      var eventBus = new EventBus();
+      var count = 0;
+      var listener1 = function () {
+        eventBus.off('test', listener2);
+        count++;
+      };
+      var listener2 = function () {
+        eventBus.off('test', listener1);
+        count++;
+      };
+      eventBus.on('test', listener1);
+      eventBus.on('test', listener2);
+      eventBus.dispatch('test');
+      eventBus.dispatch('test');
+      expect(count).toEqual(2);
+    });
   });
-  it('three boolean entries', function () {
-   expect(binarySearchFirstItem([
-    true,
-    true,
-    true
-   ], isTrue)).toEqual(0);
-   expect(binarySearchFirstItem([
-    false,
-    true,
-    true
-   ], isTrue)).toEqual(1);
-   expect(binarySearchFirstItem([
-    false,
-    false,
-    true
-   ], isTrue)).toEqual(2);
-   expect(binarySearchFirstItem([
-    false,
-    false,
-    false
-   ], isTrue)).toEqual(3);
-  });
-  it('three numeric entries', function () {
-   expect(binarySearchFirstItem([
-    0,
-    1,
-    2
-   ], isGreater3)).toEqual(3);
-   expect(binarySearchFirstItem([
-    2,
-    3,
-    4
-   ], isGreater3)).toEqual(2);
-   expect(binarySearchFirstItem([
-    4,
-    5,
-    6
-   ], isGreater3)).toEqual(0);
-  });
- });
- describe('EventBus', function () {
-  it('dispatch event', function () {
-   var eventBus = new EventBus();
-   var count = 0;
-   eventBus.on('test', function () {
-    count++;
-   });
-   eventBus.dispatch('test');
-   expect(count).toEqual(1);
-  });
-  it('dispatch different event', function () {
-   var eventBus = new EventBus();
-   var count = 0;
-   eventBus.on('test', function () {
-    count++;
-   });
-   eventBus.dispatch('nottest');
-   expect(count).toEqual(0);
-  });
-  it('dispatch event multiple times', function () {
-   var eventBus = new EventBus();
-   var count = 0;
-   eventBus.dispatch('test');
-   eventBus.on('test', function () {
-    count++;
-   });
-   eventBus.dispatch('test');
-   eventBus.dispatch('test');
-   expect(count).toEqual(2);
-  });
-  it('dispatch event to multiple handlers', function () {
-   var eventBus = new EventBus();
-   var count = 0;
-   eventBus.on('test', function () {
-    count++;
-   });
-   eventBus.on('test', function () {
-    count++;
-   });
-   eventBus.dispatch('test');
-   expect(count).toEqual(2);
-  });
-  it('dispatch to detached', function () {
-   var eventBus = new EventBus();
-   var count = 0;
-   var listener = function () {
-    count++;
-   };
-   eventBus.on('test', listener);
-   eventBus.dispatch('test');
-   eventBus.off('test', listener);
-   eventBus.dispatch('test');
-   expect(count).toEqual(1);
-  });
-  it('dispatch to wrong detached', function () {
-   var eventBus = new EventBus();
-   var count = 0;
-   eventBus.on('test', function () {
-    count++;
-   });
-   eventBus.dispatch('test');
-   eventBus.off('test', function () {
-    count++;
-   });
-   eventBus.dispatch('test');
-   expect(count).toEqual(2);
-  });
-  it('dispatch to detached during handling', function () {
-   var eventBus = new EventBus();
-   var count = 0;
-   var listener1 = function () {
-    eventBus.off('test', listener2);
-    count++;
-   };
-   var listener2 = function () {
-    eventBus.off('test', listener1);
-    count++;
-   };
-   eventBus.on('test', listener1);
-   eventBus.on('test', listener2);
-   eventBus.dispatch('test');
-   eventBus.dispatch('test');
-   expect(count).toEqual(2);
-  });
- });
 });

+ 89 - 88
lib/test/unit/unicode_spec.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var coreGlyphList = require('../../core/glyphlist.js');
 var coreUnicode = require('../../core/unicode.js');
 var getGlyphsUnicode = coreGlyphList.getGlyphsUnicode;
@@ -23,93 +24,93 @@ var getUnicodeRangeFor = coreUnicode.getUnicodeRangeFor;
 var getNormalizedUnicodes = coreUnicode.getNormalizedUnicodes;
 var reverseIfRtl = coreUnicode.reverseIfRtl;
 describe('unicode', function () {
- describe('mapSpecialUnicodeValues', function () {
-  it('should not re-map normal Unicode values', function () {
-   expect(mapSpecialUnicodeValues(0x0041)).toEqual(0x0041);
-   expect(mapSpecialUnicodeValues(0xFB01)).toEqual(0xFB01);
+  describe('mapSpecialUnicodeValues', function () {
+    it('should not re-map normal Unicode values', function () {
+      expect(mapSpecialUnicodeValues(0x0041)).toEqual(0x0041);
+      expect(mapSpecialUnicodeValues(0xFB01)).toEqual(0xFB01);
+    });
+    it('should re-map special Unicode values', function () {
+      expect(mapSpecialUnicodeValues(0xF8E9)).toEqual(0x00A9);
+      expect(mapSpecialUnicodeValues(0xFFFF)).toEqual(0);
+    });
+  });
+  describe('getUnicodeForGlyph', function () {
+    var standardMap, dingbatsMap;
+    beforeAll(function (done) {
+      standardMap = getGlyphsUnicode();
+      dingbatsMap = getDingbatsGlyphsUnicode();
+      done();
+    });
+    afterAll(function () {
+      standardMap = dingbatsMap = null;
+    });
+    it('should get Unicode values for valid glyph names', function () {
+      expect(getUnicodeForGlyph('A', standardMap)).toEqual(0x0041);
+      expect(getUnicodeForGlyph('a1', dingbatsMap)).toEqual(0x2701);
+    });
+    it('should recover Unicode values from uniXXXX/uXXXX{XX} glyph names', function () {
+      expect(getUnicodeForGlyph('uni0041', standardMap)).toEqual(0x0041);
+      expect(getUnicodeForGlyph('u0041', standardMap)).toEqual(0x0041);
+      expect(getUnicodeForGlyph('uni2701', dingbatsMap)).toEqual(0x2701);
+      expect(getUnicodeForGlyph('u2701', dingbatsMap)).toEqual(0x2701);
+    });
+    it('should not get Unicode values for invalid glyph names', function () {
+      expect(getUnicodeForGlyph('Qwerty', standardMap)).toEqual(-1);
+      expect(getUnicodeForGlyph('Qwerty', dingbatsMap)).toEqual(-1);
+    });
+  });
+  describe('getUnicodeRangeFor', function () {
+    it('should get correct Unicode range', function () {
+      expect(getUnicodeRangeFor(0x0041)).toEqual(0);
+      expect(getUnicodeRangeFor(0xFB01)).toEqual(62);
+    });
+    it('should not get a Unicode range', function () {
+      expect(getUnicodeRangeFor(0x05FF)).toEqual(-1);
+    });
+  });
+  describe('getNormalizedUnicodes', function () {
+    var NormalizedUnicodes;
+    beforeAll(function (done) {
+      NormalizedUnicodes = getNormalizedUnicodes();
+      done();
+    });
+    afterAll(function () {
+      NormalizedUnicodes = null;
+    });
+    it('should get normalized Unicode values for ligatures', function () {
+      expect(NormalizedUnicodes['\uFB01']).toEqual('fi');
+      expect(NormalizedUnicodes['\u0675']).toEqual('\u0627\u0674');
+    });
+    it('should not normalize standard characters', function () {
+      expect(NormalizedUnicodes['A']).toEqual(undefined);
+    });
+  });
+  describe('reverseIfRtl', function () {
+    var NormalizedUnicodes;
+    function getGlyphUnicode(char) {
+      if (NormalizedUnicodes[char] !== undefined) {
+        return NormalizedUnicodes[char];
+      }
+      return char;
+    }
+    beforeAll(function (done) {
+      NormalizedUnicodes = getNormalizedUnicodes();
+      done();
+    });
+    afterAll(function () {
+      NormalizedUnicodes = null;
+    });
+    it('should not reverse LTR characters', function () {
+      var A = getGlyphUnicode('A');
+      expect(reverseIfRtl(A)).toEqual('A');
+      var fi = getGlyphUnicode('\uFB01');
+      expect(reverseIfRtl(fi)).toEqual('fi');
+    });
+    it('should reverse RTL characters', function () {
+      var heAlef = getGlyphUnicode('\u05D0');
+      expect(reverseIfRtl(heAlef)).toEqual('\u05D0');
+      var arAlef = getGlyphUnicode('\u0675');
+      expect(reverseIfRtl(arAlef)).toEqual('\u0674\u0627');
+    });
   });
-  it('should re-map special Unicode values', function () {
-   expect(mapSpecialUnicodeValues(0xF8E9)).toEqual(0x00A9);
-   expect(mapSpecialUnicodeValues(0xFFFF)).toEqual(0);
-  });
- });
- describe('getUnicodeForGlyph', function () {
-  var standardMap, dingbatsMap;
-  beforeAll(function (done) {
-   standardMap = getGlyphsUnicode();
-   dingbatsMap = getDingbatsGlyphsUnicode();
-   done();
-  });
-  afterAll(function () {
-   standardMap = dingbatsMap = null;
-  });
-  it('should get Unicode values for valid glyph names', function () {
-   expect(getUnicodeForGlyph('A', standardMap)).toEqual(0x0041);
-   expect(getUnicodeForGlyph('a1', dingbatsMap)).toEqual(0x2701);
-  });
-  it('should recover Unicode values from uniXXXX/uXXXX{XX} glyph names', function () {
-   expect(getUnicodeForGlyph('uni0041', standardMap)).toEqual(0x0041);
-   expect(getUnicodeForGlyph('u0041', standardMap)).toEqual(0x0041);
-   expect(getUnicodeForGlyph('uni2701', dingbatsMap)).toEqual(0x2701);
-   expect(getUnicodeForGlyph('u2701', dingbatsMap)).toEqual(0x2701);
-  });
-  it('should not get Unicode values for invalid glyph names', function () {
-   expect(getUnicodeForGlyph('Qwerty', standardMap)).toEqual(-1);
-   expect(getUnicodeForGlyph('Qwerty', dingbatsMap)).toEqual(-1);
-  });
- });
- describe('getUnicodeRangeFor', function () {
-  it('should get correct Unicode range', function () {
-   expect(getUnicodeRangeFor(0x0041)).toEqual(0);
-   expect(getUnicodeRangeFor(0xFB01)).toEqual(62);
-  });
-  it('should not get a Unicode range', function () {
-   expect(getUnicodeRangeFor(0x05FF)).toEqual(-1);
-  });
- });
- describe('getNormalizedUnicodes', function () {
-  var NormalizedUnicodes;
-  beforeAll(function (done) {
-   NormalizedUnicodes = getNormalizedUnicodes();
-   done();
-  });
-  afterAll(function () {
-   NormalizedUnicodes = null;
-  });
-  it('should get normalized Unicode values for ligatures', function () {
-   expect(NormalizedUnicodes['\uFB01']).toEqual('fi');
-   expect(NormalizedUnicodes['\u0675']).toEqual('\u0627\u0674');
-  });
-  it('should not normalize standard characters', function () {
-   expect(NormalizedUnicodes['A']).toEqual(undefined);
-  });
- });
- describe('reverseIfRtl', function () {
-  var NormalizedUnicodes;
-  function getGlyphUnicode(char) {
-   if (NormalizedUnicodes[char] !== undefined) {
-    return NormalizedUnicodes[char];
-   }
-   return char;
-  }
-  beforeAll(function (done) {
-   NormalizedUnicodes = getNormalizedUnicodes();
-   done();
-  });
-  afterAll(function () {
-   NormalizedUnicodes = null;
-  });
-  it('should not reverse LTR characters', function () {
-   var A = getGlyphUnicode('A');
-   expect(reverseIfRtl(A)).toEqual('A');
-   var fi = getGlyphUnicode('\uFB01');
-   expect(reverseIfRtl(fi)).toEqual('fi');
-  });
-  it('should reverse RTL characters', function () {
-   var heAlef = getGlyphUnicode('\u05D0');
-   expect(reverseIfRtl(heAlef)).toEqual('\u05D0');
-   var arAlef = getGlyphUnicode('\u0675');
-   expect(reverseIfRtl(arAlef)).toEqual('\u0674\u0627');
-  });
- });
 });

+ 25 - 24
lib/test/unit/util_spec.js

@@ -13,34 +13,35 @@
  * limitations under the License.
  */
 'use strict';
+
 var sharedUtil = require('../../shared/util.js');
 var stringToPDFString = sharedUtil.stringToPDFString;
 var removeNullCharacters = sharedUtil.removeNullCharacters;
 describe('util', function () {
- describe('stringToPDFString', function () {
-  it('handles ISO Latin 1 strings', function () {
-   var str = '\x8Dstring\x8E';
-   expect(stringToPDFString(str)).toEqual('\u201Cstring\u201D');
+  describe('stringToPDFString', function () {
+    it('handles ISO Latin 1 strings', function () {
+      var str = '\x8Dstring\x8E';
+      expect(stringToPDFString(str)).toEqual('\u201Cstring\u201D');
+    });
+    it('handles UTF-16BE strings', function () {
+      var str = '\xFE\xFF\x00\x73\x00\x74\x00\x72\x00\x69\x00\x6E\x00\x67';
+      expect(stringToPDFString(str)).toEqual('string');
+    });
+    it('handles empty strings', function () {
+      var str1 = '';
+      expect(stringToPDFString(str1)).toEqual('');
+      var str2 = '\xFE\xFF';
+      expect(stringToPDFString(str2)).toEqual('');
+    });
   });
-  it('handles UTF-16BE strings', function () {
-   var str = '\xFE\xFF\x00\x73\x00\x74\x00\x72\x00\x69\x00\x6E\x00\x67';
-   expect(stringToPDFString(str)).toEqual('string');
+  describe('removeNullCharacters', function () {
+    it('should not modify string without null characters', function () {
+      var str = 'string without null chars';
+      expect(removeNullCharacters(str)).toEqual('string without null chars');
+    });
+    it('should modify string with null characters', function () {
+      var str = 'string\x00With\x00Null\x00Chars';
+      expect(removeNullCharacters(str)).toEqual('stringWithNullChars');
+    });
   });
-  it('handles empty strings', function () {
-   var str1 = '';
-   expect(stringToPDFString(str1)).toEqual('');
-   var str2 = '\xFE\xFF';
-   expect(stringToPDFString(str2)).toEqual('');
-  });
- });
- describe('removeNullCharacters', function () {
-  it('should not modify string without null characters', function () {
-   var str = 'string without null chars';
-   expect(removeNullCharacters(str)).toEqual('string without null chars');
-  });
-  it('should modify string with null characters', function () {
-   var str = 'string\x00With\x00Null\x00Chars';
-   expect(removeNullCharacters(str)).toEqual('stringWithNullChars');
-  });
- });
 });

+ 57 - 57
lib/web/annotation_layer_builder.js

@@ -13,72 +13,72 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtils = require('./ui_utils.js');
 var pdfLinkService = require('./pdf_link_service.js');
 var pdfjsLib = require('./pdfjs.js');
 var mozL10n = uiUtils.mozL10n;
 var SimpleLinkService = pdfLinkService.SimpleLinkService;
 var AnnotationLayerBuilder = function AnnotationLayerBuilderClosure() {
- function AnnotationLayerBuilder(options) {
-  this.pageDiv = options.pageDiv;
-  this.pdfPage = options.pdfPage;
-  this.renderInteractiveForms = options.renderInteractiveForms;
-  this.linkService = options.linkService;
-  this.downloadManager = options.downloadManager;
-  this.div = null;
- }
- AnnotationLayerBuilder.prototype = {
-  render: function AnnotationLayerBuilder_render(viewport, intent) {
-   var self = this;
-   var parameters = { intent: intent === undefined ? 'display' : intent };
-   this.pdfPage.getAnnotations(parameters).then(function (annotations) {
-    viewport = viewport.clone({ dontFlip: true });
-    parameters = {
-     viewport: viewport,
-     div: self.div,
-     annotations: annotations,
-     page: self.pdfPage,
-     renderInteractiveForms: self.renderInteractiveForms,
-     linkService: self.linkService,
-     downloadManager: self.downloadManager
-    };
-    if (self.div) {
-     pdfjsLib.AnnotationLayer.update(parameters);
-    } else {
-     if (annotations.length === 0) {
-      return;
-     }
-     self.div = document.createElement('div');
-     self.div.className = 'annotationLayer';
-     self.pageDiv.appendChild(self.div);
-     parameters.div = self.div;
-     pdfjsLib.AnnotationLayer.render(parameters);
-     if (typeof mozL10n !== 'undefined') {
-      mozL10n.translate(self.div);
-     }
-    }
-   });
-  },
-  hide: function AnnotationLayerBuilder_hide() {
-   if (!this.div) {
-    return;
-   }
-   this.div.setAttribute('hidden', 'true');
+  function AnnotationLayerBuilder(options) {
+    this.pageDiv = options.pageDiv;
+    this.pdfPage = options.pdfPage;
+    this.renderInteractiveForms = options.renderInteractiveForms;
+    this.linkService = options.linkService;
+    this.downloadManager = options.downloadManager;
+    this.div = null;
   }
- };
- return AnnotationLayerBuilder;
+  AnnotationLayerBuilder.prototype = {
+    render: function AnnotationLayerBuilder_render(viewport, intent) {
+      var self = this;
+      var parameters = { intent: intent === undefined ? 'display' : intent };
+      this.pdfPage.getAnnotations(parameters).then(function (annotations) {
+        viewport = viewport.clone({ dontFlip: true });
+        parameters = {
+          viewport: viewport,
+          div: self.div,
+          annotations: annotations,
+          page: self.pdfPage,
+          renderInteractiveForms: self.renderInteractiveForms,
+          linkService: self.linkService,
+          downloadManager: self.downloadManager
+        };
+        if (self.div) {
+          pdfjsLib.AnnotationLayer.update(parameters);
+        } else {
+          if (annotations.length === 0) {
+            return;
+          }
+          self.div = document.createElement('div');
+          self.div.className = 'annotationLayer';
+          self.pageDiv.appendChild(self.div);
+          parameters.div = self.div;
+          pdfjsLib.AnnotationLayer.render(parameters);
+          if (typeof mozL10n !== 'undefined') {
+            mozL10n.translate(self.div);
+          }
+        }
+      });
+    },
+    hide: function AnnotationLayerBuilder_hide() {
+      if (!this.div) {
+        return;
+      }
+      this.div.setAttribute('hidden', 'true');
+    }
+  };
+  return AnnotationLayerBuilder;
 }();
-function DefaultAnnotationLayerFactory() {
-}
+function DefaultAnnotationLayerFactory() {}
 DefaultAnnotationLayerFactory.prototype = {
- createAnnotationLayerBuilder: function (pageDiv, pdfPage, renderInteractiveForms) {
-  return new AnnotationLayerBuilder({
-   pageDiv: pageDiv,
-   pdfPage: pdfPage,
-   renderInteractiveForms: renderInteractiveForms,
-   linkService: new SimpleLinkService()
-  });
- }
+  createAnnotationLayerBuilder: function (pageDiv, pdfPage, renderInteractiveForms) {
+    return new AnnotationLayerBuilder({
+      pageDiv: pageDiv,
+      pdfPage: pdfPage,
+      renderInteractiveForms: renderInteractiveForms,
+      linkService: new SimpleLinkService()
+    });
+  }
 };
 exports.AnnotationLayerBuilder = AnnotationLayerBuilder;
 exports.DefaultAnnotationLayerFactory = DefaultAnnotationLayerFactory;

+ 1356 - 1381
lib/web/app.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtilsLib = require('./ui_utils.js');
 var downloadManagerLib = require('./download_manager.js');
 var pdfHistoryLib = require('./pdf_history.js');
@@ -75,1514 +76,1488 @@ var RendererType = uiUtilsLib.RendererType;
 var DEFAULT_SCALE_DELTA = 1.1;
 var DISABLE_AUTO_FETCH_LOADING_BAR_TIMEOUT = 5000;
 function configure(PDFJS) {
- PDFJS.imageResourcesPath = './images/';
- PDFJS.workerSrc = '../build/pdf.worker.js';
- PDFJS.cMapUrl = '../web/cmaps/';
- PDFJS.cMapPacked = true;
+  PDFJS.imageResourcesPath = './images/';
+  PDFJS.workerSrc = '../build/pdf.worker.js';
+  PDFJS.cMapUrl = '../web/cmaps/';
+  PDFJS.cMapPacked = true;
 }
 var DefaultExernalServices = {
- updateFindControlState: function (data) {
- },
- initPassiveLoading: function (callbacks) {
- },
- fallback: function (data, callback) {
- },
- reportTelemetry: function (data) {
- },
- createDownloadManager: function () {
-  return new downloadManagerLib.DownloadManager();
- },
- supportsIntegratedFind: false,
- supportsDocumentFonts: true,
- supportsDocumentColors: true,
- supportedMouseWheelZoomModifierKeys: {
-  ctrlKey: true,
-  metaKey: true
- }
+  updateFindControlState: function (data) {},
+  initPassiveLoading: function (callbacks) {},
+  fallback: function (data, callback) {},
+  reportTelemetry: function (data) {},
+  createDownloadManager: function () {
+    return new downloadManagerLib.DownloadManager();
+  },
+  supportsIntegratedFind: false,
+  supportsDocumentFonts: true,
+  supportsDocumentColors: true,
+  supportedMouseWheelZoomModifierKeys: {
+    ctrlKey: true,
+    metaKey: true
+  }
 };
 var PDFViewerApplication = {
- initialBookmark: document.location.hash.substring(1),
- initialDestination: null,
- initialized: false,
- fellback: false,
- appConfig: null,
- pdfDocument: null,
- pdfLoadingTask: null,
- printService: null,
- pdfViewer: null,
- pdfThumbnailViewer: null,
- pdfRenderingQueue: null,
- pdfPresentationMode: null,
- pdfDocumentProperties: null,
- pdfLinkService: null,
- pdfHistory: null,
- pdfSidebar: null,
- pdfOutlineViewer: null,
- pdfAttachmentViewer: null,
- store: null,
- downloadManager: null,
- toolbar: null,
- secondaryToolbar: null,
- eventBus: null,
- pageRotation: 0,
- isInitialViewSet: false,
- viewerPrefs: {
-  sidebarViewOnLoad: SidebarView.NONE,
-  pdfBugEnabled: false,
-  showPreviousViewOnLoad: true,
-  defaultZoomValue: '',
-  disablePageLabels: false,
-  renderer: 'canvas',
-  enhanceTextSelection: false,
-  renderInteractiveForms: false,
-  enablePrintAutoRotate: false
- },
- isViewerEmbedded: window.parent !== window,
- url: '',
- baseUrl: '',
- externalServices: DefaultExernalServices,
- initialize: function pdfViewInitialize(appConfig) {
-  var self = this;
-  var PDFJS = pdfjsLib.PDFJS;
-  Preferences.initialize();
-  this.preferences = Preferences;
-  configure(PDFJS);
-  this.appConfig = appConfig;
-  return this._readPreferences().then(function () {
-   return self._initializeViewerComponents();
-  }).then(function () {
-   self.bindEvents();
-   self.bindWindowEvents();
-   localized.then(function () {
-    self.eventBus.dispatch('localized');
-   });
-   if (self.isViewerEmbedded && !PDFJS.isExternalLinkTargetSet()) {
-    PDFJS.externalLinkTarget = PDFJS.LinkTarget.TOP;
-   }
-   self.initialized = true;
-  });
- },
- _readPreferences: function () {
-  var self = this;
-  var PDFJS = pdfjsLib.PDFJS;
-  return Promise.all([
-   Preferences.get('enableWebGL').then(function resolved(value) {
-    PDFJS.disableWebGL = !value;
-   }),
-   Preferences.get('sidebarViewOnLoad').then(function resolved(value) {
-    self.viewerPrefs['sidebarViewOnLoad'] = value;
-   }),
-   Preferences.get('pdfBugEnabled').then(function resolved(value) {
-    self.viewerPrefs['pdfBugEnabled'] = value;
-   }),
-   Preferences.get('showPreviousViewOnLoad').then(function resolved(value) {
-    self.viewerPrefs['showPreviousViewOnLoad'] = value;
-   }),
-   Preferences.get('defaultZoomValue').then(function resolved(value) {
-    self.viewerPrefs['defaultZoomValue'] = value;
-   }),
-   Preferences.get('enhanceTextSelection').then(function resolved(value) {
-    self.viewerPrefs['enhanceTextSelection'] = value;
-   }),
-   Preferences.get('disableTextLayer').then(function resolved(value) {
-    if (PDFJS.disableTextLayer === true) {
-     return;
+  initialBookmark: document.location.hash.substring(1),
+  initialDestination: null,
+  initialized: false,
+  fellback: false,
+  appConfig: null,
+  pdfDocument: null,
+  pdfLoadingTask: null,
+  printService: null,
+  pdfViewer: null,
+  pdfThumbnailViewer: null,
+  pdfRenderingQueue: null,
+  pdfPresentationMode: null,
+  pdfDocumentProperties: null,
+  pdfLinkService: null,
+  pdfHistory: null,
+  pdfSidebar: null,
+  pdfOutlineViewer: null,
+  pdfAttachmentViewer: null,
+  store: null,
+  downloadManager: null,
+  toolbar: null,
+  secondaryToolbar: null,
+  eventBus: null,
+  pageRotation: 0,
+  isInitialViewSet: false,
+  viewerPrefs: {
+    sidebarViewOnLoad: SidebarView.NONE,
+    pdfBugEnabled: false,
+    showPreviousViewOnLoad: true,
+    defaultZoomValue: '',
+    disablePageLabels: false,
+    renderer: 'canvas',
+    enhanceTextSelection: false,
+    renderInteractiveForms: false,
+    enablePrintAutoRotate: false
+  },
+  isViewerEmbedded: window.parent !== window,
+  url: '',
+  baseUrl: '',
+  externalServices: DefaultExernalServices,
+  initialize: function pdfViewInitialize(appConfig) {
+    var self = this;
+    var PDFJS = pdfjsLib.PDFJS;
+    Preferences.initialize();
+    this.preferences = Preferences;
+    configure(PDFJS);
+    this.appConfig = appConfig;
+    return this._readPreferences().then(function () {
+      return self._initializeViewerComponents();
+    }).then(function () {
+      self.bindEvents();
+      self.bindWindowEvents();
+      localized.then(function () {
+        self.eventBus.dispatch('localized');
+      });
+      if (self.isViewerEmbedded && !PDFJS.isExternalLinkTargetSet()) {
+        PDFJS.externalLinkTarget = PDFJS.LinkTarget.TOP;
+      }
+      self.initialized = true;
+    });
+  },
+  _readPreferences: function () {
+    var self = this;
+    var PDFJS = pdfjsLib.PDFJS;
+    return Promise.all([Preferences.get('enableWebGL').then(function resolved(value) {
+      PDFJS.disableWebGL = !value;
+    }), Preferences.get('sidebarViewOnLoad').then(function resolved(value) {
+      self.viewerPrefs['sidebarViewOnLoad'] = value;
+    }), Preferences.get('pdfBugEnabled').then(function resolved(value) {
+      self.viewerPrefs['pdfBugEnabled'] = value;
+    }), Preferences.get('showPreviousViewOnLoad').then(function resolved(value) {
+      self.viewerPrefs['showPreviousViewOnLoad'] = value;
+    }), Preferences.get('defaultZoomValue').then(function resolved(value) {
+      self.viewerPrefs['defaultZoomValue'] = value;
+    }), Preferences.get('enhanceTextSelection').then(function resolved(value) {
+      self.viewerPrefs['enhanceTextSelection'] = value;
+    }), Preferences.get('disableTextLayer').then(function resolved(value) {
+      if (PDFJS.disableTextLayer === true) {
+        return;
+      }
+      PDFJS.disableTextLayer = value;
+    }), Preferences.get('disableRange').then(function resolved(value) {
+      if (PDFJS.disableRange === true) {
+        return;
+      }
+      PDFJS.disableRange = value;
+    }), Preferences.get('disableStream').then(function resolved(value) {
+      if (PDFJS.disableStream === true) {
+        return;
+      }
+      PDFJS.disableStream = value;
+    }), Preferences.get('disableAutoFetch').then(function resolved(value) {
+      PDFJS.disableAutoFetch = value;
+    }), Preferences.get('disableFontFace').then(function resolved(value) {
+      if (PDFJS.disableFontFace === true) {
+        return;
+      }
+      PDFJS.disableFontFace = value;
+    }), Preferences.get('useOnlyCssZoom').then(function resolved(value) {
+      PDFJS.useOnlyCssZoom = value;
+    }), Preferences.get('externalLinkTarget').then(function resolved(value) {
+      if (PDFJS.isExternalLinkTargetSet()) {
+        return;
+      }
+      PDFJS.externalLinkTarget = value;
+    }), Preferences.get('renderer').then(function resolved(value) {
+      self.viewerPrefs['renderer'] = value;
+    }), Preferences.get('renderInteractiveForms').then(function resolved(value) {
+      self.viewerPrefs['renderInteractiveForms'] = value;
+    }), Preferences.get('disablePageLabels').then(function resolved(value) {
+      self.viewerPrefs['disablePageLabels'] = value;
+    }), Preferences.get('enablePrintAutoRotate').then(function resolved(value) {
+      self.viewerPrefs['enablePrintAutoRotate'] = value;
+    })]).catch(function (reason) {});
+  },
+  _initializeViewerComponents: function () {
+    var self = this;
+    var appConfig = this.appConfig;
+    return new Promise(function (resolve, reject) {
+      var eventBus = appConfig.eventBus || getGlobalEventBus();
+      self.eventBus = eventBus;
+      var pdfRenderingQueue = new PDFRenderingQueue();
+      pdfRenderingQueue.onIdle = self.cleanup.bind(self);
+      self.pdfRenderingQueue = pdfRenderingQueue;
+      var pdfLinkService = new PDFLinkService({ eventBus: eventBus });
+      self.pdfLinkService = pdfLinkService;
+      var downloadManager = self.externalServices.createDownloadManager();
+      self.downloadManager = downloadManager;
+      var container = appConfig.mainContainer;
+      var viewer = appConfig.viewerContainer;
+      self.pdfViewer = new PDFViewer({
+        container: container,
+        viewer: viewer,
+        eventBus: eventBus,
+        renderingQueue: pdfRenderingQueue,
+        linkService: pdfLinkService,
+        downloadManager: downloadManager,
+        renderer: self.viewerPrefs['renderer'],
+        enhanceTextSelection: self.viewerPrefs['enhanceTextSelection'],
+        renderInteractiveForms: self.viewerPrefs['renderInteractiveForms'],
+        enablePrintAutoRotate: self.viewerPrefs['enablePrintAutoRotate']
+      });
+      pdfRenderingQueue.setViewer(self.pdfViewer);
+      pdfLinkService.setViewer(self.pdfViewer);
+      var thumbnailContainer = appConfig.sidebar.thumbnailView;
+      self.pdfThumbnailViewer = new PDFThumbnailViewer({
+        container: thumbnailContainer,
+        renderingQueue: pdfRenderingQueue,
+        linkService: pdfLinkService
+      });
+      pdfRenderingQueue.setThumbnailViewer(self.pdfThumbnailViewer);
+      self.pdfHistory = new PDFHistory({
+        linkService: pdfLinkService,
+        eventBus: eventBus
+      });
+      pdfLinkService.setHistory(self.pdfHistory);
+      self.findController = new PDFFindController({ pdfViewer: self.pdfViewer });
+      self.findController.onUpdateResultsCount = function (matchCount) {
+        if (self.supportsIntegratedFind) {
+          return;
+        }
+        self.findBar.updateResultsCount(matchCount);
+      };
+      self.findController.onUpdateState = function (state, previous, matchCount) {
+        if (self.supportsIntegratedFind) {
+          self.externalServices.updateFindControlState({
+            result: state,
+            findPrevious: previous
+          });
+        } else {
+          self.findBar.updateUIState(state, previous, matchCount);
+        }
+      };
+      self.pdfViewer.setFindController(self.findController);
+      var findBarConfig = Object.create(appConfig.findBar);
+      findBarConfig.findController = self.findController;
+      findBarConfig.eventBus = eventBus;
+      self.findBar = new PDFFindBar(findBarConfig);
+      self.overlayManager = OverlayManager;
+      self.handTool = new HandTool({
+        container: container,
+        eventBus: eventBus
+      });
+      self.pdfDocumentProperties = new PDFDocumentProperties(appConfig.documentProperties);
+      self.toolbar = new Toolbar(appConfig.toolbar, container, eventBus);
+      self.secondaryToolbar = new SecondaryToolbar(appConfig.secondaryToolbar, container, eventBus);
+      if (self.supportsFullscreen) {
+        self.pdfPresentationMode = new PDFPresentationMode({
+          container: container,
+          viewer: viewer,
+          pdfViewer: self.pdfViewer,
+          eventBus: eventBus,
+          contextMenuItems: appConfig.fullscreen
+        });
+      }
+      self.passwordPrompt = new PasswordPrompt(appConfig.passwordOverlay);
+      self.pdfOutlineViewer = new PDFOutlineViewer({
+        container: appConfig.sidebar.outlineView,
+        eventBus: eventBus,
+        linkService: pdfLinkService
+      });
+      self.pdfAttachmentViewer = new PDFAttachmentViewer({
+        container: appConfig.sidebar.attachmentsView,
+        eventBus: eventBus,
+        downloadManager: downloadManager
+      });
+      var sidebarConfig = Object.create(appConfig.sidebar);
+      sidebarConfig.pdfViewer = self.pdfViewer;
+      sidebarConfig.pdfThumbnailViewer = self.pdfThumbnailViewer;
+      sidebarConfig.pdfOutlineViewer = self.pdfOutlineViewer;
+      sidebarConfig.eventBus = eventBus;
+      self.pdfSidebar = new PDFSidebar(sidebarConfig);
+      self.pdfSidebar.onToggled = self.forceRendering.bind(self);
+      resolve(undefined);
+    });
+  },
+  run: function pdfViewRun(config) {
+    this.initialize(config).then(webViewerInitialized);
+  },
+  zoomIn: function pdfViewZoomIn(ticks) {
+    var newScale = this.pdfViewer.currentScale;
+    do {
+      newScale = (newScale * DEFAULT_SCALE_DELTA).toFixed(2);
+      newScale = Math.ceil(newScale * 10) / 10;
+      newScale = Math.min(MAX_SCALE, newScale);
+    } while (--ticks > 0 && newScale < MAX_SCALE);
+    this.pdfViewer.currentScaleValue = newScale;
+  },
+  zoomOut: function pdfViewZoomOut(ticks) {
+    var newScale = this.pdfViewer.currentScale;
+    do {
+      newScale = (newScale / DEFAULT_SCALE_DELTA).toFixed(2);
+      newScale = Math.floor(newScale * 10) / 10;
+      newScale = Math.max(MIN_SCALE, newScale);
+    } while (--ticks > 0 && newScale > MIN_SCALE);
+    this.pdfViewer.currentScaleValue = newScale;
+  },
+  get pagesCount() {
+    return this.pdfDocument ? this.pdfDocument.numPages : 0;
+  },
+  set page(val) {
+    this.pdfViewer.currentPageNumber = val;
+  },
+  get page() {
+    return this.pdfViewer.currentPageNumber;
+  },
+  get printing() {
+    return !!this.printService;
+  },
+  get supportsPrinting() {
+    return PDFPrintServiceFactory.instance.supportsPrinting;
+  },
+  get supportsFullscreen() {
+    var support;
+    var doc = document.documentElement;
+    support = !!(doc.requestFullscreen || doc.mozRequestFullScreen || doc.webkitRequestFullScreen || doc.msRequestFullscreen);
+    if (document.fullscreenEnabled === false || document.mozFullScreenEnabled === false || document.webkitFullscreenEnabled === false || document.msFullscreenEnabled === false) {
+      support = false;
     }
-    PDFJS.disableTextLayer = value;
-   }),
-   Preferences.get('disableRange').then(function resolved(value) {
-    if (PDFJS.disableRange === true) {
-     return;
+    if (support && pdfjsLib.PDFJS.disableFullscreen === true) {
+      support = false;
     }
-    PDFJS.disableRange = value;
-   }),
-   Preferences.get('disableStream').then(function resolved(value) {
-    if (PDFJS.disableStream === true) {
-     return;
+    return pdfjsLib.shadow(this, 'supportsFullscreen', support);
+  },
+  get supportsIntegratedFind() {
+    return this.externalServices.supportsIntegratedFind;
+  },
+  get supportsDocumentFonts() {
+    return this.externalServices.supportsDocumentFonts;
+  },
+  get supportsDocumentColors() {
+    return this.externalServices.supportsDocumentColors;
+  },
+  get loadingBar() {
+    var bar = new ProgressBar('#loadingBar', {});
+    return pdfjsLib.shadow(this, 'loadingBar', bar);
+  },
+  get supportedMouseWheelZoomModifierKeys() {
+    return this.externalServices.supportedMouseWheelZoomModifierKeys;
+  },
+  initPassiveLoading: function pdfViewInitPassiveLoading() {
+    throw new Error('Not implemented: initPassiveLoading');
+  },
+  setTitleUsingUrl: function pdfViewSetTitleUsingUrl(url) {
+    this.url = url;
+    this.baseUrl = url.split('#')[0];
+    var title = getPDFFileNameFromURL(url, '');
+    if (!title) {
+      try {
+        title = decodeURIComponent(pdfjsLib.getFilenameFromUrl(url)) || url;
+      } catch (e) {
+        title = url;
+      }
     }
-    PDFJS.disableStream = value;
-   }),
-   Preferences.get('disableAutoFetch').then(function resolved(value) {
-    PDFJS.disableAutoFetch = value;
-   }),
-   Preferences.get('disableFontFace').then(function resolved(value) {
-    if (PDFJS.disableFontFace === true) {
-     return;
+    this.setTitle(title);
+  },
+  setTitle: function pdfViewSetTitle(title) {
+    if (this.isViewerEmbedded) {
+      return;
     }
-    PDFJS.disableFontFace = value;
-   }),
-   Preferences.get('useOnlyCssZoom').then(function resolved(value) {
-    PDFJS.useOnlyCssZoom = value;
-   }),
-   Preferences.get('externalLinkTarget').then(function resolved(value) {
-    if (PDFJS.isExternalLinkTargetSet()) {
-     return;
+    document.title = title;
+  },
+  close: function pdfViewClose() {
+    var errorWrapper = this.appConfig.errorWrapper.container;
+    errorWrapper.setAttribute('hidden', 'true');
+    if (!this.pdfLoadingTask) {
+      return Promise.resolve();
     }
-    PDFJS.externalLinkTarget = value;
-   }),
-   Preferences.get('renderer').then(function resolved(value) {
-    self.viewerPrefs['renderer'] = value;
-   }),
-   Preferences.get('renderInteractiveForms').then(function resolved(value) {
-    self.viewerPrefs['renderInteractiveForms'] = value;
-   }),
-   Preferences.get('disablePageLabels').then(function resolved(value) {
-    self.viewerPrefs['disablePageLabels'] = value;
-   }),
-   Preferences.get('enablePrintAutoRotate').then(function resolved(value) {
-    self.viewerPrefs['enablePrintAutoRotate'] = value;
-   })
-  ]).catch(function (reason) {
-  });
- },
- _initializeViewerComponents: function () {
-  var self = this;
-  var appConfig = this.appConfig;
-  return new Promise(function (resolve, reject) {
-   var eventBus = appConfig.eventBus || getGlobalEventBus();
-   self.eventBus = eventBus;
-   var pdfRenderingQueue = new PDFRenderingQueue();
-   pdfRenderingQueue.onIdle = self.cleanup.bind(self);
-   self.pdfRenderingQueue = pdfRenderingQueue;
-   var pdfLinkService = new PDFLinkService({ eventBus: eventBus });
-   self.pdfLinkService = pdfLinkService;
-   var downloadManager = self.externalServices.createDownloadManager();
-   self.downloadManager = downloadManager;
-   var container = appConfig.mainContainer;
-   var viewer = appConfig.viewerContainer;
-   self.pdfViewer = new PDFViewer({
-    container: container,
-    viewer: viewer,
-    eventBus: eventBus,
-    renderingQueue: pdfRenderingQueue,
-    linkService: pdfLinkService,
-    downloadManager: downloadManager,
-    renderer: self.viewerPrefs['renderer'],
-    enhanceTextSelection: self.viewerPrefs['enhanceTextSelection'],
-    renderInteractiveForms: self.viewerPrefs['renderInteractiveForms'],
-    enablePrintAutoRotate: self.viewerPrefs['enablePrintAutoRotate']
-   });
-   pdfRenderingQueue.setViewer(self.pdfViewer);
-   pdfLinkService.setViewer(self.pdfViewer);
-   var thumbnailContainer = appConfig.sidebar.thumbnailView;
-   self.pdfThumbnailViewer = new PDFThumbnailViewer({
-    container: thumbnailContainer,
-    renderingQueue: pdfRenderingQueue,
-    linkService: pdfLinkService
-   });
-   pdfRenderingQueue.setThumbnailViewer(self.pdfThumbnailViewer);
-   self.pdfHistory = new PDFHistory({
-    linkService: pdfLinkService,
-    eventBus: eventBus
-   });
-   pdfLinkService.setHistory(self.pdfHistory);
-   self.findController = new PDFFindController({ pdfViewer: self.pdfViewer });
-   self.findController.onUpdateResultsCount = function (matchCount) {
-    if (self.supportsIntegratedFind) {
-     return;
+    var promise = this.pdfLoadingTask.destroy();
+    this.pdfLoadingTask = null;
+    if (this.pdfDocument) {
+      this.pdfDocument = null;
+      this.pdfThumbnailViewer.setDocument(null);
+      this.pdfViewer.setDocument(null);
+      this.pdfLinkService.setDocument(null, null);
     }
-    self.findBar.updateResultsCount(matchCount);
-   };
-   self.findController.onUpdateState = function (state, previous, matchCount) {
-    if (self.supportsIntegratedFind) {
-     self.externalServices.updateFindControlState({
-      result: state,
-      findPrevious: previous
-     });
-    } else {
-     self.findBar.updateUIState(state, previous, matchCount);
+    this.store = null;
+    this.isInitialViewSet = false;
+    this.pdfSidebar.reset();
+    this.pdfOutlineViewer.reset();
+    this.pdfAttachmentViewer.reset();
+    this.findController.reset();
+    this.findBar.reset();
+    this.toolbar.reset();
+    this.secondaryToolbar.reset();
+    if (typeof PDFBug !== 'undefined') {
+      PDFBug.cleanup();
     }
-   };
-   self.pdfViewer.setFindController(self.findController);
-   var findBarConfig = Object.create(appConfig.findBar);
-   findBarConfig.findController = self.findController;
-   findBarConfig.eventBus = eventBus;
-   self.findBar = new PDFFindBar(findBarConfig);
-   self.overlayManager = OverlayManager;
-   self.handTool = new HandTool({
-    container: container,
-    eventBus: eventBus
-   });
-   self.pdfDocumentProperties = new PDFDocumentProperties(appConfig.documentProperties);
-   self.toolbar = new Toolbar(appConfig.toolbar, container, eventBus);
-   self.secondaryToolbar = new SecondaryToolbar(appConfig.secondaryToolbar, container, eventBus);
-   if (self.supportsFullscreen) {
-    self.pdfPresentationMode = new PDFPresentationMode({
-     container: container,
-     viewer: viewer,
-     pdfViewer: self.pdfViewer,
-     eventBus: eventBus,
-     contextMenuItems: appConfig.fullscreen
-    });
-   }
-   self.passwordPrompt = new PasswordPrompt(appConfig.passwordOverlay);
-   self.pdfOutlineViewer = new PDFOutlineViewer({
-    container: appConfig.sidebar.outlineView,
-    eventBus: eventBus,
-    linkService: pdfLinkService
-   });
-   self.pdfAttachmentViewer = new PDFAttachmentViewer({
-    container: appConfig.sidebar.attachmentsView,
-    eventBus: eventBus,
-    downloadManager: downloadManager
-   });
-   var sidebarConfig = Object.create(appConfig.sidebar);
-   sidebarConfig.pdfViewer = self.pdfViewer;
-   sidebarConfig.pdfThumbnailViewer = self.pdfThumbnailViewer;
-   sidebarConfig.pdfOutlineViewer = self.pdfOutlineViewer;
-   sidebarConfig.eventBus = eventBus;
-   self.pdfSidebar = new PDFSidebar(sidebarConfig);
-   self.pdfSidebar.onToggled = self.forceRendering.bind(self);
-   resolve(undefined);
-  });
- },
- run: function pdfViewRun(config) {
-  this.initialize(config).then(webViewerInitialized);
- },
- zoomIn: function pdfViewZoomIn(ticks) {
-  var newScale = this.pdfViewer.currentScale;
-  do {
-   newScale = (newScale * DEFAULT_SCALE_DELTA).toFixed(2);
-   newScale = Math.ceil(newScale * 10) / 10;
-   newScale = Math.min(MAX_SCALE, newScale);
-  } while (--ticks > 0 && newScale < MAX_SCALE);
-  this.pdfViewer.currentScaleValue = newScale;
- },
- zoomOut: function pdfViewZoomOut(ticks) {
-  var newScale = this.pdfViewer.currentScale;
-  do {
-   newScale = (newScale / DEFAULT_SCALE_DELTA).toFixed(2);
-   newScale = Math.floor(newScale * 10) / 10;
-   newScale = Math.max(MIN_SCALE, newScale);
-  } while (--ticks > 0 && newScale > MIN_SCALE);
-  this.pdfViewer.currentScaleValue = newScale;
- },
- get pagesCount() {
-  return this.pdfDocument ? this.pdfDocument.numPages : 0;
- },
- set page(val) {
-  this.pdfViewer.currentPageNumber = val;
- },
- get page() {
-  return this.pdfViewer.currentPageNumber;
- },
- get printing() {
-  return !!this.printService;
- },
- get supportsPrinting() {
-  return PDFPrintServiceFactory.instance.supportsPrinting;
- },
- get supportsFullscreen() {
-  var support;
-  var doc = document.documentElement;
-  support = !!(doc.requestFullscreen || doc.mozRequestFullScreen || doc.webkitRequestFullScreen || doc.msRequestFullscreen);
-  if (document.fullscreenEnabled === false || document.mozFullScreenEnabled === false || document.webkitFullscreenEnabled === false || document.msFullscreenEnabled === false) {
-   support = false;
-  }
-  if (support && pdfjsLib.PDFJS.disableFullscreen === true) {
-   support = false;
-  }
-  return pdfjsLib.shadow(this, 'supportsFullscreen', support);
- },
- get supportsIntegratedFind() {
-  return this.externalServices.supportsIntegratedFind;
- },
- get supportsDocumentFonts() {
-  return this.externalServices.supportsDocumentFonts;
- },
- get supportsDocumentColors() {
-  return this.externalServices.supportsDocumentColors;
- },
- get loadingBar() {
-  var bar = new ProgressBar('#loadingBar', {});
-  return pdfjsLib.shadow(this, 'loadingBar', bar);
- },
- get supportedMouseWheelZoomModifierKeys() {
-  return this.externalServices.supportedMouseWheelZoomModifierKeys;
- },
- initPassiveLoading: function pdfViewInitPassiveLoading() {
-  throw new Error('Not implemented: initPassiveLoading');
- },
- setTitleUsingUrl: function pdfViewSetTitleUsingUrl(url) {
-  this.url = url;
-  this.baseUrl = url.split('#')[0];
-  var title = getPDFFileNameFromURL(url, '');
-  if (!title) {
-   try {
-    title = decodeURIComponent(pdfjsLib.getFilenameFromUrl(url)) || url;
-   } catch (e) {
-    title = url;
-   }
-  }
-  this.setTitle(title);
- },
- setTitle: function pdfViewSetTitle(title) {
-  if (this.isViewerEmbedded) {
-   return;
-  }
-  document.title = title;
- },
- close: function pdfViewClose() {
-  var errorWrapper = this.appConfig.errorWrapper.container;
-  errorWrapper.setAttribute('hidden', 'true');
-  if (!this.pdfLoadingTask) {
-   return Promise.resolve();
-  }
-  var promise = this.pdfLoadingTask.destroy();
-  this.pdfLoadingTask = null;
-  if (this.pdfDocument) {
-   this.pdfDocument = null;
-   this.pdfThumbnailViewer.setDocument(null);
-   this.pdfViewer.setDocument(null);
-   this.pdfLinkService.setDocument(null, null);
-  }
-  this.store = null;
-  this.isInitialViewSet = false;
-  this.pdfSidebar.reset();
-  this.pdfOutlineViewer.reset();
-  this.pdfAttachmentViewer.reset();
-  this.findController.reset();
-  this.findBar.reset();
-  this.toolbar.reset();
-  this.secondaryToolbar.reset();
-  if (typeof PDFBug !== 'undefined') {
-   PDFBug.cleanup();
-  }
-  return promise;
- },
- open: function pdfViewOpen(file, args) {
-  if (arguments.length > 2 || typeof args === 'number') {
-   return Promise.reject(new Error('Call of open() with obsolete signature.'));
-  }
-  if (this.pdfLoadingTask) {
-   return this.close().then(function () {
-    Preferences.reload();
-    return this.open(file, args);
-   }.bind(this));
-  }
-  var parameters = Object.create(null), scale;
-  if (typeof file === 'string') {
-   this.setTitleUsingUrl(file);
-   parameters.url = file;
-  } else if (file && 'byteLength' in file) {
-   parameters.data = file;
-  } else if (file.url && file.originalUrl) {
-   this.setTitleUsingUrl(file.originalUrl);
-   parameters.url = file.url;
-  }
-  if (args) {
-   for (var prop in args) {
-    parameters[prop] = args[prop];
-   }
-   if (args.scale) {
-    scale = args.scale;
-   }
-   if (args.length) {
-    this.pdfDocumentProperties.setFileSize(args.length);
-   }
-  }
-  var self = this;
-  self.downloadComplete = false;
-  var loadingTask = pdfjsLib.getDocument(parameters);
-  this.pdfLoadingTask = loadingTask;
-  loadingTask.onPassword = function passwordNeeded(updateCallback, reason) {
-   self.passwordPrompt.setUpdateCallback(updateCallback, reason);
-   self.passwordPrompt.open();
-  };
-  loadingTask.onProgress = function getDocumentProgress(progressData) {
-   self.progress(progressData.loaded / progressData.total);
-  };
-  loadingTask.onUnsupportedFeature = this.fallback.bind(this);
-  return loadingTask.promise.then(function getDocumentCallback(pdfDocument) {
-   self.load(pdfDocument, scale);
-  }, function getDocumentError(exception) {
-   var message = exception && exception.message;
-   var loadingErrorMessage = mozL10n.get('loading_error', null, 'An error occurred while loading the PDF.');
-   if (exception instanceof pdfjsLib.InvalidPDFException) {
-    loadingErrorMessage = mozL10n.get('invalid_file_error', null, 'Invalid or corrupted PDF file.');
-   } else if (exception instanceof pdfjsLib.MissingPDFException) {
-    loadingErrorMessage = mozL10n.get('missing_file_error', null, 'Missing PDF file.');
-   } else if (exception instanceof pdfjsLib.UnexpectedResponseException) {
-    loadingErrorMessage = mozL10n.get('unexpected_response_error', null, 'Unexpected server response.');
-   }
-   var moreInfo = { message: message };
-   self.error(loadingErrorMessage, moreInfo);
-   throw new Error(loadingErrorMessage);
-  });
- },
- download: function pdfViewDownload() {
-  function downloadByUrl() {
-   downloadManager.downloadUrl(url, filename);
-  }
-  var url = this.baseUrl;
-  var filename = getPDFFileNameFromURL(this.url);
-  var downloadManager = this.downloadManager;
-  downloadManager.onerror = function (err) {
-   PDFViewerApplication.error('PDF failed to download.');
-  };
-  if (!this.pdfDocument) {
-   downloadByUrl();
-   return;
-  }
-  if (!this.downloadComplete) {
-   downloadByUrl();
-   return;
-  }
-  this.pdfDocument.getData().then(function getDataSuccess(data) {
-   var blob = pdfjsLib.createBlob(data, 'application/pdf');
-   downloadManager.download(blob, url, filename);
-  }, downloadByUrl).then(null, downloadByUrl);
- },
- fallback: function pdfViewFallback(featureId) {
- },
- error: function pdfViewError(message, moreInfo) {
-  var moreInfoText = mozL10n.get('error_version_info', {
-   version: pdfjsLib.version || '?',
-   build: pdfjsLib.build || '?'
-  }, 'PDF.js v{{version}} (build: {{build}})') + '\n';
-  if (moreInfo) {
-   moreInfoText += mozL10n.get('error_message', { message: moreInfo.message }, 'Message: {{message}}');
-   if (moreInfo.stack) {
-    moreInfoText += '\n' + mozL10n.get('error_stack', { stack: moreInfo.stack }, 'Stack: {{stack}}');
-   } else {
-    if (moreInfo.filename) {
-     moreInfoText += '\n' + mozL10n.get('error_file', { file: moreInfo.filename }, 'File: {{file}}');
+    return promise;
+  },
+  open: function pdfViewOpen(file, args) {
+    if (arguments.length > 2 || typeof args === 'number') {
+      return Promise.reject(new Error('Call of open() with obsolete signature.'));
     }
-    if (moreInfo.lineNumber) {
-     moreInfoText += '\n' + mozL10n.get('error_line', { line: moreInfo.lineNumber }, 'Line: {{line}}');
+    if (this.pdfLoadingTask) {
+      return this.close().then(function () {
+        Preferences.reload();
+        return this.open(file, args);
+      }.bind(this));
     }
-   }
-  }
-  var errorWrapperConfig = this.appConfig.errorWrapper;
-  var errorWrapper = errorWrapperConfig.container;
-  errorWrapper.removeAttribute('hidden');
-  var errorMessage = errorWrapperConfig.errorMessage;
-  errorMessage.textContent = message;
-  var closeButton = errorWrapperConfig.closeButton;
-  closeButton.onclick = function () {
-   errorWrapper.setAttribute('hidden', 'true');
-  };
-  var errorMoreInfo = errorWrapperConfig.errorMoreInfo;
-  var moreInfoButton = errorWrapperConfig.moreInfoButton;
-  var lessInfoButton = errorWrapperConfig.lessInfoButton;
-  moreInfoButton.onclick = function () {
-   errorMoreInfo.removeAttribute('hidden');
-   moreInfoButton.setAttribute('hidden', 'true');
-   lessInfoButton.removeAttribute('hidden');
-   errorMoreInfo.style.height = errorMoreInfo.scrollHeight + 'px';
-  };
-  lessInfoButton.onclick = function () {
-   errorMoreInfo.setAttribute('hidden', 'true');
-   moreInfoButton.removeAttribute('hidden');
-   lessInfoButton.setAttribute('hidden', 'true');
-  };
-  moreInfoButton.oncontextmenu = noContextMenuHandler;
-  lessInfoButton.oncontextmenu = noContextMenuHandler;
-  closeButton.oncontextmenu = noContextMenuHandler;
-  moreInfoButton.removeAttribute('hidden');
-  lessInfoButton.setAttribute('hidden', 'true');
-  errorMoreInfo.value = moreInfoText;
- },
- progress: function pdfViewProgress(level) {
-  var percent = Math.round(level * 100);
-  if (percent > this.loadingBar.percent || isNaN(percent)) {
-   this.loadingBar.percent = percent;
-   if (pdfjsLib.PDFJS.disableAutoFetch && percent) {
-    if (this.disableAutoFetchLoadingBarTimeout) {
-     clearTimeout(this.disableAutoFetchLoadingBarTimeout);
-     this.disableAutoFetchLoadingBarTimeout = null;
+    var parameters = Object.create(null),
+        scale;
+    if (typeof file === 'string') {
+      this.setTitleUsingUrl(file);
+      parameters.url = file;
+    } else if (file && 'byteLength' in file) {
+      parameters.data = file;
+    } else if (file.url && file.originalUrl) {
+      this.setTitleUsingUrl(file.originalUrl);
+      parameters.url = file.url;
     }
-    this.loadingBar.show();
-    this.disableAutoFetchLoadingBarTimeout = setTimeout(function () {
-     this.loadingBar.hide();
-     this.disableAutoFetchLoadingBarTimeout = null;
-    }.bind(this), DISABLE_AUTO_FETCH_LOADING_BAR_TIMEOUT);
-   }
-  }
- },
- load: function pdfViewLoad(pdfDocument, scale) {
-  var self = this;
-  scale = scale || UNKNOWN_SCALE;
-  this.pdfDocument = pdfDocument;
-  this.pdfDocumentProperties.setDocumentAndUrl(pdfDocument, this.url);
-  var downloadedPromise = pdfDocument.getDownloadInfo().then(function () {
-   self.downloadComplete = true;
-   self.loadingBar.hide();
-  });
-  this.toolbar.setPagesCount(pdfDocument.numPages, false);
-  this.secondaryToolbar.setPagesCount(pdfDocument.numPages);
-  var id = this.documentFingerprint = pdfDocument.fingerprint;
-  var store = this.store = new ViewHistory(id);
-  var baseDocumentUrl;
-  baseDocumentUrl = null;
-  this.pdfLinkService.setDocument(pdfDocument, baseDocumentUrl);
-  var pdfViewer = this.pdfViewer;
-  pdfViewer.currentScale = scale;
-  pdfViewer.setDocument(pdfDocument);
-  var firstPagePromise = pdfViewer.firstPagePromise;
-  var pagesPromise = pdfViewer.pagesPromise;
-  var onePageRendered = pdfViewer.onePageRendered;
-  this.pageRotation = 0;
-  var pdfThumbnailViewer = this.pdfThumbnailViewer;
-  pdfThumbnailViewer.setDocument(pdfDocument);
-  firstPagePromise.then(function (pdfPage) {
-   downloadedPromise.then(function () {
-    self.eventBus.dispatch('documentload', { source: self });
-   });
-   self.loadingBar.setWidth(self.appConfig.viewerContainer);
-   if (!pdfjsLib.PDFJS.disableHistory && !self.isViewerEmbedded) {
-    if (!self.viewerPrefs['showPreviousViewOnLoad']) {
-     self.pdfHistory.clearHistoryState();
+    if (args) {
+      for (var prop in args) {
+        parameters[prop] = args[prop];
+      }
+      if (args.scale) {
+        scale = args.scale;
+      }
+      if (args.length) {
+        this.pdfDocumentProperties.setFileSize(args.length);
+      }
     }
-    self.pdfHistory.initialize(self.documentFingerprint);
-    if (self.pdfHistory.initialDestination) {
-     self.initialDestination = self.pdfHistory.initialDestination;
-    } else if (self.pdfHistory.initialBookmark) {
-     self.initialBookmark = self.pdfHistory.initialBookmark;
+    var self = this;
+    self.downloadComplete = false;
+    var loadingTask = pdfjsLib.getDocument(parameters);
+    this.pdfLoadingTask = loadingTask;
+    loadingTask.onPassword = function passwordNeeded(updateCallback, reason) {
+      self.passwordPrompt.setUpdateCallback(updateCallback, reason);
+      self.passwordPrompt.open();
+    };
+    loadingTask.onProgress = function getDocumentProgress(progressData) {
+      self.progress(progressData.loaded / progressData.total);
+    };
+    loadingTask.onUnsupportedFeature = this.fallback.bind(this);
+    return loadingTask.promise.then(function getDocumentCallback(pdfDocument) {
+      self.load(pdfDocument, scale);
+    }, function getDocumentError(exception) {
+      var message = exception && exception.message;
+      var loadingErrorMessage = mozL10n.get('loading_error', null, 'An error occurred while loading the PDF.');
+      if (exception instanceof pdfjsLib.InvalidPDFException) {
+        loadingErrorMessage = mozL10n.get('invalid_file_error', null, 'Invalid or corrupted PDF file.');
+      } else if (exception instanceof pdfjsLib.MissingPDFException) {
+        loadingErrorMessage = mozL10n.get('missing_file_error', null, 'Missing PDF file.');
+      } else if (exception instanceof pdfjsLib.UnexpectedResponseException) {
+        loadingErrorMessage = mozL10n.get('unexpected_response_error', null, 'Unexpected server response.');
+      }
+      var moreInfo = { message: message };
+      self.error(loadingErrorMessage, moreInfo);
+      throw new Error(loadingErrorMessage);
+    });
+  },
+  download: function pdfViewDownload() {
+    function downloadByUrl() {
+      downloadManager.downloadUrl(url, filename);
     }
-   }
-   var initialParams = {
-    destination: self.initialDestination,
-    bookmark: self.initialBookmark,
-    hash: null
-   };
-   store.initializedPromise.then(function resolved() {
-    var storedHash = null, sidebarView = null;
-    if (self.viewerPrefs['showPreviousViewOnLoad'] && store.get('exists', false)) {
-     var pageNum = store.get('page', '1');
-     var zoom = self.viewerPrefs['defaultZoomValue'] || store.get('zoom', DEFAULT_SCALE_VALUE);
-     var left = store.get('scrollLeft', '0');
-     var top = store.get('scrollTop', '0');
-     storedHash = 'page=' + pageNum + '&zoom=' + zoom + ',' + left + ',' + top;
-     sidebarView = store.get('sidebarView', SidebarView.NONE);
-    } else if (self.viewerPrefs['defaultZoomValue']) {
-     storedHash = 'page=1&zoom=' + self.viewerPrefs['defaultZoomValue'];
+    var url = this.baseUrl;
+    var filename = getPDFFileNameFromURL(this.url);
+    var downloadManager = this.downloadManager;
+    downloadManager.onerror = function (err) {
+      PDFViewerApplication.error('PDF failed to download.');
+    };
+    if (!this.pdfDocument) {
+      downloadByUrl();
+      return;
     }
-    self.setInitialView(storedHash, {
-     scale: scale,
-     sidebarView: sidebarView
-    });
-    initialParams.hash = storedHash;
-    if (!self.isViewerEmbedded) {
-     self.pdfViewer.focus();
+    if (!this.downloadComplete) {
+      downloadByUrl();
+      return;
     }
-   }, function rejected(reason) {
-    console.error(reason);
-    self.setInitialView(null, { scale: scale });
-   });
-   pagesPromise.then(function resolved() {
-    if (!initialParams.destination && !initialParams.bookmark && !initialParams.hash) {
-     return;
+    this.pdfDocument.getData().then(function getDataSuccess(data) {
+      var blob = pdfjsLib.createBlob(data, 'application/pdf');
+      downloadManager.download(blob, url, filename);
+    }, downloadByUrl).then(null, downloadByUrl);
+  },
+  fallback: function pdfViewFallback(featureId) {},
+  error: function pdfViewError(message, moreInfo) {
+    var moreInfoText = mozL10n.get('error_version_info', {
+      version: pdfjsLib.version || '?',
+      build: pdfjsLib.build || '?'
+    }, 'PDF.js v{{version}} (build: {{build}})') + '\n';
+    if (moreInfo) {
+      moreInfoText += mozL10n.get('error_message', { message: moreInfo.message }, 'Message: {{message}}');
+      if (moreInfo.stack) {
+        moreInfoText += '\n' + mozL10n.get('error_stack', { stack: moreInfo.stack }, 'Stack: {{stack}}');
+      } else {
+        if (moreInfo.filename) {
+          moreInfoText += '\n' + mozL10n.get('error_file', { file: moreInfo.filename }, 'File: {{file}}');
+        }
+        if (moreInfo.lineNumber) {
+          moreInfoText += '\n' + mozL10n.get('error_line', { line: moreInfo.lineNumber }, 'Line: {{line}}');
+        }
+      }
     }
-    if (self.hasEqualPageSizes) {
-     return;
+    var errorWrapperConfig = this.appConfig.errorWrapper;
+    var errorWrapper = errorWrapperConfig.container;
+    errorWrapper.removeAttribute('hidden');
+    var errorMessage = errorWrapperConfig.errorMessage;
+    errorMessage.textContent = message;
+    var closeButton = errorWrapperConfig.closeButton;
+    closeButton.onclick = function () {
+      errorWrapper.setAttribute('hidden', 'true');
+    };
+    var errorMoreInfo = errorWrapperConfig.errorMoreInfo;
+    var moreInfoButton = errorWrapperConfig.moreInfoButton;
+    var lessInfoButton = errorWrapperConfig.lessInfoButton;
+    moreInfoButton.onclick = function () {
+      errorMoreInfo.removeAttribute('hidden');
+      moreInfoButton.setAttribute('hidden', 'true');
+      lessInfoButton.removeAttribute('hidden');
+      errorMoreInfo.style.height = errorMoreInfo.scrollHeight + 'px';
+    };
+    lessInfoButton.onclick = function () {
+      errorMoreInfo.setAttribute('hidden', 'true');
+      moreInfoButton.removeAttribute('hidden');
+      lessInfoButton.setAttribute('hidden', 'true');
+    };
+    moreInfoButton.oncontextmenu = noContextMenuHandler;
+    lessInfoButton.oncontextmenu = noContextMenuHandler;
+    closeButton.oncontextmenu = noContextMenuHandler;
+    moreInfoButton.removeAttribute('hidden');
+    lessInfoButton.setAttribute('hidden', 'true');
+    errorMoreInfo.value = moreInfoText;
+  },
+  progress: function pdfViewProgress(level) {
+    var percent = Math.round(level * 100);
+    if (percent > this.loadingBar.percent || isNaN(percent)) {
+      this.loadingBar.percent = percent;
+      if (pdfjsLib.PDFJS.disableAutoFetch && percent) {
+        if (this.disableAutoFetchLoadingBarTimeout) {
+          clearTimeout(this.disableAutoFetchLoadingBarTimeout);
+          this.disableAutoFetchLoadingBarTimeout = null;
+        }
+        this.loadingBar.show();
+        this.disableAutoFetchLoadingBarTimeout = setTimeout(function () {
+          this.loadingBar.hide();
+          this.disableAutoFetchLoadingBarTimeout = null;
+        }.bind(this), DISABLE_AUTO_FETCH_LOADING_BAR_TIMEOUT);
+      }
     }
-    self.initialDestination = initialParams.destination;
-    self.initialBookmark = initialParams.bookmark;
-    self.pdfViewer.currentScaleValue = self.pdfViewer.currentScaleValue;
-    self.setInitialView(initialParams.hash);
-   });
-  });
-  pdfDocument.getPageLabels().then(function (labels) {
-   if (!labels || self.viewerPrefs['disablePageLabels']) {
-    return;
-   }
-   var i = 0, numLabels = labels.length;
-   if (numLabels !== self.pagesCount) {
-    console.error('The number of Page Labels does not match ' + 'the number of pages in the document.');
-    return;
-   }
-   while (i < numLabels && labels[i] === (i + 1).toString()) {
-    i++;
-   }
-   if (i === numLabels) {
-    return;
-   }
-   pdfViewer.setPageLabels(labels);
-   pdfThumbnailViewer.setPageLabels(labels);
-   self.toolbar.setPagesCount(pdfDocument.numPages, true);
-   self.toolbar.setPageNumber(pdfViewer.currentPageNumber, pdfViewer.currentPageLabel);
-  });
-  pagesPromise.then(function () {
-   if (self.supportsPrinting) {
-    pdfDocument.getJavaScript().then(function (javaScript) {
-     if (javaScript.length) {
-      console.warn('Warning: JavaScript is not supported');
-      self.fallback(pdfjsLib.UNSUPPORTED_FEATURES.javaScript);
-     }
-     var regex = /\bprint\s*\(/;
-     for (var i = 0, ii = javaScript.length; i < ii; i++) {
-      var js = javaScript[i];
-      if (js && regex.test(js)) {
-       setTimeout(function () {
-        window.print();
-       });
-       return;
+  },
+  load: function pdfViewLoad(pdfDocument, scale) {
+    var self = this;
+    scale = scale || UNKNOWN_SCALE;
+    this.pdfDocument = pdfDocument;
+    this.pdfDocumentProperties.setDocumentAndUrl(pdfDocument, this.url);
+    var downloadedPromise = pdfDocument.getDownloadInfo().then(function () {
+      self.downloadComplete = true;
+      self.loadingBar.hide();
+    });
+    this.toolbar.setPagesCount(pdfDocument.numPages, false);
+    this.secondaryToolbar.setPagesCount(pdfDocument.numPages);
+    var id = this.documentFingerprint = pdfDocument.fingerprint;
+    var store = this.store = new ViewHistory(id);
+    var baseDocumentUrl;
+    baseDocumentUrl = null;
+    this.pdfLinkService.setDocument(pdfDocument, baseDocumentUrl);
+    var pdfViewer = this.pdfViewer;
+    pdfViewer.currentScale = scale;
+    pdfViewer.setDocument(pdfDocument);
+    var firstPagePromise = pdfViewer.firstPagePromise;
+    var pagesPromise = pdfViewer.pagesPromise;
+    var onePageRendered = pdfViewer.onePageRendered;
+    this.pageRotation = 0;
+    var pdfThumbnailViewer = this.pdfThumbnailViewer;
+    pdfThumbnailViewer.setDocument(pdfDocument);
+    firstPagePromise.then(function (pdfPage) {
+      downloadedPromise.then(function () {
+        self.eventBus.dispatch('documentload', { source: self });
+      });
+      self.loadingBar.setWidth(self.appConfig.viewerContainer);
+      if (!pdfjsLib.PDFJS.disableHistory && !self.isViewerEmbedded) {
+        if (!self.viewerPrefs['showPreviousViewOnLoad']) {
+          self.pdfHistory.clearHistoryState();
+        }
+        self.pdfHistory.initialize(self.documentFingerprint);
+        if (self.pdfHistory.initialDestination) {
+          self.initialDestination = self.pdfHistory.initialDestination;
+        } else if (self.pdfHistory.initialBookmark) {
+          self.initialBookmark = self.pdfHistory.initialBookmark;
+        }
       }
-     }
+      var initialParams = {
+        destination: self.initialDestination,
+        bookmark: self.initialBookmark,
+        hash: null
+      };
+      store.initializedPromise.then(function resolved() {
+        var storedHash = null,
+            sidebarView = null;
+        if (self.viewerPrefs['showPreviousViewOnLoad'] && store.get('exists', false)) {
+          var pageNum = store.get('page', '1');
+          var zoom = self.viewerPrefs['defaultZoomValue'] || store.get('zoom', DEFAULT_SCALE_VALUE);
+          var left = store.get('scrollLeft', '0');
+          var top = store.get('scrollTop', '0');
+          storedHash = 'page=' + pageNum + '&zoom=' + zoom + ',' + left + ',' + top;
+          sidebarView = store.get('sidebarView', SidebarView.NONE);
+        } else if (self.viewerPrefs['defaultZoomValue']) {
+          storedHash = 'page=1&zoom=' + self.viewerPrefs['defaultZoomValue'];
+        }
+        self.setInitialView(storedHash, {
+          scale: scale,
+          sidebarView: sidebarView
+        });
+        initialParams.hash = storedHash;
+        if (!self.isViewerEmbedded) {
+          self.pdfViewer.focus();
+        }
+      }, function rejected(reason) {
+        console.error(reason);
+        self.setInitialView(null, { scale: scale });
+      });
+      pagesPromise.then(function resolved() {
+        if (!initialParams.destination && !initialParams.bookmark && !initialParams.hash) {
+          return;
+        }
+        if (self.hasEqualPageSizes) {
+          return;
+        }
+        self.initialDestination = initialParams.destination;
+        self.initialBookmark = initialParams.bookmark;
+        self.pdfViewer.currentScaleValue = self.pdfViewer.currentScaleValue;
+        self.setInitialView(initialParams.hash);
+      });
     });
-   }
-  });
-  Promise.all([
-   onePageRendered,
-   animationStarted
-  ]).then(function () {
-   pdfDocument.getOutline().then(function (outline) {
-    self.pdfOutlineViewer.render({ outline: outline });
-   });
-   pdfDocument.getAttachments().then(function (attachments) {
-    self.pdfAttachmentViewer.render({ attachments: attachments });
-   });
-  });
-  pdfDocument.getMetadata().then(function (data) {
-   var info = data.info, metadata = data.metadata;
-   self.documentInfo = info;
-   self.metadata = metadata;
-   console.log('PDF ' + pdfDocument.fingerprint + ' [' + info.PDFFormatVersion + ' ' + (info.Producer || '-').trim() + ' / ' + (info.Creator || '-').trim() + ']' + ' (PDF.js: ' + (pdfjsLib.version || '-') + (!pdfjsLib.PDFJS.disableWebGL ? ' [WebGL]' : '') + ')');
-   var pdfTitle;
-   if (metadata && metadata.has('dc:title')) {
-    var title = metadata.get('dc:title');
-    if (title !== 'Untitled') {
-     pdfTitle = title;
+    pdfDocument.getPageLabels().then(function (labels) {
+      if (!labels || self.viewerPrefs['disablePageLabels']) {
+        return;
+      }
+      var i = 0,
+          numLabels = labels.length;
+      if (numLabels !== self.pagesCount) {
+        console.error('The number of Page Labels does not match ' + 'the number of pages in the document.');
+        return;
+      }
+      while (i < numLabels && labels[i] === (i + 1).toString()) {
+        i++;
+      }
+      if (i === numLabels) {
+        return;
+      }
+      pdfViewer.setPageLabels(labels);
+      pdfThumbnailViewer.setPageLabels(labels);
+      self.toolbar.setPagesCount(pdfDocument.numPages, true);
+      self.toolbar.setPageNumber(pdfViewer.currentPageNumber, pdfViewer.currentPageLabel);
+    });
+    pagesPromise.then(function () {
+      if (self.supportsPrinting) {
+        pdfDocument.getJavaScript().then(function (javaScript) {
+          if (javaScript.length) {
+            console.warn('Warning: JavaScript is not supported');
+            self.fallback(pdfjsLib.UNSUPPORTED_FEATURES.javaScript);
+          }
+          var regex = /\bprint\s*\(/;
+          for (var i = 0, ii = javaScript.length; i < ii; i++) {
+            var js = javaScript[i];
+            if (js && regex.test(js)) {
+              setTimeout(function () {
+                window.print();
+              });
+              return;
+            }
+          }
+        });
+      }
+    });
+    Promise.all([onePageRendered, animationStarted]).then(function () {
+      pdfDocument.getOutline().then(function (outline) {
+        self.pdfOutlineViewer.render({ outline: outline });
+      });
+      pdfDocument.getAttachments().then(function (attachments) {
+        self.pdfAttachmentViewer.render({ attachments: attachments });
+      });
+    });
+    pdfDocument.getMetadata().then(function (data) {
+      var info = data.info,
+          metadata = data.metadata;
+      self.documentInfo = info;
+      self.metadata = metadata;
+      console.log('PDF ' + pdfDocument.fingerprint + ' [' + info.PDFFormatVersion + ' ' + (info.Producer || '-').trim() + ' / ' + (info.Creator || '-').trim() + ']' + ' (PDF.js: ' + (pdfjsLib.version || '-') + (!pdfjsLib.PDFJS.disableWebGL ? ' [WebGL]' : '') + ')');
+      var pdfTitle;
+      if (metadata && metadata.has('dc:title')) {
+        var title = metadata.get('dc:title');
+        if (title !== 'Untitled') {
+          pdfTitle = title;
+        }
+      }
+      if (!pdfTitle && info && info['Title']) {
+        pdfTitle = info['Title'];
+      }
+      if (pdfTitle) {
+        self.setTitle(pdfTitle + ' - ' + document.title);
+      }
+      if (info.IsAcroFormPresent) {
+        console.warn('Warning: AcroForm/XFA is not supported');
+        self.fallback(pdfjsLib.UNSUPPORTED_FEATURES.forms);
+      }
+    });
+  },
+  setInitialView: function pdfViewSetInitialView(storedHash, options) {
+    var scale = options && options.scale;
+    var sidebarView = options && options.sidebarView;
+    this.isInitialViewSet = true;
+    this.pdfSidebar.setInitialView(this.viewerPrefs['sidebarViewOnLoad'] || sidebarView | 0);
+    if (this.initialDestination) {
+      this.pdfLinkService.navigateTo(this.initialDestination);
+      this.initialDestination = null;
+    } else if (this.initialBookmark) {
+      this.pdfLinkService.setHash(this.initialBookmark);
+      this.pdfHistory.push({ hash: this.initialBookmark }, true);
+      this.initialBookmark = null;
+    } else if (storedHash) {
+      this.pdfLinkService.setHash(storedHash);
+    } else if (scale) {
+      this.pdfViewer.currentScaleValue = scale;
+      this.page = 1;
     }
-   }
-   if (!pdfTitle && info && info['Title']) {
-    pdfTitle = info['Title'];
-   }
-   if (pdfTitle) {
-    self.setTitle(pdfTitle + ' - ' + document.title);
-   }
-   if (info.IsAcroFormPresent) {
-    console.warn('Warning: AcroForm/XFA is not supported');
-    self.fallback(pdfjsLib.UNSUPPORTED_FEATURES.forms);
-   }
-  });
- },
- setInitialView: function pdfViewSetInitialView(storedHash, options) {
-  var scale = options && options.scale;
-  var sidebarView = options && options.sidebarView;
-  this.isInitialViewSet = true;
-  this.pdfSidebar.setInitialView(this.viewerPrefs['sidebarViewOnLoad'] || sidebarView | 0);
-  if (this.initialDestination) {
-   this.pdfLinkService.navigateTo(this.initialDestination);
-   this.initialDestination = null;
-  } else if (this.initialBookmark) {
-   this.pdfLinkService.setHash(this.initialBookmark);
-   this.pdfHistory.push({ hash: this.initialBookmark }, true);
-   this.initialBookmark = null;
-  } else if (storedHash) {
-   this.pdfLinkService.setHash(storedHash);
-  } else if (scale) {
-   this.pdfViewer.currentScaleValue = scale;
-   this.page = 1;
-  }
-  this.toolbar.setPageNumber(this.pdfViewer.currentPageNumber, this.pdfViewer.currentPageLabel);
-  this.secondaryToolbar.setPageNumber(this.pdfViewer.currentPageNumber);
-  if (!this.pdfViewer.currentScaleValue) {
-   this.pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
-  }
- },
- cleanup: function pdfViewCleanup() {
-  if (!this.pdfDocument) {
-   return;
-  }
-  this.pdfViewer.cleanup();
-  this.pdfThumbnailViewer.cleanup();
-  if (this.pdfViewer.renderer !== RendererType.SVG) {
-   this.pdfDocument.cleanup();
-  }
- },
- forceRendering: function pdfViewForceRendering() {
-  this.pdfRenderingQueue.printing = this.printing;
-  this.pdfRenderingQueue.isThumbnailViewEnabled = this.pdfSidebar.isThumbnailViewVisible;
-  this.pdfRenderingQueue.renderHighestPriority();
- },
- beforePrint: function pdfViewSetupBeforePrint() {
-  if (this.printService) {
-   return;
-  }
-  if (!this.supportsPrinting) {
-   var printMessage = mozL10n.get('printing_not_supported', null, 'Warning: Printing is not fully supported by this browser.');
-   this.error(printMessage);
-   return;
-  }
-  if (!this.pdfViewer.pageViewsReady) {
-   var notReadyMessage = mozL10n.get('printing_not_ready', null, 'Warning: The PDF is not fully loaded for printing.');
-   window.alert(notReadyMessage);
-   return;
-  }
-  var pagesOverview = this.pdfViewer.getPagesOverview();
-  var printContainer = this.appConfig.printContainer;
-  var printService = PDFPrintServiceFactory.instance.createPrintService(this.pdfDocument, pagesOverview, printContainer);
-  this.printService = printService;
-  this.forceRendering();
-  printService.layout();
- },
- get hasEqualPageSizes() {
-  var firstPage = this.pdfViewer.getPageView(0);
-  for (var i = 1, ii = this.pagesCount; i < ii; ++i) {
-   var pageView = this.pdfViewer.getPageView(i);
-   if (pageView.width !== firstPage.width || pageView.height !== firstPage.height) {
-    return false;
-   }
-  }
-  return true;
- },
- afterPrint: function pdfViewSetupAfterPrint() {
-  if (this.printService) {
-   this.printService.destroy();
-   this.printService = null;
-  }
-  this.forceRendering();
- },
- rotatePages: function pdfViewRotatePages(delta) {
-  var pageNumber = this.page;
-  this.pageRotation = (this.pageRotation + 360 + delta) % 360;
-  this.pdfViewer.pagesRotation = this.pageRotation;
-  this.pdfThumbnailViewer.pagesRotation = this.pageRotation;
-  this.forceRendering();
-  this.pdfViewer.currentPageNumber = pageNumber;
- },
- requestPresentationMode: function pdfViewRequestPresentationMode() {
-  if (!this.pdfPresentationMode) {
-   return;
+    this.toolbar.setPageNumber(this.pdfViewer.currentPageNumber, this.pdfViewer.currentPageLabel);
+    this.secondaryToolbar.setPageNumber(this.pdfViewer.currentPageNumber);
+    if (!this.pdfViewer.currentScaleValue) {
+      this.pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
+    }
+  },
+  cleanup: function pdfViewCleanup() {
+    if (!this.pdfDocument) {
+      return;
+    }
+    this.pdfViewer.cleanup();
+    this.pdfThumbnailViewer.cleanup();
+    if (this.pdfViewer.renderer !== RendererType.SVG) {
+      this.pdfDocument.cleanup();
+    }
+  },
+  forceRendering: function pdfViewForceRendering() {
+    this.pdfRenderingQueue.printing = this.printing;
+    this.pdfRenderingQueue.isThumbnailViewEnabled = this.pdfSidebar.isThumbnailViewVisible;
+    this.pdfRenderingQueue.renderHighestPriority();
+  },
+  beforePrint: function pdfViewSetupBeforePrint() {
+    if (this.printService) {
+      return;
+    }
+    if (!this.supportsPrinting) {
+      var printMessage = mozL10n.get('printing_not_supported', null, 'Warning: Printing is not fully supported by this browser.');
+      this.error(printMessage);
+      return;
+    }
+    if (!this.pdfViewer.pageViewsReady) {
+      var notReadyMessage = mozL10n.get('printing_not_ready', null, 'Warning: The PDF is not fully loaded for printing.');
+      window.alert(notReadyMessage);
+      return;
+    }
+    var pagesOverview = this.pdfViewer.getPagesOverview();
+    var printContainer = this.appConfig.printContainer;
+    var printService = PDFPrintServiceFactory.instance.createPrintService(this.pdfDocument, pagesOverview, printContainer);
+    this.printService = printService;
+    this.forceRendering();
+    printService.layout();
+  },
+  get hasEqualPageSizes() {
+    var firstPage = this.pdfViewer.getPageView(0);
+    for (var i = 1, ii = this.pagesCount; i < ii; ++i) {
+      var pageView = this.pdfViewer.getPageView(i);
+      if (pageView.width !== firstPage.width || pageView.height !== firstPage.height) {
+        return false;
+      }
+    }
+    return true;
+  },
+  afterPrint: function pdfViewSetupAfterPrint() {
+    if (this.printService) {
+      this.printService.destroy();
+      this.printService = null;
+    }
+    this.forceRendering();
+  },
+  rotatePages: function pdfViewRotatePages(delta) {
+    var pageNumber = this.page;
+    this.pageRotation = (this.pageRotation + 360 + delta) % 360;
+    this.pdfViewer.pagesRotation = this.pageRotation;
+    this.pdfThumbnailViewer.pagesRotation = this.pageRotation;
+    this.forceRendering();
+    this.pdfViewer.currentPageNumber = pageNumber;
+  },
+  requestPresentationMode: function pdfViewRequestPresentationMode() {
+    if (!this.pdfPresentationMode) {
+      return;
+    }
+    this.pdfPresentationMode.request();
+  },
+  bindEvents: function pdfViewBindEvents() {
+    var eventBus = this.eventBus;
+    eventBus.on('resize', webViewerResize);
+    eventBus.on('hashchange', webViewerHashchange);
+    eventBus.on('beforeprint', this.beforePrint.bind(this));
+    eventBus.on('afterprint', this.afterPrint.bind(this));
+    eventBus.on('pagerendered', webViewerPageRendered);
+    eventBus.on('textlayerrendered', webViewerTextLayerRendered);
+    eventBus.on('updateviewarea', webViewerUpdateViewarea);
+    eventBus.on('pagechanging', webViewerPageChanging);
+    eventBus.on('scalechanging', webViewerScaleChanging);
+    eventBus.on('sidebarviewchanged', webViewerSidebarViewChanged);
+    eventBus.on('pagemode', webViewerPageMode);
+    eventBus.on('namedaction', webViewerNamedAction);
+    eventBus.on('presentationmodechanged', webViewerPresentationModeChanged);
+    eventBus.on('presentationmode', webViewerPresentationMode);
+    eventBus.on('openfile', webViewerOpenFile);
+    eventBus.on('print', webViewerPrint);
+    eventBus.on('download', webViewerDownload);
+    eventBus.on('firstpage', webViewerFirstPage);
+    eventBus.on('lastpage', webViewerLastPage);
+    eventBus.on('nextpage', webViewerNextPage);
+    eventBus.on('previouspage', webViewerPreviousPage);
+    eventBus.on('zoomin', webViewerZoomIn);
+    eventBus.on('zoomout', webViewerZoomOut);
+    eventBus.on('pagenumberchanged', webViewerPageNumberChanged);
+    eventBus.on('scalechanged', webViewerScaleChanged);
+    eventBus.on('rotatecw', webViewerRotateCw);
+    eventBus.on('rotateccw', webViewerRotateCcw);
+    eventBus.on('documentproperties', webViewerDocumentProperties);
+    eventBus.on('find', webViewerFind);
+    eventBus.on('findfromurlhash', webViewerFindFromUrlHash);
+    eventBus.on('fileinputchange', webViewerFileInputChange);
+  },
+  bindWindowEvents: function pdfViewBindWindowEvents() {
+    var eventBus = this.eventBus;
+    window.addEventListener('wheel', webViewerWheel);
+    window.addEventListener('click', webViewerClick);
+    window.addEventListener('keydown', webViewerKeyDown);
+    window.addEventListener('resize', function windowResize() {
+      eventBus.dispatch('resize');
+    });
+    window.addEventListener('hashchange', function windowHashChange() {
+      eventBus.dispatch('hashchange', { hash: document.location.hash.substring(1) });
+    });
+    window.addEventListener('beforeprint', function windowBeforePrint() {
+      eventBus.dispatch('beforeprint');
+    });
+    window.addEventListener('afterprint', function windowAfterPrint() {
+      eventBus.dispatch('afterprint');
+    });
+    window.addEventListener('change', function windowChange(evt) {
+      var files = evt.target.files;
+      if (!files || files.length === 0) {
+        return;
+      }
+      eventBus.dispatch('fileinputchange', { fileInput: evt.target });
+    });
   }
-  this.pdfPresentationMode.request();
- },
- bindEvents: function pdfViewBindEvents() {
-  var eventBus = this.eventBus;
-  eventBus.on('resize', webViewerResize);
-  eventBus.on('hashchange', webViewerHashchange);
-  eventBus.on('beforeprint', this.beforePrint.bind(this));
-  eventBus.on('afterprint', this.afterPrint.bind(this));
-  eventBus.on('pagerendered', webViewerPageRendered);
-  eventBus.on('textlayerrendered', webViewerTextLayerRendered);
-  eventBus.on('updateviewarea', webViewerUpdateViewarea);
-  eventBus.on('pagechanging', webViewerPageChanging);
-  eventBus.on('scalechanging', webViewerScaleChanging);
-  eventBus.on('sidebarviewchanged', webViewerSidebarViewChanged);
-  eventBus.on('pagemode', webViewerPageMode);
-  eventBus.on('namedaction', webViewerNamedAction);
-  eventBus.on('presentationmodechanged', webViewerPresentationModeChanged);
-  eventBus.on('presentationmode', webViewerPresentationMode);
-  eventBus.on('openfile', webViewerOpenFile);
-  eventBus.on('print', webViewerPrint);
-  eventBus.on('download', webViewerDownload);
-  eventBus.on('firstpage', webViewerFirstPage);
-  eventBus.on('lastpage', webViewerLastPage);
-  eventBus.on('nextpage', webViewerNextPage);
-  eventBus.on('previouspage', webViewerPreviousPage);
-  eventBus.on('zoomin', webViewerZoomIn);
-  eventBus.on('zoomout', webViewerZoomOut);
-  eventBus.on('pagenumberchanged', webViewerPageNumberChanged);
-  eventBus.on('scalechanged', webViewerScaleChanged);
-  eventBus.on('rotatecw', webViewerRotateCw);
-  eventBus.on('rotateccw', webViewerRotateCcw);
-  eventBus.on('documentproperties', webViewerDocumentProperties);
-  eventBus.on('find', webViewerFind);
-  eventBus.on('findfromurlhash', webViewerFindFromUrlHash);
-  eventBus.on('fileinputchange', webViewerFileInputChange);
- },
- bindWindowEvents: function pdfViewBindWindowEvents() {
-  var eventBus = this.eventBus;
-  window.addEventListener('wheel', webViewerWheel);
-  window.addEventListener('click', webViewerClick);
-  window.addEventListener('keydown', webViewerKeyDown);
-  window.addEventListener('resize', function windowResize() {
-   eventBus.dispatch('resize');
-  });
-  window.addEventListener('hashchange', function windowHashChange() {
-   eventBus.dispatch('hashchange', { hash: document.location.hash.substring(1) });
-  });
-  window.addEventListener('beforeprint', function windowBeforePrint() {
-   eventBus.dispatch('beforeprint');
-  });
-  window.addEventListener('afterprint', function windowAfterPrint() {
-   eventBus.dispatch('afterprint');
-  });
-  window.addEventListener('change', function windowChange(evt) {
-   var files = evt.target.files;
-   if (!files || files.length === 0) {
-    return;
-   }
-   eventBus.dispatch('fileinputchange', { fileInput: evt.target });
-  });
- }
 };
 var validateFileURL;
-var HOSTED_VIEWER_ORIGINS = [
- 'null',
- 'http://mozilla.github.io',
- 'https://mozilla.github.io'
-];
+var HOSTED_VIEWER_ORIGINS = ['null', 'http://mozilla.github.io', 'https://mozilla.github.io'];
 validateFileURL = function validateFileURL(file) {
- try {
-  var viewerOrigin = new URL(window.location.href).origin || 'null';
-  if (HOSTED_VIEWER_ORIGINS.indexOf(viewerOrigin) >= 0) {
-   return;
-  }
-  var fileOrigin = new URL(file, window.location.href).origin;
-  if (fileOrigin !== viewerOrigin) {
-   throw new Error('file origin does not match viewer\'s');
+  try {
+    var viewerOrigin = new URL(window.location.href).origin || 'null';
+    if (HOSTED_VIEWER_ORIGINS.indexOf(viewerOrigin) >= 0) {
+      return;
+    }
+    var fileOrigin = new URL(file, window.location.href).origin;
+    if (fileOrigin !== viewerOrigin) {
+      throw new Error('file origin does not match viewer\'s');
+    }
+  } catch (e) {
+    var message = e && e.message;
+    var loadingErrorMessage = mozL10n.get('loading_error', null, 'An error occurred while loading the PDF.');
+    var moreInfo = { message: message };
+    PDFViewerApplication.error(loadingErrorMessage, moreInfo);
+    throw e;
   }
- } catch (e) {
-  var message = e && e.message;
-  var loadingErrorMessage = mozL10n.get('loading_error', null, 'An error occurred while loading the PDF.');
-  var moreInfo = { message: message };
-  PDFViewerApplication.error(loadingErrorMessage, moreInfo);
-  throw e;
- }
 };
 function loadAndEnablePDFBug(enabledTabs) {
- return new Promise(function (resolve, reject) {
-  var appConfig = PDFViewerApplication.appConfig;
-  var script = document.createElement('script');
-  script.src = appConfig.debuggerScriptPath;
-  script.onload = function () {
-   PDFBug.enable(enabledTabs);
-   PDFBug.init(pdfjsLib, appConfig.mainContainer);
-   resolve();
-  };
-  script.onerror = function () {
-   reject(new Error('Cannot load debugger at ' + script.src));
-  };
-  (document.getElementsByTagName('head')[0] || document.body).appendChild(script);
- });
+  return new Promise(function (resolve, reject) {
+    var appConfig = PDFViewerApplication.appConfig;
+    var script = document.createElement('script');
+    script.src = appConfig.debuggerScriptPath;
+    script.onload = function () {
+      PDFBug.enable(enabledTabs);
+      PDFBug.init(pdfjsLib, appConfig.mainContainer);
+      resolve();
+    };
+    script.onerror = function () {
+      reject(new Error('Cannot load debugger at ' + script.src));
+    };
+    (document.getElementsByTagName('head')[0] || document.body).appendChild(script);
+  });
 }
 function webViewerInitialized() {
- var appConfig = PDFViewerApplication.appConfig;
- var file;
- var queryString = document.location.search.substring(1);
- var params = parseQueryString(queryString);
- file = 'file' in params ? params.file : appConfig.defaultUrl;
- validateFileURL(file);
- var waitForBeforeOpening = [];
- var fileInput = document.createElement('input');
- fileInput.id = appConfig.openFileInputName;
- fileInput.className = 'fileInput';
- fileInput.setAttribute('type', 'file');
- fileInput.oncontextmenu = noContextMenuHandler;
- document.body.appendChild(fileInput);
- if (!window.File || !window.FileReader || !window.FileList || !window.Blob) {
-  appConfig.toolbar.openFile.setAttribute('hidden', 'true');
-  appConfig.secondaryToolbar.openFileButton.setAttribute('hidden', 'true');
- } else {
-  fileInput.value = null;
- }
- var PDFJS = pdfjsLib.PDFJS;
- if (PDFViewerApplication.viewerPrefs['pdfBugEnabled']) {
-  var hash = document.location.hash.substring(1);
-  var hashParams = parseQueryString(hash);
-  if ('disableworker' in hashParams) {
-   PDFJS.disableWorker = hashParams['disableworker'] === 'true';
-  }
-  if ('disablerange' in hashParams) {
-   PDFJS.disableRange = hashParams['disablerange'] === 'true';
-  }
-  if ('disablestream' in hashParams) {
-   PDFJS.disableStream = hashParams['disablestream'] === 'true';
-  }
-  if ('disableautofetch' in hashParams) {
-   PDFJS.disableAutoFetch = hashParams['disableautofetch'] === 'true';
+  var appConfig = PDFViewerApplication.appConfig;
+  var file;
+  var queryString = document.location.search.substring(1);
+  var params = parseQueryString(queryString);
+  file = 'file' in params ? params.file : appConfig.defaultUrl;
+  validateFileURL(file);
+  var waitForBeforeOpening = [];
+  var fileInput = document.createElement('input');
+  fileInput.id = appConfig.openFileInputName;
+  fileInput.className = 'fileInput';
+  fileInput.setAttribute('type', 'file');
+  fileInput.oncontextmenu = noContextMenuHandler;
+  document.body.appendChild(fileInput);
+  if (!window.File || !window.FileReader || !window.FileList || !window.Blob) {
+    appConfig.toolbar.openFile.setAttribute('hidden', 'true');
+    appConfig.secondaryToolbar.openFileButton.setAttribute('hidden', 'true');
+  } else {
+    fileInput.value = null;
   }
-  if ('disablefontface' in hashParams) {
-   PDFJS.disableFontFace = hashParams['disablefontface'] === 'true';
+  var PDFJS = pdfjsLib.PDFJS;
+  if (PDFViewerApplication.viewerPrefs['pdfBugEnabled']) {
+    var hash = document.location.hash.substring(1);
+    var hashParams = parseQueryString(hash);
+    if ('disableworker' in hashParams) {
+      PDFJS.disableWorker = hashParams['disableworker'] === 'true';
+    }
+    if ('disablerange' in hashParams) {
+      PDFJS.disableRange = hashParams['disablerange'] === 'true';
+    }
+    if ('disablestream' in hashParams) {
+      PDFJS.disableStream = hashParams['disablestream'] === 'true';
+    }
+    if ('disableautofetch' in hashParams) {
+      PDFJS.disableAutoFetch = hashParams['disableautofetch'] === 'true';
+    }
+    if ('disablefontface' in hashParams) {
+      PDFJS.disableFontFace = hashParams['disablefontface'] === 'true';
+    }
+    if ('disablehistory' in hashParams) {
+      PDFJS.disableHistory = hashParams['disablehistory'] === 'true';
+    }
+    if ('webgl' in hashParams) {
+      PDFJS.disableWebGL = hashParams['webgl'] !== 'true';
+    }
+    if ('useonlycsszoom' in hashParams) {
+      PDFJS.useOnlyCssZoom = hashParams['useonlycsszoom'] === 'true';
+    }
+    if ('verbosity' in hashParams) {
+      PDFJS.verbosity = hashParams['verbosity'] | 0;
+    }
+    if ('ignorecurrentpositiononzoom' in hashParams) {
+      PDFJS.ignoreCurrentPositionOnZoom = hashParams['ignorecurrentpositiononzoom'] === 'true';
+    }
+    if ('locale' in hashParams) {
+      PDFJS.locale = hashParams['locale'];
+    }
+    if ('textlayer' in hashParams) {
+      switch (hashParams['textlayer']) {
+        case 'off':
+          PDFJS.disableTextLayer = true;
+          break;
+        case 'visible':
+        case 'shadow':
+        case 'hover':
+          var viewer = appConfig.viewerContainer;
+          viewer.classList.add('textLayer-' + hashParams['textlayer']);
+          break;
+      }
+    }
+    if ('pdfbug' in hashParams) {
+      PDFJS.pdfBug = true;
+      var pdfBug = hashParams['pdfbug'];
+      var enabled = pdfBug.split(',');
+      waitForBeforeOpening.push(loadAndEnablePDFBug(enabled));
+    }
   }
-  if ('disablehistory' in hashParams) {
-   PDFJS.disableHistory = hashParams['disablehistory'] === 'true';
+  mozL10n.setLanguage(PDFJS.locale);
+  if (!PDFViewerApplication.supportsPrinting) {
+    appConfig.toolbar.print.classList.add('hidden');
+    appConfig.secondaryToolbar.printButton.classList.add('hidden');
   }
-  if ('webgl' in hashParams) {
-   PDFJS.disableWebGL = hashParams['webgl'] !== 'true';
+  if (!PDFViewerApplication.supportsFullscreen) {
+    appConfig.toolbar.presentationModeButton.classList.add('hidden');
+    appConfig.secondaryToolbar.presentationModeButton.classList.add('hidden');
   }
-  if ('useonlycsszoom' in hashParams) {
-   PDFJS.useOnlyCssZoom = hashParams['useonlycsszoom'] === 'true';
+  if (PDFViewerApplication.supportsIntegratedFind) {
+    appConfig.toolbar.viewFind.classList.add('hidden');
   }
-  if ('verbosity' in hashParams) {
-   PDFJS.verbosity = hashParams['verbosity'] | 0;
+  appConfig.sidebar.mainContainer.addEventListener('transitionend', function (e) {
+    if (e.target === this) {
+      PDFViewerApplication.eventBus.dispatch('resize');
+    }
+  }, true);
+  appConfig.sidebar.toggleButton.addEventListener('click', function () {
+    PDFViewerApplication.pdfSidebar.toggle();
+  });
+  Promise.all(waitForBeforeOpening).then(function () {
+    webViewerOpenFileViaURL(file);
+  }).catch(function (reason) {
+    PDFViewerApplication.error(mozL10n.get('loading_error', null, 'An error occurred while opening.'), reason);
+  });
+}
+var webViewerOpenFileViaURL;
+webViewerOpenFileViaURL = function webViewerOpenFileViaURL(file) {
+  if (file && file.lastIndexOf('file:', 0) === 0) {
+    PDFViewerApplication.setTitleUsingUrl(file);
+    var xhr = new XMLHttpRequest();
+    xhr.onload = function () {
+      PDFViewerApplication.open(new Uint8Array(xhr.response));
+    };
+    try {
+      xhr.open('GET', file);
+      xhr.responseType = 'arraybuffer';
+      xhr.send();
+    } catch (e) {
+      PDFViewerApplication.error(mozL10n.get('loading_error', null, 'An error occurred while loading the PDF.'), e);
+    }
+    return;
   }
-  if ('ignorecurrentpositiononzoom' in hashParams) {
-   PDFJS.ignoreCurrentPositionOnZoom = hashParams['ignorecurrentpositiononzoom'] === 'true';
+  if (file) {
+    PDFViewerApplication.open(file);
   }
-  if ('locale' in hashParams) {
-   PDFJS.locale = hashParams['locale'];
+};
+function webViewerPageRendered(e) {
+  var pageNumber = e.pageNumber;
+  var pageIndex = pageNumber - 1;
+  var pageView = PDFViewerApplication.pdfViewer.getPageView(pageIndex);
+  if (pageNumber === PDFViewerApplication.page) {
+    PDFViewerApplication.toolbar.updateLoadingIndicatorState(false);
   }
-  if ('textlayer' in hashParams) {
-   switch (hashParams['textlayer']) {
-   case 'off':
-    PDFJS.disableTextLayer = true;
-    break;
-   case 'visible':
-   case 'shadow':
-   case 'hover':
-    var viewer = appConfig.viewerContainer;
-    viewer.classList.add('textLayer-' + hashParams['textlayer']);
-    break;
-   }
+  if (!pageView) {
+    return;
   }
-  if ('pdfbug' in hashParams) {
-   PDFJS.pdfBug = true;
-   var pdfBug = hashParams['pdfbug'];
-   var enabled = pdfBug.split(',');
-   waitForBeforeOpening.push(loadAndEnablePDFBug(enabled));
+  if (PDFViewerApplication.pdfSidebar.isThumbnailViewVisible) {
+    var thumbnailView = PDFViewerApplication.pdfThumbnailViewer.getThumbnail(pageIndex);
+    thumbnailView.setImage(pageView);
   }
- }
- mozL10n.setLanguage(PDFJS.locale);
- if (!PDFViewerApplication.supportsPrinting) {
-  appConfig.toolbar.print.classList.add('hidden');
-  appConfig.secondaryToolbar.printButton.classList.add('hidden');
- }
- if (!PDFViewerApplication.supportsFullscreen) {
-  appConfig.toolbar.presentationModeButton.classList.add('hidden');
-  appConfig.secondaryToolbar.presentationModeButton.classList.add('hidden');
- }
- if (PDFViewerApplication.supportsIntegratedFind) {
-  appConfig.toolbar.viewFind.classList.add('hidden');
- }
- appConfig.sidebar.mainContainer.addEventListener('transitionend', function (e) {
-  if (e.target === this) {
-   PDFViewerApplication.eventBus.dispatch('resize');
+  if (pdfjsLib.PDFJS.pdfBug && Stats.enabled && pageView.stats) {
+    Stats.add(pageNumber, pageView.stats);
   }
- }, true);
- appConfig.sidebar.toggleButton.addEventListener('click', function () {
-  PDFViewerApplication.pdfSidebar.toggle();
- });
- Promise.all(waitForBeforeOpening).then(function () {
-  webViewerOpenFileViaURL(file);
- }).catch(function (reason) {
-  PDFViewerApplication.error(mozL10n.get('loading_error', null, 'An error occurred while opening.'), reason);
- });
-}
-var webViewerOpenFileViaURL;
-webViewerOpenFileViaURL = function webViewerOpenFileViaURL(file) {
- if (file && file.lastIndexOf('file:', 0) === 0) {
-  PDFViewerApplication.setTitleUsingUrl(file);
-  var xhr = new XMLHttpRequest();
-  xhr.onload = function () {
-   PDFViewerApplication.open(new Uint8Array(xhr.response));
-  };
-  try {
-   xhr.open('GET', file);
-   xhr.responseType = 'arraybuffer';
-   xhr.send();
-  } catch (e) {
-   PDFViewerApplication.error(mozL10n.get('loading_error', null, 'An error occurred while loading the PDF.'), e);
+  if (pageView.error) {
+    PDFViewerApplication.error(mozL10n.get('rendering_error', null, 'An error occurred while rendering the page.'), pageView.error);
   }
-  return;
- }
- if (file) {
-  PDFViewerApplication.open(file);
- }
-};
-function webViewerPageRendered(e) {
- var pageNumber = e.pageNumber;
- var pageIndex = pageNumber - 1;
- var pageView = PDFViewerApplication.pdfViewer.getPageView(pageIndex);
- if (pageNumber === PDFViewerApplication.page) {
-  PDFViewerApplication.toolbar.updateLoadingIndicatorState(false);
- }
- if (!pageView) {
-  return;
- }
- if (PDFViewerApplication.pdfSidebar.isThumbnailViewVisible) {
-  var thumbnailView = PDFViewerApplication.pdfThumbnailViewer.getThumbnail(pageIndex);
-  thumbnailView.setImage(pageView);
- }
- if (pdfjsLib.PDFJS.pdfBug && Stats.enabled && pageView.stats) {
-  Stats.add(pageNumber, pageView.stats);
- }
- if (pageView.error) {
-  PDFViewerApplication.error(mozL10n.get('rendering_error', null, 'An error occurred while rendering the page.'), pageView.error);
- }
-}
-function webViewerTextLayerRendered(e) {
 }
+function webViewerTextLayerRendered(e) {}
 function webViewerPageMode(e) {
- var mode = e.mode, view;
- switch (mode) {
- case 'thumbs':
-  view = SidebarView.THUMBS;
-  break;
- case 'bookmarks':
- case 'outline':
-  view = SidebarView.OUTLINE;
-  break;
- case 'attachments':
-  view = SidebarView.ATTACHMENTS;
-  break;
- case 'none':
-  view = SidebarView.NONE;
-  break;
- default:
-  console.error('Invalid "pagemode" hash parameter: ' + mode);
-  return;
- }
- PDFViewerApplication.pdfSidebar.switchView(view, true);
+  var mode = e.mode,
+      view;
+  switch (mode) {
+    case 'thumbs':
+      view = SidebarView.THUMBS;
+      break;
+    case 'bookmarks':
+    case 'outline':
+      view = SidebarView.OUTLINE;
+      break;
+    case 'attachments':
+      view = SidebarView.ATTACHMENTS;
+      break;
+    case 'none':
+      view = SidebarView.NONE;
+      break;
+    default:
+      console.error('Invalid "pagemode" hash parameter: ' + mode);
+      return;
+  }
+  PDFViewerApplication.pdfSidebar.switchView(view, true);
 }
 function webViewerNamedAction(e) {
- var action = e.action;
- switch (action) {
- case 'GoToPage':
-  PDFViewerApplication.appConfig.toolbar.pageNumber.select();
-  break;
- case 'Find':
-  if (!PDFViewerApplication.supportsIntegratedFind) {
-   PDFViewerApplication.findBar.toggle();
+  var action = e.action;
+  switch (action) {
+    case 'GoToPage':
+      PDFViewerApplication.appConfig.toolbar.pageNumber.select();
+      break;
+    case 'Find':
+      if (!PDFViewerApplication.supportsIntegratedFind) {
+        PDFViewerApplication.findBar.toggle();
+      }
+      break;
   }
-  break;
- }
 }
 function webViewerPresentationModeChanged(e) {
- var active = e.active;
- var switchInProgress = e.switchInProgress;
- PDFViewerApplication.pdfViewer.presentationModeState = switchInProgress ? PresentationModeState.CHANGING : active ? PresentationModeState.FULLSCREEN : PresentationModeState.NORMAL;
+  var active = e.active;
+  var switchInProgress = e.switchInProgress;
+  PDFViewerApplication.pdfViewer.presentationModeState = switchInProgress ? PresentationModeState.CHANGING : active ? PresentationModeState.FULLSCREEN : PresentationModeState.NORMAL;
 }
 function webViewerSidebarViewChanged(e) {
- PDFViewerApplication.pdfRenderingQueue.isThumbnailViewEnabled = PDFViewerApplication.pdfSidebar.isThumbnailViewVisible;
- var store = PDFViewerApplication.store;
- if (!store || !PDFViewerApplication.isInitialViewSet) {
-  return;
- }
- store.initializedPromise.then(function () {
-  store.set('sidebarView', e.view).catch(function () {
+  PDFViewerApplication.pdfRenderingQueue.isThumbnailViewEnabled = PDFViewerApplication.pdfSidebar.isThumbnailViewVisible;
+  var store = PDFViewerApplication.store;
+  if (!store || !PDFViewerApplication.isInitialViewSet) {
+    return;
+  }
+  store.initializedPromise.then(function () {
+    store.set('sidebarView', e.view).catch(function () {});
   });
- });
 }
 function webViewerUpdateViewarea(e) {
- var location = e.location, store = PDFViewerApplication.store;
- if (store) {
-  store.initializedPromise.then(function () {
-   store.setMultiple({
-    'exists': true,
-    'page': location.pageNumber,
-    'zoom': location.scale,
-    'scrollLeft': location.left,
-    'scrollTop': location.top
-   }).catch(function () {
-   });
-  });
- }
- var href = PDFViewerApplication.pdfLinkService.getAnchorUrl(location.pdfOpenParams);
- PDFViewerApplication.appConfig.toolbar.viewBookmark.href = href;
- PDFViewerApplication.appConfig.secondaryToolbar.viewBookmarkButton.href = href;
- PDFViewerApplication.pdfHistory.updateCurrentBookmark(location.pdfOpenParams, location.pageNumber);
- var currentPage = PDFViewerApplication.pdfViewer.getPageView(PDFViewerApplication.page - 1);
- var loading = currentPage.renderingState !== RenderingStates.FINISHED;
- PDFViewerApplication.toolbar.updateLoadingIndicatorState(loading);
+  var location = e.location,
+      store = PDFViewerApplication.store;
+  if (store) {
+    store.initializedPromise.then(function () {
+      store.setMultiple({
+        'exists': true,
+        'page': location.pageNumber,
+        'zoom': location.scale,
+        'scrollLeft': location.left,
+        'scrollTop': location.top
+      }).catch(function () {});
+    });
+  }
+  var href = PDFViewerApplication.pdfLinkService.getAnchorUrl(location.pdfOpenParams);
+  PDFViewerApplication.appConfig.toolbar.viewBookmark.href = href;
+  PDFViewerApplication.appConfig.secondaryToolbar.viewBookmarkButton.href = href;
+  PDFViewerApplication.pdfHistory.updateCurrentBookmark(location.pdfOpenParams, location.pageNumber);
+  var currentPage = PDFViewerApplication.pdfViewer.getPageView(PDFViewerApplication.page - 1);
+  var loading = currentPage.renderingState !== RenderingStates.FINISHED;
+  PDFViewerApplication.toolbar.updateLoadingIndicatorState(loading);
 }
 function webViewerResize() {
- var currentScaleValue = PDFViewerApplication.pdfViewer.currentScaleValue;
- if (currentScaleValue === 'auto' || currentScaleValue === 'page-fit' || currentScaleValue === 'page-width') {
-  PDFViewerApplication.pdfViewer.currentScaleValue = currentScaleValue;
- } else if (!currentScaleValue) {
-  PDFViewerApplication.pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
- }
- PDFViewerApplication.pdfViewer.update();
+  var currentScaleValue = PDFViewerApplication.pdfViewer.currentScaleValue;
+  if (currentScaleValue === 'auto' || currentScaleValue === 'page-fit' || currentScaleValue === 'page-width') {
+    PDFViewerApplication.pdfViewer.currentScaleValue = currentScaleValue;
+  } else if (!currentScaleValue) {
+    PDFViewerApplication.pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
+  }
+  PDFViewerApplication.pdfViewer.update();
 }
 function webViewerHashchange(e) {
- if (PDFViewerApplication.pdfHistory.isHashChangeUnlocked) {
-  var hash = e.hash;
-  if (!hash) {
-   return;
-  }
-  if (!PDFViewerApplication.isInitialViewSet) {
-   PDFViewerApplication.initialBookmark = hash;
-  } else {
-   PDFViewerApplication.pdfLinkService.setHash(hash);
+  if (PDFViewerApplication.pdfHistory.isHashChangeUnlocked) {
+    var hash = e.hash;
+    if (!hash) {
+      return;
+    }
+    if (!PDFViewerApplication.isInitialViewSet) {
+      PDFViewerApplication.initialBookmark = hash;
+    } else {
+      PDFViewerApplication.pdfLinkService.setHash(hash);
+    }
   }
- }
 }
 var webViewerFileInputChange;
 webViewerFileInputChange = function webViewerFileInputChange(e) {
- var file = e.fileInput.files[0];
- if (!pdfjsLib.PDFJS.disableCreateObjectURL && typeof URL !== 'undefined' && URL.createObjectURL) {
-  PDFViewerApplication.open(URL.createObjectURL(file));
- } else {
-  var fileReader = new FileReader();
-  fileReader.onload = function webViewerChangeFileReaderOnload(evt) {
-   var buffer = evt.target.result;
-   var uint8Array = new Uint8Array(buffer);
-   PDFViewerApplication.open(uint8Array);
-  };
-  fileReader.readAsArrayBuffer(file);
- }
- PDFViewerApplication.setTitleUsingUrl(file.name);
- var appConfig = PDFViewerApplication.appConfig;
- appConfig.toolbar.viewBookmark.setAttribute('hidden', 'true');
- appConfig.secondaryToolbar.viewBookmarkButton.setAttribute('hidden', 'true');
- appConfig.toolbar.download.setAttribute('hidden', 'true');
- appConfig.secondaryToolbar.downloadButton.setAttribute('hidden', 'true');
+  var file = e.fileInput.files[0];
+  if (!pdfjsLib.PDFJS.disableCreateObjectURL && typeof URL !== 'undefined' && URL.createObjectURL) {
+    PDFViewerApplication.open(URL.createObjectURL(file));
+  } else {
+    var fileReader = new FileReader();
+    fileReader.onload = function webViewerChangeFileReaderOnload(evt) {
+      var buffer = evt.target.result;
+      var uint8Array = new Uint8Array(buffer);
+      PDFViewerApplication.open(uint8Array);
+    };
+    fileReader.readAsArrayBuffer(file);
+  }
+  PDFViewerApplication.setTitleUsingUrl(file.name);
+  var appConfig = PDFViewerApplication.appConfig;
+  appConfig.toolbar.viewBookmark.setAttribute('hidden', 'true');
+  appConfig.secondaryToolbar.viewBookmarkButton.setAttribute('hidden', 'true');
+  appConfig.toolbar.download.setAttribute('hidden', 'true');
+  appConfig.secondaryToolbar.downloadButton.setAttribute('hidden', 'true');
 };
 function webViewerPresentationMode() {
- PDFViewerApplication.requestPresentationMode();
+  PDFViewerApplication.requestPresentationMode();
 }
 function webViewerOpenFile() {
- var openFileInputName = PDFViewerApplication.appConfig.openFileInputName;
- document.getElementById(openFileInputName).click();
+  var openFileInputName = PDFViewerApplication.appConfig.openFileInputName;
+  document.getElementById(openFileInputName).click();
 }
 function webViewerPrint() {
- window.print();
+  window.print();
 }
 function webViewerDownload() {
- PDFViewerApplication.download();
+  PDFViewerApplication.download();
 }
 function webViewerFirstPage() {
- if (PDFViewerApplication.pdfDocument) {
-  PDFViewerApplication.page = 1;
- }
+  if (PDFViewerApplication.pdfDocument) {
+    PDFViewerApplication.page = 1;
+  }
 }
 function webViewerLastPage() {
- if (PDFViewerApplication.pdfDocument) {
-  PDFViewerApplication.page = PDFViewerApplication.pagesCount;
- }
+  if (PDFViewerApplication.pdfDocument) {
+    PDFViewerApplication.page = PDFViewerApplication.pagesCount;
+  }
 }
 function webViewerNextPage() {
- PDFViewerApplication.page++;
+  PDFViewerApplication.page++;
 }
 function webViewerPreviousPage() {
- PDFViewerApplication.page--;
+  PDFViewerApplication.page--;
 }
 function webViewerZoomIn() {
- PDFViewerApplication.zoomIn();
+  PDFViewerApplication.zoomIn();
 }
 function webViewerZoomOut() {
- PDFViewerApplication.zoomOut();
+  PDFViewerApplication.zoomOut();
 }
 function webViewerPageNumberChanged(e) {
- var pdfViewer = PDFViewerApplication.pdfViewer;
- pdfViewer.currentPageLabel = e.value;
- if (e.value !== pdfViewer.currentPageNumber.toString() && e.value !== pdfViewer.currentPageLabel) {
-  PDFViewerApplication.toolbar.setPageNumber(pdfViewer.currentPageNumber, pdfViewer.currentPageLabel);
- }
+  var pdfViewer = PDFViewerApplication.pdfViewer;
+  pdfViewer.currentPageLabel = e.value;
+  if (e.value !== pdfViewer.currentPageNumber.toString() && e.value !== pdfViewer.currentPageLabel) {
+    PDFViewerApplication.toolbar.setPageNumber(pdfViewer.currentPageNumber, pdfViewer.currentPageLabel);
+  }
 }
 function webViewerScaleChanged(e) {
- PDFViewerApplication.pdfViewer.currentScaleValue = e.value;
+  PDFViewerApplication.pdfViewer.currentScaleValue = e.value;
 }
 function webViewerRotateCw() {
- PDFViewerApplication.rotatePages(90);
+  PDFViewerApplication.rotatePages(90);
 }
 function webViewerRotateCcw() {
- PDFViewerApplication.rotatePages(-90);
+  PDFViewerApplication.rotatePages(-90);
 }
 function webViewerDocumentProperties() {
- PDFViewerApplication.pdfDocumentProperties.open();
+  PDFViewerApplication.pdfDocumentProperties.open();
 }
 function webViewerFind(e) {
- PDFViewerApplication.findController.executeCommand('find' + e.type, {
-  query: e.query,
-  phraseSearch: e.phraseSearch,
-  caseSensitive: e.caseSensitive,
-  highlightAll: e.highlightAll,
-  findPrevious: e.findPrevious
- });
+  PDFViewerApplication.findController.executeCommand('find' + e.type, {
+    query: e.query,
+    phraseSearch: e.phraseSearch,
+    caseSensitive: e.caseSensitive,
+    highlightAll: e.highlightAll,
+    findPrevious: e.findPrevious
+  });
 }
 function webViewerFindFromUrlHash(e) {
- PDFViewerApplication.findController.executeCommand('find', {
-  query: e.query,
-  phraseSearch: e.phraseSearch,
-  caseSensitive: false,
-  highlightAll: true,
-  findPrevious: false
- });
+  PDFViewerApplication.findController.executeCommand('find', {
+    query: e.query,
+    phraseSearch: e.phraseSearch,
+    caseSensitive: false,
+    highlightAll: true,
+    findPrevious: false
+  });
 }
 function webViewerScaleChanging(e) {
- PDFViewerApplication.toolbar.setPageScale(e.presetValue, e.scale);
- PDFViewerApplication.pdfViewer.update();
+  PDFViewerApplication.toolbar.setPageScale(e.presetValue, e.scale);
+  PDFViewerApplication.pdfViewer.update();
 }
 function webViewerPageChanging(e) {
- var page = e.pageNumber;
- PDFViewerApplication.toolbar.setPageNumber(page, e.pageLabel || null);
- PDFViewerApplication.secondaryToolbar.setPageNumber(page);
- if (PDFViewerApplication.pdfSidebar.isThumbnailViewVisible) {
-  PDFViewerApplication.pdfThumbnailViewer.scrollThumbnailIntoView(page);
- }
- if (pdfjsLib.PDFJS.pdfBug && Stats.enabled) {
-  var pageView = PDFViewerApplication.pdfViewer.getPageView(page - 1);
-  if (pageView.stats) {
-   Stats.add(page, pageView.stats);
+  var page = e.pageNumber;
+  PDFViewerApplication.toolbar.setPageNumber(page, e.pageLabel || null);
+  PDFViewerApplication.secondaryToolbar.setPageNumber(page);
+  if (PDFViewerApplication.pdfSidebar.isThumbnailViewVisible) {
+    PDFViewerApplication.pdfThumbnailViewer.scrollThumbnailIntoView(page);
+  }
+  if (pdfjsLib.PDFJS.pdfBug && Stats.enabled) {
+    var pageView = PDFViewerApplication.pdfViewer.getPageView(page - 1);
+    if (pageView.stats) {
+      Stats.add(page, pageView.stats);
+    }
   }
- }
 }
-var zoomDisabled = false, zoomDisabledTimeout;
+var zoomDisabled = false,
+    zoomDisabledTimeout;
 function webViewerWheel(evt) {
- var pdfViewer = PDFViewerApplication.pdfViewer;
- if (pdfViewer.isInPresentationMode) {
-  return;
- }
- if (evt.ctrlKey || evt.metaKey) {
-  var support = PDFViewerApplication.supportedMouseWheelZoomModifierKeys;
-  if (evt.ctrlKey && !support.ctrlKey || evt.metaKey && !support.metaKey) {
-   return;
-  }
-  evt.preventDefault();
-  if (zoomDisabled) {
-   return;
+  var pdfViewer = PDFViewerApplication.pdfViewer;
+  if (pdfViewer.isInPresentationMode) {
+    return;
   }
-  var previousScale = pdfViewer.currentScale;
-  var delta = normalizeWheelEventDelta(evt);
-  var MOUSE_WHEEL_DELTA_PER_PAGE_SCALE = 3.0;
-  var ticks = delta * MOUSE_WHEEL_DELTA_PER_PAGE_SCALE;
-  if (ticks < 0) {
-   PDFViewerApplication.zoomOut(-ticks);
+  if (evt.ctrlKey || evt.metaKey) {
+    var support = PDFViewerApplication.supportedMouseWheelZoomModifierKeys;
+    if (evt.ctrlKey && !support.ctrlKey || evt.metaKey && !support.metaKey) {
+      return;
+    }
+    evt.preventDefault();
+    if (zoomDisabled) {
+      return;
+    }
+    var previousScale = pdfViewer.currentScale;
+    var delta = normalizeWheelEventDelta(evt);
+    var MOUSE_WHEEL_DELTA_PER_PAGE_SCALE = 3.0;
+    var ticks = delta * MOUSE_WHEEL_DELTA_PER_PAGE_SCALE;
+    if (ticks < 0) {
+      PDFViewerApplication.zoomOut(-ticks);
+    } else {
+      PDFViewerApplication.zoomIn(ticks);
+    }
+    var currentScale = pdfViewer.currentScale;
+    if (previousScale !== currentScale) {
+      var scaleCorrectionFactor = currentScale / previousScale - 1;
+      var rect = pdfViewer.container.getBoundingClientRect();
+      var dx = evt.clientX - rect.left;
+      var dy = evt.clientY - rect.top;
+      pdfViewer.container.scrollLeft += dx * scaleCorrectionFactor;
+      pdfViewer.container.scrollTop += dy * scaleCorrectionFactor;
+    }
   } else {
-   PDFViewerApplication.zoomIn(ticks);
-  }
-  var currentScale = pdfViewer.currentScale;
-  if (previousScale !== currentScale) {
-   var scaleCorrectionFactor = currentScale / previousScale - 1;
-   var rect = pdfViewer.container.getBoundingClientRect();
-   var dx = evt.clientX - rect.left;
-   var dy = evt.clientY - rect.top;
-   pdfViewer.container.scrollLeft += dx * scaleCorrectionFactor;
-   pdfViewer.container.scrollTop += dy * scaleCorrectionFactor;
+    zoomDisabled = true;
+    clearTimeout(zoomDisabledTimeout);
+    zoomDisabledTimeout = setTimeout(function () {
+      zoomDisabled = false;
+    }, 1000);
   }
- } else {
-  zoomDisabled = true;
-  clearTimeout(zoomDisabledTimeout);
-  zoomDisabledTimeout = setTimeout(function () {
-   zoomDisabled = false;
-  }, 1000);
- }
 }
 function webViewerClick(evt) {
- if (!PDFViewerApplication.secondaryToolbar.isOpen) {
-  return;
- }
- var appConfig = PDFViewerApplication.appConfig;
- if (PDFViewerApplication.pdfViewer.containsElement(evt.target) || appConfig.toolbar.container.contains(evt.target) && evt.target !== appConfig.secondaryToolbar.toggleButton) {
-  PDFViewerApplication.secondaryToolbar.close();
- }
+  if (!PDFViewerApplication.secondaryToolbar.isOpen) {
+    return;
+  }
+  var appConfig = PDFViewerApplication.appConfig;
+  if (PDFViewerApplication.pdfViewer.containsElement(evt.target) || appConfig.toolbar.container.contains(evt.target) && evt.target !== appConfig.secondaryToolbar.toggleButton) {
+    PDFViewerApplication.secondaryToolbar.close();
+  }
 }
 function webViewerKeyDown(evt) {
- if (OverlayManager.active) {
-  return;
- }
- var handled = false, ensureViewerFocused = false;
- var cmd = (evt.ctrlKey ? 1 : 0) | (evt.altKey ? 2 : 0) | (evt.shiftKey ? 4 : 0) | (evt.metaKey ? 8 : 0);
- var pdfViewer = PDFViewerApplication.pdfViewer;
- var isViewerInPresentationMode = pdfViewer && pdfViewer.isInPresentationMode;
- if (cmd === 1 || cmd === 8 || cmd === 5 || cmd === 12) {
-  switch (evt.keyCode) {
-  case 70:
-   if (!PDFViewerApplication.supportsIntegratedFind) {
-    PDFViewerApplication.findBar.open();
-    handled = true;
-   }
-   break;
-  case 71:
-   if (!PDFViewerApplication.supportsIntegratedFind) {
-    var findState = PDFViewerApplication.findController.state;
-    if (findState) {
-     PDFViewerApplication.findController.executeCommand('findagain', {
-      query: findState.query,
-      phraseSearch: findState.phraseSearch,
-      caseSensitive: findState.caseSensitive,
-      highlightAll: findState.highlightAll,
-      findPrevious: cmd === 5 || cmd === 12
-     });
+  if (OverlayManager.active) {
+    return;
+  }
+  var handled = false,
+      ensureViewerFocused = false;
+  var cmd = (evt.ctrlKey ? 1 : 0) | (evt.altKey ? 2 : 0) | (evt.shiftKey ? 4 : 0) | (evt.metaKey ? 8 : 0);
+  var pdfViewer = PDFViewerApplication.pdfViewer;
+  var isViewerInPresentationMode = pdfViewer && pdfViewer.isInPresentationMode;
+  if (cmd === 1 || cmd === 8 || cmd === 5 || cmd === 12) {
+    switch (evt.keyCode) {
+      case 70:
+        if (!PDFViewerApplication.supportsIntegratedFind) {
+          PDFViewerApplication.findBar.open();
+          handled = true;
+        }
+        break;
+      case 71:
+        if (!PDFViewerApplication.supportsIntegratedFind) {
+          var findState = PDFViewerApplication.findController.state;
+          if (findState) {
+            PDFViewerApplication.findController.executeCommand('findagain', {
+              query: findState.query,
+              phraseSearch: findState.phraseSearch,
+              caseSensitive: findState.caseSensitive,
+              highlightAll: findState.highlightAll,
+              findPrevious: cmd === 5 || cmd === 12
+            });
+          }
+          handled = true;
+        }
+        break;
+      case 61:
+      case 107:
+      case 187:
+      case 171:
+        if (!isViewerInPresentationMode) {
+          PDFViewerApplication.zoomIn();
+        }
+        handled = true;
+        break;
+      case 173:
+      case 109:
+      case 189:
+        if (!isViewerInPresentationMode) {
+          PDFViewerApplication.zoomOut();
+        }
+        handled = true;
+        break;
+      case 48:
+      case 96:
+        if (!isViewerInPresentationMode) {
+          setTimeout(function () {
+            pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
+          });
+          handled = false;
+        }
+        break;
+      case 38:
+        if (isViewerInPresentationMode || PDFViewerApplication.page > 1) {
+          PDFViewerApplication.page = 1;
+          handled = true;
+          ensureViewerFocused = true;
+        }
+        break;
+      case 40:
+        if (isViewerInPresentationMode || PDFViewerApplication.page < PDFViewerApplication.pagesCount) {
+          PDFViewerApplication.page = PDFViewerApplication.pagesCount;
+          handled = true;
+          ensureViewerFocused = true;
+        }
+        break;
     }
-    handled = true;
-   }
-   break;
-  case 61:
-  case 107:
-  case 187:
-  case 171:
-   if (!isViewerInPresentationMode) {
-    PDFViewerApplication.zoomIn();
-   }
-   handled = true;
-   break;
-  case 173:
-  case 109:
-  case 189:
-   if (!isViewerInPresentationMode) {
-    PDFViewerApplication.zoomOut();
-   }
-   handled = true;
-   break;
-  case 48:
-  case 96:
-   if (!isViewerInPresentationMode) {
-    setTimeout(function () {
-     pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
-    });
-    handled = false;
-   }
-   break;
-  case 38:
-   if (isViewerInPresentationMode || PDFViewerApplication.page > 1) {
-    PDFViewerApplication.page = 1;
-    handled = true;
-    ensureViewerFocused = true;
-   }
-   break;
-  case 40:
-   if (isViewerInPresentationMode || PDFViewerApplication.page < PDFViewerApplication.pagesCount) {
-    PDFViewerApplication.page = PDFViewerApplication.pagesCount;
-    handled = true;
-    ensureViewerFocused = true;
-   }
-   break;
   }
- }
- if (cmd === 1 || cmd === 8) {
-  switch (evt.keyCode) {
-  case 83:
-   PDFViewerApplication.download();
-   handled = true;
-   break;
+  if (cmd === 1 || cmd === 8) {
+    switch (evt.keyCode) {
+      case 83:
+        PDFViewerApplication.download();
+        handled = true;
+        break;
+    }
   }
- }
- if (cmd === 3 || cmd === 10) {
-  switch (evt.keyCode) {
-  case 80:
-   PDFViewerApplication.requestPresentationMode();
-   handled = true;
-   break;
-  case 71:
-   PDFViewerApplication.appConfig.toolbar.pageNumber.select();
-   handled = true;
-   break;
+  if (cmd === 3 || cmd === 10) {
+    switch (evt.keyCode) {
+      case 80:
+        PDFViewerApplication.requestPresentationMode();
+        handled = true;
+        break;
+      case 71:
+        PDFViewerApplication.appConfig.toolbar.pageNumber.select();
+        handled = true;
+        break;
+    }
   }
- }
- if (handled) {
-  if (ensureViewerFocused && !isViewerInPresentationMode) {
-   pdfViewer.focus();
+  if (handled) {
+    if (ensureViewerFocused && !isViewerInPresentationMode) {
+      pdfViewer.focus();
+    }
+    evt.preventDefault();
+    return;
   }
-  evt.preventDefault();
-  return;
- }
- var curElement = document.activeElement || document.querySelector(':focus');
- var curElementTagName = curElement && curElement.tagName.toUpperCase();
- if (curElementTagName === 'INPUT' || curElementTagName === 'TEXTAREA' || curElementTagName === 'SELECT') {
-  if (evt.keyCode !== 27) {
-   return;
+  var curElement = document.activeElement || document.querySelector(':focus');
+  var curElementTagName = curElement && curElement.tagName.toUpperCase();
+  if (curElementTagName === 'INPUT' || curElementTagName === 'TEXTAREA' || curElementTagName === 'SELECT') {
+    if (evt.keyCode !== 27) {
+      return;
+    }
   }
- }
- if (cmd === 0) {
-  switch (evt.keyCode) {
-  case 38:
-  case 33:
-  case 8:
-   if (!isViewerInPresentationMode && pdfViewer.currentScaleValue !== 'page-fit') {
-    break;
-   }
-  case 37:
-   if (pdfViewer.isHorizontalScrollbarEnabled) {
-    break;
-   }
-  case 75:
-  case 80:
-   if (PDFViewerApplication.page > 1) {
-    PDFViewerApplication.page--;
-   }
-   handled = true;
-   break;
-  case 27:
-   if (PDFViewerApplication.secondaryToolbar.isOpen) {
-    PDFViewerApplication.secondaryToolbar.close();
-    handled = true;
-   }
-   if (!PDFViewerApplication.supportsIntegratedFind && PDFViewerApplication.findBar.opened) {
-    PDFViewerApplication.findBar.close();
-    handled = true;
-   }
-   break;
-  case 40:
-  case 34:
-  case 32:
-   if (!isViewerInPresentationMode && pdfViewer.currentScaleValue !== 'page-fit') {
-    break;
-   }
-  case 39:
-   if (pdfViewer.isHorizontalScrollbarEnabled) {
-    break;
-   }
-  case 74:
-  case 78:
-   if (PDFViewerApplication.page < PDFViewerApplication.pagesCount) {
-    PDFViewerApplication.page++;
-   }
-   handled = true;
-   break;
-  case 36:
-   if (isViewerInPresentationMode || PDFViewerApplication.page > 1) {
-    PDFViewerApplication.page = 1;
-    handled = true;
-    ensureViewerFocused = true;
-   }
-   break;
-  case 35:
-   if (isViewerInPresentationMode || PDFViewerApplication.page < PDFViewerApplication.pagesCount) {
-    PDFViewerApplication.page = PDFViewerApplication.pagesCount;
-    handled = true;
-    ensureViewerFocused = true;
-   }
-   break;
-  case 72:
-   if (!isViewerInPresentationMode) {
-    PDFViewerApplication.handTool.toggle();
-   }
-   break;
-  case 82:
-   PDFViewerApplication.rotatePages(90);
-   break;
+  if (cmd === 0) {
+    switch (evt.keyCode) {
+      case 38:
+      case 33:
+      case 8:
+        if (!isViewerInPresentationMode && pdfViewer.currentScaleValue !== 'page-fit') {
+          break;
+        }
+      case 37:
+        if (pdfViewer.isHorizontalScrollbarEnabled) {
+          break;
+        }
+      case 75:
+      case 80:
+        if (PDFViewerApplication.page > 1) {
+          PDFViewerApplication.page--;
+        }
+        handled = true;
+        break;
+      case 27:
+        if (PDFViewerApplication.secondaryToolbar.isOpen) {
+          PDFViewerApplication.secondaryToolbar.close();
+          handled = true;
+        }
+        if (!PDFViewerApplication.supportsIntegratedFind && PDFViewerApplication.findBar.opened) {
+          PDFViewerApplication.findBar.close();
+          handled = true;
+        }
+        break;
+      case 40:
+      case 34:
+      case 32:
+        if (!isViewerInPresentationMode && pdfViewer.currentScaleValue !== 'page-fit') {
+          break;
+        }
+      case 39:
+        if (pdfViewer.isHorizontalScrollbarEnabled) {
+          break;
+        }
+      case 74:
+      case 78:
+        if (PDFViewerApplication.page < PDFViewerApplication.pagesCount) {
+          PDFViewerApplication.page++;
+        }
+        handled = true;
+        break;
+      case 36:
+        if (isViewerInPresentationMode || PDFViewerApplication.page > 1) {
+          PDFViewerApplication.page = 1;
+          handled = true;
+          ensureViewerFocused = true;
+        }
+        break;
+      case 35:
+        if (isViewerInPresentationMode || PDFViewerApplication.page < PDFViewerApplication.pagesCount) {
+          PDFViewerApplication.page = PDFViewerApplication.pagesCount;
+          handled = true;
+          ensureViewerFocused = true;
+        }
+        break;
+      case 72:
+        if (!isViewerInPresentationMode) {
+          PDFViewerApplication.handTool.toggle();
+        }
+        break;
+      case 82:
+        PDFViewerApplication.rotatePages(90);
+        break;
+    }
   }
- }
- if (cmd === 4) {
-  switch (evt.keyCode) {
-  case 32:
-   if (!isViewerInPresentationMode && pdfViewer.currentScaleValue !== 'page-fit') {
-    break;
-   }
-   if (PDFViewerApplication.page > 1) {
-    PDFViewerApplication.page--;
-   }
-   handled = true;
-   break;
-  case 82:
-   PDFViewerApplication.rotatePages(-90);
-   break;
+  if (cmd === 4) {
+    switch (evt.keyCode) {
+      case 32:
+        if (!isViewerInPresentationMode && pdfViewer.currentScaleValue !== 'page-fit') {
+          break;
+        }
+        if (PDFViewerApplication.page > 1) {
+          PDFViewerApplication.page--;
+        }
+        handled = true;
+        break;
+      case 82:
+        PDFViewerApplication.rotatePages(-90);
+        break;
+    }
+  }
+  if (!handled && !isViewerInPresentationMode) {
+    if (evt.keyCode >= 33 && evt.keyCode <= 40 || evt.keyCode === 32 && curElementTagName !== 'BUTTON') {
+      ensureViewerFocused = true;
+    }
   }
- }
- if (!handled && !isViewerInPresentationMode) {
-  if (evt.keyCode >= 33 && evt.keyCode <= 40 || evt.keyCode === 32 && curElementTagName !== 'BUTTON') {
-   ensureViewerFocused = true;
+  if (cmd === 2) {
+    switch (evt.keyCode) {
+      case 37:
+        if (isViewerInPresentationMode) {
+          PDFViewerApplication.pdfHistory.back();
+          handled = true;
+        }
+        break;
+      case 39:
+        if (isViewerInPresentationMode) {
+          PDFViewerApplication.pdfHistory.forward();
+          handled = true;
+        }
+        break;
+    }
+  }
+  if (ensureViewerFocused && !pdfViewer.containsElement(curElement)) {
+    pdfViewer.focus();
   }
- }
- if (cmd === 2) {
-  switch (evt.keyCode) {
-  case 37:
-   if (isViewerInPresentationMode) {
-    PDFViewerApplication.pdfHistory.back();
-    handled = true;
-   }
-   break;
-  case 39:
-   if (isViewerInPresentationMode) {
-    PDFViewerApplication.pdfHistory.forward();
-    handled = true;
-   }
-   break;
+  if (handled) {
+    evt.preventDefault();
   }
- }
- if (ensureViewerFocused && !pdfViewer.containsElement(curElement)) {
-  pdfViewer.focus();
- }
- if (handled) {
-  evt.preventDefault();
- }
 }
 localized.then(function webViewerLocalized() {
- document.getElementsByTagName('html')[0].dir = mozL10n.getDirection();
+  document.getElementsByTagName('html')[0].dir = mozL10n.getDirection();
 });
 var PDFPrintServiceFactory = {
- instance: {
-  supportsPrinting: false,
-  createPrintService: function () {
-   throw new Error('Not implemented: createPrintService');
+  instance: {
+    supportsPrinting: false,
+    createPrintService: function () {
+      throw new Error('Not implemented: createPrintService');
+    }
   }
- }
 };
 exports.PDFViewerApplication = PDFViewerApplication;
 exports.DefaultExernalServices = DefaultExernalServices;

+ 1 - 0
lib/web/chromecom.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var app = require('./app.js');
 var overlayManager = require('./overlay_manager.js');
 var preferences = require('./preferences.js');

+ 530 - 533
lib/web/debugger.js

@@ -13,562 +13,559 @@
  * limitations under the License.
  */
 'use strict';
+
 var FontInspector = function FontInspectorClosure() {
- var fonts;
- var active = false;
- var fontAttribute = 'data-font-name';
- function removeSelection() {
-  var divs = document.querySelectorAll('div[' + fontAttribute + ']');
-  for (var i = 0, ii = divs.length; i < ii; ++i) {
-   var div = divs[i];
-   div.className = '';
-  }
- }
- function resetSelection() {
-  var divs = document.querySelectorAll('div[' + fontAttribute + ']');
-  for (var i = 0, ii = divs.length; i < ii; ++i) {
-   var div = divs[i];
-   div.className = 'debuggerHideText';
-  }
- }
- function selectFont(fontName, show) {
-  var divs = document.querySelectorAll('div[' + fontAttribute + '=' + fontName + ']');
-  for (var i = 0, ii = divs.length; i < ii; ++i) {
-   var div = divs[i];
-   div.className = show ? 'debuggerShowText' : 'debuggerHideText';
+  var fonts;
+  var active = false;
+  var fontAttribute = 'data-font-name';
+  function removeSelection() {
+    var divs = document.querySelectorAll('div[' + fontAttribute + ']');
+    for (var i = 0, ii = divs.length; i < ii; ++i) {
+      var div = divs[i];
+      div.className = '';
+    }
   }
- }
- function textLayerClick(e) {
-  if (!e.target.dataset.fontName || e.target.tagName.toUpperCase() !== 'DIV') {
-   return;
+  function resetSelection() {
+    var divs = document.querySelectorAll('div[' + fontAttribute + ']');
+    for (var i = 0, ii = divs.length; i < ii; ++i) {
+      var div = divs[i];
+      div.className = 'debuggerHideText';
+    }
   }
-  var fontName = e.target.dataset.fontName;
-  var selects = document.getElementsByTagName('input');
-  for (var i = 0; i < selects.length; ++i) {
-   var select = selects[i];
-   if (select.dataset.fontName !== fontName) {
-    continue;
-   }
-   select.checked = !select.checked;
-   selectFont(fontName, select.checked);
-   select.scrollIntoView();
+  function selectFont(fontName, show) {
+    var divs = document.querySelectorAll('div[' + fontAttribute + '=' + fontName + ']');
+    for (var i = 0, ii = divs.length; i < ii; ++i) {
+      var div = divs[i];
+      div.className = show ? 'debuggerShowText' : 'debuggerHideText';
+    }
   }
- }
- return {
-  id: 'FontInspector',
-  name: 'Font Inspector',
-  panel: null,
-  manager: null,
-  init: function init(pdfjsLib) {
-   var panel = this.panel;
-   panel.setAttribute('style', 'padding: 5px;');
-   var tmp = document.createElement('button');
-   tmp.addEventListener('click', resetSelection);
-   tmp.textContent = 'Refresh';
-   panel.appendChild(tmp);
-   fonts = document.createElement('div');
-   panel.appendChild(fonts);
-  },
-  cleanup: function cleanup() {
-   fonts.textContent = '';
-  },
-  enabled: false,
-  get active() {
-   return active;
-  },
-  set active(value) {
-   active = value;
-   if (active) {
-    document.body.addEventListener('click', textLayerClick, true);
-    resetSelection();
-   } else {
-    document.body.removeEventListener('click', textLayerClick, true);
-    removeSelection();
-   }
-  },
-  fontAdded: function fontAdded(fontObj, url) {
-   function properties(obj, list) {
-    var moreInfo = document.createElement('table');
-    for (var i = 0; i < list.length; i++) {
-     var tr = document.createElement('tr');
-     var td1 = document.createElement('td');
-     td1.textContent = list[i];
-     tr.appendChild(td1);
-     var td2 = document.createElement('td');
-     td2.textContent = obj[list[i]].toString();
-     tr.appendChild(td2);
-     moreInfo.appendChild(tr);
+  function textLayerClick(e) {
+    if (!e.target.dataset.fontName || e.target.tagName.toUpperCase() !== 'DIV') {
+      return;
     }
-    return moreInfo;
-   }
-   var moreInfo = properties(fontObj, [
-    'name',
-    'type'
-   ]);
-   var fontName = fontObj.loadedName;
-   var font = document.createElement('div');
-   var name = document.createElement('span');
-   name.textContent = fontName;
-   var download = document.createElement('a');
-   if (url) {
-    url = /url\(['"]?([^\)"']+)/.exec(url);
-    download.href = url[1];
-   } else if (fontObj.data) {
-    url = URL.createObjectURL(new Blob([fontObj.data], { type: fontObj.mimeType }));
-    download.href = url;
-   }
-   download.textContent = 'Download';
-   var logIt = document.createElement('a');
-   logIt.href = '';
-   logIt.textContent = 'Log';
-   logIt.addEventListener('click', function (event) {
-    event.preventDefault();
-    console.log(fontObj);
-   });
-   var select = document.createElement('input');
-   select.setAttribute('type', 'checkbox');
-   select.dataset.fontName = fontName;
-   select.addEventListener('click', function (select, fontName) {
-    return function () {
-     selectFont(fontName, select.checked);
-    };
-   }(select, fontName));
-   font.appendChild(select);
-   font.appendChild(name);
-   font.appendChild(document.createTextNode(' '));
-   font.appendChild(download);
-   font.appendChild(document.createTextNode(' '));
-   font.appendChild(logIt);
-   font.appendChild(moreInfo);
-   fonts.appendChild(font);
-   setTimeout(function () {
-    if (this.active) {
-     resetSelection();
+    var fontName = e.target.dataset.fontName;
+    var selects = document.getElementsByTagName('input');
+    for (var i = 0; i < selects.length; ++i) {
+      var select = selects[i];
+      if (select.dataset.fontName !== fontName) {
+        continue;
+      }
+      select.checked = !select.checked;
+      selectFont(fontName, select.checked);
+      select.scrollIntoView();
     }
-   }.bind(this), 2000);
   }
- };
+  return {
+    id: 'FontInspector',
+    name: 'Font Inspector',
+    panel: null,
+    manager: null,
+    init: function init(pdfjsLib) {
+      var panel = this.panel;
+      panel.setAttribute('style', 'padding: 5px;');
+      var tmp = document.createElement('button');
+      tmp.addEventListener('click', resetSelection);
+      tmp.textContent = 'Refresh';
+      panel.appendChild(tmp);
+      fonts = document.createElement('div');
+      panel.appendChild(fonts);
+    },
+    cleanup: function cleanup() {
+      fonts.textContent = '';
+    },
+    enabled: false,
+    get active() {
+      return active;
+    },
+    set active(value) {
+      active = value;
+      if (active) {
+        document.body.addEventListener('click', textLayerClick, true);
+        resetSelection();
+      } else {
+        document.body.removeEventListener('click', textLayerClick, true);
+        removeSelection();
+      }
+    },
+    fontAdded: function fontAdded(fontObj, url) {
+      function properties(obj, list) {
+        var moreInfo = document.createElement('table');
+        for (var i = 0; i < list.length; i++) {
+          var tr = document.createElement('tr');
+          var td1 = document.createElement('td');
+          td1.textContent = list[i];
+          tr.appendChild(td1);
+          var td2 = document.createElement('td');
+          td2.textContent = obj[list[i]].toString();
+          tr.appendChild(td2);
+          moreInfo.appendChild(tr);
+        }
+        return moreInfo;
+      }
+      var moreInfo = properties(fontObj, ['name', 'type']);
+      var fontName = fontObj.loadedName;
+      var font = document.createElement('div');
+      var name = document.createElement('span');
+      name.textContent = fontName;
+      var download = document.createElement('a');
+      if (url) {
+        url = /url\(['"]?([^\)"']+)/.exec(url);
+        download.href = url[1];
+      } else if (fontObj.data) {
+        url = URL.createObjectURL(new Blob([fontObj.data], { type: fontObj.mimeType }));
+        download.href = url;
+      }
+      download.textContent = 'Download';
+      var logIt = document.createElement('a');
+      logIt.href = '';
+      logIt.textContent = 'Log';
+      logIt.addEventListener('click', function (event) {
+        event.preventDefault();
+        console.log(fontObj);
+      });
+      var select = document.createElement('input');
+      select.setAttribute('type', 'checkbox');
+      select.dataset.fontName = fontName;
+      select.addEventListener('click', function (select, fontName) {
+        return function () {
+          selectFont(fontName, select.checked);
+        };
+      }(select, fontName));
+      font.appendChild(select);
+      font.appendChild(name);
+      font.appendChild(document.createTextNode(' '));
+      font.appendChild(download);
+      font.appendChild(document.createTextNode(' '));
+      font.appendChild(logIt);
+      font.appendChild(moreInfo);
+      fonts.appendChild(font);
+      setTimeout(function () {
+        if (this.active) {
+          resetSelection();
+        }
+      }.bind(this), 2000);
+    }
+  };
 }();
 var opMap;
 var StepperManager = function StepperManagerClosure() {
- var steppers = [];
- var stepperDiv = null;
- var stepperControls = null;
- var stepperChooser = null;
- var breakPoints = Object.create(null);
- return {
-  id: 'Stepper',
-  name: 'Stepper',
-  panel: null,
-  manager: null,
-  init: function init(pdfjsLib) {
-   var self = this;
-   this.panel.setAttribute('style', 'padding: 5px;');
-   stepperControls = document.createElement('div');
-   stepperChooser = document.createElement('select');
-   stepperChooser.addEventListener('change', function (event) {
-    self.selectStepper(this.value);
-   });
-   stepperControls.appendChild(stepperChooser);
-   stepperDiv = document.createElement('div');
-   this.panel.appendChild(stepperControls);
-   this.panel.appendChild(stepperDiv);
-   if (sessionStorage.getItem('pdfjsBreakPoints')) {
-    breakPoints = JSON.parse(sessionStorage.getItem('pdfjsBreakPoints'));
-   }
-   opMap = Object.create(null);
-   for (var key in pdfjsLib.OPS) {
-    opMap[pdfjsLib.OPS[key]] = key;
-   }
-  },
-  cleanup: function cleanup() {
-   stepperChooser.textContent = '';
-   stepperDiv.textContent = '';
-   steppers = [];
-  },
-  enabled: false,
-  active: false,
-  create: function create(pageIndex) {
-   var debug = document.createElement('div');
-   debug.id = 'stepper' + pageIndex;
-   debug.setAttribute('hidden', true);
-   debug.className = 'stepper';
-   stepperDiv.appendChild(debug);
-   var b = document.createElement('option');
-   b.textContent = 'Page ' + (pageIndex + 1);
-   b.value = pageIndex;
-   stepperChooser.appendChild(b);
-   var initBreakPoints = breakPoints[pageIndex] || [];
-   var stepper = new Stepper(debug, pageIndex, initBreakPoints);
-   steppers.push(stepper);
-   if (steppers.length === 1) {
-    this.selectStepper(pageIndex, false);
-   }
-   return stepper;
-  },
-  selectStepper: function selectStepper(pageIndex, selectPanel) {
-   var i;
-   pageIndex = pageIndex | 0;
-   if (selectPanel) {
-    this.manager.selectPanel(this);
-   }
-   for (i = 0; i < steppers.length; ++i) {
-    var stepper = steppers[i];
-    if (stepper.pageIndex === pageIndex) {
-     stepper.panel.removeAttribute('hidden');
-    } else {
-     stepper.panel.setAttribute('hidden', true);
+  var steppers = [];
+  var stepperDiv = null;
+  var stepperControls = null;
+  var stepperChooser = null;
+  var breakPoints = Object.create(null);
+  return {
+    id: 'Stepper',
+    name: 'Stepper',
+    panel: null,
+    manager: null,
+    init: function init(pdfjsLib) {
+      var self = this;
+      this.panel.setAttribute('style', 'padding: 5px;');
+      stepperControls = document.createElement('div');
+      stepperChooser = document.createElement('select');
+      stepperChooser.addEventListener('change', function (event) {
+        self.selectStepper(this.value);
+      });
+      stepperControls.appendChild(stepperChooser);
+      stepperDiv = document.createElement('div');
+      this.panel.appendChild(stepperControls);
+      this.panel.appendChild(stepperDiv);
+      if (sessionStorage.getItem('pdfjsBreakPoints')) {
+        breakPoints = JSON.parse(sessionStorage.getItem('pdfjsBreakPoints'));
+      }
+      opMap = Object.create(null);
+      for (var key in pdfjsLib.OPS) {
+        opMap[pdfjsLib.OPS[key]] = key;
+      }
+    },
+    cleanup: function cleanup() {
+      stepperChooser.textContent = '';
+      stepperDiv.textContent = '';
+      steppers = [];
+    },
+    enabled: false,
+    active: false,
+    create: function create(pageIndex) {
+      var debug = document.createElement('div');
+      debug.id = 'stepper' + pageIndex;
+      debug.setAttribute('hidden', true);
+      debug.className = 'stepper';
+      stepperDiv.appendChild(debug);
+      var b = document.createElement('option');
+      b.textContent = 'Page ' + (pageIndex + 1);
+      b.value = pageIndex;
+      stepperChooser.appendChild(b);
+      var initBreakPoints = breakPoints[pageIndex] || [];
+      var stepper = new Stepper(debug, pageIndex, initBreakPoints);
+      steppers.push(stepper);
+      if (steppers.length === 1) {
+        this.selectStepper(pageIndex, false);
+      }
+      return stepper;
+    },
+    selectStepper: function selectStepper(pageIndex, selectPanel) {
+      var i;
+      pageIndex = pageIndex | 0;
+      if (selectPanel) {
+        this.manager.selectPanel(this);
+      }
+      for (i = 0; i < steppers.length; ++i) {
+        var stepper = steppers[i];
+        if (stepper.pageIndex === pageIndex) {
+          stepper.panel.removeAttribute('hidden');
+        } else {
+          stepper.panel.setAttribute('hidden', true);
+        }
+      }
+      var options = stepperChooser.options;
+      for (i = 0; i < options.length; ++i) {
+        var option = options[i];
+        option.selected = (option.value | 0) === pageIndex;
+      }
+    },
+    saveBreakPoints: function saveBreakPoints(pageIndex, bps) {
+      breakPoints[pageIndex] = bps;
+      sessionStorage.setItem('pdfjsBreakPoints', JSON.stringify(breakPoints));
     }
-   }
-   var options = stepperChooser.options;
-   for (i = 0; i < options.length; ++i) {
-    var option = options[i];
-    option.selected = (option.value | 0) === pageIndex;
-   }
-  },
-  saveBreakPoints: function saveBreakPoints(pageIndex, bps) {
-   breakPoints[pageIndex] = bps;
-   sessionStorage.setItem('pdfjsBreakPoints', JSON.stringify(breakPoints));
-  }
- };
+  };
 }();
 var Stepper = function StepperClosure() {
- function c(tag, textContent) {
-  var d = document.createElement(tag);
-  if (textContent) {
-   d.textContent = textContent;
-  }
-  return d;
- }
- function simplifyArgs(args) {
-  if (typeof args === 'string') {
-   var MAX_STRING_LENGTH = 75;
-   return args.length <= MAX_STRING_LENGTH ? args : args.substr(0, MAX_STRING_LENGTH) + '...';
-  }
-  if (typeof args !== 'object' || args === null) {
-   return args;
-  }
-  if ('length' in args) {
-   var simpleArgs = [], i, ii;
-   var MAX_ITEMS = 10;
-   for (i = 0, ii = Math.min(MAX_ITEMS, args.length); i < ii; i++) {
-    simpleArgs.push(simplifyArgs(args[i]));
-   }
-   if (i < args.length) {
-    simpleArgs.push('...');
-   }
-   return simpleArgs;
-  }
-  var simpleObj = {};
-  for (var key in args) {
-   simpleObj[key] = simplifyArgs(args[key]);
-  }
-  return simpleObj;
- }
- function Stepper(panel, pageIndex, initialBreakPoints) {
-  this.panel = panel;
-  this.breakPoint = 0;
-  this.nextBreakPoint = null;
-  this.pageIndex = pageIndex;
-  this.breakPoints = initialBreakPoints;
-  this.currentIdx = -1;
-  this.operatorListIdx = 0;
- }
- Stepper.prototype = {
-  init: function init(operatorList) {
-   var panel = this.panel;
-   var content = c('div', 'c=continue, s=step');
-   var table = c('table');
-   content.appendChild(table);
-   table.cellSpacing = 0;
-   var headerRow = c('tr');
-   table.appendChild(headerRow);
-   headerRow.appendChild(c('th', 'Break'));
-   headerRow.appendChild(c('th', 'Idx'));
-   headerRow.appendChild(c('th', 'fn'));
-   headerRow.appendChild(c('th', 'args'));
-   panel.appendChild(content);
-   this.table = table;
-   this.updateOperatorList(operatorList);
-  },
-  updateOperatorList: function updateOperatorList(operatorList) {
-   var self = this;
-   function cboxOnClick() {
-    var x = +this.dataset.idx;
-    if (this.checked) {
-     self.breakPoints.push(x);
-    } else {
-     self.breakPoints.splice(self.breakPoints.indexOf(x), 1);
+  function c(tag, textContent) {
+    var d = document.createElement(tag);
+    if (textContent) {
+      d.textContent = textContent;
     }
-    StepperManager.saveBreakPoints(self.pageIndex, self.breakPoints);
-   }
-   var MAX_OPERATORS_COUNT = 15000;
-   if (this.operatorListIdx > MAX_OPERATORS_COUNT) {
-    return;
-   }
-   var chunk = document.createDocumentFragment();
-   var operatorsToDisplay = Math.min(MAX_OPERATORS_COUNT, operatorList.fnArray.length);
-   for (var i = this.operatorListIdx; i < operatorsToDisplay; i++) {
-    var line = c('tr');
-    line.className = 'line';
-    line.dataset.idx = i;
-    chunk.appendChild(line);
-    var checked = this.breakPoints.indexOf(i) !== -1;
-    var args = operatorList.argsArray[i] || [];
-    var breakCell = c('td');
-    var cbox = c('input');
-    cbox.type = 'checkbox';
-    cbox.className = 'points';
-    cbox.checked = checked;
-    cbox.dataset.idx = i;
-    cbox.onclick = cboxOnClick;
-    breakCell.appendChild(cbox);
-    line.appendChild(breakCell);
-    line.appendChild(c('td', i.toString()));
-    var fn = opMap[operatorList.fnArray[i]];
-    var decArgs = args;
-    if (fn === 'showText') {
-     var glyphs = args[0];
-     var newArgs = [];
-     var str = [];
-     for (var j = 0; j < glyphs.length; j++) {
-      var glyph = glyphs[j];
-      if (typeof glyph === 'object' && glyph !== null) {
-       str.push(glyph.fontChar);
-      } else {
-       if (str.length > 0) {
-        newArgs.push(str.join(''));
-        str = [];
-       }
-       newArgs.push(glyph);
-      }
-     }
-     if (str.length > 0) {
-      newArgs.push(str.join(''));
-     }
-     decArgs = [newArgs];
+    return d;
+  }
+  function simplifyArgs(args) {
+    if (typeof args === 'string') {
+      var MAX_STRING_LENGTH = 75;
+      return args.length <= MAX_STRING_LENGTH ? args : args.substr(0, MAX_STRING_LENGTH) + '...';
     }
-    line.appendChild(c('td', fn));
-    line.appendChild(c('td', JSON.stringify(simplifyArgs(decArgs))));
-   }
-   if (operatorsToDisplay < operatorList.fnArray.length) {
-    line = c('tr');
-    var lastCell = c('td', '...');
-    lastCell.colspan = 4;
-    chunk.appendChild(lastCell);
-   }
-   this.operatorListIdx = operatorList.fnArray.length;
-   this.table.appendChild(chunk);
-  },
-  getNextBreakPoint: function getNextBreakPoint() {
-   this.breakPoints.sort(function (a, b) {
-    return a - b;
-   });
-   for (var i = 0; i < this.breakPoints.length; i++) {
-    if (this.breakPoints[i] > this.currentIdx) {
-     return this.breakPoints[i];
+    if (typeof args !== 'object' || args === null) {
+      return args;
     }
-   }
-   return null;
-  },
-  breakIt: function breakIt(idx, callback) {
-   StepperManager.selectStepper(this.pageIndex, true);
-   var self = this;
-   var dom = document;
-   self.currentIdx = idx;
-   var listener = function (e) {
-    switch (e.keyCode) {
-    case 83:
-     dom.removeEventListener('keydown', listener);
-     self.nextBreakPoint = self.currentIdx + 1;
-     self.goTo(-1);
-     callback();
-     break;
-    case 67:
-     dom.removeEventListener('keydown', listener);
-     var breakPoint = self.getNextBreakPoint();
-     self.nextBreakPoint = breakPoint;
-     self.goTo(-1);
-     callback();
-     break;
+    if ('length' in args) {
+      var simpleArgs = [],
+          i,
+          ii;
+      var MAX_ITEMS = 10;
+      for (i = 0, ii = Math.min(MAX_ITEMS, args.length); i < ii; i++) {
+        simpleArgs.push(simplifyArgs(args[i]));
+      }
+      if (i < args.length) {
+        simpleArgs.push('...');
+      }
+      return simpleArgs;
     }
-   };
-   dom.addEventListener('keydown', listener);
-   self.goTo(idx);
-  },
-  goTo: function goTo(idx) {
-   var allRows = this.panel.getElementsByClassName('line');
-   for (var x = 0, xx = allRows.length; x < xx; ++x) {
-    var row = allRows[x];
-    if ((row.dataset.idx | 0) === idx) {
-     row.style.backgroundColor = 'rgb(251,250,207)';
-     row.scrollIntoView();
-    } else {
-     row.style.backgroundColor = null;
+    var simpleObj = {};
+    for (var key in args) {
+      simpleObj[key] = simplifyArgs(args[key]);
     }
-   }
+    return simpleObj;
   }
- };
- return Stepper;
-}();
-var Stats = function Stats() {
- var stats = [];
- function clear(node) {
-  while (node.hasChildNodes()) {
-   node.removeChild(node.lastChild);
-  }
- }
- function getStatIndex(pageNumber) {
-  for (var i = 0, ii = stats.length; i < ii; ++i) {
-   if (stats[i].pageNumber === pageNumber) {
-    return i;
-   }
-  }
-  return false;
- }
- return {
-  id: 'Stats',
-  name: 'Stats',
-  panel: null,
-  manager: null,
-  init: function init(pdfjsLib) {
-   this.panel.setAttribute('style', 'padding: 5px;');
-   pdfjsLib.PDFJS.enableStats = true;
-  },
-  enabled: false,
-  active: false,
-  add: function (pageNumber, stat) {
-   if (!stat) {
-    return;
-   }
-   var statsIndex = getStatIndex(pageNumber);
-   if (statsIndex !== false) {
-    var b = stats[statsIndex];
-    this.panel.removeChild(b.div);
-    stats.splice(statsIndex, 1);
-   }
-   var wrapper = document.createElement('div');
-   wrapper.className = 'stats';
-   var title = document.createElement('div');
-   title.className = 'title';
-   title.textContent = 'Page: ' + pageNumber;
-   var statsDiv = document.createElement('div');
-   statsDiv.textContent = stat.toString();
-   wrapper.appendChild(title);
-   wrapper.appendChild(statsDiv);
-   stats.push({
-    pageNumber: pageNumber,
-    div: wrapper
-   });
-   stats.sort(function (a, b) {
-    return a.pageNumber - b.pageNumber;
-   });
-   clear(this.panel);
-   for (var i = 0, ii = stats.length; i < ii; ++i) {
-    this.panel.appendChild(stats[i].div);
-   }
-  },
-  cleanup: function () {
-   stats = [];
-   clear(this.panel);
+  function Stepper(panel, pageIndex, initialBreakPoints) {
+    this.panel = panel;
+    this.breakPoint = 0;
+    this.nextBreakPoint = null;
+    this.pageIndex = pageIndex;
+    this.breakPoints = initialBreakPoints;
+    this.currentIdx = -1;
+    this.operatorListIdx = 0;
   }
- };
+  Stepper.prototype = {
+    init: function init(operatorList) {
+      var panel = this.panel;
+      var content = c('div', 'c=continue, s=step');
+      var table = c('table');
+      content.appendChild(table);
+      table.cellSpacing = 0;
+      var headerRow = c('tr');
+      table.appendChild(headerRow);
+      headerRow.appendChild(c('th', 'Break'));
+      headerRow.appendChild(c('th', 'Idx'));
+      headerRow.appendChild(c('th', 'fn'));
+      headerRow.appendChild(c('th', 'args'));
+      panel.appendChild(content);
+      this.table = table;
+      this.updateOperatorList(operatorList);
+    },
+    updateOperatorList: function updateOperatorList(operatorList) {
+      var self = this;
+      function cboxOnClick() {
+        var x = +this.dataset.idx;
+        if (this.checked) {
+          self.breakPoints.push(x);
+        } else {
+          self.breakPoints.splice(self.breakPoints.indexOf(x), 1);
+        }
+        StepperManager.saveBreakPoints(self.pageIndex, self.breakPoints);
+      }
+      var MAX_OPERATORS_COUNT = 15000;
+      if (this.operatorListIdx > MAX_OPERATORS_COUNT) {
+        return;
+      }
+      var chunk = document.createDocumentFragment();
+      var operatorsToDisplay = Math.min(MAX_OPERATORS_COUNT, operatorList.fnArray.length);
+      for (var i = this.operatorListIdx; i < operatorsToDisplay; i++) {
+        var line = c('tr');
+        line.className = 'line';
+        line.dataset.idx = i;
+        chunk.appendChild(line);
+        var checked = this.breakPoints.indexOf(i) !== -1;
+        var args = operatorList.argsArray[i] || [];
+        var breakCell = c('td');
+        var cbox = c('input');
+        cbox.type = 'checkbox';
+        cbox.className = 'points';
+        cbox.checked = checked;
+        cbox.dataset.idx = i;
+        cbox.onclick = cboxOnClick;
+        breakCell.appendChild(cbox);
+        line.appendChild(breakCell);
+        line.appendChild(c('td', i.toString()));
+        var fn = opMap[operatorList.fnArray[i]];
+        var decArgs = args;
+        if (fn === 'showText') {
+          var glyphs = args[0];
+          var newArgs = [];
+          var str = [];
+          for (var j = 0; j < glyphs.length; j++) {
+            var glyph = glyphs[j];
+            if (typeof glyph === 'object' && glyph !== null) {
+              str.push(glyph.fontChar);
+            } else {
+              if (str.length > 0) {
+                newArgs.push(str.join(''));
+                str = [];
+              }
+              newArgs.push(glyph);
+            }
+          }
+          if (str.length > 0) {
+            newArgs.push(str.join(''));
+          }
+          decArgs = [newArgs];
+        }
+        line.appendChild(c('td', fn));
+        line.appendChild(c('td', JSON.stringify(simplifyArgs(decArgs))));
+      }
+      if (operatorsToDisplay < operatorList.fnArray.length) {
+        line = c('tr');
+        var lastCell = c('td', '...');
+        lastCell.colspan = 4;
+        chunk.appendChild(lastCell);
+      }
+      this.operatorListIdx = operatorList.fnArray.length;
+      this.table.appendChild(chunk);
+    },
+    getNextBreakPoint: function getNextBreakPoint() {
+      this.breakPoints.sort(function (a, b) {
+        return a - b;
+      });
+      for (var i = 0; i < this.breakPoints.length; i++) {
+        if (this.breakPoints[i] > this.currentIdx) {
+          return this.breakPoints[i];
+        }
+      }
+      return null;
+    },
+    breakIt: function breakIt(idx, callback) {
+      StepperManager.selectStepper(this.pageIndex, true);
+      var self = this;
+      var dom = document;
+      self.currentIdx = idx;
+      var listener = function (e) {
+        switch (e.keyCode) {
+          case 83:
+            dom.removeEventListener('keydown', listener);
+            self.nextBreakPoint = self.currentIdx + 1;
+            self.goTo(-1);
+            callback();
+            break;
+          case 67:
+            dom.removeEventListener('keydown', listener);
+            var breakPoint = self.getNextBreakPoint();
+            self.nextBreakPoint = breakPoint;
+            self.goTo(-1);
+            callback();
+            break;
+        }
+      };
+      dom.addEventListener('keydown', listener);
+      self.goTo(idx);
+    },
+    goTo: function goTo(idx) {
+      var allRows = this.panel.getElementsByClassName('line');
+      for (var x = 0, xx = allRows.length; x < xx; ++x) {
+        var row = allRows[x];
+        if ((row.dataset.idx | 0) === idx) {
+          row.style.backgroundColor = 'rgb(251,250,207)';
+          row.scrollIntoView();
+        } else {
+          row.style.backgroundColor = null;
+        }
+      }
+    }
+  };
+  return Stepper;
 }();
-var PDFBug = function PDFBugClosure() {
- var panelWidth = 300;
- var buttons = [];
- var activePanel = null;
- return {
-  tools: [
-   FontInspector,
-   StepperManager,
-   Stats
-  ],
-  enable: function (ids) {
-   var all = false, tools = this.tools;
-   if (ids.length === 1 && ids[0] === 'all') {
-    all = true;
-   }
-   for (var i = 0; i < tools.length; ++i) {
-    var tool = tools[i];
-    if (all || ids.indexOf(tool.id) !== -1) {
-     tool.enabled = true;
+var Stats = function Stats() {
+  var stats = [];
+  function clear(node) {
+    while (node.hasChildNodes()) {
+      node.removeChild(node.lastChild);
     }
-   }
-   if (!all) {
-    tools.sort(function (a, b) {
-     var indexA = ids.indexOf(a.id);
-     indexA = indexA < 0 ? tools.length : indexA;
-     var indexB = ids.indexOf(b.id);
-     indexB = indexB < 0 ? tools.length : indexB;
-     return indexA - indexB;
-    });
-   }
-  },
-  init: function init(pdfjsLib, container) {
-   var ui = document.createElement('div');
-   ui.id = 'PDFBug';
-   var controls = document.createElement('div');
-   controls.setAttribute('class', 'controls');
-   ui.appendChild(controls);
-   var panels = document.createElement('div');
-   panels.setAttribute('class', 'panels');
-   ui.appendChild(panels);
-   container.appendChild(ui);
-   container.style.right = panelWidth + 'px';
-   var tools = this.tools;
-   var self = this;
-   for (var i = 0; i < tools.length; ++i) {
-    var tool = tools[i];
-    var panel = document.createElement('div');
-    var panelButton = document.createElement('button');
-    panelButton.textContent = tool.name;
-    panelButton.addEventListener('click', function (selected) {
-     return function (event) {
-      event.preventDefault();
-      self.selectPanel(selected);
-     };
-    }(i));
-    controls.appendChild(panelButton);
-    panels.appendChild(panel);
-    tool.panel = panel;
-    tool.manager = this;
-    if (tool.enabled) {
-     tool.init(pdfjsLib);
-    } else {
-     panel.textContent = tool.name + ' is disabled. To enable add ' + ' "' + tool.id + '" to the pdfBug parameter ' + 'and refresh (separate multiple by commas).';
+  }
+  function getStatIndex(pageNumber) {
+    for (var i = 0, ii = stats.length; i < ii; ++i) {
+      if (stats[i].pageNumber === pageNumber) {
+        return i;
+      }
     }
-    buttons.push(panelButton);
-   }
-   this.selectPanel(0);
-  },
-  cleanup: function cleanup() {
-   for (var i = 0, ii = this.tools.length; i < ii; i++) {
-    if (this.tools[i].enabled) {
-     this.tools[i].cleanup();
+    return false;
+  }
+  return {
+    id: 'Stats',
+    name: 'Stats',
+    panel: null,
+    manager: null,
+    init: function init(pdfjsLib) {
+      this.panel.setAttribute('style', 'padding: 5px;');
+      pdfjsLib.PDFJS.enableStats = true;
+    },
+    enabled: false,
+    active: false,
+    add: function (pageNumber, stat) {
+      if (!stat) {
+        return;
+      }
+      var statsIndex = getStatIndex(pageNumber);
+      if (statsIndex !== false) {
+        var b = stats[statsIndex];
+        this.panel.removeChild(b.div);
+        stats.splice(statsIndex, 1);
+      }
+      var wrapper = document.createElement('div');
+      wrapper.className = 'stats';
+      var title = document.createElement('div');
+      title.className = 'title';
+      title.textContent = 'Page: ' + pageNumber;
+      var statsDiv = document.createElement('div');
+      statsDiv.textContent = stat.toString();
+      wrapper.appendChild(title);
+      wrapper.appendChild(statsDiv);
+      stats.push({
+        pageNumber: pageNumber,
+        div: wrapper
+      });
+      stats.sort(function (a, b) {
+        return a.pageNumber - b.pageNumber;
+      });
+      clear(this.panel);
+      for (var i = 0, ii = stats.length; i < ii; ++i) {
+        this.panel.appendChild(stats[i].div);
+      }
+    },
+    cleanup: function () {
+      stats = [];
+      clear(this.panel);
     }
-   }
-  },
-  selectPanel: function selectPanel(index) {
-   if (typeof index !== 'number') {
-    index = this.tools.indexOf(index);
-   }
-   if (index === activePanel) {
-    return;
-   }
-   activePanel = index;
-   var tools = this.tools;
-   for (var j = 0; j < tools.length; ++j) {
-    if (j === index) {
-     buttons[j].setAttribute('class', 'active');
-     tools[j].active = true;
-     tools[j].panel.removeAttribute('hidden');
-    } else {
-     buttons[j].setAttribute('class', '');
-     tools[j].active = false;
-     tools[j].panel.setAttribute('hidden', 'true');
+  };
+}();
+window.PDFBug = function PDFBugClosure() {
+  var panelWidth = 300;
+  var buttons = [];
+  var activePanel = null;
+  return {
+    tools: [FontInspector, StepperManager, Stats],
+    enable: function (ids) {
+      var all = false,
+          tools = this.tools;
+      if (ids.length === 1 && ids[0] === 'all') {
+        all = true;
+      }
+      for (var i = 0; i < tools.length; ++i) {
+        var tool = tools[i];
+        if (all || ids.indexOf(tool.id) !== -1) {
+          tool.enabled = true;
+        }
+      }
+      if (!all) {
+        tools.sort(function (a, b) {
+          var indexA = ids.indexOf(a.id);
+          indexA = indexA < 0 ? tools.length : indexA;
+          var indexB = ids.indexOf(b.id);
+          indexB = indexB < 0 ? tools.length : indexB;
+          return indexA - indexB;
+        });
+      }
+    },
+    init: function init(pdfjsLib, container) {
+      var ui = document.createElement('div');
+      ui.id = 'PDFBug';
+      var controls = document.createElement('div');
+      controls.setAttribute('class', 'controls');
+      ui.appendChild(controls);
+      var panels = document.createElement('div');
+      panels.setAttribute('class', 'panels');
+      ui.appendChild(panels);
+      container.appendChild(ui);
+      container.style.right = panelWidth + 'px';
+      var tools = this.tools;
+      var self = this;
+      for (var i = 0; i < tools.length; ++i) {
+        var tool = tools[i];
+        var panel = document.createElement('div');
+        var panelButton = document.createElement('button');
+        panelButton.textContent = tool.name;
+        panelButton.addEventListener('click', function (selected) {
+          return function (event) {
+            event.preventDefault();
+            self.selectPanel(selected);
+          };
+        }(i));
+        controls.appendChild(panelButton);
+        panels.appendChild(panel);
+        tool.panel = panel;
+        tool.manager = this;
+        if (tool.enabled) {
+          tool.init(pdfjsLib);
+        } else {
+          panel.textContent = tool.name + ' is disabled. To enable add ' + ' "' + tool.id + '" to the pdfBug parameter ' + 'and refresh (separate multiple by commas).';
+        }
+        buttons.push(panelButton);
+      }
+      this.selectPanel(0);
+    },
+    cleanup: function cleanup() {
+      for (var i = 0, ii = this.tools.length; i < ii; i++) {
+        if (this.tools[i].enabled) {
+          this.tools[i].cleanup();
+        }
+      }
+    },
+    selectPanel: function selectPanel(index) {
+      if (typeof index !== 'number') {
+        index = this.tools.indexOf(index);
+      }
+      if (index === activePanel) {
+        return;
+      }
+      activePanel = index;
+      var tools = this.tools;
+      for (var j = 0; j < tools.length; ++j) {
+        if (j === index) {
+          buttons[j].setAttribute('class', 'active');
+          tools[j].active = true;
+          tools[j].panel.removeAttribute('hidden');
+        } else {
+          buttons[j].setAttribute('class', '');
+          tools[j].active = false;
+          tools[j].panel.setAttribute('hidden', 'true');
+        }
+      }
     }
-   }
-  }
- };
+  };
 }();

+ 97 - 96
lib/web/dom_events.js

@@ -13,112 +13,113 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtils = require('./ui_utils.js');
 var EventBus = uiUtils.EventBus;
 function attachDOMEventsToEventBus(eventBus) {
- eventBus.on('documentload', function () {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('documentload', true, true, {});
-  window.dispatchEvent(event);
- });
- eventBus.on('pagerendered', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('pagerendered', true, true, {
-   pageNumber: e.pageNumber,
-   cssTransform: e.cssTransform
+  eventBus.on('documentload', function () {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('documentload', true, true, {});
+    window.dispatchEvent(event);
   });
-  e.source.div.dispatchEvent(event);
- });
- eventBus.on('textlayerrendered', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('textlayerrendered', true, true, { pageNumber: e.pageNumber });
-  e.source.textLayerDiv.dispatchEvent(event);
- });
- eventBus.on('pagechange', function (e) {
-  var event = document.createEvent('UIEvents');
-  event.initUIEvent('pagechange', true, true, window, 0);
-  event.pageNumber = e.pageNumber;
-  e.source.container.dispatchEvent(event);
- });
- eventBus.on('pagesinit', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('pagesinit', true, true, null);
-  e.source.container.dispatchEvent(event);
- });
- eventBus.on('pagesloaded', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('pagesloaded', true, true, { pagesCount: e.pagesCount });
-  e.source.container.dispatchEvent(event);
- });
- eventBus.on('scalechange', function (e) {
-  var event = document.createEvent('UIEvents');
-  event.initUIEvent('scalechange', true, true, window, 0);
-  event.scale = e.scale;
-  event.presetValue = e.presetValue;
-  e.source.container.dispatchEvent(event);
- });
- eventBus.on('updateviewarea', function (e) {
-  var event = document.createEvent('UIEvents');
-  event.initUIEvent('updateviewarea', true, true, window, 0);
-  event.location = e.location;
-  e.source.container.dispatchEvent(event);
- });
- eventBus.on('find', function (e) {
-  if (e.source === window) {
-   return;
-  }
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('find' + e.type, true, true, {
-   query: e.query,
-   phraseSearch: e.phraseSearch,
-   caseSensitive: e.caseSensitive,
-   highlightAll: e.highlightAll,
-   findPrevious: e.findPrevious
+  eventBus.on('pagerendered', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('pagerendered', true, true, {
+      pageNumber: e.pageNumber,
+      cssTransform: e.cssTransform
+    });
+    e.source.div.dispatchEvent(event);
+  });
+  eventBus.on('textlayerrendered', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('textlayerrendered', true, true, { pageNumber: e.pageNumber });
+    e.source.textLayerDiv.dispatchEvent(event);
+  });
+  eventBus.on('pagechange', function (e) {
+    var event = document.createEvent('UIEvents');
+    event.initUIEvent('pagechange', true, true, window, 0);
+    event.pageNumber = e.pageNumber;
+    e.source.container.dispatchEvent(event);
+  });
+  eventBus.on('pagesinit', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('pagesinit', true, true, null);
+    e.source.container.dispatchEvent(event);
+  });
+  eventBus.on('pagesloaded', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('pagesloaded', true, true, { pagesCount: e.pagesCount });
+    e.source.container.dispatchEvent(event);
+  });
+  eventBus.on('scalechange', function (e) {
+    var event = document.createEvent('UIEvents');
+    event.initUIEvent('scalechange', true, true, window, 0);
+    event.scale = e.scale;
+    event.presetValue = e.presetValue;
+    e.source.container.dispatchEvent(event);
+  });
+  eventBus.on('updateviewarea', function (e) {
+    var event = document.createEvent('UIEvents');
+    event.initUIEvent('updateviewarea', true, true, window, 0);
+    event.location = e.location;
+    e.source.container.dispatchEvent(event);
   });
-  window.dispatchEvent(event);
- });
- eventBus.on('attachmentsloaded', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('attachmentsloaded', true, true, { attachmentsCount: e.attachmentsCount });
-  e.source.container.dispatchEvent(event);
- });
- eventBus.on('sidebarviewchanged', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('sidebarviewchanged', true, true, { view: e.view });
-  e.source.outerContainer.dispatchEvent(event);
- });
- eventBus.on('pagemode', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('pagemode', true, true, { mode: e.mode });
-  e.source.pdfViewer.container.dispatchEvent(event);
- });
- eventBus.on('namedaction', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('namedaction', true, true, { action: e.action });
-  e.source.pdfViewer.container.dispatchEvent(event);
- });
- eventBus.on('presentationmodechanged', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('presentationmodechanged', true, true, {
-   active: e.active,
-   switchInProgress: e.switchInProgress
+  eventBus.on('find', function (e) {
+    if (e.source === window) {
+      return;
+    }
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('find' + e.type, true, true, {
+      query: e.query,
+      phraseSearch: e.phraseSearch,
+      caseSensitive: e.caseSensitive,
+      highlightAll: e.highlightAll,
+      findPrevious: e.findPrevious
+    });
+    window.dispatchEvent(event);
+  });
+  eventBus.on('attachmentsloaded', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('attachmentsloaded', true, true, { attachmentsCount: e.attachmentsCount });
+    e.source.container.dispatchEvent(event);
+  });
+  eventBus.on('sidebarviewchanged', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('sidebarviewchanged', true, true, { view: e.view });
+    e.source.outerContainer.dispatchEvent(event);
+  });
+  eventBus.on('pagemode', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('pagemode', true, true, { mode: e.mode });
+    e.source.pdfViewer.container.dispatchEvent(event);
+  });
+  eventBus.on('namedaction', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('namedaction', true, true, { action: e.action });
+    e.source.pdfViewer.container.dispatchEvent(event);
+  });
+  eventBus.on('presentationmodechanged', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('presentationmodechanged', true, true, {
+      active: e.active,
+      switchInProgress: e.switchInProgress
+    });
+    window.dispatchEvent(event);
+  });
+  eventBus.on('outlineloaded', function (e) {
+    var event = document.createEvent('CustomEvent');
+    event.initCustomEvent('outlineloaded', true, true, { outlineCount: e.outlineCount });
+    e.source.container.dispatchEvent(event);
   });
-  window.dispatchEvent(event);
- });
- eventBus.on('outlineloaded', function (e) {
-  var event = document.createEvent('CustomEvent');
-  event.initCustomEvent('outlineloaded', true, true, { outlineCount: e.outlineCount });
-  e.source.container.dispatchEvent(event);
- });
 }
 var globalEventBus = null;
 function getGlobalEventBus() {
- if (globalEventBus) {
+  if (globalEventBus) {
+    return globalEventBus;
+  }
+  globalEventBus = new EventBus();
+  attachDOMEventsToEventBus(globalEventBus);
   return globalEventBus;
- }
- globalEventBus = new EventBus();
- attachDOMEventsToEventBus(globalEventBus);
- return globalEventBus;
 }
 exports.attachDOMEventsToEventBus = attachDOMEventsToEventBus;
 exports.getGlobalEventBus = getGlobalEventBus;

+ 44 - 44
lib/web/download_manager.js

@@ -13,55 +13,55 @@
  * limitations under the License.
  */
 'use strict';
+
 var pdfjsLib = require('./pdfjs.js');
 function download(blobUrl, filename) {
- var a = document.createElement('a');
- if (a.click) {
-  a.href = blobUrl;
-  a.target = '_parent';
-  if ('download' in a) {
-   a.download = filename;
+  var a = document.createElement('a');
+  if (a.click) {
+    a.href = blobUrl;
+    a.target = '_parent';
+    if ('download' in a) {
+      a.download = filename;
+    }
+    (document.body || document.documentElement).appendChild(a);
+    a.click();
+    a.parentNode.removeChild(a);
+  } else {
+    if (window.top === window && blobUrl.split('#')[0] === window.location.href.split('#')[0]) {
+      var padCharacter = blobUrl.indexOf('?') === -1 ? '?' : '&';
+      blobUrl = blobUrl.replace(/#|$/, padCharacter + '$&');
+    }
+    window.open(blobUrl, '_parent');
   }
-  (document.body || document.documentElement).appendChild(a);
-  a.click();
-  a.parentNode.removeChild(a);
- } else {
-  if (window.top === window && blobUrl.split('#')[0] === window.location.href.split('#')[0]) {
-   var padCharacter = blobUrl.indexOf('?') === -1 ? '?' : '&';
-   blobUrl = blobUrl.replace(/#|$/, padCharacter + '$&');
-  }
-  window.open(blobUrl, '_parent');
- }
-}
-function DownloadManager() {
 }
+function DownloadManager() {}
 DownloadManager.prototype = {
- downloadUrl: function DownloadManager_downloadUrl(url, filename) {
-  if (!pdfjsLib.createValidAbsoluteUrl(url, 'http://example.com')) {
-   return;
-  }
-  download(url + '#pdfjs.action=download', filename);
- },
- downloadData: function DownloadManager_downloadData(data, filename, contentType) {
-  if (navigator.msSaveBlob) {
-   return navigator.msSaveBlob(new Blob([data], { type: contentType }), filename);
-  }
-  var blobUrl = pdfjsLib.createObjectURL(data, contentType, pdfjsLib.PDFJS.disableCreateObjectURL);
-  download(blobUrl, filename);
- },
- download: function DownloadManager_download(blob, url, filename) {
-  if (navigator.msSaveBlob) {
-   if (!navigator.msSaveBlob(blob, filename)) {
-    this.downloadUrl(url, filename);
-   }
-   return;
-  }
-  if (pdfjsLib.PDFJS.disableCreateObjectURL) {
-   this.downloadUrl(url, filename);
-   return;
+  downloadUrl: function DownloadManager_downloadUrl(url, filename) {
+    if (!pdfjsLib.createValidAbsoluteUrl(url, 'http://example.com')) {
+      return;
+    }
+    download(url + '#pdfjs.action=download', filename);
+  },
+  downloadData: function DownloadManager_downloadData(data, filename, contentType) {
+    if (navigator.msSaveBlob) {
+      return navigator.msSaveBlob(new Blob([data], { type: contentType }), filename);
+    }
+    var blobUrl = pdfjsLib.createObjectURL(data, contentType, pdfjsLib.PDFJS.disableCreateObjectURL);
+    download(blobUrl, filename);
+  },
+  download: function DownloadManager_download(blob, url, filename) {
+    if (navigator.msSaveBlob) {
+      if (!navigator.msSaveBlob(blob, filename)) {
+        this.downloadUrl(url, filename);
+      }
+      return;
+    }
+    if (pdfjsLib.PDFJS.disableCreateObjectURL) {
+      this.downloadUrl(url, filename);
+      return;
+    }
+    var blobUrl = URL.createObjectURL(blob);
+    download(blobUrl, filename);
   }
-  var blobUrl = URL.createObjectURL(blob);
-  download(blobUrl, filename);
- }
 };
 exports.DownloadManager = DownloadManager;

+ 58 - 64
lib/web/firefox_print_service.js

@@ -13,82 +13,76 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtils = require('./ui_utils.js');
 var app = require('./app.js');
 var pdfjsLib = require('./pdfjs.js');
 var CSS_UNITS = uiUtils.CSS_UNITS;
 var PDFPrintServiceFactory = app.PDFPrintServiceFactory;
 function composePage(pdfDocument, pageNumber, size, printContainer) {
- var canvas = document.createElement('canvas');
- var PRINT_RESOLUTION = 150;
- var PRINT_UNITS = PRINT_RESOLUTION / 72.0;
- canvas.width = Math.floor(size.width * PRINT_UNITS);
- canvas.height = Math.floor(size.height * PRINT_UNITS);
- canvas.style.width = Math.floor(size.width * CSS_UNITS) + 'px';
- canvas.style.height = Math.floor(size.height * CSS_UNITS) + 'px';
- var canvasWrapper = document.createElement('div');
- canvasWrapper.appendChild(canvas);
- printContainer.appendChild(canvasWrapper);
- canvas.mozPrintCallback = function (obj) {
-  var ctx = obj.context;
-  ctx.save();
-  ctx.fillStyle = 'rgb(255, 255, 255)';
-  ctx.fillRect(0, 0, canvas.width, canvas.height);
-  ctx.restore();
-  pdfDocument.getPage(pageNumber).then(function (pdfPage) {
-   var renderContext = {
-    canvasContext: ctx,
-    transform: [
-     PRINT_UNITS,
-     0,
-     0,
-     PRINT_UNITS,
-     0,
-     0
-    ],
-    viewport: pdfPage.getViewport(1, size.rotation),
-    intent: 'print'
-   };
-   return pdfPage.render(renderContext).promise;
-  }).then(function () {
-   obj.done();
-  }, function (error) {
-   console.error(error);
-   if ('abort' in obj) {
-    obj.abort();
-   } else {
-    obj.done();
-   }
-  });
- };
+  var canvas = document.createElement('canvas');
+  var PRINT_RESOLUTION = 150;
+  var PRINT_UNITS = PRINT_RESOLUTION / 72.0;
+  canvas.width = Math.floor(size.width * PRINT_UNITS);
+  canvas.height = Math.floor(size.height * PRINT_UNITS);
+  canvas.style.width = Math.floor(size.width * CSS_UNITS) + 'px';
+  canvas.style.height = Math.floor(size.height * CSS_UNITS) + 'px';
+  var canvasWrapper = document.createElement('div');
+  canvasWrapper.appendChild(canvas);
+  printContainer.appendChild(canvasWrapper);
+  canvas.mozPrintCallback = function (obj) {
+    var ctx = obj.context;
+    ctx.save();
+    ctx.fillStyle = 'rgb(255, 255, 255)';
+    ctx.fillRect(0, 0, canvas.width, canvas.height);
+    ctx.restore();
+    pdfDocument.getPage(pageNumber).then(function (pdfPage) {
+      var renderContext = {
+        canvasContext: ctx,
+        transform: [PRINT_UNITS, 0, 0, PRINT_UNITS, 0, 0],
+        viewport: pdfPage.getViewport(1, size.rotation),
+        intent: 'print'
+      };
+      return pdfPage.render(renderContext).promise;
+    }).then(function () {
+      obj.done();
+    }, function (error) {
+      console.error(error);
+      if ('abort' in obj) {
+        obj.abort();
+      } else {
+        obj.done();
+      }
+    });
+  };
 }
 function FirefoxPrintService(pdfDocument, pagesOverview, printContainer) {
- this.pdfDocument = pdfDocument;
- this.pagesOverview = pagesOverview;
- this.printContainer = printContainer;
+  this.pdfDocument = pdfDocument;
+  this.pagesOverview = pagesOverview;
+  this.printContainer = printContainer;
 }
 FirefoxPrintService.prototype = {
- layout: function () {
-  var pdfDocument = this.pdfDocument;
-  var printContainer = this.printContainer;
-  var body = document.querySelector('body');
-  body.setAttribute('data-pdfjsprinting', true);
-  for (var i = 0, ii = this.pagesOverview.length; i < ii; ++i) {
-   composePage(pdfDocument, i + 1, this.pagesOverview[i], printContainer);
+  layout: function () {
+    var pdfDocument = this.pdfDocument;
+    var printContainer = this.printContainer;
+    var body = document.querySelector('body');
+    body.setAttribute('data-pdfjsprinting', true);
+    for (var i = 0, ii = this.pagesOverview.length; i < ii; ++i) {
+      composePage(pdfDocument, i + 1, this.pagesOverview[i], printContainer);
+    }
+  },
+  destroy: function () {
+    this.printContainer.textContent = '';
   }
- },
- destroy: function () {
-  this.printContainer.textContent = '';
- }
 };
 PDFPrintServiceFactory.instance = {
- get supportsPrinting() {
-  var canvas = document.createElement('canvas');
-  var value = 'mozPrintCallback' in canvas;
-  return pdfjsLib.shadow(this, 'supportsPrinting', value);
- },
- createPrintService: function (pdfDocument, pagesOverview, printContainer) {
-  return new FirefoxPrintService(pdfDocument, pagesOverview, printContainer);
- }
+  get supportsPrinting() {
+    var canvas = document.createElement('canvas');
+    var value = 'mozPrintCallback' in canvas;
+    return pdfjsLib.shadow(this, 'supportsPrinting', value);
+  },
+  createPrintService: function (pdfDocument, pagesOverview, printContainer) {
+    return new FirefoxPrintService(pdfDocument, pagesOverview, printContainer);
+  }
 };
 exports.FirefoxPrintService = FirefoxPrintService;

+ 1 - 0
lib/web/firefoxcom.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var preferences = require('./preferences.js');
 var app = require('./app.js');
 var pdfjsLib = require('./pdfjs.js');

+ 117 - 122
lib/web/grab_to_pan.js

@@ -13,139 +13,134 @@
  * limitations under the License.
  */
 'use strict';
+
 function GrabToPan(options) {
- this.element = options.element;
- this.document = options.element.ownerDocument;
- if (typeof options.ignoreTarget === 'function') {
-  this.ignoreTarget = options.ignoreTarget;
- }
- this.onActiveChanged = options.onActiveChanged;
- this.activate = this.activate.bind(this);
- this.deactivate = this.deactivate.bind(this);
- this.toggle = this.toggle.bind(this);
- this._onmousedown = this._onmousedown.bind(this);
- this._onmousemove = this._onmousemove.bind(this);
- this._endPan = this._endPan.bind(this);
- var overlay = this.overlay = document.createElement('div');
- overlay.className = 'grab-to-pan-grabbing';
+  this.element = options.element;
+  this.document = options.element.ownerDocument;
+  if (typeof options.ignoreTarget === 'function') {
+    this.ignoreTarget = options.ignoreTarget;
+  }
+  this.onActiveChanged = options.onActiveChanged;
+  this.activate = this.activate.bind(this);
+  this.deactivate = this.deactivate.bind(this);
+  this.toggle = this.toggle.bind(this);
+  this._onmousedown = this._onmousedown.bind(this);
+  this._onmousemove = this._onmousemove.bind(this);
+  this._endPan = this._endPan.bind(this);
+  var overlay = this.overlay = document.createElement('div');
+  overlay.className = 'grab-to-pan-grabbing';
 }
 GrabToPan.prototype = {
- CSS_CLASS_GRAB: 'grab-to-pan-grab',
- activate: function GrabToPan_activate() {
-  if (!this.active) {
-   this.active = true;
-   this.element.addEventListener('mousedown', this._onmousedown, true);
-   this.element.classList.add(this.CSS_CLASS_GRAB);
-   if (this.onActiveChanged) {
-    this.onActiveChanged(true);
-   }
-  }
- },
- deactivate: function GrabToPan_deactivate() {
-  if (this.active) {
-   this.active = false;
-   this.element.removeEventListener('mousedown', this._onmousedown, true);
-   this._endPan();
-   this.element.classList.remove(this.CSS_CLASS_GRAB);
-   if (this.onActiveChanged) {
-    this.onActiveChanged(false);
-   }
-  }
- },
- toggle: function GrabToPan_toggle() {
-  if (this.active) {
-   this.deactivate();
-  } else {
-   this.activate();
-  }
- },
- ignoreTarget: function GrabToPan_ignoreTarget(node) {
-  return node[matchesSelector]('a[href], a[href] *, input, textarea, button, button *, select, option');
- },
- _onmousedown: function GrabToPan__onmousedown(event) {
-  if (event.button !== 0 || this.ignoreTarget(event.target)) {
-   return;
-  }
-  if (event.originalTarget) {
-   try {
-    event.originalTarget.tagName;
-   } catch (e) {
-    return;
-   }
-  }
-  this.scrollLeftStart = this.element.scrollLeft;
-  this.scrollTopStart = this.element.scrollTop;
-  this.clientXStart = event.clientX;
-  this.clientYStart = event.clientY;
-  this.document.addEventListener('mousemove', this._onmousemove, true);
-  this.document.addEventListener('mouseup', this._endPan, true);
-  this.element.addEventListener('scroll', this._endPan, true);
-  event.preventDefault();
-  event.stopPropagation();
-  var focusedElement = document.activeElement;
-  if (focusedElement && !focusedElement.contains(event.target)) {
-   focusedElement.blur();
-  }
- },
- _onmousemove: function GrabToPan__onmousemove(event) {
-  this.element.removeEventListener('scroll', this._endPan, true);
-  if (isLeftMouseReleased(event)) {
-   this._endPan();
-   return;
+  CSS_CLASS_GRAB: 'grab-to-pan-grab',
+  activate: function GrabToPan_activate() {
+    if (!this.active) {
+      this.active = true;
+      this.element.addEventListener('mousedown', this._onmousedown, true);
+      this.element.classList.add(this.CSS_CLASS_GRAB);
+      if (this.onActiveChanged) {
+        this.onActiveChanged(true);
+      }
+    }
+  },
+  deactivate: function GrabToPan_deactivate() {
+    if (this.active) {
+      this.active = false;
+      this.element.removeEventListener('mousedown', this._onmousedown, true);
+      this._endPan();
+      this.element.classList.remove(this.CSS_CLASS_GRAB);
+      if (this.onActiveChanged) {
+        this.onActiveChanged(false);
+      }
+    }
+  },
+  toggle: function GrabToPan_toggle() {
+    if (this.active) {
+      this.deactivate();
+    } else {
+      this.activate();
+    }
+  },
+  ignoreTarget: function GrabToPan_ignoreTarget(node) {
+    return node[matchesSelector]('a[href], a[href] *, input, textarea, button, button *, select, option');
+  },
+  _onmousedown: function GrabToPan__onmousedown(event) {
+    if (event.button !== 0 || this.ignoreTarget(event.target)) {
+      return;
+    }
+    if (event.originalTarget) {
+      try {
+        event.originalTarget.tagName;
+      } catch (e) {
+        return;
+      }
+    }
+    this.scrollLeftStart = this.element.scrollLeft;
+    this.scrollTopStart = this.element.scrollTop;
+    this.clientXStart = event.clientX;
+    this.clientYStart = event.clientY;
+    this.document.addEventListener('mousemove', this._onmousemove, true);
+    this.document.addEventListener('mouseup', this._endPan, true);
+    this.element.addEventListener('scroll', this._endPan, true);
+    event.preventDefault();
+    event.stopPropagation();
+    var focusedElement = document.activeElement;
+    if (focusedElement && !focusedElement.contains(event.target)) {
+      focusedElement.blur();
+    }
+  },
+  _onmousemove: function GrabToPan__onmousemove(event) {
+    this.element.removeEventListener('scroll', this._endPan, true);
+    if (isLeftMouseReleased(event)) {
+      this._endPan();
+      return;
+    }
+    var xDiff = event.clientX - this.clientXStart;
+    var yDiff = event.clientY - this.clientYStart;
+    var scrollTop = this.scrollTopStart - yDiff;
+    var scrollLeft = this.scrollLeftStart - xDiff;
+    if (this.element.scrollTo) {
+      this.element.scrollTo({
+        top: scrollTop,
+        left: scrollLeft,
+        behavior: 'instant'
+      });
+    } else {
+      this.element.scrollTop = scrollTop;
+      this.element.scrollLeft = scrollLeft;
+    }
+    if (!this.overlay.parentNode) {
+      document.body.appendChild(this.overlay);
+    }
+  },
+  _endPan: function GrabToPan__endPan() {
+    this.element.removeEventListener('scroll', this._endPan, true);
+    this.document.removeEventListener('mousemove', this._onmousemove, true);
+    this.document.removeEventListener('mouseup', this._endPan, true);
+    this.overlay.remove();
   }
-  var xDiff = event.clientX - this.clientXStart;
-  var yDiff = event.clientY - this.clientYStart;
-  var scrollTop = this.scrollTopStart - yDiff;
-  var scrollLeft = this.scrollLeftStart - xDiff;
-  if (this.element.scrollTo) {
-   this.element.scrollTo({
-    top: scrollTop,
-    left: scrollLeft,
-    behavior: 'instant'
-   });
-  } else {
-   this.element.scrollTop = scrollTop;
-   this.element.scrollLeft = scrollLeft;
-  }
-  if (!this.overlay.parentNode) {
-   document.body.appendChild(this.overlay);
-  }
- },
- _endPan: function GrabToPan__endPan() {
-  this.element.removeEventListener('scroll', this._endPan, true);
-  this.document.removeEventListener('mousemove', this._onmousemove, true);
-  this.document.removeEventListener('mouseup', this._endPan, true);
-  this.overlay.remove();
- }
 };
 var matchesSelector;
-[
- 'webkitM',
- 'mozM',
- 'msM',
- 'oM',
- 'm'
-].some(function (prefix) {
- var name = prefix + 'atches';
- if (name in document.documentElement) {
-  matchesSelector = name;
- }
- name += 'Selector';
- if (name in document.documentElement) {
-  matchesSelector = name;
- }
- return matchesSelector;
+['webkitM', 'mozM', 'msM', 'oM', 'm'].some(function (prefix) {
+  var name = prefix + 'atches';
+  if (name in document.documentElement) {
+    matchesSelector = name;
+  }
+  name += 'Selector';
+  if (name in document.documentElement) {
+    matchesSelector = name;
+  }
+  return matchesSelector;
 });
 var isNotIEorIsIE10plus = !document.documentMode || document.documentMode > 9;
 var chrome = window.chrome;
 var isChrome15OrOpera15plus = chrome && (chrome.webstore || chrome.app);
 var isSafari6plus = /Apple/.test(navigator.vendor) && /Version\/([6-9]\d*|[1-5]\d+)/.test(navigator.userAgent);
 function isLeftMouseReleased(event) {
- if ('buttons' in event && isNotIEorIsIE10plus) {
-  return !(event.buttons & 1);
- }
- if (isChrome15OrOpera15plus || isSafari6plus) {
-  return event.which === 0;
- }
+  if ('buttons' in event && isNotIEorIsIE10plus) {
+    return !(event.buttons & 1);
+  }
+  if (isChrome15OrOpera15plus || isSafari6plus) {
+    return event.which === 0;
+  }
 }
 exports.GrabToPan = GrabToPan;

+ 47 - 50
lib/web/hand_tool.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var grabToPan = require('./grab_to_pan.js');
 var preferences = require('./preferences.js');
 var uiUtils = require('./ui_utils.js');
@@ -20,57 +21,53 @@ var GrabToPan = grabToPan.GrabToPan;
 var Preferences = preferences.Preferences;
 var localized = uiUtils.localized;
 var HandTool = function HandToolClosure() {
- function HandTool(options) {
-  this.container = options.container;
-  this.eventBus = options.eventBus;
-  this.wasActive = false;
-  this.handTool = new GrabToPan({
-   element: this.container,
-   onActiveChanged: function (isActive) {
-    this.eventBus.dispatch('handtoolchanged', { isActive: isActive });
-   }.bind(this)
-  });
-  this.eventBus.on('togglehandtool', this.toggle.bind(this));
-  Promise.all([
-   localized,
-   Preferences.get('enableHandToolOnLoad')
-  ]).then(function resolved(values) {
-   if (values[1] === true) {
-    this.handTool.activate();
-   }
-  }.bind(this)).catch(function rejected(reason) {
-  });
-  this.eventBus.on('presentationmodechanged', function (e) {
-   if (e.switchInProgress) {
-    return;
-   }
-   if (e.active) {
-    this.enterPresentationMode();
-   } else {
-    this.exitPresentationMode();
-   }
-  }.bind(this));
- }
- HandTool.prototype = {
-  get isActive() {
-   return !!this.handTool.active;
-  },
-  toggle: function HandTool_toggle() {
-   this.handTool.toggle();
-  },
-  enterPresentationMode: function HandTool_enterPresentationMode() {
-   if (this.isActive) {
-    this.wasActive = true;
-    this.handTool.deactivate();
-   }
-  },
-  exitPresentationMode: function HandTool_exitPresentationMode() {
-   if (this.wasActive) {
+  function HandTool(options) {
+    this.container = options.container;
+    this.eventBus = options.eventBus;
     this.wasActive = false;
-    this.handTool.activate();
-   }
+    this.handTool = new GrabToPan({
+      element: this.container,
+      onActiveChanged: function (isActive) {
+        this.eventBus.dispatch('handtoolchanged', { isActive: isActive });
+      }.bind(this)
+    });
+    this.eventBus.on('togglehandtool', this.toggle.bind(this));
+    Promise.all([localized, Preferences.get('enableHandToolOnLoad')]).then(function resolved(values) {
+      if (values[1] === true) {
+        this.handTool.activate();
+      }
+    }.bind(this)).catch(function rejected(reason) {});
+    this.eventBus.on('presentationmodechanged', function (e) {
+      if (e.switchInProgress) {
+        return;
+      }
+      if (e.active) {
+        this.enterPresentationMode();
+      } else {
+        this.exitPresentationMode();
+      }
+    }.bind(this));
   }
- };
- return HandTool;
+  HandTool.prototype = {
+    get isActive() {
+      return !!this.handTool.active;
+    },
+    toggle: function HandTool_toggle() {
+      this.handTool.toggle();
+    },
+    enterPresentationMode: function HandTool_enterPresentationMode() {
+      if (this.isActive) {
+        this.wasActive = true;
+        this.handTool.deactivate();
+      }
+    },
+    exitPresentationMode: function HandTool_exitPresentationMode() {
+      if (this.wasActive) {
+        this.wasActive = false;
+        this.handTool.activate();
+      }
+    }
+  };
+  return HandTool;
 }();
 exports.HandTool = HandTool;

+ 24 - 46
lib/web/interfaces.js

@@ -13,59 +13,37 @@
  * limitations under the License.
  */
 'use strict';
-function IPDFLinkService() {
-}
+
+function IPDFLinkService() {}
 IPDFLinkService.prototype = {
- get page() {
- },
- set page(value) {
- },
- navigateTo: function (dest) {
- },
- getDestinationHash: function (dest) {
- },
- getAnchorUrl: function (hash) {
- },
- setHash: function (hash) {
- },
- executeNamedAction: function (action) {
- },
- cachePageRef: function (pageNum, pageRef) {
- }
+  get page() {},
+  set page(value) {},
+  navigateTo: function (dest) {},
+  getDestinationHash: function (dest) {},
+  getAnchorUrl: function (hash) {},
+  setHash: function (hash) {},
+  executeNamedAction: function (action) {},
+  cachePageRef: function (pageNum, pageRef) {}
 };
-function IPDFHistory() {
-}
+function IPDFHistory() {}
 IPDFHistory.prototype = {
- forward: function () {
- },
- back: function () {
- },
- push: function (params) {
- },
- updateNextHashParam: function (hash) {
- }
+  forward: function () {},
+  back: function () {},
+  push: function (params) {},
+  updateNextHashParam: function (hash) {}
 };
-function IRenderableView() {
-}
+function IRenderableView() {}
 IRenderableView.prototype = {
- get renderingId() {
- },
- get renderingState() {
- },
- draw: function () {
- },
- resume: function () {
- }
+  get renderingId() {},
+  get renderingState() {},
+  draw: function () {},
+  resume: function () {}
 };
-function IPDFTextLayerFactory() {
-}
+function IPDFTextLayerFactory() {}
 IPDFTextLayerFactory.prototype = {
- createTextLayerBuilder: function (textLayerDiv, pageIndex, viewport, enhanceTextSelection) {
- }
+  createTextLayerBuilder: function (textLayerDiv, pageIndex, viewport, enhanceTextSelection) {}
 };
-function IPDFAnnotationLayerFactory() {
-}
+function IPDFAnnotationLayerFactory() {}
 IPDFAnnotationLayerFactory.prototype = {
- createAnnotationLayerBuilder: function (pageDiv, pdfPage, renderInteractiveForms) {
- }
+  createAnnotationLayerBuilder: function (pageDiv, pdfPage, renderInteractiveForms) {}
 };

+ 80 - 79
lib/web/overlay_manager.js

@@ -13,87 +13,88 @@
  * limitations under the License.
  */
 'use strict';
+
 var OverlayManager = {
- overlays: {},
- active: null,
- register: function overlayManagerRegister(name, element, callerCloseMethod, canForceClose) {
-  return new Promise(function (resolve) {
-   var container;
-   if (!name || !element || !(container = element.parentNode)) {
-    throw new Error('Not enough parameters.');
-   } else if (this.overlays[name]) {
-    throw new Error('The overlay is already registered.');
-   }
-   this.overlays[name] = {
-    element: element,
-    container: container,
-    callerCloseMethod: callerCloseMethod || null,
-    canForceClose: canForceClose || false
-   };
-   resolve();
-  }.bind(this));
- },
- unregister: function overlayManagerUnregister(name) {
-  return new Promise(function (resolve) {
-   if (!this.overlays[name]) {
-    throw new Error('The overlay does not exist.');
-   } else if (this.active === name) {
-    throw new Error('The overlay cannot be removed while it is active.');
-   }
-   delete this.overlays[name];
-   resolve();
-  }.bind(this));
- },
- open: function overlayManagerOpen(name) {
-  return new Promise(function (resolve) {
-   if (!this.overlays[name]) {
-    throw new Error('The overlay does not exist.');
-   } else if (this.active) {
-    if (this.overlays[name].canForceClose) {
-     this._closeThroughCaller();
-    } else if (this.active === name) {
-     throw new Error('The overlay is already active.');
-    } else {
-     throw new Error('Another overlay is currently active.');
+  overlays: {},
+  active: null,
+  register: function overlayManagerRegister(name, element, callerCloseMethod, canForceClose) {
+    return new Promise(function (resolve) {
+      var container;
+      if (!name || !element || !(container = element.parentNode)) {
+        throw new Error('Not enough parameters.');
+      } else if (this.overlays[name]) {
+        throw new Error('The overlay is already registered.');
+      }
+      this.overlays[name] = {
+        element: element,
+        container: container,
+        callerCloseMethod: callerCloseMethod || null,
+        canForceClose: canForceClose || false
+      };
+      resolve();
+    }.bind(this));
+  },
+  unregister: function overlayManagerUnregister(name) {
+    return new Promise(function (resolve) {
+      if (!this.overlays[name]) {
+        throw new Error('The overlay does not exist.');
+      } else if (this.active === name) {
+        throw new Error('The overlay cannot be removed while it is active.');
+      }
+      delete this.overlays[name];
+      resolve();
+    }.bind(this));
+  },
+  open: function overlayManagerOpen(name) {
+    return new Promise(function (resolve) {
+      if (!this.overlays[name]) {
+        throw new Error('The overlay does not exist.');
+      } else if (this.active) {
+        if (this.overlays[name].canForceClose) {
+          this._closeThroughCaller();
+        } else if (this.active === name) {
+          throw new Error('The overlay is already active.');
+        } else {
+          throw new Error('Another overlay is currently active.');
+        }
+      }
+      this.active = name;
+      this.overlays[this.active].element.classList.remove('hidden');
+      this.overlays[this.active].container.classList.remove('hidden');
+      window.addEventListener('keydown', this._keyDown);
+      resolve();
+    }.bind(this));
+  },
+  close: function overlayManagerClose(name) {
+    return new Promise(function (resolve) {
+      if (!this.overlays[name]) {
+        throw new Error('The overlay does not exist.');
+      } else if (!this.active) {
+        throw new Error('The overlay is currently not active.');
+      } else if (this.active !== name) {
+        throw new Error('Another overlay is currently active.');
+      }
+      this.overlays[this.active].container.classList.add('hidden');
+      this.overlays[this.active].element.classList.add('hidden');
+      this.active = null;
+      window.removeEventListener('keydown', this._keyDown);
+      resolve();
+    }.bind(this));
+  },
+  _keyDown: function overlayManager_keyDown(evt) {
+    var self = OverlayManager;
+    if (self.active && evt.keyCode === 27) {
+      self._closeThroughCaller();
+      evt.preventDefault();
+    }
+  },
+  _closeThroughCaller: function overlayManager_closeThroughCaller() {
+    if (this.overlays[this.active].callerCloseMethod) {
+      this.overlays[this.active].callerCloseMethod();
+    }
+    if (this.active) {
+      this.close(this.active);
     }
-   }
-   this.active = name;
-   this.overlays[this.active].element.classList.remove('hidden');
-   this.overlays[this.active].container.classList.remove('hidden');
-   window.addEventListener('keydown', this._keyDown);
-   resolve();
-  }.bind(this));
- },
- close: function overlayManagerClose(name) {
-  return new Promise(function (resolve) {
-   if (!this.overlays[name]) {
-    throw new Error('The overlay does not exist.');
-   } else if (!this.active) {
-    throw new Error('The overlay is currently not active.');
-   } else if (this.active !== name) {
-    throw new Error('Another overlay is currently active.');
-   }
-   this.overlays[this.active].container.classList.add('hidden');
-   this.overlays[this.active].element.classList.add('hidden');
-   this.active = null;
-   window.removeEventListener('keydown', this._keyDown);
-   resolve();
-  }.bind(this));
- },
- _keyDown: function overlayManager_keyDown(evt) {
-  var self = OverlayManager;
-  if (self.active && evt.keyCode === 27) {
-   self._closeThroughCaller();
-   evt.preventDefault();
-  }
- },
- _closeThroughCaller: function overlayManager_closeThroughCaller() {
-  if (this.overlays[this.active].callerCloseMethod) {
-   this.overlays[this.active].callerCloseMethod();
-  }
-  if (this.active) {
-   this.close(this.active);
   }
- }
 };
 exports.OverlayManager = OverlayManager;

+ 49 - 48
lib/web/password_prompt.js

@@ -13,60 +13,61 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtils = require('./ui_utils.js');
 var overlayManager = require('./overlay_manager.js');
 var pdfjsLib = require('./pdfjs.js');
 var mozL10n = uiUtils.mozL10n;
 var OverlayManager = overlayManager.OverlayManager;
 var PasswordPrompt = function PasswordPromptClosure() {
- function PasswordPrompt(options) {
-  this.overlayName = options.overlayName;
-  this.container = options.container;
-  this.label = options.label;
-  this.input = options.input;
-  this.submitButton = options.submitButton;
-  this.cancelButton = options.cancelButton;
-  this.updateCallback = null;
-  this.reason = null;
-  this.submitButton.addEventListener('click', this.verify.bind(this));
-  this.cancelButton.addEventListener('click', this.close.bind(this));
-  this.input.addEventListener('keydown', function (e) {
-   if (e.keyCode === 13) {
-    this.verify();
-   }
-  }.bind(this));
-  OverlayManager.register(this.overlayName, this.container, this.close.bind(this), true);
- }
- PasswordPrompt.prototype = {
-  open: function PasswordPrompt_open() {
-   OverlayManager.open(this.overlayName).then(function () {
-    this.input.type = 'password';
-    this.input.focus();
-    var promptString = mozL10n.get('password_label', null, 'Enter the password to open this PDF file.');
-    if (this.reason === pdfjsLib.PasswordResponses.INCORRECT_PASSWORD) {
-     promptString = mozL10n.get('password_invalid', null, 'Invalid password. Please try again.');
-    }
-    this.label.textContent = promptString;
-   }.bind(this));
-  },
-  close: function PasswordPrompt_close() {
-   OverlayManager.close(this.overlayName).then(function () {
-    this.input.value = '';
-    this.input.type = '';
-   }.bind(this));
-  },
-  verify: function PasswordPrompt_verify() {
-   var password = this.input.value;
-   if (password && password.length > 0) {
-    this.close();
-    return this.updateCallback(password);
-   }
-  },
-  setUpdateCallback: function PasswordPrompt_setUpdateCallback(updateCallback, reason) {
-   this.updateCallback = updateCallback;
-   this.reason = reason;
+  function PasswordPrompt(options) {
+    this.overlayName = options.overlayName;
+    this.container = options.container;
+    this.label = options.label;
+    this.input = options.input;
+    this.submitButton = options.submitButton;
+    this.cancelButton = options.cancelButton;
+    this.updateCallback = null;
+    this.reason = null;
+    this.submitButton.addEventListener('click', this.verify.bind(this));
+    this.cancelButton.addEventListener('click', this.close.bind(this));
+    this.input.addEventListener('keydown', function (e) {
+      if (e.keyCode === 13) {
+        this.verify();
+      }
+    }.bind(this));
+    OverlayManager.register(this.overlayName, this.container, this.close.bind(this), true);
   }
- };
- return PasswordPrompt;
+  PasswordPrompt.prototype = {
+    open: function PasswordPrompt_open() {
+      OverlayManager.open(this.overlayName).then(function () {
+        this.input.type = 'password';
+        this.input.focus();
+        var promptString = mozL10n.get('password_label', null, 'Enter the password to open this PDF file.');
+        if (this.reason === pdfjsLib.PasswordResponses.INCORRECT_PASSWORD) {
+          promptString = mozL10n.get('password_invalid', null, 'Invalid password. Please try again.');
+        }
+        this.label.textContent = promptString;
+      }.bind(this));
+    },
+    close: function PasswordPrompt_close() {
+      OverlayManager.close(this.overlayName).then(function () {
+        this.input.value = '';
+        this.input.type = '';
+      }.bind(this));
+    },
+    verify: function PasswordPrompt_verify() {
+      var password = this.input.value;
+      if (password && password.length > 0) {
+        this.close();
+        return this.updateCallback(password);
+      }
+    },
+    setUpdateCallback: function PasswordPrompt_setUpdateCallback(updateCallback, reason) {
+      this.updateCallback = updateCallback;
+      this.reason = reason;
+    }
+  };
+  return PasswordPrompt;
 }();
 exports.PasswordPrompt = PasswordPrompt;

+ 98 - 97
lib/web/pdf_attachment_viewer.js

@@ -13,107 +13,108 @@
  * limitations under the License.
  */
 'use strict';
+
 var pdfjsLib = require('./pdfjs.js');
 var PDFAttachmentViewer = function PDFAttachmentViewerClosure() {
- function PDFAttachmentViewer(options) {
-  this.attachments = null;
-  this.container = options.container;
-  this.eventBus = options.eventBus;
-  this.downloadManager = options.downloadManager;
-  this._renderedCapability = pdfjsLib.createPromiseCapability();
-  this.eventBus.on('fileattachmentannotation', this._appendAttachment.bind(this));
- }
- PDFAttachmentViewer.prototype = {
-  reset: function PDFAttachmentViewer_reset(keepRenderedCapability) {
-   this.attachments = null;
-   this.container.textContent = '';
-   if (!keepRenderedCapability) {
+  function PDFAttachmentViewer(options) {
+    this.attachments = null;
+    this.container = options.container;
+    this.eventBus = options.eventBus;
+    this.downloadManager = options.downloadManager;
     this._renderedCapability = pdfjsLib.createPromiseCapability();
-   }
-  },
-  _dispatchEvent: function PDFAttachmentViewer_dispatchEvent(attachmentsCount) {
-   this.eventBus.dispatch('attachmentsloaded', {
-    source: this,
-    attachmentsCount: attachmentsCount
-   });
-   this._renderedCapability.resolve();
-  },
-  _bindPdfLink: function PDFAttachmentViewer_bindPdfLink(button, content, filename) {
-   var blobUrl;
-   button.onclick = function () {
-    if (!blobUrl) {
-     blobUrl = pdfjsLib.createObjectURL(content, 'application/pdf', pdfjsLib.PDFJS.disableCreateObjectURL);
-    }
-    var viewerUrl;
-    viewerUrl = '?file=' + encodeURIComponent(blobUrl + '#' + filename);
-    window.open(viewerUrl);
-    return false;
-   };
-  },
-  _bindLink: function PDFAttachmentViewer_bindLink(button, content, filename) {
-   button.onclick = function downloadFile(e) {
-    this.downloadManager.downloadData(content, filename, '');
-    return false;
-   }.bind(this);
-  },
-  render: function PDFAttachmentViewer_render(params) {
-   params = params || {};
-   var attachments = params.attachments || null;
-   var attachmentsCount = 0;
-   if (this.attachments) {
-    var keepRenderedCapability = params.keepRenderedCapability === true;
-    this.reset(keepRenderedCapability);
-   }
-   this.attachments = attachments;
-   if (!attachments) {
-    this._dispatchEvent(attachmentsCount);
-    return;
-   }
-   var names = Object.keys(attachments).sort(function (a, b) {
-    return a.toLowerCase().localeCompare(b.toLowerCase());
-   });
-   attachmentsCount = names.length;
-   for (var i = 0; i < attachmentsCount; i++) {
-    var item = attachments[names[i]];
-    var filename = pdfjsLib.getFilenameFromUrl(item.filename);
-    filename = pdfjsLib.removeNullCharacters(filename);
-    var div = document.createElement('div');
-    div.className = 'attachmentsItem';
-    var button = document.createElement('button');
-    button.textContent = filename;
-    if (/\.pdf$/i.test(filename)) {
-     this._bindPdfLink(button, item.content, filename);
-    } else {
-     this._bindLink(button, item.content, filename);
-    }
-    div.appendChild(button);
-    this.container.appendChild(div);
-   }
-   this._dispatchEvent(attachmentsCount);
-  },
-  _appendAttachment: function PDFAttachmentViewer_appendAttachment(item) {
-   this._renderedCapability.promise.then(function (id, filename, content) {
-    var attachments = this.attachments;
-    if (!attachments) {
-     attachments = Object.create(null);
-    } else {
-     for (var name in attachments) {
-      if (id === name) {
-       return;
+    this.eventBus.on('fileattachmentannotation', this._appendAttachment.bind(this));
+  }
+  PDFAttachmentViewer.prototype = {
+    reset: function PDFAttachmentViewer_reset(keepRenderedCapability) {
+      this.attachments = null;
+      this.container.textContent = '';
+      if (!keepRenderedCapability) {
+        this._renderedCapability = pdfjsLib.createPromiseCapability();
+      }
+    },
+    _dispatchEvent: function PDFAttachmentViewer_dispatchEvent(attachmentsCount) {
+      this.eventBus.dispatch('attachmentsloaded', {
+        source: this,
+        attachmentsCount: attachmentsCount
+      });
+      this._renderedCapability.resolve();
+    },
+    _bindPdfLink: function PDFAttachmentViewer_bindPdfLink(button, content, filename) {
+      var blobUrl;
+      button.onclick = function () {
+        if (!blobUrl) {
+          blobUrl = pdfjsLib.createObjectURL(content, 'application/pdf', pdfjsLib.PDFJS.disableCreateObjectURL);
+        }
+        var viewerUrl;
+        viewerUrl = '?file=' + encodeURIComponent(blobUrl + '#' + filename);
+        window.open(viewerUrl);
+        return false;
+      };
+    },
+    _bindLink: function PDFAttachmentViewer_bindLink(button, content, filename) {
+      button.onclick = function downloadFile(e) {
+        this.downloadManager.downloadData(content, filename, '');
+        return false;
+      }.bind(this);
+    },
+    render: function PDFAttachmentViewer_render(params) {
+      params = params || {};
+      var attachments = params.attachments || null;
+      var attachmentsCount = 0;
+      if (this.attachments) {
+        var keepRenderedCapability = params.keepRenderedCapability === true;
+        this.reset(keepRenderedCapability);
       }
-     }
+      this.attachments = attachments;
+      if (!attachments) {
+        this._dispatchEvent(attachmentsCount);
+        return;
+      }
+      var names = Object.keys(attachments).sort(function (a, b) {
+        return a.toLowerCase().localeCompare(b.toLowerCase());
+      });
+      attachmentsCount = names.length;
+      for (var i = 0; i < attachmentsCount; i++) {
+        var item = attachments[names[i]];
+        var filename = pdfjsLib.getFilenameFromUrl(item.filename);
+        filename = pdfjsLib.removeNullCharacters(filename);
+        var div = document.createElement('div');
+        div.className = 'attachmentsItem';
+        var button = document.createElement('button');
+        button.textContent = filename;
+        if (/\.pdf$/i.test(filename)) {
+          this._bindPdfLink(button, item.content, filename);
+        } else {
+          this._bindLink(button, item.content, filename);
+        }
+        div.appendChild(button);
+        this.container.appendChild(div);
+      }
+      this._dispatchEvent(attachmentsCount);
+    },
+    _appendAttachment: function PDFAttachmentViewer_appendAttachment(item) {
+      this._renderedCapability.promise.then(function (id, filename, content) {
+        var attachments = this.attachments;
+        if (!attachments) {
+          attachments = Object.create(null);
+        } else {
+          for (var name in attachments) {
+            if (id === name) {
+              return;
+            }
+          }
+        }
+        attachments[id] = {
+          filename: filename,
+          content: content
+        };
+        this.render({
+          attachments: attachments,
+          keepRenderedCapability: true
+        });
+      }.bind(this, item.id, item.filename, item.content));
     }
-    attachments[id] = {
-     filename: filename,
-     content: content
-    };
-    this.render({
-     attachments: attachments,
-     keepRenderedCapability: true
-    });
-   }.bind(this, item.id, item.filename, item.content));
-  }
- };
- return PDFAttachmentViewer;
+  };
+  return PDFAttachmentViewer;
 }();
 exports.PDFAttachmentViewer = PDFAttachmentViewer;

+ 119 - 120
lib/web/pdf_document_properties.js

@@ -13,133 +13,132 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtils = require('./ui_utils.js');
 var overlayManager = require('./overlay_manager.js');
 var getPDFFileNameFromURL = uiUtils.getPDFFileNameFromURL;
 var mozL10n = uiUtils.mozL10n;
 var OverlayManager = overlayManager.OverlayManager;
 var PDFDocumentProperties = function PDFDocumentPropertiesClosure() {
- function PDFDocumentProperties(options) {
-  this.fields = options.fields;
-  this.overlayName = options.overlayName;
-  this.container = options.container;
-  this.rawFileSize = 0;
-  this.url = null;
-  this.pdfDocument = null;
-  if (options.closeButton) {
-   options.closeButton.addEventListener('click', this.close.bind(this));
-  }
-  this.dataAvailablePromise = new Promise(function (resolve) {
-   this.resolveDataAvailable = resolve;
-  }.bind(this));
-  OverlayManager.register(this.overlayName, this.container, this.close.bind(this));
- }
- PDFDocumentProperties.prototype = {
-  open: function PDFDocumentProperties_open() {
-   Promise.all([
-    OverlayManager.open(this.overlayName),
-    this.dataAvailablePromise
-   ]).then(function () {
-    this._getProperties();
-   }.bind(this));
-  },
-  close: function PDFDocumentProperties_close() {
-   OverlayManager.close(this.overlayName);
-  },
-  setFileSize: function PDFDocumentProperties_setFileSize(fileSize) {
-   if (fileSize > 0) {
-    this.rawFileSize = fileSize;
-   }
-  },
-  setDocumentAndUrl: function PDFDocumentProperties_setDocumentAndUrl(pdfDocument, url) {
-   this.pdfDocument = pdfDocument;
-   this.url = url;
-   this.resolveDataAvailable();
-  },
-  _getProperties: function PDFDocumentProperties_getProperties() {
-   if (!OverlayManager.active) {
-    return;
-   }
-   this.pdfDocument.getDownloadInfo().then(function (data) {
-    if (data.length === this.rawFileSize) {
-     return;
-    }
-    this.setFileSize(data.length);
-    this._updateUI(this.fields['fileSize'], this._parseFileSize());
-   }.bind(this));
-   this.pdfDocument.getMetadata().then(function (data) {
-    var content = {
-     'fileName': getPDFFileNameFromURL(this.url),
-     'fileSize': this._parseFileSize(),
-     'title': data.info.Title,
-     'author': data.info.Author,
-     'subject': data.info.Subject,
-     'keywords': data.info.Keywords,
-     'creationDate': this._parseDate(data.info.CreationDate),
-     'modificationDate': this._parseDate(data.info.ModDate),
-     'creator': data.info.Creator,
-     'producer': data.info.Producer,
-     'version': data.info.PDFFormatVersion,
-     'pageCount': this.pdfDocument.numPages
-    };
-    for (var identifier in content) {
-     this._updateUI(this.fields[identifier], content[identifier]);
+  function PDFDocumentProperties(options) {
+    this.fields = options.fields;
+    this.overlayName = options.overlayName;
+    this.container = options.container;
+    this.rawFileSize = 0;
+    this.url = null;
+    this.pdfDocument = null;
+    if (options.closeButton) {
+      options.closeButton.addEventListener('click', this.close.bind(this));
     }
-   }.bind(this));
-  },
-  _updateUI: function PDFDocumentProperties_updateUI(field, content) {
-   if (field && content !== undefined && content !== '') {
-    field.textContent = content;
-   }
-  },
-  _parseFileSize: function PDFDocumentProperties_parseFileSize() {
-   var fileSize = this.rawFileSize, kb = fileSize / 1024;
-   if (!kb) {
-    return;
-   } else if (kb < 1024) {
-    return mozL10n.get('document_properties_kb', {
-     size_kb: (+kb.toPrecision(3)).toLocaleString(),
-     size_b: fileSize.toLocaleString()
-    }, '{{size_kb}} KB ({{size_b}} bytes)');
-   }
-   return mozL10n.get('document_properties_mb', {
-    size_mb: (+(kb / 1024).toPrecision(3)).toLocaleString(),
-    size_b: fileSize.toLocaleString()
-   }, '{{size_mb}} MB ({{size_b}} bytes)');
-  },
-  _parseDate: function PDFDocumentProperties_parseDate(inputDate) {
-   var dateToParse = inputDate;
-   if (dateToParse === undefined) {
-    return '';
-   }
-   if (dateToParse.substring(0, 2) === 'D:') {
-    dateToParse = dateToParse.substring(2);
-   }
-   var year = parseInt(dateToParse.substring(0, 4), 10);
-   var month = parseInt(dateToParse.substring(4, 6), 10) - 1;
-   var day = parseInt(dateToParse.substring(6, 8), 10);
-   var hours = parseInt(dateToParse.substring(8, 10), 10);
-   var minutes = parseInt(dateToParse.substring(10, 12), 10);
-   var seconds = parseInt(dateToParse.substring(12, 14), 10);
-   var utRel = dateToParse.substring(14, 15);
-   var offsetHours = parseInt(dateToParse.substring(15, 17), 10);
-   var offsetMinutes = parseInt(dateToParse.substring(18, 20), 10);
-   if (utRel === '-') {
-    hours += offsetHours;
-    minutes += offsetMinutes;
-   } else if (utRel === '+') {
-    hours -= offsetHours;
-    minutes -= offsetMinutes;
-   }
-   var date = new Date(Date.UTC(year, month, day, hours, minutes, seconds));
-   var dateString = date.toLocaleDateString();
-   var timeString = date.toLocaleTimeString();
-   return mozL10n.get('document_properties_date_string', {
-    date: dateString,
-    time: timeString
-   }, '{{date}}, {{time}}');
+    this.dataAvailablePromise = new Promise(function (resolve) {
+      this.resolveDataAvailable = resolve;
+    }.bind(this));
+    OverlayManager.register(this.overlayName, this.container, this.close.bind(this));
   }
- };
- return PDFDocumentProperties;
+  PDFDocumentProperties.prototype = {
+    open: function PDFDocumentProperties_open() {
+      Promise.all([OverlayManager.open(this.overlayName), this.dataAvailablePromise]).then(function () {
+        this._getProperties();
+      }.bind(this));
+    },
+    close: function PDFDocumentProperties_close() {
+      OverlayManager.close(this.overlayName);
+    },
+    setFileSize: function PDFDocumentProperties_setFileSize(fileSize) {
+      if (fileSize > 0) {
+        this.rawFileSize = fileSize;
+      }
+    },
+    setDocumentAndUrl: function PDFDocumentProperties_setDocumentAndUrl(pdfDocument, url) {
+      this.pdfDocument = pdfDocument;
+      this.url = url;
+      this.resolveDataAvailable();
+    },
+    _getProperties: function PDFDocumentProperties_getProperties() {
+      if (!OverlayManager.active) {
+        return;
+      }
+      this.pdfDocument.getDownloadInfo().then(function (data) {
+        if (data.length === this.rawFileSize) {
+          return;
+        }
+        this.setFileSize(data.length);
+        this._updateUI(this.fields['fileSize'], this._parseFileSize());
+      }.bind(this));
+      this.pdfDocument.getMetadata().then(function (data) {
+        var content = {
+          'fileName': getPDFFileNameFromURL(this.url),
+          'fileSize': this._parseFileSize(),
+          'title': data.info.Title,
+          'author': data.info.Author,
+          'subject': data.info.Subject,
+          'keywords': data.info.Keywords,
+          'creationDate': this._parseDate(data.info.CreationDate),
+          'modificationDate': this._parseDate(data.info.ModDate),
+          'creator': data.info.Creator,
+          'producer': data.info.Producer,
+          'version': data.info.PDFFormatVersion,
+          'pageCount': this.pdfDocument.numPages
+        };
+        for (var identifier in content) {
+          this._updateUI(this.fields[identifier], content[identifier]);
+        }
+      }.bind(this));
+    },
+    _updateUI: function PDFDocumentProperties_updateUI(field, content) {
+      if (field && content !== undefined && content !== '') {
+        field.textContent = content;
+      }
+    },
+    _parseFileSize: function PDFDocumentProperties_parseFileSize() {
+      var fileSize = this.rawFileSize,
+          kb = fileSize / 1024;
+      if (!kb) {
+        return;
+      } else if (kb < 1024) {
+        return mozL10n.get('document_properties_kb', {
+          size_kb: (+kb.toPrecision(3)).toLocaleString(),
+          size_b: fileSize.toLocaleString()
+        }, '{{size_kb}} KB ({{size_b}} bytes)');
+      }
+      return mozL10n.get('document_properties_mb', {
+        size_mb: (+(kb / 1024).toPrecision(3)).toLocaleString(),
+        size_b: fileSize.toLocaleString()
+      }, '{{size_mb}} MB ({{size_b}} bytes)');
+    },
+    _parseDate: function PDFDocumentProperties_parseDate(inputDate) {
+      var dateToParse = inputDate;
+      if (dateToParse === undefined) {
+        return '';
+      }
+      if (dateToParse.substring(0, 2) === 'D:') {
+        dateToParse = dateToParse.substring(2);
+      }
+      var year = parseInt(dateToParse.substring(0, 4), 10);
+      var month = parseInt(dateToParse.substring(4, 6), 10) - 1;
+      var day = parseInt(dateToParse.substring(6, 8), 10);
+      var hours = parseInt(dateToParse.substring(8, 10), 10);
+      var minutes = parseInt(dateToParse.substring(10, 12), 10);
+      var seconds = parseInt(dateToParse.substring(12, 14), 10);
+      var utRel = dateToParse.substring(14, 15);
+      var offsetHours = parseInt(dateToParse.substring(15, 17), 10);
+      var offsetMinutes = parseInt(dateToParse.substring(18, 20), 10);
+      if (utRel === '-') {
+        hours += offsetHours;
+        minutes += offsetMinutes;
+      } else if (utRel === '+') {
+        hours -= offsetHours;
+        minutes -= offsetMinutes;
+      }
+      var date = new Date(Date.UTC(year, month, day, hours, minutes, seconds));
+      var dateString = date.toLocaleDateString();
+      var timeString = date.toLocaleTimeString();
+      return mozL10n.get('document_properties_date_string', {
+        date: dateString,
+        time: timeString
+      }, '{{date}}, {{time}}');
+    }
+  };
+  return PDFDocumentProperties;
 }();
 exports.PDFDocumentProperties = PDFDocumentProperties;

+ 146 - 145
lib/web/pdf_find_bar.js

@@ -13,157 +13,158 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtils = require('./ui_utils.js');
 var pdfFindController = require('./pdf_find_controller.js');
 var mozL10n = uiUtils.mozL10n;
 var FindStates = pdfFindController.FindStates;
 var PDFFindBar = function PDFFindBarClosure() {
- function PDFFindBar(options) {
-  this.opened = false;
-  this.bar = options.bar || null;
-  this.toggleButton = options.toggleButton || null;
-  this.findField = options.findField || null;
-  this.highlightAll = options.highlightAllCheckbox || null;
-  this.caseSensitive = options.caseSensitiveCheckbox || null;
-  this.findMsg = options.findMsg || null;
-  this.findResultsCount = options.findResultsCount || null;
-  this.findStatusIcon = options.findStatusIcon || null;
-  this.findPreviousButton = options.findPreviousButton || null;
-  this.findNextButton = options.findNextButton || null;
-  this.findController = options.findController || null;
-  this.eventBus = options.eventBus;
-  if (this.findController === null) {
-   throw new Error('PDFFindBar cannot be used without a ' + 'PDFFindController instance.');
-  }
-  var self = this;
-  this.toggleButton.addEventListener('click', function () {
-   self.toggle();
-  });
-  this.findField.addEventListener('input', function () {
-   self.dispatchEvent('');
-  });
-  this.bar.addEventListener('keydown', function (evt) {
-   switch (evt.keyCode) {
-   case 13:
-    if (evt.target === self.findField) {
-     self.dispatchEvent('again', evt.shiftKey);
-    }
-    break;
-   case 27:
-    self.close();
-    break;
-   }
-  });
-  this.findPreviousButton.addEventListener('click', function () {
-   self.dispatchEvent('again', true);
-  });
-  this.findNextButton.addEventListener('click', function () {
-   self.dispatchEvent('again', false);
-  });
-  this.highlightAll.addEventListener('click', function () {
-   self.dispatchEvent('highlightallchange');
-  });
-  this.caseSensitive.addEventListener('click', function () {
-   self.dispatchEvent('casesensitivitychange');
-  });
-  this.eventBus.on('resize', this._adjustWidth.bind(this));
- }
- PDFFindBar.prototype = {
-  reset: function PDFFindBar_reset() {
-   this.updateUIState();
-  },
-  dispatchEvent: function PDFFindBar_dispatchEvent(type, findPrev) {
-   this.eventBus.dispatch('find', {
-    source: this,
-    type: type,
-    query: this.findField.value,
-    caseSensitive: this.caseSensitive.checked,
-    phraseSearch: true,
-    highlightAll: this.highlightAll.checked,
-    findPrevious: findPrev
-   });
-  },
-  updateUIState: function PDFFindBar_updateUIState(state, previous, matchCount) {
-   var notFound = false;
-   var findMsg = '';
-   var status = '';
-   switch (state) {
-   case FindStates.FIND_FOUND:
-    break;
-   case FindStates.FIND_PENDING:
-    status = 'pending';
-    break;
-   case FindStates.FIND_NOTFOUND:
-    findMsg = mozL10n.get('find_not_found', null, 'Phrase not found');
-    notFound = true;
-    break;
-   case FindStates.FIND_WRAPPED:
-    if (previous) {
-     findMsg = mozL10n.get('find_reached_top', null, 'Reached top of document, continued from bottom');
-    } else {
-     findMsg = mozL10n.get('find_reached_bottom', null, 'Reached end of document, continued from top');
+  function PDFFindBar(options) {
+    this.opened = false;
+    this.bar = options.bar || null;
+    this.toggleButton = options.toggleButton || null;
+    this.findField = options.findField || null;
+    this.highlightAll = options.highlightAllCheckbox || null;
+    this.caseSensitive = options.caseSensitiveCheckbox || null;
+    this.findMsg = options.findMsg || null;
+    this.findResultsCount = options.findResultsCount || null;
+    this.findStatusIcon = options.findStatusIcon || null;
+    this.findPreviousButton = options.findPreviousButton || null;
+    this.findNextButton = options.findNextButton || null;
+    this.findController = options.findController || null;
+    this.eventBus = options.eventBus;
+    if (this.findController === null) {
+      throw new Error('PDFFindBar cannot be used without a ' + 'PDFFindController instance.');
     }
-    break;
-   }
-   if (notFound) {
-    this.findField.classList.add('notFound');
-   } else {
-    this.findField.classList.remove('notFound');
-   }
-   this.findField.setAttribute('data-status', status);
-   this.findMsg.textContent = findMsg;
-   this.updateResultsCount(matchCount);
-   this._adjustWidth();
-  },
-  updateResultsCount: function (matchCount) {
-   if (!this.findResultsCount) {
-    return;
-   }
-   if (!matchCount) {
-    this.findResultsCount.classList.add('hidden');
-    return;
-   }
-   this.findResultsCount.textContent = matchCount.toLocaleString();
-   this.findResultsCount.classList.remove('hidden');
-  },
-  open: function PDFFindBar_open() {
-   if (!this.opened) {
-    this.opened = true;
-    this.toggleButton.classList.add('toggled');
-    this.bar.classList.remove('hidden');
-   }
-   this.findField.select();
-   this.findField.focus();
-   this._adjustWidth();
-  },
-  close: function PDFFindBar_close() {
-   if (!this.opened) {
-    return;
-   }
-   this.opened = false;
-   this.toggleButton.classList.remove('toggled');
-   this.bar.classList.add('hidden');
-   this.findController.active = false;
-  },
-  toggle: function PDFFindBar_toggle() {
-   if (this.opened) {
-    this.close();
-   } else {
-    this.open();
-   }
-  },
-  _adjustWidth: function PDFFindBar_adjustWidth() {
-   if (!this.opened) {
-    return;
-   }
-   this.bar.classList.remove('wrapContainers');
-   var findbarHeight = this.bar.clientHeight;
-   var inputContainerHeight = this.bar.firstElementChild.clientHeight;
-   if (findbarHeight > inputContainerHeight) {
-    this.bar.classList.add('wrapContainers');
-   }
+    var self = this;
+    this.toggleButton.addEventListener('click', function () {
+      self.toggle();
+    });
+    this.findField.addEventListener('input', function () {
+      self.dispatchEvent('');
+    });
+    this.bar.addEventListener('keydown', function (evt) {
+      switch (evt.keyCode) {
+        case 13:
+          if (evt.target === self.findField) {
+            self.dispatchEvent('again', evt.shiftKey);
+          }
+          break;
+        case 27:
+          self.close();
+          break;
+      }
+    });
+    this.findPreviousButton.addEventListener('click', function () {
+      self.dispatchEvent('again', true);
+    });
+    this.findNextButton.addEventListener('click', function () {
+      self.dispatchEvent('again', false);
+    });
+    this.highlightAll.addEventListener('click', function () {
+      self.dispatchEvent('highlightallchange');
+    });
+    this.caseSensitive.addEventListener('click', function () {
+      self.dispatchEvent('casesensitivitychange');
+    });
+    this.eventBus.on('resize', this._adjustWidth.bind(this));
   }
- };
- return PDFFindBar;
+  PDFFindBar.prototype = {
+    reset: function PDFFindBar_reset() {
+      this.updateUIState();
+    },
+    dispatchEvent: function PDFFindBar_dispatchEvent(type, findPrev) {
+      this.eventBus.dispatch('find', {
+        source: this,
+        type: type,
+        query: this.findField.value,
+        caseSensitive: this.caseSensitive.checked,
+        phraseSearch: true,
+        highlightAll: this.highlightAll.checked,
+        findPrevious: findPrev
+      });
+    },
+    updateUIState: function PDFFindBar_updateUIState(state, previous, matchCount) {
+      var notFound = false;
+      var findMsg = '';
+      var status = '';
+      switch (state) {
+        case FindStates.FIND_FOUND:
+          break;
+        case FindStates.FIND_PENDING:
+          status = 'pending';
+          break;
+        case FindStates.FIND_NOTFOUND:
+          findMsg = mozL10n.get('find_not_found', null, 'Phrase not found');
+          notFound = true;
+          break;
+        case FindStates.FIND_WRAPPED:
+          if (previous) {
+            findMsg = mozL10n.get('find_reached_top', null, 'Reached top of document, continued from bottom');
+          } else {
+            findMsg = mozL10n.get('find_reached_bottom', null, 'Reached end of document, continued from top');
+          }
+          break;
+      }
+      if (notFound) {
+        this.findField.classList.add('notFound');
+      } else {
+        this.findField.classList.remove('notFound');
+      }
+      this.findField.setAttribute('data-status', status);
+      this.findMsg.textContent = findMsg;
+      this.updateResultsCount(matchCount);
+      this._adjustWidth();
+    },
+    updateResultsCount: function (matchCount) {
+      if (!this.findResultsCount) {
+        return;
+      }
+      if (!matchCount) {
+        this.findResultsCount.classList.add('hidden');
+        return;
+      }
+      this.findResultsCount.textContent = matchCount.toLocaleString();
+      this.findResultsCount.classList.remove('hidden');
+    },
+    open: function PDFFindBar_open() {
+      if (!this.opened) {
+        this.opened = true;
+        this.toggleButton.classList.add('toggled');
+        this.bar.classList.remove('hidden');
+      }
+      this.findField.select();
+      this.findField.focus();
+      this._adjustWidth();
+    },
+    close: function PDFFindBar_close() {
+      if (!this.opened) {
+        return;
+      }
+      this.opened = false;
+      this.toggleButton.classList.remove('toggled');
+      this.bar.classList.add('hidden');
+      this.findController.active = false;
+    },
+    toggle: function PDFFindBar_toggle() {
+      if (this.opened) {
+        this.close();
+      } else {
+        this.open();
+      }
+    },
+    _adjustWidth: function PDFFindBar_adjustWidth() {
+      if (!this.opened) {
+        return;
+      }
+      this.bar.classList.remove('wrapContainers');
+      var findbarHeight = this.bar.clientHeight;
+      var inputContainerHeight = this.bar.firstElementChild.clientHeight;
+      if (findbarHeight > inputContainerHeight) {
+        this.bar.classList.add('wrapContainers');
+      }
+    }
+  };
+  return PDFFindBar;
 }();
 exports.PDFFindBar = PDFFindBar;

+ 344 - 343
lib/web/pdf_find_controller.js

@@ -13,360 +13,361 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtils = require('./ui_utils.js');
 var scrollIntoView = uiUtils.scrollIntoView;
 var FindStates = {
- FIND_FOUND: 0,
- FIND_NOTFOUND: 1,
- FIND_WRAPPED: 2,
- FIND_PENDING: 3
+  FIND_FOUND: 0,
+  FIND_NOTFOUND: 1,
+  FIND_WRAPPED: 2,
+  FIND_PENDING: 3
 };
 var FIND_SCROLL_OFFSET_TOP = -50;
 var FIND_SCROLL_OFFSET_LEFT = -400;
 var CHARACTERS_TO_NORMALIZE = {
- '\u2018': '\'',
- '\u2019': '\'',
- '\u201A': '\'',
- '\u201B': '\'',
- '\u201C': '"',
- '\u201D': '"',
- '\u201E': '"',
- '\u201F': '"',
- '\u00BC': '1/4',
- '\u00BD': '1/2',
- '\u00BE': '3/4'
+  '\u2018': '\'',
+  '\u2019': '\'',
+  '\u201A': '\'',
+  '\u201B': '\'',
+  '\u201C': '"',
+  '\u201D': '"',
+  '\u201E': '"',
+  '\u201F': '"',
+  '\u00BC': '1/4',
+  '\u00BD': '1/2',
+  '\u00BE': '3/4'
 };
 var PDFFindController = function PDFFindControllerClosure() {
- function PDFFindController(options) {
-  this.pdfViewer = options.pdfViewer || null;
-  this.onUpdateResultsCount = null;
-  this.onUpdateState = null;
-  this.reset();
-  var replace = Object.keys(CHARACTERS_TO_NORMALIZE).join('');
-  this.normalizationRegex = new RegExp('[' + replace + ']', 'g');
- }
- PDFFindController.prototype = {
-  reset: function PDFFindController_reset() {
-   this.startedTextExtraction = false;
-   this.extractTextPromises = [];
-   this.pendingFindMatches = Object.create(null);
-   this.active = false;
-   this.pageContents = [];
-   this.pageMatches = [];
-   this.pageMatchesLength = null;
-   this.matchCount = 0;
-   this.selected = {
-    pageIdx: -1,
-    matchIdx: -1
-   };
-   this.offset = {
-    pageIdx: null,
-    matchIdx: null
-   };
-   this.pagesToSearch = null;
-   this.resumePageIdx = null;
-   this.state = null;
-   this.dirtyMatch = false;
-   this.findTimeout = null;
-   this.firstPagePromise = new Promise(function (resolve) {
-    this.resolveFirstPage = resolve;
-   }.bind(this));
-  },
-  normalize: function PDFFindController_normalize(text) {
-   return text.replace(this.normalizationRegex, function (ch) {
-    return CHARACTERS_TO_NORMALIZE[ch];
-   });
-  },
-  _prepareMatches: function PDFFindController_prepareMatches(matchesWithLength, matches, matchesLength) {
-   function isSubTerm(matchesWithLength, currentIndex) {
-    var currentElem, prevElem, nextElem;
-    currentElem = matchesWithLength[currentIndex];
-    nextElem = matchesWithLength[currentIndex + 1];
-    if (currentIndex < matchesWithLength.length - 1 && currentElem.match === nextElem.match) {
-     currentElem.skipped = true;
-     return true;
-    }
-    for (var i = currentIndex - 1; i >= 0; i--) {
-     prevElem = matchesWithLength[i];
-     if (prevElem.skipped) {
-      continue;
-     }
-     if (prevElem.match + prevElem.matchLength < currentElem.match) {
-      break;
-     }
-     if (prevElem.match + prevElem.matchLength >= currentElem.match + currentElem.matchLength) {
-      currentElem.skipped = true;
-      return true;
-     }
-    }
-    return false;
-   }
-   var i, len;
-   matchesWithLength.sort(function (a, b) {
-    return a.match === b.match ? a.matchLength - b.matchLength : a.match - b.match;
-   });
-   for (i = 0, len = matchesWithLength.length; i < len; i++) {
-    if (isSubTerm(matchesWithLength, i)) {
-     continue;
-    }
-    matches.push(matchesWithLength[i].match);
-    matchesLength.push(matchesWithLength[i].matchLength);
-   }
-  },
-  calcFindPhraseMatch: function PDFFindController_calcFindPhraseMatch(query, pageIndex, pageContent) {
-   var matches = [];
-   var queryLen = query.length;
-   var matchIdx = -queryLen;
-   while (true) {
-    matchIdx = pageContent.indexOf(query, matchIdx + queryLen);
-    if (matchIdx === -1) {
-     break;
-    }
-    matches.push(matchIdx);
-   }
-   this.pageMatches[pageIndex] = matches;
-  },
-  calcFindWordMatch: function PDFFindController_calcFindWordMatch(query, pageIndex, pageContent) {
-   var matchesWithLength = [];
-   var queryArray = query.match(/\S+/g);
-   var subquery, subqueryLen, matchIdx;
-   for (var i = 0, len = queryArray.length; i < len; i++) {
-    subquery = queryArray[i];
-    subqueryLen = subquery.length;
-    matchIdx = -subqueryLen;
-    while (true) {
-     matchIdx = pageContent.indexOf(subquery, matchIdx + subqueryLen);
-     if (matchIdx === -1) {
-      break;
-     }
-     matchesWithLength.push({
-      match: matchIdx,
-      matchLength: subqueryLen,
-      skipped: false
-     });
-    }
-   }
-   if (!this.pageMatchesLength) {
-    this.pageMatchesLength = [];
-   }
-   this.pageMatchesLength[pageIndex] = [];
-   this.pageMatches[pageIndex] = [];
-   this._prepareMatches(matchesWithLength, this.pageMatches[pageIndex], this.pageMatchesLength[pageIndex]);
-  },
-  calcFindMatch: function PDFFindController_calcFindMatch(pageIndex) {
-   var pageContent = this.normalize(this.pageContents[pageIndex]);
-   var query = this.normalize(this.state.query);
-   var caseSensitive = this.state.caseSensitive;
-   var phraseSearch = this.state.phraseSearch;
-   var queryLen = query.length;
-   if (queryLen === 0) {
-    return;
-   }
-   if (!caseSensitive) {
-    pageContent = pageContent.toLowerCase();
-    query = query.toLowerCase();
-   }
-   if (phraseSearch) {
-    this.calcFindPhraseMatch(query, pageIndex, pageContent);
-   } else {
-    this.calcFindWordMatch(query, pageIndex, pageContent);
-   }
-   this.updatePage(pageIndex);
-   if (this.resumePageIdx === pageIndex) {
-    this.resumePageIdx = null;
-    this.nextPageMatch();
-   }
-   if (this.pageMatches[pageIndex].length > 0) {
-    this.matchCount += this.pageMatches[pageIndex].length;
-    this.updateUIResultsCount();
-   }
-  },
-  extractText: function PDFFindController_extractText() {
-   if (this.startedTextExtraction) {
-    return;
-   }
-   this.startedTextExtraction = true;
-   this.pageContents = [];
-   var extractTextPromisesResolves = [];
-   var numPages = this.pdfViewer.pagesCount;
-   for (var i = 0; i < numPages; i++) {
-    this.extractTextPromises.push(new Promise(function (resolve) {
-     extractTextPromisesResolves.push(resolve);
-    }));
-   }
-   var self = this;
-   function extractPageText(pageIndex) {
-    self.pdfViewer.getPageTextContent(pageIndex).then(function textContentResolved(textContent) {
-     var textItems = textContent.items;
-     var str = [];
-     for (var i = 0, len = textItems.length; i < len; i++) {
-      str.push(textItems[i].str);
-     }
-     self.pageContents.push(str.join(''));
-     extractTextPromisesResolves[pageIndex](pageIndex);
-     if (pageIndex + 1 < self.pdfViewer.pagesCount) {
-      extractPageText(pageIndex + 1);
-     }
-    });
-   }
-   extractPageText(0);
-  },
-  executeCommand: function PDFFindController_executeCommand(cmd, state) {
-   if (this.state === null || cmd !== 'findagain') {
-    this.dirtyMatch = true;
-   }
-   this.state = state;
-   this.updateUIState(FindStates.FIND_PENDING);
-   this.firstPagePromise.then(function () {
-    this.extractText();
-    clearTimeout(this.findTimeout);
-    if (cmd === 'find') {
-     this.findTimeout = setTimeout(this.nextMatch.bind(this), 250);
-    } else {
-     this.nextMatch();
-    }
-   }.bind(this));
-  },
-  updatePage: function PDFFindController_updatePage(index) {
-   if (this.selected.pageIdx === index) {
-    this.pdfViewer.currentPageNumber = index + 1;
-   }
-   var page = this.pdfViewer.getPageView(index);
-   if (page.textLayer) {
-    page.textLayer.updateMatches();
-   }
-  },
-  nextMatch: function PDFFindController_nextMatch() {
-   var previous = this.state.findPrevious;
-   var currentPageIndex = this.pdfViewer.currentPageNumber - 1;
-   var numPages = this.pdfViewer.pagesCount;
-   this.active = true;
-   if (this.dirtyMatch) {
-    this.dirtyMatch = false;
-    this.selected.pageIdx = this.selected.matchIdx = -1;
-    this.offset.pageIdx = currentPageIndex;
-    this.offset.matchIdx = null;
-    this.hadMatch = false;
-    this.resumePageIdx = null;
-    this.pageMatches = [];
-    this.matchCount = 0;
-    this.pageMatchesLength = null;
-    var self = this;
-    for (var i = 0; i < numPages; i++) {
-     this.updatePage(i);
-     if (!(i in this.pendingFindMatches)) {
-      this.pendingFindMatches[i] = true;
-      this.extractTextPromises[i].then(function (pageIdx) {
-       delete self.pendingFindMatches[pageIdx];
-       self.calcFindMatch(pageIdx);
+  function PDFFindController(options) {
+    this.pdfViewer = options.pdfViewer || null;
+    this.onUpdateResultsCount = null;
+    this.onUpdateState = null;
+    this.reset();
+    var replace = Object.keys(CHARACTERS_TO_NORMALIZE).join('');
+    this.normalizationRegex = new RegExp('[' + replace + ']', 'g');
+  }
+  PDFFindController.prototype = {
+    reset: function PDFFindController_reset() {
+      this.startedTextExtraction = false;
+      this.extractTextPromises = [];
+      this.pendingFindMatches = Object.create(null);
+      this.active = false;
+      this.pageContents = [];
+      this.pageMatches = [];
+      this.pageMatchesLength = null;
+      this.matchCount = 0;
+      this.selected = {
+        pageIdx: -1,
+        matchIdx: -1
+      };
+      this.offset = {
+        pageIdx: null,
+        matchIdx: null
+      };
+      this.pagesToSearch = null;
+      this.resumePageIdx = null;
+      this.state = null;
+      this.dirtyMatch = false;
+      this.findTimeout = null;
+      this.firstPagePromise = new Promise(function (resolve) {
+        this.resolveFirstPage = resolve;
+      }.bind(this));
+    },
+    normalize: function PDFFindController_normalize(text) {
+      return text.replace(this.normalizationRegex, function (ch) {
+        return CHARACTERS_TO_NORMALIZE[ch];
       });
-     }
-    }
-   }
-   if (this.state.query === '') {
-    this.updateUIState(FindStates.FIND_FOUND);
-    return;
-   }
-   if (this.resumePageIdx) {
-    return;
-   }
-   var offset = this.offset;
-   this.pagesToSearch = numPages;
-   if (offset.matchIdx !== null) {
-    var numPageMatches = this.pageMatches[offset.pageIdx].length;
-    if (!previous && offset.matchIdx + 1 < numPageMatches || previous && offset.matchIdx > 0) {
-     this.hadMatch = true;
-     offset.matchIdx = previous ? offset.matchIdx - 1 : offset.matchIdx + 1;
-     this.updateMatch(true);
-     return;
-    }
-    this.advanceOffsetPage(previous);
-   }
-   this.nextPageMatch();
-  },
-  matchesReady: function PDFFindController_matchesReady(matches) {
-   var offset = this.offset;
-   var numMatches = matches.length;
-   var previous = this.state.findPrevious;
-   if (numMatches) {
-    this.hadMatch = true;
-    offset.matchIdx = previous ? numMatches - 1 : 0;
-    this.updateMatch(true);
-    return true;
-   }
-   this.advanceOffsetPage(previous);
-   if (offset.wrapped) {
-    offset.matchIdx = null;
-    if (this.pagesToSearch < 0) {
-     this.updateMatch(false);
-     return true;
-    }
-   }
-   return false;
-  },
-  updateMatchPosition: function PDFFindController_updateMatchPosition(pageIndex, index, elements, beginIdx) {
-   if (this.selected.matchIdx === index && this.selected.pageIdx === pageIndex) {
-    var spot = {
-     top: FIND_SCROLL_OFFSET_TOP,
-     left: FIND_SCROLL_OFFSET_LEFT
-    };
-    scrollIntoView(elements[beginIdx], spot, true);
-   }
-  },
-  nextPageMatch: function PDFFindController_nextPageMatch() {
-   if (this.resumePageIdx !== null) {
-    console.error('There can only be one pending page.');
-   }
-   do {
-    var pageIdx = this.offset.pageIdx;
-    var matches = this.pageMatches[pageIdx];
-    if (!matches) {
-     this.resumePageIdx = pageIdx;
-     break;
-    }
-   } while (!this.matchesReady(matches));
-  },
-  advanceOffsetPage: function PDFFindController_advanceOffsetPage(previous) {
-   var offset = this.offset;
-   var numPages = this.extractTextPromises.length;
-   offset.pageIdx = previous ? offset.pageIdx - 1 : offset.pageIdx + 1;
-   offset.matchIdx = null;
-   this.pagesToSearch--;
-   if (offset.pageIdx >= numPages || offset.pageIdx < 0) {
-    offset.pageIdx = previous ? numPages - 1 : 0;
-    offset.wrapped = true;
-   }
-  },
-  updateMatch: function PDFFindController_updateMatch(found) {
-   var state = FindStates.FIND_NOTFOUND;
-   var wrapped = this.offset.wrapped;
-   this.offset.wrapped = false;
-   if (found) {
-    var previousPage = this.selected.pageIdx;
-    this.selected.pageIdx = this.offset.pageIdx;
-    this.selected.matchIdx = this.offset.matchIdx;
-    state = wrapped ? FindStates.FIND_WRAPPED : FindStates.FIND_FOUND;
-    if (previousPage !== -1 && previousPage !== this.selected.pageIdx) {
-     this.updatePage(previousPage);
+    },
+    _prepareMatches: function PDFFindController_prepareMatches(matchesWithLength, matches, matchesLength) {
+      function isSubTerm(matchesWithLength, currentIndex) {
+        var currentElem, prevElem, nextElem;
+        currentElem = matchesWithLength[currentIndex];
+        nextElem = matchesWithLength[currentIndex + 1];
+        if (currentIndex < matchesWithLength.length - 1 && currentElem.match === nextElem.match) {
+          currentElem.skipped = true;
+          return true;
+        }
+        for (var i = currentIndex - 1; i >= 0; i--) {
+          prevElem = matchesWithLength[i];
+          if (prevElem.skipped) {
+            continue;
+          }
+          if (prevElem.match + prevElem.matchLength < currentElem.match) {
+            break;
+          }
+          if (prevElem.match + prevElem.matchLength >= currentElem.match + currentElem.matchLength) {
+            currentElem.skipped = true;
+            return true;
+          }
+        }
+        return false;
+      }
+      var i, len;
+      matchesWithLength.sort(function (a, b) {
+        return a.match === b.match ? a.matchLength - b.matchLength : a.match - b.match;
+      });
+      for (i = 0, len = matchesWithLength.length; i < len; i++) {
+        if (isSubTerm(matchesWithLength, i)) {
+          continue;
+        }
+        matches.push(matchesWithLength[i].match);
+        matchesLength.push(matchesWithLength[i].matchLength);
+      }
+    },
+    calcFindPhraseMatch: function PDFFindController_calcFindPhraseMatch(query, pageIndex, pageContent) {
+      var matches = [];
+      var queryLen = query.length;
+      var matchIdx = -queryLen;
+      while (true) {
+        matchIdx = pageContent.indexOf(query, matchIdx + queryLen);
+        if (matchIdx === -1) {
+          break;
+        }
+        matches.push(matchIdx);
+      }
+      this.pageMatches[pageIndex] = matches;
+    },
+    calcFindWordMatch: function PDFFindController_calcFindWordMatch(query, pageIndex, pageContent) {
+      var matchesWithLength = [];
+      var queryArray = query.match(/\S+/g);
+      var subquery, subqueryLen, matchIdx;
+      for (var i = 0, len = queryArray.length; i < len; i++) {
+        subquery = queryArray[i];
+        subqueryLen = subquery.length;
+        matchIdx = -subqueryLen;
+        while (true) {
+          matchIdx = pageContent.indexOf(subquery, matchIdx + subqueryLen);
+          if (matchIdx === -1) {
+            break;
+          }
+          matchesWithLength.push({
+            match: matchIdx,
+            matchLength: subqueryLen,
+            skipped: false
+          });
+        }
+      }
+      if (!this.pageMatchesLength) {
+        this.pageMatchesLength = [];
+      }
+      this.pageMatchesLength[pageIndex] = [];
+      this.pageMatches[pageIndex] = [];
+      this._prepareMatches(matchesWithLength, this.pageMatches[pageIndex], this.pageMatchesLength[pageIndex]);
+    },
+    calcFindMatch: function PDFFindController_calcFindMatch(pageIndex) {
+      var pageContent = this.normalize(this.pageContents[pageIndex]);
+      var query = this.normalize(this.state.query);
+      var caseSensitive = this.state.caseSensitive;
+      var phraseSearch = this.state.phraseSearch;
+      var queryLen = query.length;
+      if (queryLen === 0) {
+        return;
+      }
+      if (!caseSensitive) {
+        pageContent = pageContent.toLowerCase();
+        query = query.toLowerCase();
+      }
+      if (phraseSearch) {
+        this.calcFindPhraseMatch(query, pageIndex, pageContent);
+      } else {
+        this.calcFindWordMatch(query, pageIndex, pageContent);
+      }
+      this.updatePage(pageIndex);
+      if (this.resumePageIdx === pageIndex) {
+        this.resumePageIdx = null;
+        this.nextPageMatch();
+      }
+      if (this.pageMatches[pageIndex].length > 0) {
+        this.matchCount += this.pageMatches[pageIndex].length;
+        this.updateUIResultsCount();
+      }
+    },
+    extractText: function PDFFindController_extractText() {
+      if (this.startedTextExtraction) {
+        return;
+      }
+      this.startedTextExtraction = true;
+      this.pageContents = [];
+      var extractTextPromisesResolves = [];
+      var numPages = this.pdfViewer.pagesCount;
+      for (var i = 0; i < numPages; i++) {
+        this.extractTextPromises.push(new Promise(function (resolve) {
+          extractTextPromisesResolves.push(resolve);
+        }));
+      }
+      var self = this;
+      function extractPageText(pageIndex) {
+        self.pdfViewer.getPageTextContent(pageIndex).then(function textContentResolved(textContent) {
+          var textItems = textContent.items;
+          var str = [];
+          for (var i = 0, len = textItems.length; i < len; i++) {
+            str.push(textItems[i].str);
+          }
+          self.pageContents.push(str.join(''));
+          extractTextPromisesResolves[pageIndex](pageIndex);
+          if (pageIndex + 1 < self.pdfViewer.pagesCount) {
+            extractPageText(pageIndex + 1);
+          }
+        });
+      }
+      extractPageText(0);
+    },
+    executeCommand: function PDFFindController_executeCommand(cmd, state) {
+      if (this.state === null || cmd !== 'findagain') {
+        this.dirtyMatch = true;
+      }
+      this.state = state;
+      this.updateUIState(FindStates.FIND_PENDING);
+      this.firstPagePromise.then(function () {
+        this.extractText();
+        clearTimeout(this.findTimeout);
+        if (cmd === 'find') {
+          this.findTimeout = setTimeout(this.nextMatch.bind(this), 250);
+        } else {
+          this.nextMatch();
+        }
+      }.bind(this));
+    },
+    updatePage: function PDFFindController_updatePage(index) {
+      if (this.selected.pageIdx === index) {
+        this.pdfViewer.currentPageNumber = index + 1;
+      }
+      var page = this.pdfViewer.getPageView(index);
+      if (page.textLayer) {
+        page.textLayer.updateMatches();
+      }
+    },
+    nextMatch: function PDFFindController_nextMatch() {
+      var previous = this.state.findPrevious;
+      var currentPageIndex = this.pdfViewer.currentPageNumber - 1;
+      var numPages = this.pdfViewer.pagesCount;
+      this.active = true;
+      if (this.dirtyMatch) {
+        this.dirtyMatch = false;
+        this.selected.pageIdx = this.selected.matchIdx = -1;
+        this.offset.pageIdx = currentPageIndex;
+        this.offset.matchIdx = null;
+        this.hadMatch = false;
+        this.resumePageIdx = null;
+        this.pageMatches = [];
+        this.matchCount = 0;
+        this.pageMatchesLength = null;
+        var self = this;
+        for (var i = 0; i < numPages; i++) {
+          this.updatePage(i);
+          if (!(i in this.pendingFindMatches)) {
+            this.pendingFindMatches[i] = true;
+            this.extractTextPromises[i].then(function (pageIdx) {
+              delete self.pendingFindMatches[pageIdx];
+              self.calcFindMatch(pageIdx);
+            });
+          }
+        }
+      }
+      if (this.state.query === '') {
+        this.updateUIState(FindStates.FIND_FOUND);
+        return;
+      }
+      if (this.resumePageIdx) {
+        return;
+      }
+      var offset = this.offset;
+      this.pagesToSearch = numPages;
+      if (offset.matchIdx !== null) {
+        var numPageMatches = this.pageMatches[offset.pageIdx].length;
+        if (!previous && offset.matchIdx + 1 < numPageMatches || previous && offset.matchIdx > 0) {
+          this.hadMatch = true;
+          offset.matchIdx = previous ? offset.matchIdx - 1 : offset.matchIdx + 1;
+          this.updateMatch(true);
+          return;
+        }
+        this.advanceOffsetPage(previous);
+      }
+      this.nextPageMatch();
+    },
+    matchesReady: function PDFFindController_matchesReady(matches) {
+      var offset = this.offset;
+      var numMatches = matches.length;
+      var previous = this.state.findPrevious;
+      if (numMatches) {
+        this.hadMatch = true;
+        offset.matchIdx = previous ? numMatches - 1 : 0;
+        this.updateMatch(true);
+        return true;
+      }
+      this.advanceOffsetPage(previous);
+      if (offset.wrapped) {
+        offset.matchIdx = null;
+        if (this.pagesToSearch < 0) {
+          this.updateMatch(false);
+          return true;
+        }
+      }
+      return false;
+    },
+    updateMatchPosition: function PDFFindController_updateMatchPosition(pageIndex, index, elements, beginIdx) {
+      if (this.selected.matchIdx === index && this.selected.pageIdx === pageIndex) {
+        var spot = {
+          top: FIND_SCROLL_OFFSET_TOP,
+          left: FIND_SCROLL_OFFSET_LEFT
+        };
+        scrollIntoView(elements[beginIdx], spot, true);
+      }
+    },
+    nextPageMatch: function PDFFindController_nextPageMatch() {
+      if (this.resumePageIdx !== null) {
+        console.error('There can only be one pending page.');
+      }
+      do {
+        var pageIdx = this.offset.pageIdx;
+        var matches = this.pageMatches[pageIdx];
+        if (!matches) {
+          this.resumePageIdx = pageIdx;
+          break;
+        }
+      } while (!this.matchesReady(matches));
+    },
+    advanceOffsetPage: function PDFFindController_advanceOffsetPage(previous) {
+      var offset = this.offset;
+      var numPages = this.extractTextPromises.length;
+      offset.pageIdx = previous ? offset.pageIdx - 1 : offset.pageIdx + 1;
+      offset.matchIdx = null;
+      this.pagesToSearch--;
+      if (offset.pageIdx >= numPages || offset.pageIdx < 0) {
+        offset.pageIdx = previous ? numPages - 1 : 0;
+        offset.wrapped = true;
+      }
+    },
+    updateMatch: function PDFFindController_updateMatch(found) {
+      var state = FindStates.FIND_NOTFOUND;
+      var wrapped = this.offset.wrapped;
+      this.offset.wrapped = false;
+      if (found) {
+        var previousPage = this.selected.pageIdx;
+        this.selected.pageIdx = this.offset.pageIdx;
+        this.selected.matchIdx = this.offset.matchIdx;
+        state = wrapped ? FindStates.FIND_WRAPPED : FindStates.FIND_FOUND;
+        if (previousPage !== -1 && previousPage !== this.selected.pageIdx) {
+          this.updatePage(previousPage);
+        }
+      }
+      this.updateUIState(state, this.state.findPrevious);
+      if (this.selected.pageIdx !== -1) {
+        this.updatePage(this.selected.pageIdx);
+      }
+    },
+    updateUIResultsCount: function PDFFindController_updateUIResultsCount() {
+      if (this.onUpdateResultsCount) {
+        this.onUpdateResultsCount(this.matchCount);
+      }
+    },
+    updateUIState: function PDFFindController_updateUIState(state, previous) {
+      if (this.onUpdateState) {
+        this.onUpdateState(state, previous, this.matchCount);
+      }
     }
-   }
-   this.updateUIState(state, this.state.findPrevious);
-   if (this.selected.pageIdx !== -1) {
-    this.updatePage(this.selected.pageIdx);
-   }
-  },
-  updateUIResultsCount: function PDFFindController_updateUIResultsCount() {
-   if (this.onUpdateResultsCount) {
-    this.onUpdateResultsCount(this.matchCount);
-   }
-  },
-  updateUIState: function PDFFindController_updateUIState(state, previous) {
-   if (this.onUpdateState) {
-    this.onUpdateState(state, previous, this.matchCount);
-   }
-  }
- };
- return PDFFindController;
+  };
+  return PDFFindController;
 }();
 exports.FindStates = FindStates;
 exports.PDFFindController = PDFFindController;

+ 282 - 281
lib/web/pdf_history.js

@@ -13,296 +13,297 @@
  * limitations under the License.
  */
 'use strict';
+
 var domEvents = require('./dom_events.js');
 function PDFHistory(options) {
- this.linkService = options.linkService;
- this.eventBus = options.eventBus || domEvents.getGlobalEventBus();
- this.initialized = false;
- this.initialDestination = null;
- this.initialBookmark = null;
+  this.linkService = options.linkService;
+  this.eventBus = options.eventBus || domEvents.getGlobalEventBus();
+  this.initialized = false;
+  this.initialDestination = null;
+  this.initialBookmark = null;
 }
 PDFHistory.prototype = {
- initialize: function pdfHistoryInitialize(fingerprint) {
-  this.initialized = true;
-  this.reInitialized = false;
-  this.allowHashChange = true;
-  this.historyUnlocked = true;
-  this.isViewerInPresentationMode = false;
-  this.previousHash = window.location.hash.substring(1);
-  this.currentBookmark = '';
-  this.currentPage = 0;
-  this.updatePreviousBookmark = false;
-  this.previousBookmark = '';
-  this.previousPage = 0;
-  this.nextHashParam = '';
-  this.fingerprint = fingerprint;
-  this.currentUid = this.uid = 0;
-  this.current = {};
-  var state = window.history.state;
-  if (this._isStateObjectDefined(state)) {
-   if (state.target.dest) {
-    this.initialDestination = state.target.dest;
-   } else {
-    this.initialBookmark = state.target.hash;
-   }
-   this.currentUid = state.uid;
-   this.uid = state.uid + 1;
-   this.current = state.target;
-  } else {
-   if (state && state.fingerprint && this.fingerprint !== state.fingerprint) {
-    this.reInitialized = true;
-   }
-   this._pushOrReplaceState({ fingerprint: this.fingerprint }, true);
-  }
-  var self = this;
-  window.addEventListener('popstate', function pdfHistoryPopstate(evt) {
-   if (!self.historyUnlocked) {
-    return;
-   }
-   if (evt.state) {
-    self._goTo(evt.state);
-    return;
-   }
-   if (self.uid === 0) {
-    var previousParams = self.previousHash && self.currentBookmark && self.previousHash !== self.currentBookmark ? {
-     hash: self.currentBookmark,
-     page: self.currentPage
-    } : { page: 1 };
-    replacePreviousHistoryState(previousParams, function () {
-     updateHistoryWithCurrentHash();
-    });
-   } else {
-    updateHistoryWithCurrentHash();
-   }
-  });
-  function updateHistoryWithCurrentHash() {
-   self.previousHash = window.location.hash.slice(1);
-   self._pushToHistory({ hash: self.previousHash }, false, true);
-   self._updatePreviousBookmark();
-  }
-  function replacePreviousHistoryState(params, callback) {
-   self.historyUnlocked = false;
-   self.allowHashChange = false;
-   window.addEventListener('popstate', rewriteHistoryAfterBack);
-   history.back();
-   function rewriteHistoryAfterBack() {
-    window.removeEventListener('popstate', rewriteHistoryAfterBack);
-    window.addEventListener('popstate', rewriteHistoryAfterForward);
-    self._pushToHistory(params, false, true);
-    history.forward();
-   }
-   function rewriteHistoryAfterForward() {
-    window.removeEventListener('popstate', rewriteHistoryAfterForward);
-    self.allowHashChange = true;
-    self.historyUnlocked = true;
-    callback();
-   }
-  }
-  function pdfHistoryBeforeUnload() {
-   var previousParams = self._getPreviousParams(null, true);
-   if (previousParams) {
-    var replacePrevious = !self.current.dest && self.current.hash !== self.previousHash;
-    self._pushToHistory(previousParams, false, replacePrevious);
-    self._updatePreviousBookmark();
-   }
-   window.removeEventListener('beforeunload', pdfHistoryBeforeUnload);
-  }
-  window.addEventListener('beforeunload', pdfHistoryBeforeUnload);
-  window.addEventListener('pageshow', function pdfHistoryPageShow(evt) {
-   window.addEventListener('beforeunload', pdfHistoryBeforeUnload);
-  });
-  self.eventBus.on('presentationmodechanged', function (e) {
-   self.isViewerInPresentationMode = e.active;
-  });
- },
- clearHistoryState: function pdfHistory_clearHistoryState() {
-  this._pushOrReplaceState(null, true);
- },
- _isStateObjectDefined: function pdfHistory_isStateObjectDefined(state) {
-  return state && state.uid >= 0 && state.fingerprint && this.fingerprint === state.fingerprint && state.target && state.target.hash ? true : false;
- },
- _pushOrReplaceState: function pdfHistory_pushOrReplaceState(stateObj, replace) {
-  if (replace) {
-   window.history.replaceState(stateObj, '', document.URL);
-  } else {
-   window.history.pushState(stateObj, '', document.URL);
-  }
- },
- get isHashChangeUnlocked() {
-  if (!this.initialized) {
-   return true;
-  }
-  return this.allowHashChange;
- },
- _updatePreviousBookmark: function pdfHistory_updatePreviousBookmark() {
-  if (this.updatePreviousBookmark && this.currentBookmark && this.currentPage) {
-   this.previousBookmark = this.currentBookmark;
-   this.previousPage = this.currentPage;
-   this.updatePreviousBookmark = false;
-  }
- },
- updateCurrentBookmark: function pdfHistoryUpdateCurrentBookmark(bookmark, pageNum) {
-  if (this.initialized) {
-   this.currentBookmark = bookmark.substring(1);
-   this.currentPage = pageNum | 0;
-   this._updatePreviousBookmark();
-  }
- },
- updateNextHashParam: function pdfHistoryUpdateNextHashParam(param) {
-  if (this.initialized) {
-   this.nextHashParam = param;
-  }
- },
- push: function pdfHistoryPush(params, isInitialBookmark) {
-  if (!(this.initialized && this.historyUnlocked)) {
-   return;
-  }
-  if (params.dest && !params.hash) {
-   params.hash = this.current.hash && this.current.dest && this.current.dest === params.dest ? this.current.hash : this.linkService.getDestinationHash(params.dest).split('#')[1];
-  }
-  if (params.page) {
-   params.page |= 0;
-  }
-  if (isInitialBookmark) {
-   var target = window.history.state.target;
-   if (!target) {
-    this._pushToHistory(params, false);
+  initialize: function pdfHistoryInitialize(fingerprint) {
+    this.initialized = true;
+    this.reInitialized = false;
+    this.allowHashChange = true;
+    this.historyUnlocked = true;
+    this.isViewerInPresentationMode = false;
     this.previousHash = window.location.hash.substring(1);
-   }
-   this.updatePreviousBookmark = this.nextHashParam ? false : true;
-   if (target) {
-    this._updatePreviousBookmark();
-   }
-   return;
-  }
-  if (this.nextHashParam) {
-   if (this.nextHashParam === params.hash) {
-    this.nextHashParam = null;
+    this.currentBookmark = '';
+    this.currentPage = 0;
+    this.updatePreviousBookmark = false;
+    this.previousBookmark = '';
+    this.previousPage = 0;
+    this.nextHashParam = '';
+    this.fingerprint = fingerprint;
+    this.currentUid = this.uid = 0;
+    this.current = {};
+    var state = window.history.state;
+    if (this._isStateObjectDefined(state)) {
+      if (state.target.dest) {
+        this.initialDestination = state.target.dest;
+      } else {
+        this.initialBookmark = state.target.hash;
+      }
+      this.currentUid = state.uid;
+      this.uid = state.uid + 1;
+      this.current = state.target;
+    } else {
+      if (state && state.fingerprint && this.fingerprint !== state.fingerprint) {
+        this.reInitialized = true;
+      }
+      this._pushOrReplaceState({ fingerprint: this.fingerprint }, true);
+    }
+    var self = this;
+    window.addEventListener('popstate', function pdfHistoryPopstate(evt) {
+      if (!self.historyUnlocked) {
+        return;
+      }
+      if (evt.state) {
+        self._goTo(evt.state);
+        return;
+      }
+      if (self.uid === 0) {
+        var previousParams = self.previousHash && self.currentBookmark && self.previousHash !== self.currentBookmark ? {
+          hash: self.currentBookmark,
+          page: self.currentPage
+        } : { page: 1 };
+        replacePreviousHistoryState(previousParams, function () {
+          updateHistoryWithCurrentHash();
+        });
+      } else {
+        updateHistoryWithCurrentHash();
+      }
+    });
+    function updateHistoryWithCurrentHash() {
+      self.previousHash = window.location.hash.slice(1);
+      self._pushToHistory({ hash: self.previousHash }, false, true);
+      self._updatePreviousBookmark();
+    }
+    function replacePreviousHistoryState(params, callback) {
+      self.historyUnlocked = false;
+      self.allowHashChange = false;
+      window.addEventListener('popstate', rewriteHistoryAfterBack);
+      history.back();
+      function rewriteHistoryAfterBack() {
+        window.removeEventListener('popstate', rewriteHistoryAfterBack);
+        window.addEventListener('popstate', rewriteHistoryAfterForward);
+        self._pushToHistory(params, false, true);
+        history.forward();
+      }
+      function rewriteHistoryAfterForward() {
+        window.removeEventListener('popstate', rewriteHistoryAfterForward);
+        self.allowHashChange = true;
+        self.historyUnlocked = true;
+        callback();
+      }
+    }
+    function pdfHistoryBeforeUnload() {
+      var previousParams = self._getPreviousParams(null, true);
+      if (previousParams) {
+        var replacePrevious = !self.current.dest && self.current.hash !== self.previousHash;
+        self._pushToHistory(previousParams, false, replacePrevious);
+        self._updatePreviousBookmark();
+      }
+      window.removeEventListener('beforeunload', pdfHistoryBeforeUnload);
+    }
+    window.addEventListener('beforeunload', pdfHistoryBeforeUnload);
+    window.addEventListener('pageshow', function pdfHistoryPageShow(evt) {
+      window.addEventListener('beforeunload', pdfHistoryBeforeUnload);
+    });
+    self.eventBus.on('presentationmodechanged', function (e) {
+      self.isViewerInPresentationMode = e.active;
+    });
+  },
+  clearHistoryState: function pdfHistory_clearHistoryState() {
+    this._pushOrReplaceState(null, true);
+  },
+  _isStateObjectDefined: function pdfHistory_isStateObjectDefined(state) {
+    return state && state.uid >= 0 && state.fingerprint && this.fingerprint === state.fingerprint && state.target && state.target.hash ? true : false;
+  },
+  _pushOrReplaceState: function pdfHistory_pushOrReplaceState(stateObj, replace) {
+    if (replace) {
+      window.history.replaceState(stateObj, '', document.URL);
+    } else {
+      window.history.pushState(stateObj, '', document.URL);
+    }
+  },
+  get isHashChangeUnlocked() {
+    if (!this.initialized) {
+      return true;
+    }
+    return this.allowHashChange;
+  },
+  _updatePreviousBookmark: function pdfHistory_updatePreviousBookmark() {
+    if (this.updatePreviousBookmark && this.currentBookmark && this.currentPage) {
+      this.previousBookmark = this.currentBookmark;
+      this.previousPage = this.currentPage;
+      this.updatePreviousBookmark = false;
+    }
+  },
+  updateCurrentBookmark: function pdfHistoryUpdateCurrentBookmark(bookmark, pageNum) {
+    if (this.initialized) {
+      this.currentBookmark = bookmark.substring(1);
+      this.currentPage = pageNum | 0;
+      this._updatePreviousBookmark();
+    }
+  },
+  updateNextHashParam: function pdfHistoryUpdateNextHashParam(param) {
+    if (this.initialized) {
+      this.nextHashParam = param;
+    }
+  },
+  push: function pdfHistoryPush(params, isInitialBookmark) {
+    if (!(this.initialized && this.historyUnlocked)) {
+      return;
+    }
+    if (params.dest && !params.hash) {
+      params.hash = this.current.hash && this.current.dest && this.current.dest === params.dest ? this.current.hash : this.linkService.getDestinationHash(params.dest).split('#')[1];
+    }
+    if (params.page) {
+      params.page |= 0;
+    }
+    if (isInitialBookmark) {
+      var target = window.history.state.target;
+      if (!target) {
+        this._pushToHistory(params, false);
+        this.previousHash = window.location.hash.substring(1);
+      }
+      this.updatePreviousBookmark = this.nextHashParam ? false : true;
+      if (target) {
+        this._updatePreviousBookmark();
+      }
+      return;
+    }
+    if (this.nextHashParam) {
+      if (this.nextHashParam === params.hash) {
+        this.nextHashParam = null;
+        this.updatePreviousBookmark = true;
+        return;
+      }
+      this.nextHashParam = null;
+    }
+    if (params.hash) {
+      if (this.current.hash) {
+        if (this.current.hash !== params.hash) {
+          this._pushToHistory(params, true);
+        } else {
+          if (!this.current.page && params.page) {
+            this._pushToHistory(params, false, true);
+          }
+          this.updatePreviousBookmark = true;
+        }
+      } else {
+        this._pushToHistory(params, true);
+      }
+    } else if (this.current.page && params.page && this.current.page !== params.page) {
+      this._pushToHistory(params, true);
+    }
+  },
+  _getPreviousParams: function pdfHistory_getPreviousParams(onlyCheckPage, beforeUnload) {
+    if (!(this.currentBookmark && this.currentPage)) {
+      return null;
+    } else if (this.updatePreviousBookmark) {
+      this.updatePreviousBookmark = false;
+    }
+    if (this.uid > 0 && !(this.previousBookmark && this.previousPage)) {
+      return null;
+    }
+    if (!this.current.dest && !onlyCheckPage || beforeUnload) {
+      if (this.previousBookmark === this.currentBookmark) {
+        return null;
+      }
+    } else if (this.current.page || onlyCheckPage) {
+      if (this.previousPage === this.currentPage) {
+        return null;
+      }
+    } else {
+      return null;
+    }
+    var params = {
+      hash: this.currentBookmark,
+      page: this.currentPage
+    };
+    if (this.isViewerInPresentationMode) {
+      params.hash = null;
+    }
+    return params;
+  },
+  _stateObj: function pdfHistory_stateObj(params) {
+    return {
+      fingerprint: this.fingerprint,
+      uid: this.uid,
+      target: params
+    };
+  },
+  _pushToHistory: function pdfHistory_pushToHistory(params, addPrevious, overwrite) {
+    if (!this.initialized) {
+      return;
+    }
+    if (!params.hash && params.page) {
+      params.hash = 'page=' + params.page;
+    }
+    if (addPrevious && !overwrite) {
+      var previousParams = this._getPreviousParams();
+      if (previousParams) {
+        var replacePrevious = !this.current.dest && this.current.hash !== this.previousHash;
+        this._pushToHistory(previousParams, false, replacePrevious);
+      }
+    }
+    this._pushOrReplaceState(this._stateObj(params), overwrite || this.uid === 0);
+    this.currentUid = this.uid++;
+    this.current = params;
     this.updatePreviousBookmark = true;
-    return;
-   }
-   this.nextHashParam = null;
-  }
-  if (params.hash) {
-   if (this.current.hash) {
-    if (this.current.hash !== params.hash) {
-     this._pushToHistory(params, true);
+  },
+  _goTo: function pdfHistory_goTo(state) {
+    if (!(this.initialized && this.historyUnlocked && this._isStateObjectDefined(state))) {
+      return;
+    }
+    if (!this.reInitialized && state.uid < this.currentUid) {
+      var previousParams = this._getPreviousParams(true);
+      if (previousParams) {
+        this._pushToHistory(this.current, false);
+        this._pushToHistory(previousParams, false);
+        this.currentUid = state.uid;
+        window.history.back();
+        return;
+      }
+    }
+    this.historyUnlocked = false;
+    if (state.target.dest) {
+      this.linkService.navigateTo(state.target.dest);
     } else {
-     if (!this.current.page && params.page) {
-      this._pushToHistory(params, false, true);
-     }
-     this.updatePreviousBookmark = true;
+      this.linkService.setHash(state.target.hash);
     }
-   } else {
-    this._pushToHistory(params, true);
-   }
-  } else if (this.current.page && params.page && this.current.page !== params.page) {
-   this._pushToHistory(params, true);
-  }
- },
- _getPreviousParams: function pdfHistory_getPreviousParams(onlyCheckPage, beforeUnload) {
-  if (!(this.currentBookmark && this.currentPage)) {
-   return null;
-  } else if (this.updatePreviousBookmark) {
-   this.updatePreviousBookmark = false;
-  }
-  if (this.uid > 0 && !(this.previousBookmark && this.previousPage)) {
-   return null;
-  }
-  if (!this.current.dest && !onlyCheckPage || beforeUnload) {
-   if (this.previousBookmark === this.currentBookmark) {
-    return null;
-   }
-  } else if (this.current.page || onlyCheckPage) {
-   if (this.previousPage === this.currentPage) {
-    return null;
-   }
-  } else {
-   return null;
-  }
-  var params = {
-   hash: this.currentBookmark,
-   page: this.currentPage
-  };
-  if (this.isViewerInPresentationMode) {
-   params.hash = null;
-  }
-  return params;
- },
- _stateObj: function pdfHistory_stateObj(params) {
-  return {
-   fingerprint: this.fingerprint,
-   uid: this.uid,
-   target: params
-  };
- },
- _pushToHistory: function pdfHistory_pushToHistory(params, addPrevious, overwrite) {
-  if (!this.initialized) {
-   return;
-  }
-  if (!params.hash && params.page) {
-   params.hash = 'page=' + params.page;
-  }
-  if (addPrevious && !overwrite) {
-   var previousParams = this._getPreviousParams();
-   if (previousParams) {
-    var replacePrevious = !this.current.dest && this.current.hash !== this.previousHash;
-    this._pushToHistory(previousParams, false, replacePrevious);
-   }
-  }
-  this._pushOrReplaceState(this._stateObj(params), overwrite || this.uid === 0);
-  this.currentUid = this.uid++;
-  this.current = params;
-  this.updatePreviousBookmark = true;
- },
- _goTo: function pdfHistory_goTo(state) {
-  if (!(this.initialized && this.historyUnlocked && this._isStateObjectDefined(state))) {
-   return;
-  }
-  if (!this.reInitialized && state.uid < this.currentUid) {
-   var previousParams = this._getPreviousParams(true);
-   if (previousParams) {
-    this._pushToHistory(this.current, false);
-    this._pushToHistory(previousParams, false);
     this.currentUid = state.uid;
-    window.history.back();
-    return;
-   }
-  }
-  this.historyUnlocked = false;
-  if (state.target.dest) {
-   this.linkService.navigateTo(state.target.dest);
-  } else {
-   this.linkService.setHash(state.target.hash);
-  }
-  this.currentUid = state.uid;
-  if (state.uid > this.uid) {
-   this.uid = state.uid;
-  }
-  this.current = state.target;
-  this.updatePreviousBookmark = true;
-  var currentHash = window.location.hash.substring(1);
-  if (this.previousHash !== currentHash) {
-   this.allowHashChange = false;
-  }
-  this.previousHash = currentHash;
-  this.historyUnlocked = true;
- },
- back: function pdfHistoryBack() {
-  this.go(-1);
- },
- forward: function pdfHistoryForward() {
-  this.go(1);
- },
- go: function pdfHistoryGo(direction) {
-  if (this.initialized && this.historyUnlocked) {
-   var state = window.history.state;
-   if (direction === -1 && state && state.uid > 0) {
-    window.history.back();
-   } else if (direction === 1 && state && state.uid < this.uid - 1) {
-    window.history.forward();
-   }
+    if (state.uid > this.uid) {
+      this.uid = state.uid;
+    }
+    this.current = state.target;
+    this.updatePreviousBookmark = true;
+    var currentHash = window.location.hash.substring(1);
+    if (this.previousHash !== currentHash) {
+      this.allowHashChange = false;
+    }
+    this.previousHash = currentHash;
+    this.historyUnlocked = true;
+  },
+  back: function pdfHistoryBack() {
+    this.go(-1);
+  },
+  forward: function pdfHistoryForward() {
+    this.go(1);
+  },
+  go: function pdfHistoryGo(direction) {
+    if (this.initialized && this.historyUnlocked) {
+      var state = window.history.state;
+      if (direction === -1 && state && state.uid > 0) {
+        window.history.back();
+      } else if (direction === 1 && state && state.uid < this.uid - 1) {
+        window.history.forward();
+      }
+    }
   }
- }
 };
 exports.PDFHistory = PDFHistory;

+ 292 - 318
lib/web/pdf_link_service.js

@@ -13,344 +13,318 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtils = require('./ui_utils.js');
 var domEvents = require('./dom_events.js');
 var parseQueryString = uiUtils.parseQueryString;
 var PageNumberRegExp = /^\d+$/;
 function isPageNumber(str) {
- return PageNumberRegExp.test(str);
+  return PageNumberRegExp.test(str);
 }
 var PDFLinkService = function PDFLinkServiceClosure() {
- function PDFLinkService(options) {
-  options = options || {};
-  this.eventBus = options.eventBus || domEvents.getGlobalEventBus();
-  this.baseUrl = null;
-  this.pdfDocument = null;
-  this.pdfViewer = null;
-  this.pdfHistory = null;
-  this._pagesRefCache = null;
- }
- PDFLinkService.prototype = {
-  setDocument: function PDFLinkService_setDocument(pdfDocument, baseUrl) {
-   this.baseUrl = baseUrl;
-   this.pdfDocument = pdfDocument;
-   this._pagesRefCache = Object.create(null);
-  },
-  setViewer: function PDFLinkService_setViewer(pdfViewer) {
-   this.pdfViewer = pdfViewer;
-  },
-  setHistory: function PDFLinkService_setHistory(pdfHistory) {
-   this.pdfHistory = pdfHistory;
-  },
-  get pagesCount() {
-   return this.pdfDocument ? this.pdfDocument.numPages : 0;
-  },
-  get page() {
-   return this.pdfViewer.currentPageNumber;
-  },
-  set page(value) {
-   this.pdfViewer.currentPageNumber = value;
-  },
-  navigateTo: function PDFLinkService_navigateTo(dest) {
-   var destString = '';
-   var self = this;
-   var goToDestination = function (destRef) {
-    var pageNumber;
-    if (destRef instanceof Object) {
-     pageNumber = self._cachedPageNumber(destRef);
-    } else if ((destRef | 0) === destRef) {
-     pageNumber = destRef + 1;
-    } else {
-     console.error('PDFLinkService_navigateTo: "' + destRef + '" is not a valid destination reference.');
-     return;
-    }
-    if (pageNumber) {
-     if (pageNumber < 1 || pageNumber > self.pagesCount) {
-      console.error('PDFLinkService_navigateTo: "' + pageNumber + '" is a non-existent page number.');
-      return;
-     }
-     self.pdfViewer.scrollPageIntoView({
-      pageNumber: pageNumber,
-      destArray: dest
-     });
-     if (self.pdfHistory) {
-      self.pdfHistory.push({
-       dest: dest,
-       hash: destString,
-       page: pageNumber
+  function PDFLinkService(options) {
+    options = options || {};
+    this.eventBus = options.eventBus || domEvents.getGlobalEventBus();
+    this.baseUrl = null;
+    this.pdfDocument = null;
+    this.pdfViewer = null;
+    this.pdfHistory = null;
+    this._pagesRefCache = null;
+  }
+  PDFLinkService.prototype = {
+    setDocument: function PDFLinkService_setDocument(pdfDocument, baseUrl) {
+      this.baseUrl = baseUrl;
+      this.pdfDocument = pdfDocument;
+      this._pagesRefCache = Object.create(null);
+    },
+    setViewer: function PDFLinkService_setViewer(pdfViewer) {
+      this.pdfViewer = pdfViewer;
+    },
+    setHistory: function PDFLinkService_setHistory(pdfHistory) {
+      this.pdfHistory = pdfHistory;
+    },
+    get pagesCount() {
+      return this.pdfDocument ? this.pdfDocument.numPages : 0;
+    },
+    get page() {
+      return this.pdfViewer.currentPageNumber;
+    },
+    set page(value) {
+      this.pdfViewer.currentPageNumber = value;
+    },
+    navigateTo: function PDFLinkService_navigateTo(dest) {
+      var destString = '';
+      var self = this;
+      var goToDestination = function (destRef) {
+        var pageNumber;
+        if (destRef instanceof Object) {
+          pageNumber = self._cachedPageNumber(destRef);
+        } else if ((destRef | 0) === destRef) {
+          pageNumber = destRef + 1;
+        } else {
+          console.error('PDFLinkService_navigateTo: "' + destRef + '" is not a valid destination reference.');
+          return;
+        }
+        if (pageNumber) {
+          if (pageNumber < 1 || pageNumber > self.pagesCount) {
+            console.error('PDFLinkService_navigateTo: "' + pageNumber + '" is a non-existent page number.');
+            return;
+          }
+          self.pdfViewer.scrollPageIntoView({
+            pageNumber: pageNumber,
+            destArray: dest
+          });
+          if (self.pdfHistory) {
+            self.pdfHistory.push({
+              dest: dest,
+              hash: destString,
+              page: pageNumber
+            });
+          }
+        } else {
+          self.pdfDocument.getPageIndex(destRef).then(function (pageIndex) {
+            self.cachePageRef(pageIndex + 1, destRef);
+            goToDestination(destRef);
+          }).catch(function () {
+            console.error('PDFLinkService_navigateTo: "' + destRef + '" is not a valid page reference.');
+          });
+        }
+      };
+      var destinationPromise;
+      if (typeof dest === 'string') {
+        destString = dest;
+        destinationPromise = this.pdfDocument.getDestination(dest);
+      } else {
+        destinationPromise = Promise.resolve(dest);
+      }
+      destinationPromise.then(function (destination) {
+        dest = destination;
+        if (!(destination instanceof Array)) {
+          console.error('PDFLinkService_navigateTo: "' + destination + '" is not a valid destination array.');
+          return;
+        }
+        goToDestination(destination[0]);
       });
-     }
-    } else {
-     self.pdfDocument.getPageIndex(destRef).then(function (pageIndex) {
-      self.cachePageRef(pageIndex + 1, destRef);
-      goToDestination(destRef);
-     }).catch(function () {
-      console.error('PDFLinkService_navigateTo: "' + destRef + '" is not a valid page reference.');
-     });
-    }
-   };
-   var destinationPromise;
-   if (typeof dest === 'string') {
-    destString = dest;
-    destinationPromise = this.pdfDocument.getDestination(dest);
-   } else {
-    destinationPromise = Promise.resolve(dest);
-   }
-   destinationPromise.then(function (destination) {
-    dest = destination;
-    if (!(destination instanceof Array)) {
-     console.error('PDFLinkService_navigateTo: "' + destination + '" is not a valid destination array.');
-     return;
-    }
-    goToDestination(destination[0]);
-   });
-  },
-  getDestinationHash: function PDFLinkService_getDestinationHash(dest) {
-   if (typeof dest === 'string') {
-    return this.getAnchorUrl('#' + (isPageNumber(dest) ? 'nameddest=' : '') + escape(dest));
-   }
-   if (dest instanceof Array) {
-    var str = JSON.stringify(dest);
-    return this.getAnchorUrl('#' + escape(str));
-   }
-   return this.getAnchorUrl('');
-  },
-  getAnchorUrl: function PDFLinkService_getAnchorUrl(anchor) {
-   return (this.baseUrl || '') + anchor;
-  },
-  setHash: function PDFLinkService_setHash(hash) {
-   var pageNumber, dest;
-   if (hash.indexOf('=') >= 0) {
-    var params = parseQueryString(hash);
-    if ('search' in params) {
-     this.eventBus.dispatch('findfromurlhash', {
-      source: this,
-      query: params['search'].replace(/"/g, ''),
-      phraseSearch: params['phrase'] === 'true'
-     });
-    }
-    if ('nameddest' in params) {
-     if (this.pdfHistory) {
-      this.pdfHistory.updateNextHashParam(params.nameddest);
-     }
-     this.navigateTo(params.nameddest);
-     return;
-    }
-    if ('page' in params) {
-     pageNumber = params.page | 0 || 1;
-    }
-    if ('zoom' in params) {
-     var zoomArgs = params.zoom.split(',');
-     var zoomArg = zoomArgs[0];
-     var zoomArgNumber = parseFloat(zoomArg);
-     if (zoomArg.indexOf('Fit') === -1) {
-      dest = [
-       null,
-       { name: 'XYZ' },
-       zoomArgs.length > 1 ? zoomArgs[1] | 0 : null,
-       zoomArgs.length > 2 ? zoomArgs[2] | 0 : null,
-       zoomArgNumber ? zoomArgNumber / 100 : zoomArg
-      ];
-     } else {
-      if (zoomArg === 'Fit' || zoomArg === 'FitB') {
-       dest = [
-        null,
-        { name: zoomArg }
-       ];
-      } else if (zoomArg === 'FitH' || zoomArg === 'FitBH' || (zoomArg === 'FitV' || zoomArg === 'FitBV')) {
-       dest = [
-        null,
-        { name: zoomArg },
-        zoomArgs.length > 1 ? zoomArgs[1] | 0 : null
-       ];
-      } else if (zoomArg === 'FitR') {
-       if (zoomArgs.length !== 5) {
-        console.error('PDFLinkService_setHash: ' + 'Not enough parameters for \'FitR\'.');
-       } else {
-        dest = [
-         null,
-         { name: zoomArg },
-         zoomArgs[1] | 0,
-         zoomArgs[2] | 0,
-         zoomArgs[3] | 0,
-         zoomArgs[4] | 0
-        ];
-       }
+    },
+    getDestinationHash: function PDFLinkService_getDestinationHash(dest) {
+      if (typeof dest === 'string') {
+        return this.getAnchorUrl('#' + (isPageNumber(dest) ? 'nameddest=' : '') + escape(dest));
+      }
+      if (dest instanceof Array) {
+        var str = JSON.stringify(dest);
+        return this.getAnchorUrl('#' + escape(str));
+      }
+      return this.getAnchorUrl('');
+    },
+    getAnchorUrl: function PDFLinkService_getAnchorUrl(anchor) {
+      return (this.baseUrl || '') + anchor;
+    },
+    setHash: function PDFLinkService_setHash(hash) {
+      var pageNumber, dest;
+      if (hash.indexOf('=') >= 0) {
+        var params = parseQueryString(hash);
+        if ('search' in params) {
+          this.eventBus.dispatch('findfromurlhash', {
+            source: this,
+            query: params['search'].replace(/"/g, ''),
+            phraseSearch: params['phrase'] === 'true'
+          });
+        }
+        if ('nameddest' in params) {
+          if (this.pdfHistory) {
+            this.pdfHistory.updateNextHashParam(params.nameddest);
+          }
+          this.navigateTo(params.nameddest);
+          return;
+        }
+        if ('page' in params) {
+          pageNumber = params.page | 0 || 1;
+        }
+        if ('zoom' in params) {
+          var zoomArgs = params.zoom.split(',');
+          var zoomArg = zoomArgs[0];
+          var zoomArgNumber = parseFloat(zoomArg);
+          if (zoomArg.indexOf('Fit') === -1) {
+            dest = [null, { name: 'XYZ' }, zoomArgs.length > 1 ? zoomArgs[1] | 0 : null, zoomArgs.length > 2 ? zoomArgs[2] | 0 : null, zoomArgNumber ? zoomArgNumber / 100 : zoomArg];
+          } else {
+            if (zoomArg === 'Fit' || zoomArg === 'FitB') {
+              dest = [null, { name: zoomArg }];
+            } else if (zoomArg === 'FitH' || zoomArg === 'FitBH' || zoomArg === 'FitV' || zoomArg === 'FitBV') {
+              dest = [null, { name: zoomArg }, zoomArgs.length > 1 ? zoomArgs[1] | 0 : null];
+            } else if (zoomArg === 'FitR') {
+              if (zoomArgs.length !== 5) {
+                console.error('PDFLinkService_setHash: ' + 'Not enough parameters for \'FitR\'.');
+              } else {
+                dest = [null, { name: zoomArg }, zoomArgs[1] | 0, zoomArgs[2] | 0, zoomArgs[3] | 0, zoomArgs[4] | 0];
+              }
+            } else {
+              console.error('PDFLinkService_setHash: \'' + zoomArg + '\' is not a valid zoom value.');
+            }
+          }
+        }
+        if (dest) {
+          this.pdfViewer.scrollPageIntoView({
+            pageNumber: pageNumber || this.page,
+            destArray: dest,
+            allowNegativeOffset: true
+          });
+        } else if (pageNumber) {
+          this.page = pageNumber;
+        }
+        if ('pagemode' in params) {
+          this.eventBus.dispatch('pagemode', {
+            source: this,
+            mode: params.pagemode
+          });
+        }
       } else {
-       console.error('PDFLinkService_setHash: \'' + zoomArg + '\' is not a valid zoom value.');
+        if (isPageNumber(hash) && hash <= this.pagesCount) {
+          console.warn('PDFLinkService_setHash: specifying a page number ' + 'directly after the hash symbol (#) is deprecated, ' + 'please use the "#page=' + hash + '" form instead.');
+          this.page = hash | 0;
+        }
+        dest = unescape(hash);
+        try {
+          dest = JSON.parse(dest);
+          if (!(dest instanceof Array)) {
+            dest = dest.toString();
+          }
+        } catch (ex) {}
+        if (typeof dest === 'string' || isValidExplicitDestination(dest)) {
+          if (this.pdfHistory) {
+            this.pdfHistory.updateNextHashParam(dest);
+          }
+          this.navigateTo(dest);
+          return;
+        }
+        console.error('PDFLinkService_setHash: \'' + unescape(hash) + '\' is not a valid destination.');
       }
-     }
-    }
-    if (dest) {
-     this.pdfViewer.scrollPageIntoView({
-      pageNumber: pageNumber || this.page,
-      destArray: dest,
-      allowNegativeOffset: true
-     });
-    } else if (pageNumber) {
-     this.page = pageNumber;
-    }
-    if ('pagemode' in params) {
-     this.eventBus.dispatch('pagemode', {
-      source: this,
-      mode: params.pagemode
-     });
-    }
-   } else {
-    if (isPageNumber(hash) && hash <= this.pagesCount) {
-     console.warn('PDFLinkService_setHash: specifying a page number ' + 'directly after the hash symbol (#) is deprecated, ' + 'please use the "#page=' + hash + '" form instead.');
-     this.page = hash | 0;
+    },
+    executeNamedAction: function PDFLinkService_executeNamedAction(action) {
+      switch (action) {
+        case 'GoBack':
+          if (this.pdfHistory) {
+            this.pdfHistory.back();
+          }
+          break;
+        case 'GoForward':
+          if (this.pdfHistory) {
+            this.pdfHistory.forward();
+          }
+          break;
+        case 'NextPage':
+          if (this.page < this.pagesCount) {
+            this.page++;
+          }
+          break;
+        case 'PrevPage':
+          if (this.page > 1) {
+            this.page--;
+          }
+          break;
+        case 'LastPage':
+          this.page = this.pagesCount;
+          break;
+        case 'FirstPage':
+          this.page = 1;
+          break;
+        default:
+          break;
+      }
+      this.eventBus.dispatch('namedaction', {
+        source: this,
+        action: action
+      });
+    },
+    onFileAttachmentAnnotation: function (params) {
+      this.eventBus.dispatch('fileattachmentannotation', {
+        source: this,
+        id: params.id,
+        filename: params.filename,
+        content: params.content
+      });
+    },
+    cachePageRef: function PDFLinkService_cachePageRef(pageNum, pageRef) {
+      var refStr = pageRef.num + ' ' + pageRef.gen + ' R';
+      this._pagesRefCache[refStr] = pageNum;
+    },
+    _cachedPageNumber: function PDFLinkService_cachedPageNumber(pageRef) {
+      var refStr = pageRef.num + ' ' + pageRef.gen + ' R';
+      return this._pagesRefCache && this._pagesRefCache[refStr] || null;
     }
-    dest = unescape(hash);
-    try {
-     dest = JSON.parse(dest);
-     if (!(dest instanceof Array)) {
-      dest = dest.toString();
-     }
-    } catch (ex) {
+  };
+  function isValidExplicitDestination(dest) {
+    if (!(dest instanceof Array)) {
+      return false;
     }
-    if (typeof dest === 'string' || isValidExplicitDestination(dest)) {
-     if (this.pdfHistory) {
-      this.pdfHistory.updateNextHashParam(dest);
-     }
-     this.navigateTo(dest);
-     return;
+    var destLength = dest.length,
+        allowNull = true;
+    if (destLength < 2) {
+      return false;
     }
-    console.error('PDFLinkService_setHash: \'' + unescape(hash) + '\' is not a valid destination.');
-   }
-  },
-  executeNamedAction: function PDFLinkService_executeNamedAction(action) {
-   switch (action) {
-   case 'GoBack':
-    if (this.pdfHistory) {
-     this.pdfHistory.back();
+    var page = dest[0];
+    if (!(typeof page === 'object' && typeof page.num === 'number' && (page.num | 0) === page.num && typeof page.gen === 'number' && (page.gen | 0) === page.gen) && !(typeof page === 'number' && (page | 0) === page && page >= 0)) {
+      return false;
     }
-    break;
-   case 'GoForward':
-    if (this.pdfHistory) {
-     this.pdfHistory.forward();
+    var zoom = dest[1];
+    if (!(typeof zoom === 'object' && typeof zoom.name === 'string')) {
+      return false;
     }
-    break;
-   case 'NextPage':
-    if (this.page < this.pagesCount) {
-     this.page++;
+    switch (zoom.name) {
+      case 'XYZ':
+        if (destLength !== 5) {
+          return false;
+        }
+        break;
+      case 'Fit':
+      case 'FitB':
+        return destLength === 2;
+      case 'FitH':
+      case 'FitBH':
+      case 'FitV':
+      case 'FitBV':
+        if (destLength !== 3) {
+          return false;
+        }
+        break;
+      case 'FitR':
+        if (destLength !== 6) {
+          return false;
+        }
+        allowNull = false;
+        break;
+      default:
+        return false;
     }
-    break;
-   case 'PrevPage':
-    if (this.page > 1) {
-     this.page--;
+    for (var i = 2; i < destLength; i++) {
+      var param = dest[i];
+      if (!(typeof param === 'number' || allowNull && param === null)) {
+        return false;
+      }
     }
-    break;
-   case 'LastPage':
-    this.page = this.pagesCount;
-    break;
-   case 'FirstPage':
-    this.page = 1;
-    break;
-   default:
-    break;
-   }
-   this.eventBus.dispatch('namedaction', {
-    source: this,
-    action: action
-   });
-  },
-  onFileAttachmentAnnotation: function (params) {
-   this.eventBus.dispatch('fileattachmentannotation', {
-    source: this,
-    id: params.id,
-    filename: params.filename,
-    content: params.content
-   });
-  },
-  cachePageRef: function PDFLinkService_cachePageRef(pageNum, pageRef) {
-   var refStr = pageRef.num + ' ' + pageRef.gen + ' R';
-   this._pagesRefCache[refStr] = pageNum;
-  },
-  _cachedPageNumber: function PDFLinkService_cachedPageNumber(pageRef) {
-   var refStr = pageRef.num + ' ' + pageRef.gen + ' R';
-   return this._pagesRefCache && this._pagesRefCache[refStr] || null;
-  }
- };
- function isValidExplicitDestination(dest) {
-  if (!(dest instanceof Array)) {
-   return false;
+    return true;
   }
-  var destLength = dest.length, allowNull = true;
-  if (destLength < 2) {
-   return false;
-  }
-  var page = dest[0];
-  if (!(typeof page === 'object' && typeof page.num === 'number' && (page.num | 0) === page.num && typeof page.gen === 'number' && (page.gen | 0) === page.gen) && !(typeof page === 'number' && (page | 0) === page && page >= 0)) {
-   return false;
-  }
-  var zoom = dest[1];
-  if (!(typeof zoom === 'object' && typeof zoom.name === 'string')) {
-   return false;
-  }
-  switch (zoom.name) {
-  case 'XYZ':
-   if (destLength !== 5) {
-    return false;
-   }
-   break;
-  case 'Fit':
-  case 'FitB':
-   return destLength === 2;
-  case 'FitH':
-  case 'FitBH':
-  case 'FitV':
-  case 'FitBV':
-   if (destLength !== 3) {
-    return false;
-   }
-   break;
-  case 'FitR':
-   if (destLength !== 6) {
-    return false;
-   }
-   allowNull = false;
-   break;
-  default:
-   return false;
-  }
-  for (var i = 2; i < destLength; i++) {
-   var param = dest[i];
-   if (!(typeof param === 'number' || allowNull && param === null)) {
-    return false;
-   }
-  }
-  return true;
- }
- return PDFLinkService;
+  return PDFLinkService;
 }();
 var SimpleLinkService = function SimpleLinkServiceClosure() {
- function SimpleLinkService() {
- }
- SimpleLinkService.prototype = {
-  get page() {
-   return 0;
-  },
-  set page(value) {
-  },
-  navigateTo: function (dest) {
-  },
-  getDestinationHash: function (dest) {
-   return '#';
-  },
-  getAnchorUrl: function (hash) {
-   return '#';
-  },
-  setHash: function (hash) {
-  },
-  executeNamedAction: function (action) {
-  },
-  onFileAttachmentAnnotation: function (params) {
-  },
-  cachePageRef: function (pageNum, pageRef) {
-  }
- };
- return SimpleLinkService;
+  function SimpleLinkService() {}
+  SimpleLinkService.prototype = {
+    get page() {
+      return 0;
+    },
+    set page(value) {},
+    navigateTo: function (dest) {},
+    getDestinationHash: function (dest) {
+      return '#';
+    },
+    getAnchorUrl: function (hash) {
+      return '#';
+    },
+    setHash: function (hash) {},
+    executeNamedAction: function (action) {},
+    onFileAttachmentAnnotation: function (params) {},
+    cachePageRef: function (pageNum, pageRef) {}
+  };
+  return SimpleLinkService;
 }();
 exports.PDFLinkService = PDFLinkService;
 exports.SimpleLinkService = SimpleLinkService;

+ 126 - 124
lib/web/pdf_outline_viewer.js

@@ -13,135 +13,137 @@
  * limitations under the License.
  */
 'use strict';
+
 var pdfjsLib = require('./pdfjs.js');
 var PDFJS = pdfjsLib.PDFJS;
 var DEFAULT_TITLE = '\u2013';
 var PDFOutlineViewer = function PDFOutlineViewerClosure() {
- function PDFOutlineViewer(options) {
-  this.outline = null;
-  this.lastToggleIsShow = true;
-  this.container = options.container;
-  this.linkService = options.linkService;
-  this.eventBus = options.eventBus;
- }
- PDFOutlineViewer.prototype = {
-  reset: function PDFOutlineViewer_reset() {
-   this.outline = null;
-   this.lastToggleIsShow = true;
-   this.container.textContent = '';
-   this.container.classList.remove('outlineWithDeepNesting');
-  },
-  _dispatchEvent: function PDFOutlineViewer_dispatchEvent(outlineCount) {
-   this.eventBus.dispatch('outlineloaded', {
-    source: this,
-    outlineCount: outlineCount
-   });
-  },
-  _bindLink: function PDFOutlineViewer_bindLink(element, item) {
-   if (item.url) {
-    pdfjsLib.addLinkAttributes(element, {
-     url: item.url,
-     target: item.newWindow ? PDFJS.LinkTarget.BLANK : undefined
-    });
-    return;
-   }
-   var self = this, destination = item.dest;
-   element.href = self.linkService.getDestinationHash(destination);
-   element.onclick = function () {
-    if (destination) {
-     self.linkService.navigateTo(destination);
-    }
-    return false;
-   };
-  },
-  _setStyles: function PDFOutlineViewer_setStyles(element, item) {
-   var styleStr = '';
-   if (item.bold) {
-    styleStr += 'font-weight: bold;';
-   }
-   if (item.italic) {
-    styleStr += 'font-style: italic;';
-   }
-   if (styleStr) {
-    element.setAttribute('style', styleStr);
-   }
-  },
-  _addToggleButton: function PDFOutlineViewer_addToggleButton(div) {
-   var toggler = document.createElement('div');
-   toggler.className = 'outlineItemToggler';
-   toggler.onclick = function (event) {
-    event.stopPropagation();
-    toggler.classList.toggle('outlineItemsHidden');
-    if (event.shiftKey) {
-     var shouldShowAll = !toggler.classList.contains('outlineItemsHidden');
-     this._toggleOutlineItem(div, shouldShowAll);
-    }
-   }.bind(this);
-   div.insertBefore(toggler, div.firstChild);
-  },
-  _toggleOutlineItem: function PDFOutlineViewer_toggleOutlineItem(root, show) {
-   this.lastToggleIsShow = show;
-   var togglers = root.querySelectorAll('.outlineItemToggler');
-   for (var i = 0, ii = togglers.length; i < ii; ++i) {
-    togglers[i].classList[show ? 'remove' : 'add']('outlineItemsHidden');
-   }
-  },
-  toggleOutlineTree: function PDFOutlineViewer_toggleOutlineTree() {
-   if (!this.outline) {
-    return;
-   }
-   this._toggleOutlineItem(this.container, !this.lastToggleIsShow);
-  },
-  render: function PDFOutlineViewer_render(params) {
-   var outline = params && params.outline || null;
-   var outlineCount = 0;
-   if (this.outline) {
-    this.reset();
-   }
-   this.outline = outline;
-   if (!outline) {
-    this._dispatchEvent(outlineCount);
-    return;
-   }
-   var fragment = document.createDocumentFragment();
-   var queue = [{
-     parent: fragment,
-     items: this.outline
-    }];
-   var hasAnyNesting = false;
-   while (queue.length > 0) {
-    var levelData = queue.shift();
-    for (var i = 0, len = levelData.items.length; i < len; i++) {
-     var item = levelData.items[i];
-     var div = document.createElement('div');
-     div.className = 'outlineItem';
-     var element = document.createElement('a');
-     this._bindLink(element, item);
-     this._setStyles(element, item);
-     element.textContent = pdfjsLib.removeNullCharacters(item.title) || DEFAULT_TITLE;
-     div.appendChild(element);
-     if (item.items.length > 0) {
-      hasAnyNesting = true;
-      this._addToggleButton(div);
-      var itemsDiv = document.createElement('div');
-      itemsDiv.className = 'outlineItems';
-      div.appendChild(itemsDiv);
-      queue.push({
-       parent: itemsDiv,
-       items: item.items
+  function PDFOutlineViewer(options) {
+    this.outline = null;
+    this.lastToggleIsShow = true;
+    this.container = options.container;
+    this.linkService = options.linkService;
+    this.eventBus = options.eventBus;
+  }
+  PDFOutlineViewer.prototype = {
+    reset: function PDFOutlineViewer_reset() {
+      this.outline = null;
+      this.lastToggleIsShow = true;
+      this.container.textContent = '';
+      this.container.classList.remove('outlineWithDeepNesting');
+    },
+    _dispatchEvent: function PDFOutlineViewer_dispatchEvent(outlineCount) {
+      this.eventBus.dispatch('outlineloaded', {
+        source: this,
+        outlineCount: outlineCount
       });
-     }
-     levelData.parent.appendChild(div);
-     outlineCount++;
+    },
+    _bindLink: function PDFOutlineViewer_bindLink(element, item) {
+      if (item.url) {
+        pdfjsLib.addLinkAttributes(element, {
+          url: item.url,
+          target: item.newWindow ? PDFJS.LinkTarget.BLANK : undefined
+        });
+        return;
+      }
+      var self = this,
+          destination = item.dest;
+      element.href = self.linkService.getDestinationHash(destination);
+      element.onclick = function () {
+        if (destination) {
+          self.linkService.navigateTo(destination);
+        }
+        return false;
+      };
+    },
+    _setStyles: function PDFOutlineViewer_setStyles(element, item) {
+      var styleStr = '';
+      if (item.bold) {
+        styleStr += 'font-weight: bold;';
+      }
+      if (item.italic) {
+        styleStr += 'font-style: italic;';
+      }
+      if (styleStr) {
+        element.setAttribute('style', styleStr);
+      }
+    },
+    _addToggleButton: function PDFOutlineViewer_addToggleButton(div) {
+      var toggler = document.createElement('div');
+      toggler.className = 'outlineItemToggler';
+      toggler.onclick = function (event) {
+        event.stopPropagation();
+        toggler.classList.toggle('outlineItemsHidden');
+        if (event.shiftKey) {
+          var shouldShowAll = !toggler.classList.contains('outlineItemsHidden');
+          this._toggleOutlineItem(div, shouldShowAll);
+        }
+      }.bind(this);
+      div.insertBefore(toggler, div.firstChild);
+    },
+    _toggleOutlineItem: function PDFOutlineViewer_toggleOutlineItem(root, show) {
+      this.lastToggleIsShow = show;
+      var togglers = root.querySelectorAll('.outlineItemToggler');
+      for (var i = 0, ii = togglers.length; i < ii; ++i) {
+        togglers[i].classList[show ? 'remove' : 'add']('outlineItemsHidden');
+      }
+    },
+    toggleOutlineTree: function PDFOutlineViewer_toggleOutlineTree() {
+      if (!this.outline) {
+        return;
+      }
+      this._toggleOutlineItem(this.container, !this.lastToggleIsShow);
+    },
+    render: function PDFOutlineViewer_render(params) {
+      var outline = params && params.outline || null;
+      var outlineCount = 0;
+      if (this.outline) {
+        this.reset();
+      }
+      this.outline = outline;
+      if (!outline) {
+        this._dispatchEvent(outlineCount);
+        return;
+      }
+      var fragment = document.createDocumentFragment();
+      var queue = [{
+        parent: fragment,
+        items: this.outline
+      }];
+      var hasAnyNesting = false;
+      while (queue.length > 0) {
+        var levelData = queue.shift();
+        for (var i = 0, len = levelData.items.length; i < len; i++) {
+          var item = levelData.items[i];
+          var div = document.createElement('div');
+          div.className = 'outlineItem';
+          var element = document.createElement('a');
+          this._bindLink(element, item);
+          this._setStyles(element, item);
+          element.textContent = pdfjsLib.removeNullCharacters(item.title) || DEFAULT_TITLE;
+          div.appendChild(element);
+          if (item.items.length > 0) {
+            hasAnyNesting = true;
+            this._addToggleButton(div);
+            var itemsDiv = document.createElement('div');
+            itemsDiv.className = 'outlineItems';
+            div.appendChild(itemsDiv);
+            queue.push({
+              parent: itemsDiv,
+              items: item.items
+            });
+          }
+          levelData.parent.appendChild(div);
+          outlineCount++;
+        }
+      }
+      if (hasAnyNesting) {
+        this.container.classList.add('outlineWithDeepNesting');
+      }
+      this.container.appendChild(fragment);
+      this._dispatchEvent(outlineCount);
     }
-   }
-   if (hasAnyNesting) {
-    this.container.classList.add('outlineWithDeepNesting');
-   }
-   this.container.appendChild(fragment);
-   this._dispatchEvent(outlineCount);
-  }
- };
- return PDFOutlineViewer;
+  };
+  return PDFOutlineViewer;
 }();
 exports.PDFOutlineViewer = PDFOutlineViewer;

+ 452 - 457
lib/web/pdf_page_view.js

@@ -13,6 +13,7 @@
  * limitations under the License.
  */
 'use strict';
+
 var uiUtils = require('./ui_utils.js');
 var pdfRenderingQueue = require('./pdf_rendering_queue.js');
 var domEvents = require('./dom_events.js');
@@ -26,469 +27,463 @@ var RendererType = uiUtils.RendererType;
 var RenderingStates = pdfRenderingQueue.RenderingStates;
 var TEXT_LAYER_RENDER_DELAY = 200;
 var PDFPageView = function PDFPageViewClosure() {
- function PDFPageView(options) {
-  var container = options.container;
-  var id = options.id;
-  var scale = options.scale;
-  var defaultViewport = options.defaultViewport;
-  var renderingQueue = options.renderingQueue;
-  var textLayerFactory = options.textLayerFactory;
-  var annotationLayerFactory = options.annotationLayerFactory;
-  var enhanceTextSelection = options.enhanceTextSelection || false;
-  var renderInteractiveForms = options.renderInteractiveForms || false;
-  this.id = id;
-  this.renderingId = 'page' + id;
-  this.pageLabel = null;
-  this.rotation = 0;
-  this.scale = scale || DEFAULT_SCALE;
-  this.viewport = defaultViewport;
-  this.pdfPageRotate = defaultViewport.rotation;
-  this.hasRestrictedScaling = false;
-  this.enhanceTextSelection = enhanceTextSelection;
-  this.renderInteractiveForms = renderInteractiveForms;
-  this.eventBus = options.eventBus || domEvents.getGlobalEventBus();
-  this.renderingQueue = renderingQueue;
-  this.textLayerFactory = textLayerFactory;
-  this.annotationLayerFactory = annotationLayerFactory;
-  this.renderer = options.renderer || RendererType.CANVAS;
-  this.paintTask = null;
-  this.paintedViewportMap = new WeakMap();
-  this.renderingState = RenderingStates.INITIAL;
-  this.resume = null;
-  this.error = null;
-  this.onBeforeDraw = null;
-  this.onAfterDraw = null;
-  this.textLayer = null;
-  this.zoomLayer = null;
-  this.annotationLayer = null;
-  var div = document.createElement('div');
-  div.className = 'page';
-  div.style.width = Math.floor(this.viewport.width) + 'px';
-  div.style.height = Math.floor(this.viewport.height) + 'px';
-  div.setAttribute('data-page-number', this.id);
-  this.div = div;
-  container.appendChild(div);
- }
- PDFPageView.prototype = {
-  setPdfPage: function PDFPageView_setPdfPage(pdfPage) {
-   this.pdfPage = pdfPage;
-   this.pdfPageRotate = pdfPage.rotate;
-   var totalRotation = (this.rotation + this.pdfPageRotate) % 360;
-   this.viewport = pdfPage.getViewport(this.scale * CSS_UNITS, totalRotation);
-   this.stats = pdfPage.stats;
-   this.reset();
-  },
-  destroy: function PDFPageView_destroy() {
-   this.zoomLayer = null;
-   this.reset();
-   if (this.pdfPage) {
-    this.pdfPage.cleanup();
-   }
-  },
-  reset: function PDFPageView_reset(keepZoomLayer, keepAnnotations) {
-   this.cancelRendering();
-   var div = this.div;
-   div.style.width = Math.floor(this.viewport.width) + 'px';
-   div.style.height = Math.floor(this.viewport.height) + 'px';
-   var childNodes = div.childNodes;
-   var currentZoomLayerNode = keepZoomLayer && this.zoomLayer || null;
-   var currentAnnotationNode = keepAnnotations && this.annotationLayer && this.annotationLayer.div || null;
-   for (var i = childNodes.length - 1; i >= 0; i--) {
-    var node = childNodes[i];
-    if (currentZoomLayerNode === node || currentAnnotationNode === node) {
-     continue;
-    }
-    div.removeChild(node);
-   }
-   div.removeAttribute('data-loaded');
-   if (currentAnnotationNode) {
-    this.annotationLayer.hide();
-   } else {
-    this.annotationLayer = null;
-   }
-   if (this.canvas && !currentZoomLayerNode) {
-    this.paintedViewportMap.delete(this.canvas);
-    this.canvas.width = 0;
-    this.canvas.height = 0;
-    delete this.canvas;
-   }
-   if (this.svg) {
-    this.paintedViewportMap.delete(this.svg);
-    delete this.svg;
-   }
-   this.loadingIconDiv = document.createElement('div');
-   this.loadingIconDiv.className = 'loadingIcon';
-   div.appendChild(this.loadingIconDiv);
-  },
-  update: function PDFPageView_update(scale, rotation) {
-   this.scale = scale || this.scale;
-   if (typeof rotation !== 'undefined') {
-    this.rotation = rotation;
-   }
-   var totalRotation = (this.rotation + this.pdfPageRotate) % 360;
-   this.viewport = this.viewport.clone({
-    scale: this.scale * CSS_UNITS,
-    rotation: totalRotation
-   });
-   if (this.svg) {
-    this.cssTransform(this.svg, true);
-    this.eventBus.dispatch('pagerendered', {
-     source: this,
-     pageNumber: this.id,
-     cssTransform: true
-    });
-    return;
-   }
-   var isScalingRestricted = false;
-   if (this.canvas && pdfjsLib.PDFJS.maxCanvasPixels > 0) {
-    var outputScale = this.outputScale;
-    if ((Math.floor(this.viewport.width) * outputScale.sx | 0) * (Math.floor(this.viewport.height) * outputScale.sy | 0) > pdfjsLib.PDFJS.maxCanvasPixels) {
-     isScalingRestricted = true;
-    }
-   }
-   if (this.canvas) {
-    if (pdfjsLib.PDFJS.useOnlyCssZoom || this.hasRestrictedScaling && isScalingRestricted) {
-     this.cssTransform(this.canvas, true);
-     this.eventBus.dispatch('pagerendered', {
-      source: this,
-      pageNumber: this.id,
-      cssTransform: true
-     });
-     return;
-    }
-    if (!this.zoomLayer) {
-     this.zoomLayer = this.canvas.parentNode;
-     this.zoomLayer.style.position = 'absolute';
-    }
-   }
-   if (this.zoomLayer) {
-    this.cssTransform(this.zoomLayer.firstChild);
-   }
-   this.reset(true, true);
-  },
-  cancelRendering: function PDFPageView_cancelRendering() {
-   if (this.paintTask) {
-    this.paintTask.cancel();
+  function PDFPageView(options) {
+    var container = options.container;
+    var id = options.id;
+    var scale = options.scale;
+    var defaultViewport = options.defaultViewport;
+    var renderingQueue = options.renderingQueue;
+    var textLayerFactory = options.textLayerFactory;
+    var annotationLayerFactory = options.annotationLayerFactory;
+    var enhanceTextSelection = options.enhanceTextSelection || false;
+    var renderInteractiveForms = options.renderInteractiveForms || false;
+    this.id = id;
+    this.renderingId = 'page' + id;
+    this.pageLabel = null;
+    this.rotation = 0;
+    this.scale = scale || DEFAULT_SCALE;
+    this.viewport = defaultViewport;
+    this.pdfPageRotate = defaultViewport.rotation;
+    this.hasRestrictedScaling = false;
+    this.enhanceTextSelection = enhanceTextSelection;
+    this.renderInteractiveForms = renderInteractiveForms;
+    this.eventBus = options.eventBus || domEvents.getGlobalEventBus();
+    this.renderingQueue = renderingQueue;
+    this.textLayerFactory = textLayerFactory;
+    this.annotationLayerFactory = annotationLayerFactory;
+    this.renderer = options.renderer || RendererType.CANVAS;
     this.paintTask = null;
-   }
-   this.renderingState = RenderingStates.INITIAL;
-   this.resume = null;
-   if (this.textLayer) {
-    this.textLayer.cancel();
+    this.paintedViewportMap = new WeakMap();
+    this.renderingState = RenderingStates.INITIAL;
+    this.resume = null;
+    this.error = null;
+    this.onBeforeDraw = null;
+    this.onAfterDraw = null;
     this.textLayer = null;
-   }
-  },
-  updatePosition: function PDFPageView_updatePosition() {
-   if (this.textLayer) {
-    this.textLayer.render(TEXT_LAYER_RENDER_DELAY);
-   }
-  },
-  cssTransform: function PDFPageView_transform(target, redrawAnnotations) {
-   var CustomStyle = pdfjsLib.CustomStyle;
-   var width = this.viewport.width;
-   var height = this.viewport.height;
-   var div = this.div;
-   target.style.width = target.parentNode.style.width = div.style.width = Math.floor(width) + 'px';
-   target.style.height = target.parentNode.style.height = div.style.height = Math.floor(height) + 'px';
-   var relativeRotation = this.viewport.rotation - this.paintedViewportMap.get(target).rotation;
-   var absRotation = Math.abs(relativeRotation);
-   var scaleX = 1, scaleY = 1;
-   if (absRotation === 90 || absRotation === 270) {
-    scaleX = height / width;
-    scaleY = width / height;
-   }
-   var cssTransform = 'rotate(' + relativeRotation + 'deg) ' + 'scale(' + scaleX + ',' + scaleY + ')';
-   CustomStyle.setProp('transform', target, cssTransform);
-   if (this.textLayer) {
-    var textLayerViewport = this.textLayer.viewport;
-    var textRelativeRotation = this.viewport.rotation - textLayerViewport.rotation;
-    var textAbsRotation = Math.abs(textRelativeRotation);
-    var scale = width / textLayerViewport.width;
-    if (textAbsRotation === 90 || textAbsRotation === 270) {
-     scale = width / textLayerViewport.height;
-    }
-    var textLayerDiv = this.textLayer.textLayerDiv;
-    var transX, transY;
-    switch (textAbsRotation) {
-    case 0:
-     transX = transY = 0;
-     break;
-    case 90:
-     transX = 0;
-     transY = '-' + textLayerDiv.style.height;
-     break;
-    case 180:
-     transX = '-' + textLayerDiv.style.width;
-     transY = '-' + textLayerDiv.style.height;
-     break;
-    case 270:
-     transX = '-' + textLayerDiv.style.width;
-     transY = 0;
-     break;
-    default:
-     console.error('Bad rotation value.');
-     break;
-    }
-    CustomStyle.setProp('transform', textLayerDiv, 'rotate(' + textAbsRotation + 'deg) ' + 'scale(' + scale + ', ' + scale + ') ' + 'translate(' + transX + ', ' + transY + ')');
-    CustomStyle.setProp('transformOrigin', textLayerDiv, '0% 0%');
-   }
-   if (redrawAnnotations && this.annotationLayer) {
-    this.annotationLayer.render(this.viewport, 'display');
-   }
-  },
-  get width() {
-   return this.viewport.width;
-  },
-  get height() {
-   return this.viewport.height;
-  },
-  getPagePoint: function PDFPageView_getPagePoint(x, y) {
-   return this.viewport.convertToPdfPoint(x, y);
-  },
-  draw: function PDFPageView_draw() {
-   if (this.renderingState !== RenderingStates.INITIAL) {
-    console.error('Must be in new state before drawing');
-    this.reset();
-   }
-   this.renderingState = RenderingStates.RUNNING;
-   var self = this;
-   var pdfPage = this.pdfPage;
-   var div = this.div;
-   var canvasWrapper = document.createElement('div');
-   canvasWrapper.style.width = div.style.width;
-   canvasWrapper.style.height = div.style.height;
-   canvasWrapper.classList.add('canvasWrapper');
-   if (this.annotationLayer && this.annotationLayer.div) {
-    div.insertBefore(canvasWrapper, this.annotationLayer.div);
-   } else {
-    div.appendChild(canvasWrapper);
-   }
-   var textLayerDiv = null;
-   var textLayer = null;
-   if (this.textLayerFactory) {
-    textLayerDiv = document.createElement('div');
-    textLayerDiv.className = 'textLayer';
-    textLayerDiv.style.width = canvasWrapper.style.width;
-    textLayerDiv.style.height = canvasWrapper.style.height;
-    if (this.annotationLayer && this.annotationLayer.div) {
-     div.insertBefore(textLayerDiv, this.annotationLayer.div);
-    } else {
-     div.appendChild(textLayerDiv);
-    }
-    textLayer = this.textLayerFactory.createTextLayerBuilder(textLayerDiv, this.id - 1, this.viewport, this.enhanceTextSelection);
-   }
-   this.textLayer = textLayer;
-   var renderContinueCallback = null;
-   if (this.renderingQueue) {
-    renderContinueCallback = function renderContinueCallback(cont) {
-     if (!self.renderingQueue.isHighestPriority(self)) {
-      self.renderingState = RenderingStates.PAUSED;
-      self.resume = function resumeCallback() {
-       self.renderingState = RenderingStates.RUNNING;
-       cont();
+    this.zoomLayer = null;
+    this.annotationLayer = null;
+    var div = document.createElement('div');
+    div.className = 'page';
+    div.style.width = Math.floor(this.viewport.width) + 'px';
+    div.style.height = Math.floor(this.viewport.height) + 'px';
+    div.setAttribute('data-page-number', this.id);
+    this.div = div;
+    container.appendChild(div);
+  }
+  PDFPageView.prototype = {
+    setPdfPage: function PDFPageView_setPdfPage(pdfPage) {
+      this.pdfPage = pdfPage;
+      this.pdfPageRotate = pdfPage.rotate;
+      var totalRotation = (this.rotation + this.pdfPageRotate) % 360;
+      this.viewport = pdfPage.getViewport(this.scale * CSS_UNITS, totalRotation);
+      this.stats = pdfPage.stats;
+      this.reset();
+    },
+    destroy: function PDFPageView_destroy() {
+      this.zoomLayer = null;
+      this.reset();
+      if (this.pdfPage) {
+        this.pdfPage.cleanup();
+      }
+    },
+    reset: function PDFPageView_reset(keepZoomLayer, keepAnnotations) {
+      this.cancelRendering();
+      var div = this.div;
+      div.style.width = Math.floor(this.viewport.width) + 'px';
+      div.style.height = Math.floor(this.viewport.height) + 'px';
+      var childNodes = div.childNodes;
+      var currentZoomLayerNode = keepZoomLayer && this.zoomLayer || null;
+      var currentAnnotationNode = keepAnnotations && this.annotationLayer && this.annotationLayer.div || null;
+      for (var i = childNodes.length - 1; i >= 0; i--) {
+        var node = childNodes[i];
+        if (currentZoomLayerNode === node || currentAnnotationNode === node) {
+          continue;
+        }
+        div.removeChild(node);
+      }
+      div.removeAttribute('data-loaded');
+      if (currentAnnotationNode) {
+        this.annotationLayer.hide();
+      } else {
+        this.annotationLayer = null;
+      }
+      if (this.canvas && !currentZoomLayerNode) {
+        this.paintedViewportMap.delete(this.canvas);
+        this.canvas.width = 0;
+        this.canvas.height = 0;
+        delete this.canvas;
+      }
+      if (this.svg) {
+        this.paintedViewportMap.delete(this.svg);
+        delete this.svg;
+      }
+      this.loadingIconDiv = document.createElement('div');
+      this.loadingIconDiv.className = 'loadingIcon';
+      div.appendChild(this.loadingIconDiv);
+    },
+    update: function PDFPageView_update(scale, rotation) {
+      this.scale = scale || this.scale;
+      if (typeof rotation !== 'undefined') {
+        this.rotation = rotation;
+      }
+      var totalRotation = (this.rotation + this.pdfPageRotate) % 360;
+      this.viewport = this.viewport.clone({
+        scale: this.scale * CSS_UNITS,
+        rotation: totalRotation
+      });
+      if (this.svg) {
+        this.cssTransform(this.svg, true);
+        this.eventBus.dispatch('pagerendered', {
+          source: this,
+          pageNumber: this.id,
+          cssTransform: true
+        });
+        return;
+      }
+      var isScalingRestricted = false;
+      if (this.canvas && pdfjsLib.PDFJS.maxCanvasPixels > 0) {
+        var outputScale = this.outputScale;
+        if ((Math.floor(this.viewport.width) * outputScale.sx | 0) * (Math.floor(this.viewport.height) * outputScale.sy | 0) > pdfjsLib.PDFJS.maxCanvasPixels) {
+          isScalingRestricted = true;
+        }
+      }
+      if (this.canvas) {
+        if (pdfjsLib.PDFJS.useOnlyCssZoom || this.hasRestrictedScaling && isScalingRestricted) {
+          this.cssTransform(this.canvas, true);
+          this.eventBus.dispatch('pagerendered', {
+            source: this,
+            pageNumber: this.id,
+            cssTransform: true
+          });
+          return;
+        }
+        if (!this.zoomLayer) {
+          this.zoomLayer = this.canvas.parentNode;
+          this.zoomLayer.style.position = 'absolute';
+        }
+      }
+      if (this.zoomLayer) {
+        this.cssTransform(this.zoomLayer.firstChild);
+      }
+      this.reset(true, true);
+    },
+    cancelRendering: function PDFPageView_cancelRendering() {
+      if (this.paintTask) {
+        this.paintTask.cancel();
+        this.paintTask = null;
+      }
+      this.renderingState = RenderingStates.INITIAL;
+      this.resume = null;
+      if (this.textLayer) {
+        this.textLayer.cancel();
+        this.textLayer = null;
+      }
+    },
+    updatePosition: function PDFPageView_updatePosition() {
+      if (this.textLayer) {
+        this.textLayer.render(TEXT_LAYER_RENDER_DELAY);
+      }
+    },
+    cssTransform: function PDFPageView_transform(target, redrawAnnotations) {
+      var CustomStyle = pdfjsLib.CustomStyle;
+      var width = this.viewport.width;
+      var height = this.viewport.height;
+      var div = this.div;
+      target.style.width = target.parentNode.style.width = div.style.width = Math.floor(width) + 'px';
+      target.style.height = target.parentNode.style.height = div.style.height = Math.floor(height) + 'px';
+      var relativeRotation = this.viewport.rotation - this.paintedViewportMap.get(target).rotation;
+      var absRotation = Math.abs(relativeRotation);
+      var scaleX = 1,
+          scaleY = 1;
+      if (absRotation === 90 || absRotation === 270) {
+        scaleX = height / width;
+        scaleY = width / height;
+      }
+      var cssTransform = 'rotate(' + relativeRotation + 'deg) ' + 'scale(' + scaleX + ',' + scaleY + ')';
+      CustomStyle.setProp('transform', target, cssTransform);
+      if (this.textLayer) {
+        var textLayerViewport = this.textLayer.viewport;
+        var textRelativeRotation = this.viewport.rotation - textLayerViewport.rotation;
+        var textAbsRotation = Math.abs(textRelativeRotation);
+        var scale = width / textLayerViewport.width;
+        if (textAbsRotation === 90 || textAbsRotation === 270) {
+          scale = width / textLayerViewport.height;
+        }
+        var textLayerDiv = this.textLayer.textLayerDiv;
+        var transX, transY;
+        switch (textAbsRotation) {
+          case 0:
+            transX = transY = 0;
+            break;
+          case 90:
+            transX = 0;
+            transY = '-' + textLayerDiv.style.height;
+            break;
+          case 180:
+            transX = '-' + textLayerDiv.style.width;
+            transY = '-' + textLayerDiv.style.height;
+            break;
+          case 270:
+            transX = '-' + textLayerDiv.style.width;
+            transY = 0;
+            break;
+          default:
+            console.error('Bad rotation value.');
+            break;
+        }
+        CustomStyle.setProp('transform', textLayerDiv, 'rotate(' + textAbsRotation + 'deg) ' + 'scale(' + scale + ', ' + scale + ') ' + 'translate(' + transX + ', ' + transY + ')');
+        CustomStyle.setProp('transformOrigin', textLayerDiv, '0% 0%');
+      }
+      if (redrawAnnotations && this.annotationLayer) {
+        this.annotationLayer.render(this.viewport, 'display');
+      }
+    },
+    get width() {
+      return this.viewport.width;
+    },
+    get height() {
+      return this.viewport.height;
+    },
+    getPagePoint: function PDFPageView_getPagePoint(x, y) {
+      return this.viewport.convertToPdfPoint(x, y);
+    },
+    draw: function PDFPageView_draw() {
+      if (this.renderingState !== RenderingStates.INITIAL) {
+        console.error('Must be in new state before drawing');
+        this.reset();
+      }
+      this.renderingState = RenderingStates.RUNNING;
+      var self = this;
+      var pdfPage = this.pdfPage;
+      var div = this.div;
+      var canvasWrapper = document.createElement('div');
+      canvasWrapper.style.width = div.style.width;
+      canvasWrapper.style.height = div.style.height;
+      canvasWrapper.classList.add('canvasWrapper');
+      if (this.annotationLayer && this.annotationLayer.div) {
+        div.insertBefore(canvasWrapper, this.annotationLayer.div);
+      } else {
+        div.appendChild(canvasWrapper);
+      }
+      var textLayerDiv = null;
+      var textLayer = null;
+      if (this.textLayerFactory) {
+        textLayerDiv = document.createElement('div');
+        textLayerDiv.className = 'textLayer';
+        textLayerDiv.style.width = canvasWrapper.style.width;
+        textLayerDiv.style.height = canvasWrapper.style.height;
+        if (this.annotationLayer && this.annotationLayer.div) {
+          div.insertBefore(textLayerDiv, this.annotationLayer.div);
+        } else {
+          div.appendChild(textLayerDiv);
+        }
+        textLayer = this.textLayerFactory.createTextLayerBuilder(textLayerDiv, this.id - 1, this.viewport, this.enhanceTextSelection);
+      }
+      this.textLayer = textLayer;
+      var renderContinueCallback = null;
+      if (this.renderingQueue) {
+        renderContinueCallback = function renderContinueCallback(cont) {
+          if (!self.renderingQueue.isHighestPriority(self)) {
+            self.renderingState = RenderingStates.PAUSED;
+            self.resume = function resumeCallback() {
+              self.renderingState = RenderingStates.RUNNING;
+              cont();
+            };
+            return;
+          }
+          cont();
+        };
+      }
+      var finishPaintTask = function finishPaintTask(error) {
+        if (paintTask === self.paintTask) {
+          self.paintTask = null;
+        }
+        if (error === 'cancelled') {
+          self.error = null;
+          return Promise.resolve(undefined);
+        }
+        self.renderingState = RenderingStates.FINISHED;
+        if (self.loadingIconDiv) {
+          div.removeChild(self.loadingIconDiv);
+          delete self.loadingIconDiv;
+        }
+        if (self.zoomLayer) {
+          var zoomLayerCanvas = self.zoomLayer.firstChild;
+          self.paintedViewportMap.delete(zoomLayerCanvas);
+          zoomLayerCanvas.width = 0;
+          zoomLayerCanvas.height = 0;
+          if (div.contains(self.zoomLayer)) {
+            div.removeChild(self.zoomLayer);
+          }
+          self.zoomLayer = null;
+        }
+        self.error = error;
+        self.stats = pdfPage.stats;
+        if (self.onAfterDraw) {
+          self.onAfterDraw();
+        }
+        self.eventBus.dispatch('pagerendered', {
+          source: self,
+          pageNumber: self.id,
+          cssTransform: false
+        });
+        if (error) {
+          return Promise.reject(error);
+        }
+        return Promise.resolve(undefined);
       };
-      return;
-     }
-     cont();
-    };
-   }
-   var finishPaintTask = function finishPaintTask(error) {
-    if (paintTask === self.paintTask) {
-     self.paintTask = null;
-    }
-    if (error === 'cancelled') {
-     self.error = null;
-     return Promise.resolve(undefined);
-    }
-    self.renderingState = RenderingStates.FINISHED;
-    if (self.loadingIconDiv) {
-     div.removeChild(self.loadingIconDiv);
-     delete self.loadingIconDiv;
-    }
-    if (self.zoomLayer) {
-     var zoomLayerCanvas = self.zoomLayer.firstChild;
-     self.paintedViewportMap.delete(zoomLayerCanvas);
-     zoomLayerCanvas.width = 0;
-     zoomLayerCanvas.height = 0;
-     if (div.contains(self.zoomLayer)) {
-      div.removeChild(self.zoomLayer);
-     }
-     self.zoomLayer = null;
-    }
-    self.error = error;
-    self.stats = pdfPage.stats;
-    if (self.onAfterDraw) {
-     self.onAfterDraw();
-    }
-    self.eventBus.dispatch('pagerendered', {
-     source: self,
-     pageNumber: self.id,
-     cssTransform: false
-    });
-    if (error) {
-     return Promise.reject(error);
-    }
-    return Promise.resolve(undefined);
-   };
-   var paintTask = this.renderer === RendererType.SVG ? this.paintOnSvg(canvasWrapper) : this.paintOnCanvas(canvasWrapper);
-   paintTask.onRenderContinue = renderContinueCallback;
-   this.paintTask = paintTask;
-   var resultPromise = paintTask.promise.then(function () {
-    return finishPaintTask(null).then(function () {
-     if (textLayer) {
-      pdfPage.getTextContent({ normalizeWhitespace: true }).then(function textContentResolved(textContent) {
-       textLayer.setTextContent(textContent);
-       textLayer.render(TEXT_LAYER_RENDER_DELAY);
+      var paintTask = this.renderer === RendererType.SVG ? this.paintOnSvg(canvasWrapper) : this.paintOnCanvas(canvasWrapper);
+      paintTask.onRenderContinue = renderContinueCallback;
+      this.paintTask = paintTask;
+      var resultPromise = paintTask.promise.then(function () {
+        return finishPaintTask(null).then(function () {
+          if (textLayer) {
+            pdfPage.getTextContent({ normalizeWhitespace: true }).then(function textContentResolved(textContent) {
+              textLayer.setTextContent(textContent);
+              textLayer.render(TEXT_LAYER_RENDER_DELAY);
+            });
+          }
+        });
+      }, function (reason) {
+        return finishPaintTask(reason);
       });
-     }
-    });
-   }, function (reason) {
-    return finishPaintTask(reason);
-   });
-   if (this.annotationLayerFactory) {
-    if (!this.annotationLayer) {
-     this.annotationLayer = this.annotationLayerFactory.createAnnotationLayerBuilder(div, pdfPage, this.renderInteractiveForms);
-    }
-    this.annotationLayer.render(this.viewport, 'display');
-   }
-   div.setAttribute('data-loaded', true);
-   if (this.onBeforeDraw) {
-    this.onBeforeDraw();
-   }
-   return resultPromise;
-  },
-  paintOnCanvas: function (canvasWrapper) {
-   var resolveRenderPromise, rejectRenderPromise;
-   var promise = new Promise(function (resolve, reject) {
-    resolveRenderPromise = resolve;
-    rejectRenderPromise = reject;
-   });
-   var result = {
-    promise: promise,
-    onRenderContinue: function (cont) {
-     cont();
+      if (this.annotationLayerFactory) {
+        if (!this.annotationLayer) {
+          this.annotationLayer = this.annotationLayerFactory.createAnnotationLayerBuilder(div, pdfPage, this.renderInteractiveForms);
+        }
+        this.annotationLayer.render(this.viewport, 'display');
+      }
+      div.setAttribute('data-loaded', true);
+      if (this.onBeforeDraw) {
+        this.onBeforeDraw();
+      }
+      return resultPromise;
     },
-    cancel: function () {
-     renderTask.cancel();
-    }
-   };
-   var viewport = this.viewport;
-   var canvas = document.createElement('canvas');
-   canvas.id = 'page' + this.id;
-   canvas.setAttribute('hidden', 'hidden');
-   var isCanvasHidden = true;
-   var showCanvas = function () {
-    if (isCanvasHidden) {
-     canvas.removeAttribute('hidden');
-     isCanvasHidden = false;
-    }
-   };
-   canvasWrapper.appendChild(canvas);
-   this.canvas = canvas;
-   canvas.mozOpaque = true;
-   var ctx = canvas.getContext('2d', { alpha: false });
-   var outputScale = getOutputScale(ctx);
-   this.outputScale = outputScale;
-   if (pdfjsLib.PDFJS.useOnlyCssZoom) {
-    var actualSizeViewport = viewport.clone({ scale: CSS_UNITS });
-    outputScale.sx *= actualSizeViewport.width / viewport.width;
-    outputScale.sy *= actualSizeViewport.height / viewport.height;
-    outputScale.scaled = true;
-   }
-   if (pdfjsLib.PDFJS.maxCanvasPixels > 0) {
-    var pixelsInViewport = viewport.width * viewport.height;
-    var maxScale = Math.sqrt(pdfjsLib.PDFJS.maxCanvasPixels / pixelsInViewport);
-    if (outputScale.sx > maxScale || outputScale.sy > maxScale) {
-     outputScale.sx = maxScale;
-     outputScale.sy = maxScale;
-     outputScale.scaled = true;
-     this.hasRestrictedScaling = true;
-    } else {
-     this.hasRestrictedScaling = false;
-    }
-   }
-   var sfx = approximateFraction(outputScale.sx);
-   var sfy = approximateFraction(outputScale.sy);
-   canvas.width = roundToDivide(viewport.width * outputScale.sx, sfx[0]);
-   canvas.height = roundToDivide(viewport.height * outputScale.sy, sfy[0]);
-   canvas.style.width = roundToDivide(viewport.width, sfx[1]) + 'px';
-   canvas.style.height = roundToDivide(viewport.height, sfy[1]) + 'px';
-   this.paintedViewportMap.set(canvas, viewport);
-   var transform = !outputScale.scaled ? null : [
-    outputScale.sx,
-    0,
-    0,
-    outputScale.sy,
-    0,
-    0
-   ];
-   var renderContext = {
-    canvasContext: ctx,
-    transform: transform,
-    viewport: this.viewport,
-    renderInteractiveForms: this.renderInteractiveForms
-   };
-   var renderTask = this.pdfPage.render(renderContext);
-   renderTask.onContinue = function (cont) {
-    showCanvas();
-    if (result.onRenderContinue) {
-     result.onRenderContinue(cont);
-    } else {
-     cont();
-    }
-   };
-   renderTask.promise.then(function pdfPageRenderCallback() {
-    showCanvas();
-    resolveRenderPromise(undefined);
-   }, function pdfPageRenderError(error) {
-    showCanvas();
-    rejectRenderPromise(error);
-   });
-   return result;
-  },
-  paintOnSvg: function PDFPageView_paintOnSvg(wrapper) {
-   var cancelled = false;
-   var ensureNotCancelled = function () {
-    if (cancelled) {
-     throw 'cancelled';
-    }
-   };
-   var self = this;
-   var pdfPage = this.pdfPage;
-   var SVGGraphics = pdfjsLib.SVGGraphics;
-   var actualSizeViewport = this.viewport.clone({ scale: CSS_UNITS });
-   var promise = pdfPage.getOperatorList().then(function (opList) {
-    ensureNotCancelled();
-    var svgGfx = new SVGGraphics(pdfPage.commonObjs, pdfPage.objs);
-    return svgGfx.getSVG(opList, actualSizeViewport).then(function (svg) {
-     ensureNotCancelled();
-     self.svg = svg;
-     self.paintedViewportMap.set(svg, actualSizeViewport);
-     svg.style.width = wrapper.style.width;
-     svg.style.height = wrapper.style.height;
-     self.renderingState = RenderingStates.FINISHED;
-     wrapper.appendChild(svg);
-    });
-   });
-   return {
-    promise: promise,
-    onRenderContinue: function (cont) {
-     cont();
+    paintOnCanvas: function (canvasWrapper) {
+      var resolveRenderPromise, rejectRenderPromise;
+      var promise = new Promise(function (resolve, reject) {
+        resolveRenderPromise = resolve;
+        rejectRenderPromise = reject;
+      });
+      var result = {
+        promise: promise,
+        onRenderContinue: function (cont) {
+          cont();
+        },
+        cancel: function () {
+          renderTask.cancel();
+        }
+      };
+      var viewport = this.viewport;
+      var canvas = document.createElement('canvas');
+      canvas.id = 'page' + this.id;
+      canvas.setAttribute('hidden', 'hidden');
+      var isCanvasHidden = true;
+      var showCanvas = function () {
+        if (isCanvasHidden) {
+          canvas.removeAttribute('hidden');
+          isCanvasHidden = false;
+        }
+      };
+      canvasWrapper.appendChild(canvas);
+      this.canvas = canvas;
+      canvas.mozOpaque = true;
+      var ctx = canvas.getContext('2d', { alpha: false });
+      var outputScale = getOutputScale(ctx);
+      this.outputScale = outputScale;
+      if (pdfjsLib.PDFJS.useOnlyCssZoom) {
+        var actualSizeViewport = viewport.clone({ scale: CSS_UNITS });
+        outputScale.sx *= actualSizeViewport.width / viewport.width;
+        outputScale.sy *= actualSizeViewport.height / viewport.height;
+        outputScale.scaled = true;
+      }
+      if (pdfjsLib.PDFJS.maxCanvasPixels > 0) {
+        var pixelsInViewport = viewport.width * viewport.height;
+        var maxScale = Math.sqrt(pdfjsLib.PDFJS.maxCanvasPixels / pixelsInViewport);
+        if (outputScale.sx > maxScale || outputScale.sy > maxScale) {
+          outputScale.sx = maxScale;
+          outputScale.sy = maxScale;
+          outputScale.scaled = true;
+          this.hasRestrictedScaling = true;
+        } else {
+          this.hasRestrictedScaling = false;
+        }
+      }
+      var sfx = approximateFraction(outputScale.sx);
+      var sfy = approximateFraction(outputScale.sy);
+      canvas.width = roundToDivide(viewport.width * outputScale.sx, sfx[0]);
+      canvas.height = roundToDivide(viewport.height * outputScale.sy, sfy[0]);
+      canvas.style.width = roundToDivide(viewport.width, sfx[1]) + 'px';
+      canvas.style.height = roundToDivide(viewport.height, sfy[1]) + 'px';
+      this.paintedViewportMap.set(canvas, viewport);
+      var transform = !outputScale.scaled ? null : [outputScale.sx, 0, 0, outputScale.sy, 0, 0];
+      var renderContext = {
+        canvasContext: ctx,
+        transform: transform,
+        viewport: this.viewport,
+        renderInteractiveForms: this.renderInteractiveForms
+      };
+      var renderTask = this.pdfPage.render(renderContext);
+      renderTask.onContinue = function (cont) {
+        showCanvas();
+        if (result.onRenderContinue) {
+          result.onRenderContinue(cont);
+        } else {
+          cont();
+        }
+      };
+      renderTask.promise.then(function pdfPageRenderCallback() {
+        showCanvas();
+        resolveRenderPromise(undefined);
+      }, function pdfPageRenderError(error) {
+        showCanvas();
+        rejectRenderPromise(error);
+      });
+      return result;
     },
-    cancel: function () {
-     cancelled = true;
+    paintOnSvg: function PDFPageView_paintOnSvg(wrapper) {
+      var cancelled = false;
+      var ensureNotCancelled = function () {
+        if (cancelled) {
+          throw 'cancelled';
+        }
+      };
+      var self = this;
+      var pdfPage = this.pdfPage;
+      var SVGGraphics = pdfjsLib.SVGGraphics;
+      var actualSizeViewport = this.viewport.clone({ scale: CSS_UNITS });
+      var promise = pdfPage.getOperatorList().then(function (opList) {
+        ensureNotCancelled();
+        var svgGfx = new SVGGraphics(pdfPage.commonObjs, pdfPage.objs);
+        return svgGfx.getSVG(opList, actualSizeViewport).then(function (svg) {
+          ensureNotCancelled();
+          self.svg = svg;
+          self.paintedViewportMap.set(svg, actualSizeViewport);
+          svg.style.width = wrapper.style.width;
+          svg.style.height = wrapper.style.height;
+          self.renderingState = RenderingStates.FINISHED;
+          wrapper.appendChild(svg);
+        });
+      });
+      return {
+        promise: promise,
+        onRenderContinue: function (cont) {
+          cont();
+        },
+        cancel: function () {
+          cancelled = true;
+        }
+      };
+    },
+    setPageLabel: function PDFView_setPageLabel(label) {
+      this.pageLabel = typeof label === 'string' ? label : null;
+      if (this.pageLabel !== null) {
+        this.div.setAttribute('data-page-label', this.pageLabel);
+      } else {
+        this.div.removeAttribute('data-page-label');
+      }
     }
-   };
-  },
-  setPageLabel: function PDFView_setPageLabel(label) {
-   this.pageLabel = typeof label === 'string' ? label : null;
-   if (this.pageLabel !== null) {
-    this.div.setAttribute('data-page-label', this.pageLabel);
-   } else {
-    this.div.removeAttribute('data-page-label');
-   }
-  }
- };
- return PDFPageView;
+  };
+  return PDFPageView;
 }();
 exports.PDFPageView = PDFPageView;

Some files were not shown because too many files changed in this diff