| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129 |  
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
 
 
 
 
 
 
 
1
336
 
336
 
 
 
 
 
 
 
 
336
336
 
336
 
1
 
 
1
 
13
 
13
 
13
 
13
 
 
 
13
 
 
1
491
162
 
162
162
 
162
 
491
 
 
1
 
164
164
164
 
164
 
 
 
 
 
 
 
 
1
162
 
 
 
162
1
 
 
161
 
162
 
 
1
 
162
1
 
 
161
161
 
 
 
 
161
 
 
 
 
 
 
161
 
  | // Copyright (c) 2012 Kuba Niegowski
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
 
'use strict';
 
 
var util = require('util'),
    zlib = require('zlib'),
    ChunkStream = require('./chunkstream'),
    FilterAsync = require('./filter-async'),
    Parser = require('./parser'),
    bitmapper = require('./bitmapper');
 
 
var ParserAsync = module.exports = function(options) {
    ChunkStream.call(this);
 
    this._parser = new Parser(options, {
        read: this.read.bind(this),
        error: this._handleError.bind(this),
        metadata: this.emit.bind(this, "metadata"),
        gamma: this.emit.bind(this, "gamma"),
        finished: this._finished.bind(this),
        inflateData: this._inflateData.bind(this),
        createData: this._createData.bind(this)
    });
    this._options = options;
    this.writable = true;
 
    this._parser.start();
};
util.inherits(ParserAsync, ChunkStream);
 
 
ParserAsync.prototype._handleError = function(err) {
 
    this.emit('error', err);
 
    this.writable = false;
 
    this.destroy();
 
    Iif (this._inflate && this._inflate.destroy) {
        this._inflate.destroy();
    }
 
    this.errord = true;
};
 
ParserAsync.prototype._inflateData = function(data) {
    if (!this._inflate) {
        this._inflate = zlib.createInflate();
 
        this._inflate.on('error', this.emit.bind(this, 'error'));
        this._filter.on('complete', this._complete.bind(this));
 
        this._inflate.pipe(this._filter);
    }
    this._inflate.write(data);
};
 
ParserAsync.prototype._createData = function(width, height, bpp, depth, interlace) {
 
    this._bpp = bpp;
    this._depth = depth;
    this._interlace = interlace;
 
    this._filter = new FilterAsync(
      width, height,
      bpp,
      depth,
      interlace,
      this._options
    );
};
 
ParserAsync.prototype._finished = function(data) {
    Iif (this.errord) {
        return;
    }
 
    if (!this._inflate) {
        this.emit('error', 'No Inflate block');
    } else {
        // no more data to inflate
        this._inflate.end();
    }
    this.destroySoon();
};
 
ParserAsync.prototype._complete = function(data, width, height) {
 
    if (this.errord) {
        return;
    }
 
    try {
        data = bitmapper.dataToBitMap(data, width, height,
          this._bpp,
          this._depth,
          this._interlace);
 
        data = this._parser.reverseFiltered(data, this._depth, width, height);
    }
    catch(e) {
        this._handleError();
        return;
    }
 
    this.emit('parsed', data);
};
  |