731 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			731 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 
								 | 
							
								var x11 = Require('x11/core/x11');
							 | 
						||
| 
								 | 
							
								var xutil = Require('x11/core/xutil');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// adding XRender functions manually from
							 | 
						||
| 
								 | 
							
								//     http://cgit.freedesktop.org/xcb/proto/tree/src/render.xml?id=HEAD
							 | 
						||
| 
								 | 
							
								// and http://www.x.org/releases/X11R7.6/doc/renderproto/renderproto.txt
							 | 
						||
| 
								 | 
							
								// TODO: move to templates
							 | 
						||
| 
								 | 
							
								exports.requireExt = function(display, callback)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        var X = display.client;
							 | 
						||
| 
								 | 
							
								        X.QueryExtension('RENDER', function(err, ext) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (!ext.present)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                return callback(new Error('extension not available'));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.QueryVersion = function(clientMaj, clientMin, callback)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack('CCSLL', [ext.majorOpcode, 0, 3, clientMaj, clientMin]);
							 | 
						||
| 
								 | 
							
								                X.replies[X.seq_num] = [
							 | 
						||
| 
								 | 
							
								                    function(buf, opt) {
							 | 
						||
| 
								 | 
							
								                        var res = buf.unpack('LL');
							 | 
						||
| 
								 | 
							
								                        return res;
							 | 
						||
| 
								 | 
							
								                    },
							 | 
						||
| 
								 | 
							
								                    callback
							 | 
						||
| 
								 | 
							
								                ];
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.QueryPictFormat = function(callback)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack('CCS', [ext.majorOpcode, 1, 1]);
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								                X.replies[X.seq_num] = [
							 | 
						||
| 
								 | 
							
								                    function (buf, opt) {
							 | 
						||
| 
								 | 
							
								                        var res = {};
							 | 
						||
| 
								 | 
							
								                        var res1 = buf.unpack('LLLLL');
							 | 
						||
| 
								 | 
							
								                        var num_formats = res1[0];
							 | 
						||
| 
								 | 
							
								                        var num_screens = res1[1];
							 | 
						||
| 
								 | 
							
								                        var num_depths = res1[2];
							 | 
						||
| 
								 | 
							
								                        var num_visuals = res1[3];
							 | 
						||
| 
								 | 
							
								                        var num_subpixel = res1[4];
							 | 
						||
| 
								 | 
							
								                        // formats list:
							 | 
						||
| 
								 | 
							
								                        var offset = 24;
							 | 
						||
| 
								 | 
							
								                        res.formats = [];
							 | 
						||
| 
								 | 
							
								                        for (var i=0; i < num_formats; ++i)
							 | 
						||
| 
								 | 
							
								                        {
							 | 
						||
| 
								 | 
							
								                            var format = {};
							 | 
						||
| 
								 | 
							
								                            var f = buf.unpack('LCCxxSSSSSSSSL', offset);
							 | 
						||
| 
								 | 
							
								                            res.formats.push(f);
							 | 
						||
| 
								 | 
							
								                            offset += 28;
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								                        return res;
							 | 
						||
| 
								 | 
							
								                    },
							 | 
						||
| 
								 | 
							
								                    callback
							 | 
						||
| 
								 | 
							
								                ];
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.QueryFilters = function(callback)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack('CCSL', [ext.majorOpcode, 29, 2, display.screen[0].root]);
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								                X.replies[X.seq_num] = [
							 | 
						||
| 
								 | 
							
								                    function(buf, opt) {
							 | 
						||
| 
								 | 
							
								                        var h = buf.unpack('LL');
							 | 
						||
| 
								 | 
							
								                        var num_aliases = h[0];
							 | 
						||
| 
								 | 
							
								                        var num_filters = h[1];
							 | 
						||
| 
								 | 
							
								                        var aliases = [];
							 | 
						||
| 
								 | 
							
								                        var offset = 24; // LL + 16 bytes pad
							 | 
						||
| 
								 | 
							
								                        for (var i=0; i < num_aliases; ++i)
							 | 
						||
| 
								 | 
							
								                        {
							 | 
						||
| 
								 | 
							
								                            aliases.push(buf.unpack('S', offset)[0]);
							 | 
						||
| 
								 | 
							
								                            offset+=2;
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								                        var filters = [];
							 | 
						||
| 
								 | 
							
								                        for (var i=0; i < num_filters; ++i)
							 | 
						||
| 
								 | 
							
								                        {
							 | 
						||
| 
								 | 
							
								                            var len = buf.unpack('C', offset)[0];
							 | 
						||
| 
								 | 
							
								                            //if (!len) break;
							 | 
						||
| 
								 | 
							
								                            offset++;
							 | 
						||
| 
								 | 
							
								                            filters.push(buf.toString('ascii', offset, offset+len));
							 | 
						||
| 
								 | 
							
								                            offset+=len;
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								                        return [aliases, filters];
							 | 
						||
| 
								 | 
							
								                    },
							 | 
						||
| 
								 | 
							
								                    callback
							 | 
						||
| 
								 | 
							
								                ];
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            var valueList = [
							 | 
						||
| 
								 | 
							
								                ['repeat', 'Cxxx'],
							 | 
						||
| 
								 | 
							
								                ['alphaMap', 'L'],
							 | 
						||
| 
								 | 
							
								                ['alphaXOrigin', 'sxx'],
							 | 
						||
| 
								 | 
							
								                ['alphaYOrigin', 'sxx'],
							 | 
						||
| 
								 | 
							
								                ['clipXOrigin', 'sxx'],
							 | 
						||
| 
								 | 
							
								                ['clipYOrigin', 'sxx'],
							 | 
						||
| 
								 | 
							
								                ['clipMask', 'L'],
							 | 
						||
| 
								 | 
							
								                ['graphicsExposures', 'Cxxx'],
							 | 
						||
| 
								 | 
							
								                ['subwindowMode', 'Cxxx'],
							 | 
						||
| 
								 | 
							
								                ['polyEdge', 'Cxxx'],
							 | 
						||
| 
								 | 
							
								                ['polyMode', 'Cxxx'],
							 | 
						||
| 
								 | 
							
								                ['dither', 'L'],
							 | 
						||
| 
								 | 
							
								                ['componentAlpha', 'Cxxx']
							 | 
						||
| 
								 | 
							
								            ];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            var argumentLength = {
							 | 
						||
| 
								 | 
							
								                C: 1,
							 | 
						||
| 
								 | 
							
								                S: 2,
							 | 
						||
| 
								 | 
							
								                s: 2,
							 | 
						||
| 
								 | 
							
								                L: 4,
							 | 
						||
| 
								 | 
							
								                x: 1
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.CreatePicture = function(pid, drawable, pictformat, values)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                var mask = 0;
							 | 
						||
| 
								 | 
							
								                var reqLen = 5; // + (values + pad)/4
							 | 
						||
| 
								 | 
							
								                var format = 'CCSLLLL';
							 | 
						||
| 
								 | 
							
								                var params = [ext.majorOpcode, 4, reqLen, pid, drawable, pictformat, mask];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if (values)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    var valuesLength = 0;
							 | 
						||
| 
								 | 
							
								                    for (var i=0; i < valueList.length; ++i)
							 | 
						||
| 
								 | 
							
								                    {
							 | 
						||
| 
								 | 
							
								                        var name = valueList[i][0];
							 | 
						||
| 
								 | 
							
								                        var val = values[name];
							 | 
						||
| 
								 | 
							
								                        if (val) {
							 | 
						||
| 
								 | 
							
								                            mask |= (1 << i);
							 | 
						||
| 
								 | 
							
								                            params.push(val);
							 | 
						||
| 
								 | 
							
								                            var valueFormat = valueList[i][1];
							 | 
						||
| 
								 | 
							
								                            format += valueFormat;
							 | 
						||
| 
								 | 
							
								                            valuesLength += 4; //argumentLength[valueFormat];
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                    var pad4 = (valuesLength + 3) >> 2;
							 | 
						||
| 
								 | 
							
								                    var toPad = (pad4 << 2) - valuesLength;
							 | 
						||
| 
								 | 
							
								                    for (var i=0; i < toPad; ++i)
							 | 
						||
| 
								 | 
							
								                        format += 'x';
							 | 
						||
| 
								 | 
							
								                    reqLen += pad4;
							 | 
						||
| 
								 | 
							
								                    params[2] = reqLen;
							 | 
						||
| 
								 | 
							
								                    params[6] = mask;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.FreePicture = function(pid) {
							 | 
						||
| 
								 | 
							
								              X.pack_stream.pack('CCSL', [ext.majorOpcode, 7, 2, pid]);
							 | 
						||
| 
								 | 
							
								              X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								              X.seq_num++;
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            function floatToFix(f)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                return parseInt(f*65536);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            function colorToFix(f)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								              if (f < 0) f = 0;
							 | 
						||
| 
								 | 
							
								              if (f > 1) f = 1;
							 | 
						||
| 
								 | 
							
								              return parseInt(f*65535);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.SetPictureTransform = function(pid, matrix) {
							 | 
						||
| 
								 | 
							
								              var format = 'CCSLLLLLLLLLL';
							 | 
						||
| 
								 | 
							
								              if (matrix.length !== 9)
							 | 
						||
| 
								 | 
							
								                throw 'Render.SetPictureTransform: incorrect transform matrix. Must be array of 9 numbers';
							 | 
						||
| 
								 | 
							
								              var params = [ext.majorOpcode, 28, 11, pid];
							 | 
						||
| 
								 | 
							
								              for (var i=0; i < 9; ++i) {
							 | 
						||
| 
								 | 
							
								                if (typeof matrix[i] !== 'number')
							 | 
						||
| 
								 | 
							
								                  throw 'Render.SetPictureTransform: matrix element must be a number';
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(matrix[i]));
							 | 
						||
| 
								 | 
							
								              }
							 | 
						||
| 
								 | 
							
								              X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								              X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								              X.seq_num++;
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // see example of blur filter here: https://github.com/richoH/rxvt-unicode/blob/master/src/background.C
							 | 
						||
| 
								 | 
							
								            ext.SetPictureFilter = function(pid, name, filterParams)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                if (filterParams === 0)
							 | 
						||
| 
								 | 
							
								                  filterParams = [0];
							 | 
						||
| 
								 | 
							
								                if (!filterParams)
							 | 
						||
| 
								 | 
							
								                  filterParams = [];
							 | 
						||
| 
								 | 
							
								                if (!Array.isArray(filterParams))
							 | 
						||
| 
								 | 
							
								                  filterParams = [filterParams];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                var reqLen = 2;
							 | 
						||
| 
								 | 
							
								                var format = 'CCSLSxxp';
							 | 
						||
| 
								 | 
							
								                var params = [ext.majorOpcode, 30, reqLen, pid, name.length, name];
							 | 
						||
| 
								 | 
							
								                reqLen += xutil.padded_length(name.length+3)/4 + filterParams.length;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if (name == 'nearest' || name == 'bilinear' || name == 'fast' || name == 'good' || name == 'best') {
							 | 
						||
| 
								 | 
							
								                  if (filterParams.length !== 0) {
							 | 
						||
| 
								 | 
							
								                    throw 'Render.SetPictureFilter: "' + name + '" - unexpected parameters for filters';
							 | 
						||
| 
								 | 
							
								                  }
							 | 
						||
| 
								 | 
							
								                } else if (name == 'convolution') {
							 | 
						||
| 
								 | 
							
								                   if (filterParams.length < 2 || ((filterParams[0]*filterParams[1] + 2) !== filterParams.length) ) {
							 | 
						||
| 
								 | 
							
								                     throw 'Render.SetPictureFilter: "convolution" - incorrect matrix dimensions. Must be flat array [ w, h, elem1, elem2, ... ]';
							 | 
						||
| 
								 | 
							
								                   }
							 | 
						||
| 
								 | 
							
								                   for (var i=0; i < filterParams.length; ++i) {
							 | 
						||
| 
								 | 
							
								                      format += 'L';
							 | 
						||
| 
								 | 
							
								                      params.push(floatToFix(filterParams[i]));
							 | 
						||
| 
								 | 
							
								                   }
							 | 
						||
| 
								 | 
							
								                } else if (name == 'binomial' || name == 'gaussian') {
							 | 
						||
| 
								 | 
							
								                   if (filterParams.length !== 1) {
							 | 
						||
| 
								 | 
							
								                     throw 'Render.SetPictureFilter: "' + name + '" - incorrect number of parameters, must be exactly 1 number, instead got: ' + filterParams;
							 | 
						||
| 
								 | 
							
								                   }
							 | 
						||
| 
								 | 
							
								                  format += 'L';
							 | 
						||
| 
								 | 
							
								                  params.push(floatToFix(filterParams[0]));
							 | 
						||
| 
								 | 
							
								                } else {
							 | 
						||
| 
								 | 
							
								                    throw 'Render.SetPictureFilter: unknown filter "' + name + '"';
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                params[2] = reqLen;
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.CreateSolidFill = function(pid, r, g, b, a)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack('CCSLSSSS', [ext.majorOpcode, 33, 4, pid, colorToFix(r), colorToFix(g), colorToFix(b), colorToFix(a)]);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.RadialGradient = function(pid, p1, p2, r1, r2, stops)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                var reqLen = 9+stops.length*3;  //header + params + 1xStopfix+2xColors
							 | 
						||
| 
								 | 
							
								                var format = 'CCSLLLLLLLL';
							 | 
						||
| 
								 | 
							
								                var params = [ext.majorOpcode, 35, reqLen, pid];
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(p1[0])); // L
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(p1[1]));
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(p2[0]));
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(p2[1])); // L
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(r1)); // L
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(r2)); // L
							 | 
						||
| 
								 | 
							
								                params.push(stops.length);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                // [ [float stopDist, [float r, g, b, a] ], ...]
							 | 
						||
| 
								 | 
							
								                // stop distances
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < stops.length; ++i)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    format += 'L';
							 | 
						||
| 
								 | 
							
								                    // TODO: we know total params length in advance. ? params[index] =
							 | 
						||
| 
								 | 
							
								                    params.push(floatToFix(stops[i][0]))
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                // colors
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < stops.length; ++i)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    format += 'SSSS';
							 | 
						||
| 
								 | 
							
								                    for (var j=0; j < 4; ++j)
							 | 
						||
| 
								 | 
							
								                        params.push(colorToFix(stops[i][1][j]));
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.LinearGradient = function(pid, p1, p2, stops)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                var reqLen = 7+stops.length*3;  //header + params + 1xStopfix+2xColors
							 | 
						||
| 
								 | 
							
								                var format = 'CCSLLLLLL';
							 | 
						||
| 
								 | 
							
								                var params = [ext.majorOpcode, 34, reqLen, pid];
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(p1[0])); // L
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(p1[1]));
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(p2[0]));
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(p2[1])); // L
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                params.push(stops.length);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                // [ [float stopDist, [float r, g, b, a] ], ...]
							 | 
						||
| 
								 | 
							
								                // stop distances
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < stops.length; ++i)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    format += 'L';
							 | 
						||
| 
								 | 
							
								                    // TODO: we know total params length in advance. ? params[index] =
							 | 
						||
| 
								 | 
							
								                    params.push(floatToFix(stops[i][0]))
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                // colors
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < stops.length; ++i)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    format += 'SSSS';
							 | 
						||
| 
								 | 
							
								                    for (var j=0; j < 4; ++j)
							 | 
						||
| 
								 | 
							
								                        params.push(colorToFix(stops[i][1][j]));
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.ConicalGradient = function(pid, center, angle, stops)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                var reqLen = 6+stops.length*3;  //header + params + 1xStopfix+2xColors
							 | 
						||
| 
								 | 
							
								                var format = 'CCSLLLLL';
							 | 
						||
| 
								 | 
							
								                var params = [ext.majorOpcode, 36, reqLen, pid];
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(center[0])); // L
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(center[1]));
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(angle)); // L
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                params.push(stops.length);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                // [ [float stopDist, [float r, g, b, a] ], ...]
							 | 
						||
| 
								 | 
							
								                // stop distances
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < stops.length; ++i)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    format += 'L';
							 | 
						||
| 
								 | 
							
								                    // TODO: we know total params length in advance. ? params[index] =
							 | 
						||
| 
								 | 
							
								                    params.push(floatToFix(stops[i][0]))
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                // colors
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < stops.length; ++i)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    format += 'SSSS';
							 | 
						||
| 
								 | 
							
								                    for (var j=0; j < 4; ++j)
							 | 
						||
| 
								 | 
							
								                        params.push(colorToFix(stops[i][1][j]));
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.FillRectangles = function(op, pid, color, rects)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                var reqLen = 5+rects.length/2;
							 | 
						||
| 
								 | 
							
								                var format = 'CCSCxxxLSSSS';
							 | 
						||
| 
								 | 
							
								                var params = [ext.majorOpcode, 26, reqLen, op, pid];
							 | 
						||
| 
								 | 
							
								                for (var j=0; j < 4; ++j)
							 | 
						||
| 
								 | 
							
								                    params.push(colorToFix(color[j]));
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < rects.length; i+=4)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    format += 'ssSS';
							 | 
						||
| 
								 | 
							
								                    params.push(rects[i*4]);
							 | 
						||
| 
								 | 
							
								                    params.push(rects[i*4 + 1]);
							 | 
						||
| 
								 | 
							
								                    params.push(rects[i*4 + 2]);
							 | 
						||
| 
								 | 
							
								                    params.push(rects[i*4 + 3]);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.Composite = function(op, src, mask, dst, srcX, srcY, maskX, maskY, dstX, dstY, width, height)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(
							 | 
						||
| 
								 | 
							
								                    'CCSCxxxLLLssssssSS',
							 | 
						||
| 
								 | 
							
								                    [ext.majorOpcode, 8, 9, op, src, mask, dst, srcX, srcY, maskX, maskY, dstX, dstY, width, height]
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								                .flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // note that Trapezoids is considered deprecated by Render extension
							 | 
						||
| 
								 | 
							
								            ext.Trapezoids = function(op, src, srcX, srcY, dst, maskFormat, trapz)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                var format = 'CCSCxxxLLLss';
							 | 
						||
| 
								 | 
							
								                var params = [ext.majorOpcode, 10, 6+trapz.length, op, src, dst, maskFormat, srcX, srcY];
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < trapz.length; i++)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    format += 'llllllllll';
							 | 
						||
| 
								 | 
							
								                    for (var j=0; j < 10; ++j)
							 | 
						||
| 
								 | 
							
								                        params.push(floatToFix(trapz[i*10 + j]));
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.AddTraps = function(pic, offX, offY, trapList) {
							 | 
						||
| 
								 | 
							
								              var format = 'CCSLss';
							 | 
						||
| 
								 | 
							
								              var params = [ext.majorOpcode, 32, 3+trapList.length, pic, offX, offY];
							 | 
						||
| 
								 | 
							
								              for (var i=0; i < trapList.length; i++)
							 | 
						||
| 
								 | 
							
								              {
							 | 
						||
| 
								 | 
							
								                format += 'l';
							 | 
						||
| 
								 | 
							
								                params.push(floatToFix(trapList[i]));
							 | 
						||
| 
								 | 
							
								              }
							 | 
						||
| 
								 | 
							
								              X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								              X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								              X.seq_num++;
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.Triangles = function(op, src, srcX, srcY, dst, maskFormat, tris)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                var format = 'CCSCxxxLLLss';
							 | 
						||
| 
								 | 
							
								                var params = [ext.majorOpcode, 11, 6+tris.length, op, src, dst, maskFormat, srcX, srcY];
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < tris.length; i+=6)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    format += 'llllll';
							 | 
						||
| 
								 | 
							
								                    //TODO: Array.copy
							 | 
						||
| 
								 | 
							
								                    params.push(floatToFix(tris[i + 0])); // x1
							 | 
						||
| 
								 | 
							
								                    params.push(floatToFix(tris[i + 1])); // y1
							 | 
						||
| 
								 | 
							
								                    params.push(floatToFix(tris[i + 2])); // x2
							 | 
						||
| 
								 | 
							
								                    params.push(floatToFix(tris[i + 3])); // y2
							 | 
						||
| 
								 | 
							
								                    params.push(floatToFix(tris[i + 4])); // x3
							 | 
						||
| 
								 | 
							
								                    params.push(floatToFix(tris[i + 5])); // y3
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(format, params);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.CreateGlyphSet = function(gsid, format) {
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack('CCSLL', [ext.majorOpcode, 17, 3, gsid, format]);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.ReferenceGlyphSet = function(gsid, existing) {
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack('CCSLL', [ext.majorOpcode, 18, 3, gsid, existing]);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.FreeGlyphSet = function(gsid) {
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack('CCSL', [ext.majorOpcode, 19, 2, gsid]);
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.AddGlyphs = function(gsid, glyphs) {
							 | 
						||
| 
								 | 
							
								                var numGlyphs = glyphs.length;
							 | 
						||
| 
								 | 
							
								                var imageBytes = 0;
							 | 
						||
| 
								 | 
							
								                var glyphPaddedLength;
							 | 
						||
| 
								 | 
							
								                var glyphLength;
							 | 
						||
| 
								 | 
							
								                var stride;
							 | 
						||
| 
								 | 
							
								                var glyph;
							 | 
						||
| 
								 | 
							
								                for (var i = 0; i < numGlyphs; i++) {
							 | 
						||
| 
								 | 
							
								                  glyph = glyphs[i];
							 | 
						||
| 
								 | 
							
								                  if (glyph.width % 4 !== 0) {
							 | 
						||
| 
								 | 
							
								                    var stride = (glyph.width+3)&~3;
							 | 
						||
| 
								 | 
							
								                    var res = new Buffer(glyph.height*stride);
							 | 
						||
| 
								 | 
							
								                    res.fill(0);
							 | 
						||
| 
								 | 
							
								                    for (var y=0; y < glyph.height; ++y) {
							 | 
						||
| 
								 | 
							
								                      glyph.image.copy(res, y*stride, y*glyph.width, y*glyph.width + glyph.width);
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                    glyph.image = res;
							 | 
						||
| 
								 | 
							
								                    glyph.width = stride;
							 | 
						||
| 
								 | 
							
								                  }
							 | 
						||
| 
								 | 
							
								                  glyphLength = glyphs[i].image.length;
							 | 
						||
| 
								 | 
							
								                  imageBytes += glyphLength;
							 | 
						||
| 
								 | 
							
								                  glyph.offX = glyph.offX / 64;
							 | 
						||
| 
								 | 
							
								                  glyph.offY = glyph.offY / 64;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                var len = numGlyphs * 4 + imageBytes/4 + 3;
							 | 
						||
| 
								 | 
							
								                // TODO: check length, use bigReq
							 | 
						||
| 
								 | 
							
								                // X.pack_stream.pack('CCSLL', [ext.majorOpcode, 20, len, gsid, glyphs.length]);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                // BigReq: S + [ length ] replaced with SL + [ 0, length+1 ]
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack('CCSLLL', [ext.majorOpcode, 20, 0, len+1, gsid, glyphs.length]);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                // glyph ids
							 | 
						||
| 
								 | 
							
								                for (i = 0; i < numGlyphs; i++) {
							 | 
						||
| 
								 | 
							
								                  X.pack_stream.pack('L', [glyphs[i].id]);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                // width + heiht + origin xy + advance xy
							 | 
						||
| 
								 | 
							
								                for (i = 0; i < numGlyphs; i++) {
							 | 
						||
| 
								 | 
							
								                  X.pack_stream.pack('SSssss', [glyphs[i].width, glyphs[i].height, -glyphs[i].x, glyphs[i].y, glyphs[i].offX, glyphs[i].offY]);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                // image
							 | 
						||
| 
								 | 
							
								                for (i = 0; i < numGlyphs; i++) {
							 | 
						||
| 
								 | 
							
								                  X.pack_stream.write_queue.push(glyphs[i].image);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            //AddGlyphsFromPicture, opcode=21 (not in spec)
							 | 
						||
| 
								 | 
							
								            // FreeGlyps - opcode 22
							 | 
						||
| 
								 | 
							
								            // gsid(L) , glyphs.length (L) + each glyph id (L)
							 | 
						||
| 
								 | 
							
								            //
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // each GlyphEle:
							 | 
						||
| 
								 | 
							
								            // 1 byte - number of glyphs
							 | 
						||
| 
								 | 
							
								            // xxx
							 | 
						||
| 
								 | 
							
								            // int16 deltax, deltay
							 | 
						||
| 
								 | 
							
								            // + list of 8/16/32 byte indexesext.CompositeGlyphs
							 | 
						||
| 
								 | 
							
								            //  OR
							 | 
						||
| 
								 | 
							
								            //  255 + 0 + 0 + glyphsetId / font:
							 | 
						||
| 
								 | 
							
								            //  CxxxssL, [255, 0, 0, glyphable]
							 | 
						||
| 
								 | 
							
								            //
							 | 
						||
| 
								 | 
							
								            //  Each GlyphEle must be padded to 4 byte boundary
							 | 
						||
| 
								 | 
							
								            //
							 | 
						||
| 
								 | 
							
								            // glyphs as input:
							 | 
						||
| 
								 | 
							
								            // [ "just string (0,0) offset is used", [ 10, 10, "string offseted 10,10 from previous pen position" ], 1234567 ] 1234567 is glypfset id or FONT
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // TODO: pre-process input so strings larger than 254 chars are supported
							 | 
						||
| 
								 | 
							
								            // (split them into multiple entries with 0,0 offset)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            var formatFromBits = [,,,,,,,,'C',,,,,,,,'S',,,,,,,,,,,,,,,,'L'];
							 | 
						||
| 
								 | 
							
								            var bufferWriteBits = [,,,,,,,,'writeUInt8',,,,,,,,'writeUInt16LE',,,,,,,,,,,,,,,,'writeUInt32LE'];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // 8/16/32 bit string + 4-byte pad
							 | 
						||
| 
								 | 
							
								            function wstring(bits, s) {
							 | 
						||
| 
								 | 
							
								              var charLength = bits / 8;
							 | 
						||
| 
								 | 
							
								              var dataLength = s.length*charLength;
							 | 
						||
| 
								 | 
							
								              var res = new Buffer(xutil.padded_length(dataLength));
							 | 
						||
| 
								 | 
							
								              debugger;
							 | 
						||
| 
								 | 
							
								              var write = res[bufferWriteBits[bits]]
							 | 
						||
| 
								 | 
							
								              res.fill(0);
							 | 
						||
| 
								 | 
							
								              for(var i=0; i < s.length; i++)
							 | 
						||
| 
								 | 
							
								                write.call(res, s.charCodeAt(i), i*charLength);
							 | 
						||
| 
								 | 
							
								              return res;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            var compositeGlyphsOpcodeFromBits = [,,,,,,,,23,,,,,,,,24,,,,,,,,,,,,,,,,25];
							 | 
						||
| 
								 | 
							
								            ext.CompositeGlyphs = function(glyphBits, op, src, dst, maskFormat, gsid, srcX, srcY, glyphs)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                var opcode = compositeGlyphsOpcodeFromBits[glyphBits];
							 | 
						||
| 
								 | 
							
								                var charFormat = formatFromBits[glyphBits];
							 | 
						||
| 
								 | 
							
								                var charLength = glyphBits / 8;
							 | 
						||
| 
								 | 
							
								                var length = 7;
							 | 
						||
| 
								 | 
							
								                var glyphs_length_split = [];
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < glyphs.length; ++i) {
							 | 
						||
| 
								 | 
							
								                  var g = glyphs[i];
							 | 
						||
| 
								 | 
							
								                  switch (typeof g) {
							 | 
						||
| 
								 | 
							
								                    case 'string':
							 | 
						||
| 
								 | 
							
								                      length += xutil.padded_length(g.length*charLength)/4 + 2;
							 | 
						||
| 
								 | 
							
								                      break;
							 | 
						||
| 
								 | 
							
								                    case 'object':
							 | 
						||
| 
								 | 
							
								                      length += xutil.padded_length(g[2].length*charLength)/4 + 2;
							 | 
						||
| 
								 | 
							
								                      break;
							 | 
						||
| 
								 | 
							
								                    case 'number': // glyphset id
							 | 
						||
| 
								 | 
							
								                      length += 3;
							 | 
						||
| 
								 | 
							
								                      break;
							 | 
						||
| 
								 | 
							
								                  }
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.pack(
							 | 
						||
| 
								 | 
							
								                    'CCSCxxxLLLLss',
							 | 
						||
| 
								 | 
							
								                    [ext.majorOpcode, opcode, length, op, src, dst, maskFormat, gsid, srcX, srcY]
							 | 
						||
| 
								 | 
							
								                );
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < glyphs.length; ++i) {
							 | 
						||
| 
								 | 
							
								                  var g = glyphs[i];
							 | 
						||
| 
								 | 
							
								                  switch (typeof g) {
							 | 
						||
| 
								 | 
							
								                    case 'string':
							 | 
						||
| 
								 | 
							
								                      X.pack_stream.pack('Cxxxssa', [g.length, 0, 0, wstring(glyphBits, g)]);
							 | 
						||
| 
								 | 
							
								                      break;
							 | 
						||
| 
								 | 
							
								                    case 'object': // array
							 | 
						||
| 
								 | 
							
								                      X.pack_stream.pack('Cxxxssa', [g[2].length, g[0], g[1], wstring(glyphBits, g[2])]);
							 | 
						||
| 
								 | 
							
								                      break;
							 | 
						||
| 
								 | 
							
								                    case 'number': // glyphset id
							 | 
						||
| 
								 | 
							
								                      X.pack_stream.pack('CxxxSSL', [0xff, 0, 0, g]);
							 | 
						||
| 
								 | 
							
								                      break;
							 | 
						||
| 
								 | 
							
								                  }
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                X.pack_stream.flush();
							 | 
						||
| 
								 | 
							
								                X.seq_num++;
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      ext.CompositeGlyphs8 = function(op, src, dst, maskFormat, gsid, srcX, srcY, glyphs)
							 | 
						||
| 
								 | 
							
								      {
							 | 
						||
| 
								 | 
							
								         return ext.CompositeGlyphs(8, op, src, dst, maskFormat, gsid, srcX, srcY, glyphs);
							 | 
						||
| 
								 | 
							
								      };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      ext.CompositeGlyphs16 = function(op, src, dst, maskFormat, gsid, srcX, srcY, glyphs)
							 | 
						||
| 
								 | 
							
								      {
							 | 
						||
| 
								 | 
							
								         return ext.CompositeGlyphs(16, op, src, dst, maskFormat, gsid, srcX, srcY, glyphs);
							 | 
						||
| 
								 | 
							
								      };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      ext.CompositeGlyphs32 = function(op, src, dst, maskFormat, gsid, srcX, srcY, glyphs)
							 | 
						||
| 
								 | 
							
								      {
							 | 
						||
| 
								 | 
							
								         return ext.CompositeGlyphs(32, op, src, dst, maskFormat, gsid, srcX, srcY, glyphs);
							 | 
						||
| 
								 | 
							
								      };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // TODO: implement xutil-like code https://github.com/alexer/python-xlib-render/blob/master/xutil.py
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // TODO: name format fields
							 | 
						||
| 
								 | 
							
								            // 0 - id
							 | 
						||
| 
								 | 
							
								            // 1 - type ( direct / ? /)
							 | 
						||
| 
								 | 
							
								            // 2 - depth
							 | 
						||
| 
								 | 
							
								            //
							 | 
						||
| 
								 | 
							
								            // 3 - red shift
							 | 
						||
| 
								 | 
							
								            // 4 - red mask
							 | 
						||
| 
								 | 
							
								            // 5 - green shift
							 | 
						||
| 
								 | 
							
								            // 6 - green mask
							 | 
						||
| 
								 | 
							
								            // 7 - blue shift
							 | 
						||
| 
								 | 
							
								            // 8 - blue mask
							 | 
						||
| 
								 | 
							
								            // 9 - alpha shift
							 | 
						||
| 
								 | 
							
								            // 10 - alpha mask
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // 11 - colormap or none
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.QueryPictFormat(function(err, formats) {
							 | 
						||
| 
								 | 
							
								                if (err)
							 | 
						||
| 
								 | 
							
								                    return callback(err);
							 | 
						||
| 
								 | 
							
								                for (var i=0; i < formats.formats.length; ++i) {
							 | 
						||
| 
								 | 
							
								                    var f = formats.formats[i];
							 | 
						||
| 
								 | 
							
								                    if (f[2] == 1 && f[10] == 1)
							 | 
						||
| 
								 | 
							
								                        ext.mono1 = f[0] ;
							 | 
						||
| 
								 | 
							
								                    if (f[2] == 24 && f[3] == 16 && f[5] == 8 && f[7] == 0)
							 | 
						||
| 
								 | 
							
								                        ext.rgb24 = f[0];
							 | 
						||
| 
								 | 
							
								                    // 1, 32, 16, 255, 8, 255, 0, 255, 24, 255, 0
							 | 
						||
| 
								 | 
							
								                    if (f[2] == 32 && f[3] == 16 && f[4] == 255 && f[5] == 8 && f[6] == 255 && f[7] == 0 && f[9] == 24)
							 | 
						||
| 
								 | 
							
								                        ext.rgba32 = f[0] ;
							 | 
						||
| 
								 | 
							
								                    if (f[2] == 8 && f[10] == 255)
							 | 
						||
| 
								 | 
							
								                        ext.a8 = f[0];
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                callback(null, ext);
							 | 
						||
| 
								 | 
							
								            });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            [
							 | 
						||
| 
								 | 
							
								              "PICTFORMAT argument does not name a defined PICTFORMAT",
							 | 
						||
| 
								 | 
							
								              "PICTURE argument does not name a defined PICTURE",
							 | 
						||
| 
								 | 
							
								              "PICTOP argument does not name a defined PICTOP",
							 | 
						||
| 
								 | 
							
								              "GLYPHSET argument does not name a defined GLYPHSET",
							 | 
						||
| 
								 | 
							
								              "GLYPH argument does not name a defined GLYPH in the glyphset"
							 | 
						||
| 
								 | 
							
								            ].forEach(function(desc, code) {
							 | 
						||
| 
								 | 
							
								              X.errorParsers[ext.firstError + code] = function(err) {
							 | 
						||
| 
								 | 
							
								                err.message = "XRender: a value for a " + desc;
							 | 
						||
| 
								 | 
							
								              };
							 | 
						||
| 
								 | 
							
								            });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ext.PictOp = {
							 | 
						||
| 
								 | 
							
								              Minimum:    0,
							 | 
						||
| 
								 | 
							
								              Clear:    0,
							 | 
						||
| 
								 | 
							
								              Src:    1,
							 | 
						||
| 
								 | 
							
								              Dst:    2,
							 | 
						||
| 
								 | 
							
								              Over:    3,
							 | 
						||
| 
								 | 
							
								              OverReverse:   4,
							 | 
						||
| 
								 | 
							
								              In:    5,
							 | 
						||
| 
								 | 
							
								              InReverse:    6,
							 | 
						||
| 
								 | 
							
								              Out:    7,
							 | 
						||
| 
								 | 
							
								              OutReverse:    8,
							 | 
						||
| 
								 | 
							
								              Atop:    9,
							 | 
						||
| 
								 | 
							
								              AtopReverse:    10,
							 | 
						||
| 
								 | 
							
								              Xor:    11,
							 | 
						||
| 
								 | 
							
								              Add:    12,
							 | 
						||
| 
								 | 
							
								              Saturate:    13,
							 | 
						||
| 
								 | 
							
								              Maximum:    13,
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            /*,
							 | 
						||
| 
								 | 
							
								             * Operators only available in version 0.2,
							 | 
						||
| 
								 | 
							
								             */
							 | 
						||
| 
								 | 
							
								              DisjointMinimum:    0x10,
							 | 
						||
| 
								 | 
							
								              DisjointClear:    0x10,
							 | 
						||
| 
								 | 
							
								              DisjointSrc:    0x11,
							 | 
						||
| 
								 | 
							
								              DisjointDst:    0x12,
							 | 
						||
| 
								 | 
							
								              DisjointOver:    0x13,
							 | 
						||
| 
								 | 
							
								              DisjointOverReverse:    0x14,
							 | 
						||
| 
								 | 
							
								              DisjointIn:    0x15,
							 | 
						||
| 
								 | 
							
								              DisjointInReverse:    0x16,
							 | 
						||
| 
								 | 
							
								              DisjointOut:    0x17,
							 | 
						||
| 
								 | 
							
								              DisjointOutReverse:    0x18,
							 | 
						||
| 
								 | 
							
								              DisjointAtop:    0x19,
							 | 
						||
| 
								 | 
							
								              DisjointAtopReverse:    0x1a,
							 | 
						||
| 
								 | 
							
								              DisjointXor:    0x1b,
							 | 
						||
| 
								 | 
							
								              DisjointMaximum:    0x1b,
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								              ConjointMinimum:    0x20,
							 | 
						||
| 
								 | 
							
								              ConjointClear:    0x20,
							 | 
						||
| 
								 | 
							
								              ConjointSrc:    0x21,
							 | 
						||
| 
								 | 
							
								              ConjointDst:    0x22,
							 | 
						||
| 
								 | 
							
								              ConjointOver:    0x23,
							 | 
						||
| 
								 | 
							
								              ConjointOverReverse:    0x24,
							 | 
						||
| 
								 | 
							
								              ConjointIn:    0x25,
							 | 
						||
| 
								 | 
							
								              ConjointInReverse:    0x26,
							 | 
						||
| 
								 | 
							
								              ConjointOut:    0x27,
							 | 
						||
| 
								 | 
							
								              ConjointOutReverse:    0x28,
							 | 
						||
| 
								 | 
							
								              ConjointAtop:    0x29,
							 | 
						||
| 
								 | 
							
								              ConjointAtopReverse:    0x2a,
							 | 
						||
| 
								 | 
							
								              ConjointXor:    0x2b,
							 | 
						||
| 
								 | 
							
								              ConjointMaximum:    0x2b,
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            /*,
							 | 
						||
| 
								 | 
							
								             * Operators only available in version 0.11,
							 | 
						||
| 
								 | 
							
								             */
							 | 
						||
| 
								 | 
							
								              BlendMinimum :    0x30,
							 | 
						||
| 
								 | 
							
								              Multiply     :    0x30,
							 | 
						||
| 
								 | 
							
								              Screen       :    0x31,
							 | 
						||
| 
								 | 
							
								              Overlay      :    0x32,
							 | 
						||
| 
								 | 
							
								              Darken       :    0x33,
							 | 
						||
| 
								 | 
							
								              Lighten      :    0x34,
							 | 
						||
| 
								 | 
							
								              ColorDodge   :    0x35,
							 | 
						||
| 
								 | 
							
								              ColorBurn    :    0x36,
							 | 
						||
| 
								 | 
							
								              HardLight    :    0x37,
							 | 
						||
| 
								 | 
							
								              SoftLight    :    0x38,
							 | 
						||
| 
								 | 
							
								              Difference   :    0x39,
							 | 
						||
| 
								 | 
							
								              Exclusion    :    0x3a,
							 | 
						||
| 
								 | 
							
								              HSLHue       :    0x3b,
							 | 
						||
| 
								 | 
							
								              HSLSaturation:    0x3c,
							 | 
						||
| 
								 | 
							
								              HSLColor     :    0x3d,
							 | 
						||
| 
								 | 
							
								              HSLLuminosity:    0x3e,
							 | 
						||
| 
								 | 
							
								              BlendMaximum :    0x3e
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								          ext.PolyEdge = {
							 | 
						||
| 
								 | 
							
								            Sharp: 0,
							 | 
						||
| 
								 | 
							
								            Smooth: 1
							 | 
						||
| 
								 | 
							
								          };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								          ext.PolyMode = {
							 | 
						||
| 
								 | 
							
								            Precise: 0,
							 | 
						||
| 
								 | 
							
								            Imprecise: 1
							 | 
						||
| 
								 | 
							
								          };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								          ext.Repeat = {
							 | 
						||
| 
								 | 
							
								            None: 0,
							 | 
						||
| 
								 | 
							
								            Normal: 1,
							 | 
						||
| 
								 | 
							
								            Pad: 2,
							 | 
						||
| 
								 | 
							
								            Reflect: 3
							 | 
						||
| 
								 | 
							
								          };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								          ext.Subpixel = {
							 | 
						||
| 
								 | 
							
								            Unknown:       0,
							 | 
						||
| 
								 | 
							
								            HorizontalRGB: 1,
							 | 
						||
| 
								 | 
							
								            HorizontalBGR: 2,
							 | 
						||
| 
								 | 
							
								            VerticalRGB  : 3,
							 | 
						||
| 
								 | 
							
								            VerticalBGR  : 4,
							 | 
						||
| 
								 | 
							
								            None         : 5
							 | 
						||
| 
								 | 
							
								          };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								          ext.Filters = {
							 | 
						||
| 
								 | 
							
								            Nearest: 'nearest',
							 | 
						||
| 
								 | 
							
								            Bilinear: 'bilinear',
							 | 
						||
| 
								 | 
							
								            Convolution: 'convolution',
							 | 
						||
| 
								 | 
							
								            Fast: 'fast',
							 | 
						||
| 
								 | 
							
								            Good: 'good',
							 | 
						||
| 
								 | 
							
								            Best: 'best'
							 | 
						||
| 
								 | 
							
								          };
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								}
							 |