You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							108 lines
						
					
					
						
							3.5 KiB
						
					
					
				
			
		
		
	
	
							108 lines
						
					
					
						
							3.5 KiB
						
					
					
				const Stream = require('./'); const { DEFAULT_ENCODING, getEncoding } = Stream;
 | 
						|
const { end_of_stream, finished, stringToCodePoints } = require('../utils');
 | 
						|
const { encoders } = require('../table');
 | 
						|
 | 
						|
// 8.2 Interface TextEncoder
 | 
						|
 | 
						|
class TextEncoder {
 | 
						|
  /**
 | 
						|
   * @param {string=} label The label of the encoding. NONSTANDARD.
 | 
						|
   * @param {Object=} [options] NONSTANDARD.
 | 
						|
   */
 | 
						|
  constructor(label, options = {}) {
 | 
						|
    // A TextEncoder object has an associated encoding and encoder.
 | 
						|
 | 
						|
    /** @private */
 | 
						|
    this._encoding = null
 | 
						|
    /** @private @type {?Encoder} */
 | 
						|
    this._encoder = null
 | 
						|
 | 
						|
    // Non-standard
 | 
						|
    /** @private @type {boolean} */
 | 
						|
    this._do_not_flush = false
 | 
						|
    /** @private @type {string} */
 | 
						|
    this._fatal = options['fatal'] ? 'fatal' : 'replacement'
 | 
						|
 | 
						|
    // 2. Set enc's encoding to UTF-8's encoder.
 | 
						|
    if (options['NONSTANDARD_allowLegacyEncoding']) {
 | 
						|
      // NONSTANDARD behavior.
 | 
						|
      label = label !== undefined ? String(label) : DEFAULT_ENCODING
 | 
						|
      var encoding = getEncoding(label)
 | 
						|
      if (encoding === null || encoding.name === 'replacement')
 | 
						|
        throw RangeError('Unknown encoding: ' + label)
 | 
						|
      if (!encoders[encoding.name]) {
 | 
						|
        throw Error('Encoder not present.' +
 | 
						|
                    ' Did you forget to include encoding-indexes.js first?')
 | 
						|
      }
 | 
						|
      this._encoding = encoding
 | 
						|
    } else {
 | 
						|
      // Standard behavior.
 | 
						|
      this._encoding = getEncoding('utf-8')
 | 
						|
 | 
						|
      if (label !== undefined && 'console' in global) {
 | 
						|
        console.warn('TextEncoder constructor called with encoding label, '
 | 
						|
                      + 'which is ignored.')
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
  get encoding() {
 | 
						|
    return this._encoding.name.toLowerCase()
 | 
						|
  }
 | 
						|
  /**
 | 
						|
   * @param {string=} opt_string The string to encode.
 | 
						|
   * @param {Object=} options
 | 
						|
   */
 | 
						|
  encode(opt_string = '', options = {}) {
 | 
						|
    // NOTE: This option is nonstandard. None of the encodings
 | 
						|
    // permitted for encoding (i.e. UTF-8, UTF-16) are stateful when
 | 
						|
    // the input is a USVString so streaming is not necessary.
 | 
						|
    if (!this._do_not_flush)
 | 
						|
      this._encoder = encoders[this._encoding.name]({
 | 
						|
        fatal: this._fatal === 'fatal' })
 | 
						|
    this._do_not_flush = Boolean(options['stream'])
 | 
						|
 | 
						|
    // 1. Convert input to a stream.
 | 
						|
    const input = new Stream(stringToCodePoints(opt_string))
 | 
						|
 | 
						|
    // 2. Let output be a new stream
 | 
						|
    const output = []
 | 
						|
 | 
						|
    /** @type {?(number|!Array.<number>)} */
 | 
						|
    var result
 | 
						|
    // 3. While true, run these substeps:
 | 
						|
    while (true) {
 | 
						|
      // 1. Let token be the result of reading from input.
 | 
						|
      var token = input.read()
 | 
						|
      if (token === end_of_stream)
 | 
						|
        break
 | 
						|
      // 2. Let result be the result of processing token for encoder,
 | 
						|
      // input, output.
 | 
						|
      result = this._encoder.handler(input, token)
 | 
						|
      if (result === finished)
 | 
						|
        break
 | 
						|
      if (Array.isArray(result))
 | 
						|
        output.push.apply(output, /**@type {!Array.<number>}*/(result))
 | 
						|
      else
 | 
						|
        output.push(result)
 | 
						|
    }
 | 
						|
    // TODO: Align with spec algorithm.
 | 
						|
    if (!this._do_not_flush) {
 | 
						|
      while (true) {
 | 
						|
        result = this._encoder.handler(input, input.read())
 | 
						|
        if (result === finished)
 | 
						|
          break
 | 
						|
        if (Array.isArray(result))
 | 
						|
          output.push.apply(output, /**@type {!Array.<number>}*/(result))
 | 
						|
        else
 | 
						|
          output.push(result)
 | 
						|
      }
 | 
						|
      this._encoder = null
 | 
						|
    }
 | 
						|
    // 3. If result is finished, convert output into a byte sequence,
 | 
						|
    // and then return a Uint8Array object wrapping an ArrayBuffer
 | 
						|
    // containing output.
 | 
						|
    return new Uint8Array(output)
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
module.exports = TextEncoder |