Compare commits

..

No commits in common. "master" and "v2.0.5" have entirely different histories.

44 changed files with 933 additions and 1959 deletions

4
.gitignore vendored
View file

@ -1,4 +0,0 @@
*.log
yarn.lock
package-lock.json
node_modules

View file

@ -1,16 +1,14 @@
before_script: before_script:
- "export XAUTHORITY=/tmp/.Xauthority-Xvfb" - "export DISPLAY=:99.0"
- "xauth add :99 . $(mcookie)" - "/sbin/start-stop-daemon --start --quiet --pidfile /tmp/custom_xvfb_99.pid --make-pidfile --background --exec /usr/bin/Xvfb -- :99 -nolisten $NOLISTEN"
- "xauth add 127.0.0.2:99 . $(mcookie)"
- "/sbin/start-stop-daemon --start --quiet --pidfile /tmp/custom_xvfb_99.pid --make-pidfile --background --exec /usr/bin/Xvfb -- :99 -nolisten $NOLISTEN -auth $XAUTHORITY"
- "sleep 1"
env: env:
- NOLISTEN=tcp DISPLAY=:99.0 - NOLISTEN=tcp
# - NOLISTEN=unix DISPLAY=:99.0 - NOLISTEN=unix
# - NOLISTEN=unix DISPLAY=127.0.0.2:99.0
language: node_js language: node_js
node_js: node_js:
- '12' - '0.10'
- '14' - '0.12'
- '4.2'
- '5.0'

View file

@ -1,20 +1,21 @@
# node-x11 # About
[![Gitter](https://badges.gitter.im/Join Chat.svg)](https://gitter.im/sidorares/node-x11?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
X11 protocol client for node.js
X11 protocol client for Node.js: implements the core X11 protocol, as well as Xrender, Damage, Composite, Big-Requests, Dpms, Screensaver, XFixes, Shape, XTest, XC-Misc, GLX, and Apple-WM extensions. Implements core X11 protocol, as well as Xrender, Damage, Composite, Big-Requests, Dpms, Screensaver, XFixes, Shape, XTest, XC-Misc, GLX and Apple-WM extensions.
# install
[![Gitter](https://badges.gitter.im/Join%20Chat.svg)](https://gitter.im/sidorares/node-x11?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge) `npm install x11`
[![Build Status](https://secure.travis-ci.org/sidorares/node-x11.png)](http://travis-ci.org/sidorares/node-x11)
## Install
npm install x11
Windows users: Windows users:
1) install [XMing](http://www.straightrunning.com/XmingNotes/) or [Cygwin/X](http://x.cygwin.com/)
2) get node-x11 copy (using [git](http://code.google.com/p/msysgit/downloads/list?can=3) or from [Github](https://github.com/sidorares/node-x11/archives/master ))
1. install [XMing](http://www.straightrunning.com/XmingNotes/) or [Cygwin/X](http://x.cygwin.com/) #CI build status:
2. get node-x11 copy (using [git](http://code.google.com/p/msysgit/downloads/list?can=3) or from [Github](https://github.com/sidorares/node-x11/archives/master))
## Example [![Build Status](https://secure.travis-ci.org/sidorares/node-x11.png)](http://travis-ci.org/sidorares/node-x11)
# example
Core requests usage: Core requests usage:
@ -30,31 +31,18 @@ x11.createClient(function(err, display) {
var root = display.screen[0].root; var root = display.screen[0].root;
var wid = X.AllocID(); var wid = X.AllocID();
X.CreateWindow( X.CreateWindow(
wid, wid, root, // new window id, parent
root, // new window id, parent 0, 0, 100, 100, // x, y, w, h
0, 0, 0, 0, 0, // border, depth, class, visual
0,
500,
500, // x, y, w, h
0,
0,
0,
0, // border, depth, class, visual
{ eventMask: Exposure|PointerMotion } // other parameters { eventMask: Exposure|PointerMotion } // other parameters
); );
X.MapWindow(wid); X.MapWindow(wid);
var gc = X.AllocID(); var gc = X.AllocID();
X.CreateGC(gc, wid); X.CreateGC(gc, wid);
var white = display.screen[0].white_pixel;
var black = display.screen[0].black_pixel;
cidBlack = X.AllocID();
cidWhite = X.AllocID();
X.CreateGC(cidBlack, wid, { foreground: black, background: white });
X.CreateGC(cidWhite, wid, { foreground: white, background: black });
X.on('event', function(ev) { X.on('event', function(ev) {
if (ev.type == 12) { if (ev.type == 12)
X.PolyFillRectangle(wid, cidWhite, [0, 0, 500, 500]); {
X.PolyText8(wid, cidBlack, 50, 50, ['Hello, Node.JS!']); X.PolyText8(wid, gc, 50, 50, ['Hello, Node.JS!']);
} }
}); });
X.on('error', function(e) { X.on('error', function(e) {
@ -66,15 +54,14 @@ x11.createClient(function(err, display) {
}); });
``` ```
## Screenshots # Screenshots
![tetris game](https://lh6.googleusercontent.com/-RCRY9A7WwnA/Tlww0FHP7NI/AAAAAAAAAwo/nxfSxsw6xow/s400/tetris.png) ![tetris game](https://lh6.googleusercontent.com/-RCRY9A7WwnA/Tlww0FHP7NI/AAAAAAAAAwo/nxfSxsw6xow/s400/tetris.png)
![XRENDER gradients](https://lh4.googleusercontent.com/-VS0BMYYmq6M/Tlww0Y1ij0I/AAAAAAAAAws/pVWsPZ63Yeo/s400/render-gradients.png) ![XRENDER gradients](https://lh4.googleusercontent.com/-VS0BMYYmq6M/Tlww0Y1ij0I/AAAAAAAAAws/pVWsPZ63Yeo/s400/render-gradients.png)
![OpenGL glxgears](http://img-fotki.yandex.ru/get/4123/37511094.30/0_81712_6c2ebb11_L) ![OpenGL glxgears](http://img-fotki.yandex.ru/get/4123/37511094.30/0_81712_6c2ebb11_L)
![OpenGL teapot](http://img-fotki.yandex.ru/get/4132/37511094.30/0_81713_82a5ac48_L) ![OpenGL teapot](http://img-fotki.yandex.ru/get/4132/37511094.30/0_81713_82a5ac48_L)
## In use # In use
- [ntk](https://github.com/sidorares/ntk) - higher level toolkit on top of X11 - [ntk](https://github.com/sidorares/ntk) - higher level toolkit on top of X11
- [node-remote](https://github.com/AndrewSwerlick/node-remote) - media center controller - [node-remote](https://github.com/AndrewSwerlick/node-remote) - media center controller
- [tiles](https://github.com/dominictarr/tiles) - tiling window manager - [tiles](https://github.com/dominictarr/tiles) - tiling window manager
@ -87,7 +74,7 @@ x11.createClient(function(err, display) {
- [tinywm](https://github.com/Airblader/node-tinywm) The famous [TinyWM](https://github.com/mackstann/tinywm) written in node.js - [tinywm](https://github.com/Airblader/node-tinywm) The famous [TinyWM](https://github.com/mackstann/tinywm) written in node.js
- [basedwm](https://github.com/anko/basedwm) Infinite-desktop panning X window manager in LiveScript - [basedwm](https://github.com/anko/basedwm) Infinite-desktop panning X window manager in LiveScript
## X11 resources/documentation: # X11 resources/documentation:
- [Xplain](https://github.com/magcius/xplain) - A series of articles to help explain the X Window System http://magcius.github.io/xplain/article/ - [Xplain](https://github.com/magcius/xplain) - A series of articles to help explain the X Window System http://magcius.github.io/xplain/article/
- [Official X11 docs](http://www.x.org/releases/X11R7.6/doc/) - [Official X11 docs](http://www.x.org/releases/X11R7.6/doc/)
@ -96,7 +83,7 @@ x11.createClient(function(err, display) {
- [How to write composite manager](http://www.talisman.org/~erlkonig/misc/x11-composite-tutorial/) - [How to write composite manager](http://www.talisman.org/~erlkonig/misc/x11-composite-tutorial/)
- [Extended Window Manager Hints specification](http://standards.freedesktop.org/wm-spec/wm-spec-1.3.html) - [Extended Window Manager Hints specification](http://standards.freedesktop.org/wm-spec/wm-spec-1.3.html)
## Other implementations # Other implementations
- C: XLib - http://www.sbin.org/doc/Xlib/ http://www.tronche.com/gui/x/xlib/ http://www.x.org/docs/X11/xlib.pdf - C: XLib - http://www.sbin.org/doc/Xlib/ http://www.tronche.com/gui/x/xlib/ http://www.x.org/docs/X11/xlib.pdf
- C: XCB - http://xcb.freedesktop.org/ - C: XCB - http://xcb.freedesktop.org/
@ -109,9 +96,8 @@ x11.createClient(function(err, display) {
- Ruby: https://github.com/dj2/x-ruby-bindings - Ruby: https://github.com/dj2/x-ruby-bindings
- Clojure: https://github.com/noodlewiz/xcljb - Clojure: https://github.com/noodlewiz/xcljb
- Guile: https://github.com/mwitmer/guile-xcb - Guile: https://github.com/mwitmer/guile-xcb
- Emacs lisp: https://github.com/ch11ng/xelb ( autogenerated from XCB XML )
## Server side (protocol + functionality) implementations for js + DOM # Server side (protocol + functionality) implementations for js + DOM
would be really great to make completely web based playground page, connecting node-x11 api to DOM based implementation would be really great to make completely web based playground page, connecting node-x11 api to DOM based implementation

View file

@ -252,7 +252,7 @@ function genReq(req, last)
if (req.body.length == 0) if (req.body.length == 0)
{ {
result.push(' function() {'); result.push(' function() {');
result.push(' return Buffer.from([' + req.opcode + ', 0, 1, 0]);'); result.push(' return new Buffer([' + req.opcode + ', 0, 1, 0]);');
} else { } else {
result.push(' function (args) {'); result.push(' function (args) {');
result.push(' var extraLength = 0;'); result.push(' var extraLength = 0;');
@ -287,10 +287,10 @@ function genReq(req, last)
var reqLen4 = ((reqLen + 3) >> 2); var reqLen4 = ((reqLen + 3) >> 2);
if (extraLength) if (extraLength)
result.push(' var data = Buffer.alloc(' + reqLen + ' + extraLength);'); result.push(' var data = new Buffer(' + reqLen + ' + extraLength);');
else { else {
result.pop(); result.pop();
result.push(' var data = Buffer.alloc(' + reqLen + ');'); result.push(' var data = new Buffer(' + reqLen + ');');
} }
result.push(' data[0] = ' + req.opcode + ';'); result.push(' data[0] = ' + req.opcode + ';');
if (req.body.length != 0) { if (req.body.length != 0) {

View file

@ -571,7 +571,7 @@ module.exports.readJpeg = function(path)
var imageData = {}; var imageData = {};
imageData.width = j.width; imageData.width = j.width;
imageData.height = j.height; imageData.height = j.height;
imageData.data = Buffer.from(j.width*j.height*4); imageData.data = new Buffer(j.width*j.height*4);
j.copyToImageData(imageData); j.copyToImageData(imageData);
return imageData; return imageData;
} }

View file

@ -874,7 +874,7 @@ module.exports.readPng = function(path)
var imageData = {}; var imageData = {};
imageData.width = j.width; imageData.width = j.width;
imageData.height = j.height; imageData.height = j.height;
imageData.data = Buffer.alloc(j.width*j.height*4); imageData.data = new Buffer(j.width*j.height*4);
j.render(imageData); j.render(imageData);
return imageData; return imageData;
} }

View file

@ -1,344 +0,0 @@
/*
* Simple png decoder
* https://github.com/b37t1td/png-decoder
*/
var CRCTable = new Int32Array([
0x00000000, 0x77073096, 0xee0e612c, 0x990951ba, 0x076dc419, 0x706af48f,
0xe963a535, 0x9e6495a3, 0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988,
0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91, 0x1db71064, 0x6ab020f2,
0xf3b97148, 0x84be41de, 0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec, 0x14015c4f, 0x63066cd9,
0xfa0f3d63, 0x8d080df5, 0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172,
0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b, 0x35b5a8fa, 0x42b2986c,
0xdbbbc9d6, 0xacbcf940, 0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116, 0x21b4f4b5, 0x56b3c423,
0xcfba9599, 0xb8bda50f, 0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924,
0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d, 0x76dc4190, 0x01db7106,
0x98d220bc, 0xefd5102a, 0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818, 0x7f6a0dbb, 0x086d3d2d,
0x91646c97, 0xe6635c01, 0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e,
0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457, 0x65b0d9c6, 0x12b7e950,
0x8bbeb8ea, 0xfcb9887c, 0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2, 0x4adfa541, 0x3dd895d7,
0xa4d1c46d, 0xd3d6f4fb, 0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0,
0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9, 0x5005713c, 0x270241aa,
0xbe0b1010, 0xc90c2086, 0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4, 0x59b33d17, 0x2eb40d81,
0xb7bd5c3b, 0xc0ba6cad, 0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a,
0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683, 0xe3630b12, 0x94643b84,
0x0d6d6a3e, 0x7a6a5aa8, 0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe, 0xf762575d, 0x806567cb,
0x196c3671, 0x6e6b06e7, 0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc,
0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5, 0xd6d6a3e8, 0xa1d1937e,
0x38d8c2c4, 0x4fdff252, 0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60, 0xdf60efc3, 0xa867df55,
0x316e8eef, 0x4669be79, 0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236,
0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f, 0xc5ba3bbe, 0xb2bd0b28,
0x2bb45a92, 0x5cb36a04, 0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a, 0x9c0906a9, 0xeb0e363f,
0x72076785, 0x05005713, 0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38,
0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21, 0x86d3d2d4, 0xf1d4e242,
0x68ddb3f8, 0x1fda836e, 0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c, 0x8f659eff, 0xf862ae69,
0x616bffd3, 0x166ccf45, 0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2,
0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db, 0xaed16a4a, 0xd9d65adc,
0x40df0b66, 0x37d83bf0, 0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6, 0xbad03605, 0xcdd70693,
0x54de5729, 0x23d967bf, 0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94,
0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d
]);
var crc32 = function(buf) {
var crc = -1;
for (var i = 0; i < buf.length; i++) {
crc = CRCTable[(crc ^ buf[i]) & 0xff] ^ (crc >>> 8);
}
return crc ^ -1;
};
/*
* BE byteArray implementation
* by Svetlana Linuxenko <linuxenko@yahoo.com>
*/
/* eslint no-undef: 0 */
var byteArray = Uint8Array;
var cmp = function(a, b) {
if (!b) b = a; a = this;
return a.filter(function(c,i) { return c === b[i]; }).length === a.length;
};
var toInt = function(a) {
if (!a) a = this.slice(this.off, 4);
return (a[0] << 24) | (a[1] << 16) | (a[2] << 8) | a[3];
//return a[0] | (a[1] << 8) | (a[2] << 16) | (a[3] << 24);
};
var toBytes = function(int) {
return new byteArray([
(int >> 24) & 0xff,
(int >> 16) & 0xff,
(int >> 8) & 0xff,
int & 0xff
]);
};
var nextInt = function() {
return this.toInt(this.slice(this.off, (this.off += 4)));
};
var nextIntBytes = function() {
return this.nextBytes(4);
};
var nextBytes = function(size) {
return this.slice(this.off, (this.off += size));
};
var nextByte = function() {
return this.nextBytes(1)[0];
};
var insertInt = function(int) {
this.insertBytes(this.toBytes(int));
};
var insertBytes = function(bytes, length) {
length = length || 4;
this.set(bytes, this.off, length);
this.off += length;
};
var insertByte = function(byte) {
this.set([byte], this.off, (this.off += 1));
};
byteArray.prototype.cmp = cmp;
byteArray.prototype.toInt = toInt;
byteArray.prototype.nextInt = nextInt;
byteArray.prototype.nextIntBytes = nextIntBytes;
byteArray.prototype.toBytes = toBytes;
byteArray.prototype.insertInt = insertInt;
byteArray.prototype.insertBytes = insertBytes;
byteArray.prototype.nextBytes = nextBytes;
byteArray.prototype.nextByte = nextByte;
byteArray.prototype.insertByte = insertByte;
Object.defineProperty(byteArray.prototype , 'off', {
enumerable: false,
configurable: false,
writable: true,
value : 0
});
var SIGNATURE = new byteArray([0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a]);
var IHDR = new byteArray([0x49, 0x48, 0x44, 0x52]);
var IDAT = new byteArray([0x49, 0x44, 0x41, 0x54]);
var IEND = new byteArray([0x49, 0x45, 0x4e, 0x44]);
/*
* Decoder
*/
var zlib = require('zlib');
var inflateFunction = function(data) {
return zlib.inflateSync(Buffer.from(data));
};
var Decoder = function() { };
Decoder.prototype.parse = function(data) {
if (!(data instanceof byteArray)) {
data = new byteArray(data);
}
if (!SIGNATURE.cmp(data.nextBytes(SIGNATURE.length))) {
throw new Error('Not png');
}
while (data.off < data.length) {
var len = data.nextInt();
var hdr = data.nextBytes(len + 4);
if (crc32(hdr) !== data.nextInt()) {
throw new Error('Crc error');
}
if (IHDR.cmp(hdr)) {
this._IHDR = this._chunkIHDR(hdr.slice(4, len + 4));
if (this._IHDR.palette !== 8) {
throw new Error('Depth error');
}
if (this._IHDR.compression !== 0) {
throw new Error('Compression error');
}
if (this._IHDR.filter !== 0) {
throw new Error('Filter error');
}
if (this._IHDR.interlace !== 0) {
throw new Error('Interlace error');
}
switch (this._IHDR.colorType){
case 0: this.bpp = 1; break;
case 2: this.bpp = 3; break;
case 3: this.bpp = 1; break;
case 4: this.bpp = 2; break;
case 6: this.bpp = 4; break;
default: throw new Error('ColorType error');
}
this.chunks = [];
}
if (IDAT.cmp(hdr)) {
if (!this._IHDR) {
throw new Error('IHDR error');
}
this._chunkIDAT(hdr.slice(4, len + 4));
}
if (IEND.cmp(hdr)) {
return this._chunkIEND();
}
}
throw new Error('Data error');
};
Decoder.prototype._chunkIEND = function() {
var tmp = [];
for (var i = 0; i < this.chunks.length; i++) {
for (var j = 0; j < this.chunks[i].length; j++) {
tmp.push(this.chunks[i][j]);
}
}
return this.filter(inflateFunction(tmp));
};
Decoder.prototype._chunkIDAT = function(chunk) {
this.chunks.push(chunk);
};
Decoder.prototype._chunkIHDR = function(chunk) {
return {
width : chunk.nextInt(),
height : chunk.nextInt(),
palette : chunk.nextByte(),
colorType : chunk.nextByte(),
compression : chunk.nextByte(),
filter : chunk.nextByte(),
interlace : chunk.nextByte()
};
};
Decoder.prototype.filter = function(data) {
var bpp = this.bpp;
var width = this._IHDR.width, height = this._IHDR.height;
var pixels = new byteArray((width * height) * bpp);
var filter, line, left, leftup, up, pixel;
var lineWidth = width * bpp, byte, off;
for (var y = 0; y < height; y++) {
filter = data.nextByte();
line = data.nextBytes(lineWidth);
for (var x = 0; x < lineWidth; x++) {
if (filter !== 0) {
off = (y * lineWidth) + x;
}
byte = line.nextByte();
switch(filter) {
case 0: //None
pixel = byte;
break;
case 1: // Sub Raw(x) + Raw(x - bpp)
if (x < bpp) {
pixel = byte;
break;
}
pixel = pixels[off - bpp] + byte & 0xff;
break;
case 2: // Up(x) = Raw(x) + Prior(x)
if (y === 0) {
pixel = byte;
break;
}
pixel = pixels[off - lineWidth] + byte & 0xff;
break;
case 3: // Average(x) = Raw(x) + floor((Raw(x-bpp)+Prior(x))/2)
if (y === 0) {
if (x < bpp) {
pixel = byte;
} else {
pixel = (byte + (pixels[off - bpp] >> 1)) & 0xff;
}
break;
}
if (x < bpp) {
pixel = (byte + (pixels[off - lineWidth] >> 1)) & 0xff;
break;
}
pixel = (byte + (pixels[off - bpp] + pixels[off - lineWidth] >> 1)) & 0xff;
break;
case 4: // Paeth
if (y === 0) {
if (x < bpp) {
pixel = byte;
} else {
pixel = (byte + (pixels[off - bpp])) & 0xff;
}
break;
}
if (x < bpp) {
pixel = (byte + (pixels[off - lineWidth])) & 0xff;
break;
}
up = pixels[off - lineWidth];
left = pixels[off - bpp];
leftup = pixels[(off - lineWidth) - bpp];
var p = left + up - leftup,
pleft = Math.abs(p - left),
pup = Math.abs(p - up),
pleftup = Math.abs(p - leftup);
if (pleft <= pup && pleft <= pleftup){
pixel = byte + left & 0xff;
break;
} else if (pup <= pleftup) {
pixel = byte + up & 0xff;
break;
}
pixel = byte + leftup & 0xff;
break;
default:
throw new Error('Filter error: ' + filter);
}
pixels.insertByte(pixel);
}
}
return pixels;
};
module.exports = Decoder;

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.3 KiB

View file

@ -1,80 +0,0 @@
//var logo = require('./node-png').readPng('./node-logo.png');
var fs = require('fs');
var Decoder = require('./png-decoder');
var decoder = new Decoder();
var logo = {
data : Buffer.from(decoder.parse(fs.readFileSync('./screen.png'))),
width : decoder._IHDR.width,
height : decoder._IHDR.height
};
var x11 = require('../../lib');
var Exposure = x11.eventMask.Exposure;
x11.createClient(function(err, display)
{
var X = display.client;
X.require('render', function(err, Render) {
var root = display.screen[0].root;
main(root, X, Render, display);
});
});
function main(root, X, Render, display) {
var win, picWin, pic, gc;
win = X.AllocID();
X.CreateWindow(
win, root,
0, 0, logo.width, logo.height,
0, 0, 0, 0,
{ eventMask: Exposure }
);
X.MapWindow(win);
gc = X.AllocID();
X.CreateGC(gc, win);
var logoPixmap = X.AllocID();
X.CreatePixmap(logoPixmap, win, 24, logo.width, logo.height);
var rscreen = display.screen[0];
var screen =
rscreen.depths[rscreen.root_depth][
Object.keys(rscreen.depths[rscreen.root_depth])[0]];
var rmask = parseInt(screen.red_mask, 10);
var gmask = parseInt(screen.green_mask, 10);
var bmask = parseInt(screen.blue_mask, 10);
for (var y = 0; y < logo.height; y++) {
for (var x = 0; x < logo.width; x++) {
var pixel = Buffer.from([
logo.data[(x + logo.width * y) * 4],
logo.data[(x + logo.width * y) * 4 + 1],
logo.data[(x + logo.width * y) * 4 + 2], 0]).readInt32LE();
logo.data[(x + logo.width * y) * 4 ] = (pixel & rmask) >> 16;
logo.data[(x + logo.width * y) * 4 + 1] = (pixel & gmask) >> 8;
logo.data[(x + logo.width * y) * 4 + 2] = (pixel & bmask) >> 0;
logo.data[(x + logo.width * y) * 4 + 3] = 0x00;
}
}
X.PutImage(2, logoPixmap, gc, logo.width, logo.height, 0, 0, 0, 24, logo.data);
var logoPicture = X.AllocID();
Render.CreatePicture(logoPicture, logoPixmap, Render.rgb24);
var winPicture = X.AllocID();
Render.CreatePicture(winPicture, win, Render.rgb24);
X.on('event', function(ev) {
if (ev.name == 'Expose') {
Render.Composite(3, logoPicture, 0, winPicture, 0, 0, 0, 0, 0, 0, logo.width, logo.height);
}
});
}

8
examples/screenshot.js Normal file → Executable file
View file

@ -17,7 +17,7 @@ x11.createClient(function(err, display) {
var dispwin = X.AllocID(); var dispwin = X.AllocID();
X.CreateWindow(dispwin, root, 0, 0, width, height, 0, 0, 0, 0, { eventMask: x11.eventMask.Exposure }); X.CreateWindow(dispwin, root, 0, 0, width, height, 1, 1, 0, { eventMask: x11.eventMask.Exposure });
X.MapWindow(dispwin); X.MapWindow(dispwin);
//X.CopyArea(idScreenshot, dispwin, gc, 0, 0, 0, 0, width, height); //X.CopyArea(idScreenshot, dispwin, gc, 0, 0, 0, 0, width, height);
@ -71,11 +71,7 @@ x11.createClient(function(err, display) {
}); });
*/ */
X.GetImage(2, id, 0, 0, width, height, 0xffffffff, function(err, image) { X.GetImage(2, root, 0, 0, width, height, 0xffffffff, function(image) {
if (err) {
console.log(err);
process.exit(1);
}
console.log(image); console.log(image);
// format, drawable, gc, width, height, dstX, dstY, leftPad, depth, data // format, drawable, gc, width, height, dstX, dstY, leftPad, depth, data
X.PutImage(2, dispwin, gc, width, height, 0, 0, 0, 24, image.data); X.PutImage(2, dispwin, gc, width, height, 0, 0, 0, 24, image.data);

View file

@ -1,36 +0,0 @@
/*
* GCFunction usage example
*/
var x11 = require('../../lib');
x11.createClient(function(err, display) {
var X = display.client;
var root = display.screen[0].root;
var white = display.screen[0].white_pixel;
var black = display.screen[0].black_pixel;
var wid = X.AllocID();
X.CreateWindow(wid, root, 0, 0, 400, 300, 0, 0, 0, 0, {
backgroundPixel: black,
eventMask: x11.eventMask.ButtonPress|x11.eventMask.Exposure });
var gc = X.AllocID();
X.CreateGC(gc, wid, {foreground : white, 'function' : x11.gcFunction.GXinvert});
X.MapWindow(wid);
X.on('event', function(ev) {
if (ev.type === 12) {
X.PolyFillRectangle(wid, gc, [0, 0, 400, 300]);
}
if (ev.type === 4) {
var x = ev.x;
var y = ev.y;
X.PolyFillRectangle(wid, gc, [x - 25, y - 25, 50, 50]);
}
});
});

View file

@ -26,7 +26,7 @@ function padWidth(buf, width) {
return buf; return buf;
else { else {
var stride = (width+3)&~3; var stride = (width+3)&~3;
var res = Buffer.alloc(height*stride); var res = new Buffer(height*stride);
res.fill(0); res.fill(0);
for (var y=0; y < height; ++y) { for (var y=0; y < height; ++y) {
// memcpy(tmpbitmap+y*stride, bitmap->buffer+y*ginfo.width, ginfo.width); // memcpy(tmpbitmap+y*stride, bitmap->buffer+y*ginfo.width, ginfo.width);
@ -88,7 +88,7 @@ var xclient = x11.createClient({ debug: true }, function(err, display) {
var glyphFromCode = []; var glyphFromCode = [];
glyphs.forEach(function(g) { glyphs.forEach(function(g) {
if (!g.image || (g.image.length == 0)) { if (!g.image || (g.image.length == 0)) {
g.image = Buffer.alloc(64); g.image = new Buffer(64);
g.image.fill(0); g.image.fill(0);
g.width = 8; g.width = 8;
g.height = 8; g.height = 8;

View file

@ -6,7 +6,7 @@ var Pixmap = require('./pixmap').Pixmap;
var Buffer = require('buffer').Buffer; var Buffer = require('buffer').Buffer;
require('../../lib/unpackbuffer').addUnpack(Buffer); require('../../lib/unpackbuffer').addUnpack(Buffer);
var reversed = Buffer.alloc(256); var reversed = new Buffer(256);
for (var i=0; i < 256; ++i) for (var i=0; i < 256; ++i)
{ {
var res = 0; var res = 0;

View file

@ -8,7 +8,7 @@ var xclient = x11.createClient();
var Exposure = x11.eventMask.Exposure; var Exposure = x11.eventMask.Exposure;
var PointerMotion = x11.eventMask.PointerMotion; var PointerMotion = x11.eventMask.PointerMotion;
var bitmap = Buffer.alloc(128*128/8); // 16384 bits, 2048 bytes bitmap var bitmap = new Buffer(128*128/8); // 16384 bits, 2048 bytes bitmap
for (var i=0; i < bitmap.length; ++i) for (var i=0; i < bitmap.length; ++i)
{ {
bitmap[i] = i % 256; bitmap[i] = i % 256;

View file

@ -4,7 +4,7 @@ var x11 = require('../../lib');
var Exposure = x11.eventMask.Exposure; var Exposure = x11.eventMask.Exposure;
var PointerMotion = x11.eventMask.PointerMotion; var PointerMotion = x11.eventMask.PointerMotion;
var bitmap = Buffer.alloc(128*128*4); // 16384 bits, 2048 bytes bitmap var bitmap = new Buffer(128*128*4); // 16384 bits, 2048 bytes bitmap
for (var i=0; i < bitmap.length; ++i) for (var i=0; i < bitmap.length; ++i)
{ {
var byteNum = i%4; var byteNum = i%4;

View file

@ -25,7 +25,7 @@ var Buffer = require('buffer').Buffer;
var startpos = [4, 15]; var startpos = [4, 15];
var cupsize = [10, 20]; var cupsize = [10, 20];
var cup = Buffer.alloc(cupsize[0]*cupsize[1]); var cup = new Buffer(cupsize[0]*cupsize[1]);
var moveInterval; var moveInterval;
function clearCup() function clearCup()

View file

@ -56,8 +56,8 @@ var pc2 = [
function deskey(key, edf) function deskey(key, edf)
{ {
var i, j, l, m, n; var i, j, l, m, n;
var pc1m = Buffer.alloc(56); var pc1m = new Buffer(56);
var pcr = Buffer.alloc(56); var pcr = new Buffer(56);
var kn = new Array(32); var kn = new Array(32);
for ( j = 0; j < 56; j++ ) { for ( j = 0; j < 56; j++ ) {
@ -388,17 +388,17 @@ function desfunc(block, keys)
module.exports.response = function(challenge, password) module.exports.response = function(challenge, password)
{ {
var key = Buffer.alloc(8); var key = new Buffer(8);
key.fill(0); key.fill(0);
key.write(password.substring(0,8)); key.write(password.substring(0,8));
var in1 = challenge.slice(0,8); var in1 = challenge.slice(0,8);
var in2 = challenge.slice(8,16); var in2 = challenge.slice(8,16);
var res1 = Buffer.alloc(8); var res1 = new Buffer(8);
var res2 = Buffer.alloc(8); var res2 = new Buffer(8);
deskey(key, EN0); deskey(key, EN0);
des(in1, res1); des(in1, res1);
des(in2, res2); des(in2, res2);
var resp = Buffer.alloc(16); var resp = new Buffer(16);
res1.copy(resp); res1.copy(resp);
res2.copy(resp, 8); res2.copy(resp, 8);
return resp; return resp;

View file

@ -7,7 +7,7 @@
// It should create a pleasant looking hex dumb by default: // It should create a pleasant looking hex dumb by default:
// //
// var hexy = require('hexy.js'), // var hexy = require('hexy.js'),
// b = Buffer.from("\000\001\003\005\037\012\011bcdefghijklmnopqrstuvwxyz0123456789") // b = new Buffer("\000\001\003\005\037\012\011bcdefghijklmnopqrstuvwxyz0123456789")
// //
// console.log(hexy.hexy(b)) // console.log(hexy.hexy(b))
// //

View file

@ -400,7 +400,7 @@ RfbClient.prototype.readHextileTile = function(rect, cb)
}); });
return; return;
} }
tile.buffer = Buffer.alloc(tilebuflen); tile.buffer = new Buffer(tilebuflen);
function solidBackground() { function solidBackground() {
clog('solidBackground'); clog('solidBackground');
@ -667,8 +667,8 @@ function createConnection(params)
var wstream = fs.createWriteStream(params.rfbFileOut); var wstream = fs.createWriteStream(params.rfbFileOut);
wstream.write('FBS 001.001\n'); wstream.write('FBS 001.001\n');
stream.on('data', function(data) { stream.on('data', function(data) {
var sizeBuf = Buffer.alloc(4); var sizeBuf = new Buffer(4);
var timeBuf = Buffer.alloc(4); var timeBuf = new Buffer(4);
var size = data.length; var size = data.length;
sizeBuf.writeInt32BE(size, 0); sizeBuf.writeInt32BE(size, 0);
wstream.write(sizeBuf); wstream.write(sizeBuf);
@ -676,7 +676,7 @@ function createConnection(params)
timeBuf.writeInt32BE(+new Date() - start, 0); timeBuf.writeInt32BE(+new Date() - start, 0);
wstream.write(timeBuf); wstream.write(timeBuf);
var padding = 3 - ((size - 1) & 0x03); var padding = 3 - ((size - 1) & 0x03);
var pbuf = Buffer.alloc(padding); var pbuf = new Buffer(padding);
wstream.write(pbuf); wstream.write(pbuf);
}).on('end', function() { }).on('end', function() {
wstream.end(); wstream.end();

View file

@ -62,7 +62,7 @@ RfbServer.prototype.processSecurity = function()
break; break;
case rfb.security.VNC: case rfb.security.VNC:
// generate random 16 byte challenge // generate random 16 byte challenge
serv.challenge = Buffer.alloc(16); serv.challenge = new Buffer(16);
serv.challenge.write('1234567890abcdef'); serv.challenge.write('1234567890abcdef');
console.log(['sending challenge', serv.challenge]); console.log(['sending challenge', serv.challenge]);
serv.pack_stream.pack('a', [serv.challenge]).flush(); serv.pack_stream.pack('a', [serv.challenge]).flush();

View file

@ -34,7 +34,7 @@ function ReadFixedRequest(length, callback)
this.length = length; this.length = length;
this.callback = callback; this.callback = callback;
//clog(length); //clog(length);
this.data = Buffer.alloc(length); this.data = new Buffer(length);
this.received_bytes = 0; this.received_bytes = 0;
} }
@ -328,7 +328,7 @@ UnpackStream.prototype.pack = function(format, args)
} }
} }
var buf = Buffer.alloc(packetlength); var buf = new Buffer(packetlength);
var offset = 0; var offset = 0;
var arg = 0; var arg = 0;
for (var i = 0; i < format.length; ++i) for (var i = 0; i < format.length; ++i)

View file

@ -95,7 +95,7 @@ PixmapFromFile.prototype.mapColors = function(content,size){
} }
PixmapFromFile.prototype.toBuffer = function (colors,content,size) { PixmapFromFile.prototype.toBuffer = function (colors,content,size) {
var buf = Buffer.alloc(size.width*size.height*4); var buf = new Buffer(size.width*size.height*4);
var offset = 0, byte,color; var offset = 0, byte,color;
var copy; var copy;
if( !this.options.format || this.options.format.toUpperCase() === "BGRA"){ if( !this.options.format || this.options.format.toUpperCase() === "BGRA"){

View file

@ -1,24 +1,10 @@
// TODO: differentiate between auth types (i.e., MIT-MAGIC-COOKIE-1 and XDM-AUTHORIZATION-1) // TODO: http://en.wikipedia.org/wiki/X_Window_authorization
// and choose the best based on the algorithm in libXau's XauGetBestAuthByAddr
var fs = require('fs'); var fs = require('fs');
var Buffer = require('buffer').Buffer; var Buffer = require('buffer').Buffer;
// add 'unpack' method for buffer // add 'unpack' method for buffer
require('./unpackbuffer').addUnpack(Buffer); require('./unpackbuffer').addUnpack(Buffer);
var typeToName = {
256: 'Local',
65535: 'Wild',
254: 'Netname',
253: 'Krb5Principal',
252: 'LocalHost',
0: 'Internet',
1: 'DECnet',
2: 'Chaos',
5: 'ServerInterpreted',
6: 'Internet6'
};
function parseXauth( buf ) function parseXauth( buf )
{ {
var offset = 0; var offset = 0;
@ -28,7 +14,20 @@ function parseXauth( buf )
while (offset < buf.length) while (offset < buf.length)
{ {
var cookie = {}; var cookie = {};
cookie.type = buf.readUInt16BE(offset); var typeToName = {
256: 'Local',
65535: 'Wild',
254: 'Netname',
253: 'Krb5Principal',
252: 'LocalHost',
0: 'Internet',
1: 'DECnet',
2: 'Chaos',
5: 'ServerInterpreted',
6: 'InternetV6'
};
cookie.type = buf.readUInt16LE(offset);
console.log('Cookie type: ');
if (!typeToName[cookie.type]) { if (!typeToName[cookie.type]) {
console.warn('Unknown address type'); console.warn('Unknown address type');
} }
@ -50,61 +49,36 @@ function parseXauth( buf )
return auth; return auth;
} }
var homedir = require('os').homedir; var os = require('os');
var path = require('path'); var path = require('path');
function readXauthority(cb) { function readXauthority(cb) {
var filename = process.env.XAUTHORITY || path.join(homedir(), '.Xauthority'); var filename = process.env.XAUTHORITY || path.join(os.homedir(), '.Xauthority');
fs.readFile(filename, function(err, data) { fs.readFile(filename, function(err, data) {
if (!err) if (!err)
return cb(null, data); return cb(null, data);
if(err.code == 'ENOENT') { if(err.code == 'ENOENT') {
// Xming/windows uses %HOME%/Xauthority ( .Xauthority with no dot ) - try with this name // Xming/windows uses %HOME%/Xauthority ( .Xauthority with no dot ) - try with this name
filename = process.env.XAUTHORITY || path.join(homedir(), 'Xauthority'); filename = process.env.XAUTHORITY || path.join(os.homedir(), 'Xauthority');
fs.readFile(filename, function (err, data) { return fs.readFile(filename, cb);
if (err.code == 'ENOENT') {
cb(null, null);
} else {
cb(err);
}
});
} else { } else {
cb(err); cb(err);
} }
}); });
} }
module.exports = function( display, host, socketFamily, cb ) module.exports = function( display, host, cb )
{ {
var family;
if (socketFamily === 'IPv4') {
family = 0; // Internet
} else if (socketFamily === 'IPv6') {
family = 6; // Internet6
} else {
family = 256; // Local
}
readXauthority(function(err, data) { readXauthority(function(err, data) {
if(err) return cb(err); if(err) return cb(err);
if (!data) {
return cb(null, {
authName: '',
authData: ''
});
}
var auth = parseXauth(data); var auth = parseXauth(data);
for (var cookieNum in auth) for (var cookieNum in auth)
{ {
var cookie = auth[cookieNum]; var cookie = auth[cookieNum];
if ((typeToName[cookie.family] === 'Wild' || (cookie.type === family && cookie.address === host)) && if (cookie.display === display && cookie.address === host)
(cookie.display.length === 0 || cookie.display === display))
return cb( null, cookie ); return cb( null, cookie );
} }
// If no cookie is found, proceed without authentication cb(new Error('No auth cookie matching display=' + display + ' and host=' + host));
cb(null, {
authName: '',
authData: ''
});
}); });
}; };

View file

@ -231,10 +231,9 @@ function packValueMask(reqname, values)
}); });
var args = []; var args = [];
for (var i=0,length=masksList.length;i<length;i++) for (m in masksList)
{ {
var value = masksList[i]; var valueName = reqValueMaskName[masksList[m]];
var valueName = reqValueMaskName[value];
format += reqValueMask[valueName].format format += reqValueMask[valueName].format
args.push( values[valueName] ); args.push( values[valueName] );
} }
@ -465,7 +464,7 @@ var templates = {
function(mode, wid, name, type, units, data) function(mode, wid, name, type, units, data)
{ {
var padded4 = (data.length + 3) >> 2; var padded4 = (data.length + 3) >> 2;
var pad = Buffer.alloc( (padded4<<2) - data.length); var pad = new Buffer( (padded4<<2) - data.length);
var format = 'CCSLLLCxxxLaa'; var format = 'CCSLLLCxxxLaa';
var requestLength = 6 + padded4; var requestLength = 6 + padded4;
var dataLenInFormatUnits = data.length / (units >> 3); var dataLenInFormatUnits = data.length / (units >> 3);
@ -733,6 +732,12 @@ var templates = {
} }
], ],
FreePixmap: [
function (pixmap) {
return [ 'CxSL', [54, 2, pixmap] ];
}
],
CreateCursor: [ CreateCursor: [
function(cid, source, mask, foreRGB, backRGB, x, y) { function(cid, source, mask, foreRGB, backRGB, x, y) {
foreR = foreRGB.R foreR = foreRGB.R
@ -766,9 +771,13 @@ var templates = {
var vals = packValueMask('CreateGC', values); var vals = packValueMask('CreateGC', values);
var packetLength = 3 + (values ? vals[2].length : 0); var packetLength = 3 + (values ? vals[2].length : 0);
var args = [56, packetLength, cid]; var args = [56, packetLength, cid];
format += vals[0] args.push(vals[0]); // values bitmask
args.push(vals[1]); // values bitmask var valArr = vals[1];
args = args.concat(vals[2]) for (var v in valArr)
{
format += 'L'; // TODO: we know format string length in advance and += inefficient for string
args.push(valArr[v]);
}
return [format, args]; return [format, args];
} }
], ],
@ -851,7 +860,7 @@ var templates = {
var padded = xutil.padded_length(data.length); var padded = xutil.padded_length(data.length);
var reqLen = 6 + padded/4; // (length + 3) >> 2 ??? var reqLen = 6 + padded/4; // (length + 3) >> 2 ???
var padLength = padded - data.length; var padLength = padded - data.length;
var pad = Buffer.alloc(padLength); // TODO: new pack format 'X' - skip amount of bytes supplied in numerical argument var pad = new Buffer(padLength); // TODO: new pack format 'X' - skip amount of bytes supplied in numerical argument
// TODO: move code to calculate reqLength and use BigReq if needed outside of corereq.js // TODO: move code to calculate reqLength and use BigReq if needed outside of corereq.js
// NOTE: big req is used here (first 'L' in format, 0 and +1 in params), won't work if not enabled // NOTE: big req is used here (first 'L' in format, 0 and +1 in params), won't work if not enabled
@ -1021,12 +1030,6 @@ var templates = {
} }
], ],
Bell: [
function(percent) {
return ["CxCs",[108,1]];
}
],
ForceScreenSaver: [ ForceScreenSaver: [
function(activate) { function(activate) {
return [ 'CCS', [115, activate?1:0, 1] ]; return [ 'CCS', [115, activate?1:0, 1] ];

View file

@ -258,7 +258,7 @@ exports.requireExt = function(display, callback)
ext.BindTexImage = function(ctx, drawable, buffer, attribs) { ext.BindTexImage = function(ctx, drawable, buffer, attribs) {
if (!attribs) if (!attribs)
attribs = []; attribs = [];
var data = Buffer.alloc(12 + attribs.length*4); var data = new Buffer(12 + attribs.length*4);
data.writeUInt32LE(drawable, 0); data.writeUInt32LE(drawable, 0);
data.writeUInt32LE(buffer, 4); data.writeUInt32LE(buffer, 4);
data.writeUInt32LE(attribs.length, 8); data.writeUInt32LE(attribs.length, 8);
@ -268,7 +268,7 @@ exports.requireExt = function(display, callback)
} }
ext.ReleaseTexImage = function(ctx, drawable, buffer) { ext.ReleaseTexImage = function(ctx, drawable, buffer) {
var data = Buffer.alloc(8); var data = new Buffer(8);
data.writeUint32LE(drawable, 0); data.writeUint32LE(drawable, 0);
data.writeUint32LE(buffer, 4); data.writeUint32LE(buffer, 4);
ext.VendorPrivate(ctx, 1331, data); ext.VendorPrivate(ctx, 1331, data);
@ -287,7 +287,7 @@ exports.requireExt = function(display, callback)
X.pack_stream.pack('CCSLSSL', [ext.majorOpcode, 2, length, ctx, requestNum, requestTotal, data.length]); X.pack_stream.pack('CCSLSSL', [ext.majorOpcode, 2, length, ctx, requestNum, requestTotal, data.length]);
X.pack_stream.write_queue.push(data); X.pack_stream.write_queue.push(data);
var pad = Buffer.alloc(padLength); var pad = new Buffer(padLength);
pad.fill(0); pad.fill(0);
X.pack_stream.write_queue.push(pad); X.pack_stream.write_queue.push(pad);
X.pack_stream.flush(); X.pack_stream.flush();

View file

@ -16,7 +16,7 @@ module.exports = function(GLX, ctx) {
throw Error('Buffer too big. Make sure you are using RenderLarge for large commands'); throw Error('Buffer too big. Make sure you are using RenderLarge for large commands');
currentLength += len; currentLength += len;
var res = Buffer.alloc(len); var res = Buffer(len);
res.writeUInt16LE(len, 0); res.writeUInt16LE(len, 0);
res.writeUInt16LE(opcode, 2); res.writeUInt16LE(opcode, 2);
return res; return res;
@ -266,7 +266,7 @@ module.exports = function(GLX, ctx) {
typeSize[constants.BYTE] = 1; typeSize[constants.BYTE] = 1;
typeSize[constants.UNSIGNED_BYTE] = 1; typeSize[constants.UNSIGNED_BYTE] = 1;
var res = Buffer.alloc(60 + data.length*typeSize[type]); var res = new Buffer(60 + data.length*typeSize[type]);
res.writeUInt32LE(res.length, 0); res.writeUInt32LE(res.length, 0);
res.writeUInt32LE(110, 4); res.writeUInt32LE(110, 4);

View file

@ -204,43 +204,7 @@ exports.requireExt = function(display, callback)
X.pack_stream.flush(); X.pack_stream.flush();
}, },
ext.GetOutputInfo = function(output, ts, cb)
{
X.seq_num ++;
X.pack_stream.pack('CCSLL', [ext.majorOpcode, 9, 3, output, ts ]);
X.replies[X.seq_num] = [
function(buf, opt) {
var i;
var pos = 0;
var res = buf.unpack('LLLLCCSSSSS');
var info = {
timestamp : res[0],
crtc : res[1],
mm_width : res[2],
mm_height : res[3],
connection : res[4],
subpixelOrder : res[5],
preferredModes: res[8]
};
pos += 28;
var format = Array(res[6] + 1).join('L');
info.crtcs = buf.unpack(format, pos);
pos += res[6] << 2;
format = Array(res[7] + 1).join('L');
info.modes = buf.unpack(format, pos);
pos += res[7] << 2;
format = Array(res[9] + 1).join('L');
info.clones = buf.unpack(format, pos);
pos += res[9] << 2;
info.name = buf.slice(pos, pos + res_modes[10]).toString('binary');
return info;
},
cb
];
X.pack_stream.flush();
},
ext.GetCrtcInfo = function(crtc, configTs, cb) { ext.GetCrtcInfo = function(crtc, configTs, cb) {
X.seq_num ++; X.seq_num ++;
X.pack_stream.pack('CCSLL', [ext.majorOpcode, 20, 3, crtc, configTs ]); X.pack_stream.pack('CCSLL', [ext.majorOpcode, 20, 3, crtc, configTs ]);

View file

@ -5,14 +5,19 @@ var xutil = require('../xutil');
// http://cgit.freedesktop.org/xcb/proto/tree/src/render.xml?id=HEAD // http://cgit.freedesktop.org/xcb/proto/tree/src/render.xml?id=HEAD
// and http://www.x.org/releases/X11R7.6/doc/renderproto/renderproto.txt // and http://www.x.org/releases/X11R7.6/doc/renderproto/renderproto.txt
// TODO: move to templates // TODO: move to templates
exports.requireExt = function(display, callback) { exports.requireExt = function(display, callback)
{
var X = display.client; var X = display.client;
X.QueryExtension('RENDER', function(err, ext) { X.QueryExtension('RENDER', function(err, ext) {
if (!ext.present) {
if (!ext.present)
{
return callback(new Error('extension not available')); return callback(new Error('extension not available'));
} }
ext.QueryVersion = function(clientMaj, clientMin, callback) { ext.QueryVersion = function(clientMaj, clientMin, callback)
{
X.seq_num++; X.seq_num++;
X.pack_stream.pack('CCSLL', [ext.majorOpcode, 0, 3, clientMaj, clientMin]); X.pack_stream.pack('CCSLL', [ext.majorOpcode, 0, 3, clientMaj, clientMin]);
X.replies[X.seq_num] = [ X.replies[X.seq_num] = [
@ -23,9 +28,10 @@ exports.requireExt = function(display, callback) {
callback callback
]; ];
X.pack_stream.flush(); X.pack_stream.flush();
}; }
ext.QueryPictFormat = function(callback) { ext.QueryPictFormat = function(callback)
{
X.pack_stream.pack('CCS', [ext.majorOpcode, 1, 1]); X.pack_stream.pack('CCS', [ext.majorOpcode, 1, 1]);
X.seq_num++; X.seq_num++;
X.replies[X.seq_num] = [ X.replies[X.seq_num] = [
@ -40,7 +46,8 @@ exports.requireExt = function(display, callback) {
// formats list: // formats list:
var offset = 24; var offset = 24;
res.formats = []; res.formats = [];
for (var i = 0; i < num_formats; ++i) { for (var i=0; i < num_formats; ++i)
{
var format = {}; var format = {};
var f = buf.unpack('LCCxxSSSSSSSSL', offset); var f = buf.unpack('LCCxxSSSSSSSSL', offset);
res.formats.push(f); res.formats.push(f);
@ -51,9 +58,10 @@ exports.requireExt = function(display, callback) {
callback callback
]; ];
X.pack_stream.flush(); X.pack_stream.flush();
}; }
ext.QueryFilters = function(callback) { ext.QueryFilters = function(callback)
{
X.pack_stream.pack('CCSL', [ext.majorOpcode, 29, 2, display.screen[0].root]); X.pack_stream.pack('CCSL', [ext.majorOpcode, 29, 2, display.screen[0].root]);
X.seq_num++; X.seq_num++;
X.replies[X.seq_num] = [ X.replies[X.seq_num] = [
@ -63,12 +71,14 @@ exports.requireExt = function(display, callback) {
var num_filters = h[1]; var num_filters = h[1];
var aliases = []; var aliases = [];
var offset = 24; // LL + 16 bytes pad var offset = 24; // LL + 16 bytes pad
for (var i = 0; i < num_aliases; ++i) { for (var i=0; i < num_aliases; ++i)
{
aliases.push(buf.unpack('S', offset)[0]); aliases.push(buf.unpack('S', offset)[0]);
offset+=2; offset+=2;
} }
var filters = []; var filters = [];
for (var i = 0; i < num_filters; ++i) { for (var i=0; i < num_filters; ++i)
{
var len = buf.unpack('C', offset)[0]; var len = buf.unpack('C', offset)[0];
//if (!len) break; //if (!len) break;
offset++; offset++;
@ -80,7 +90,7 @@ exports.requireExt = function(display, callback) {
callback callback
]; ];
X.pack_stream.flush(); X.pack_stream.flush();
}; }
var valueList = [ var valueList = [
['repeat', 'Cxxx'], ['repeat', 'Cxxx'],
@ -106,19 +116,22 @@ exports.requireExt = function(display, callback) {
x: 1 x: 1
}; };
ext.CreatePicture = function(pid, drawable, pictformat, values) { ext.CreatePicture = function(pid, drawable, pictformat, values)
{
var mask = 0; var mask = 0;
var reqLen = 5; // + (values + pad)/4 var reqLen = 5; // + (values + pad)/4
var format = 'CCSLLLL'; var format = 'CCSLLLL';
var params = [ext.majorOpcode, 4, reqLen, pid, drawable, pictformat, mask]; var params = [ext.majorOpcode, 4, reqLen, pid, drawable, pictformat, mask];
if (values) { if (values)
{
var valuesLength = 0; var valuesLength = 0;
for (var i = 0; i < valueList.length; ++i) { for (var i=0; i < valueList.length; ++i)
{
var name = valueList[i][0]; var name = valueList[i][0];
var val = values[name]; var val = values[name];
if (val) { if (val) {
mask |= 1 << i; mask |= (1 << i);
params.push(val); params.push(val);
var valueFormat = valueList[i][1]; var valueFormat = valueList[i][1];
format += valueFormat; format += valueFormat;
@ -127,7 +140,8 @@ exports.requireExt = function(display, callback) {
} }
var pad4 = (valuesLength + 3) >> 2; var pad4 = (valuesLength + 3) >> 2;
var toPad = (pad4 << 2) - valuesLength; var toPad = (pad4 << 2) - valuesLength;
for (var i = 0; i < toPad; ++i) format += 'x'; for (var i=0; i < toPad; ++i)
format += 'x';
reqLen += pad4; reqLen += pad4;
params[2] = reqLen; params[2] = reqLen;
params[6] = mask; params[6] = mask;
@ -135,7 +149,7 @@ exports.requireExt = function(display, callback) {
X.pack_stream.pack(format, params); X.pack_stream.pack(format, params);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; }
ext.FreePicture = function(pid) { ext.FreePicture = function(pid) {
X.pack_stream.pack('CCSL', [ext.majorOpcode, 7, 2, pid]); X.pack_stream.pack('CCSL', [ext.majorOpcode, 7, 2, pid]);
@ -143,11 +157,13 @@ exports.requireExt = function(display, callback) {
X.seq_num++; X.seq_num++;
}; };
function floatToFix(f) { function floatToFix(f)
{
return parseInt(f*65536); return parseInt(f*65536);
} }
function colorToFix(f) { function colorToFix(f)
{
if (f < 0) f = 0; if (f < 0) f = 0;
if (f > 1) f = 1; if (f > 1) f = 1;
return parseInt(f*65535); return parseInt(f*65535);
@ -169,31 +185,26 @@ exports.requireExt = function(display, callback) {
}; };
// see example of blur filter here: https://github.com/richoH/rxvt-unicode/blob/master/src/background.C // see example of blur filter here: https://github.com/richoH/rxvt-unicode/blob/master/src/background.C
ext.SetPictureFilter = function(pid, name, filterParams) { ext.SetPictureFilter = function(pid, name, filterParams)
if (filterParams === 0) filterParams = [0]; {
if (!filterParams) filterParams = []; if (filterParams === 0)
if (!Array.isArray(filterParams)) filterParams = [filterParams]; filterParams = [0];
if (!filterParams)
filterParams = [];
if (!Array.isArray(filterParams))
filterParams = [filterParams];
var reqLen = 2; var reqLen = 2;
var format = 'CCSLSxxp'; var format = 'CCSLSxxp';
var params = [ext.majorOpcode, 30, reqLen, pid, name.length, name]; var params = [ext.majorOpcode, 30, reqLen, pid, name.length, name];
reqLen += xutil.padded_length(name.length+3)/4 + filterParams.length; reqLen += xutil.padded_length(name.length+3)/4 + filterParams.length;
if ( if (name == 'nearest' || name == 'bilinear' || name == 'fast' || name == 'good' || name == 'best') {
name == 'nearest' ||
name == 'bilinear' ||
name == 'fast' ||
name == 'good' ||
name == 'best'
) {
if (filterParams.length !== 0) { if (filterParams.length !== 0) {
throw 'Render.SetPictureFilter: "' + name + '" - unexpected parameters for filters'; throw 'Render.SetPictureFilter: "' + name + '" - unexpected parameters for filters';
} }
} else if (name == 'convolution') { } else if (name == 'convolution') {
if ( if (filterParams.length < 2 || ((filterParams[0]*filterParams[1] + 2) !== filterParams.length) ) {
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, ... ]'; throw 'Render.SetPictureFilter: "convolution" - incorrect matrix dimensions. Must be flat array [ w, h, elem1, elem2, ... ]';
} }
for (var i=0; i < filterParams.length; ++i) { for (var i=0; i < filterParams.length; ++i) {
@ -202,10 +213,7 @@ exports.requireExt = function(display, callback) {
} }
} else if (name == 'binomial' || name == 'gaussian') { } else if (name == 'binomial' || name == 'gaussian') {
if (filterParams.length !== 1) { if (filterParams.length !== 1) {
throw 'Render.SetPictureFilter: "' + throw 'Render.SetPictureFilter: "' + name + '" - incorrect number of parameters, must be exactly 1 number, instead got: ' + filterParams;
name +
'" - incorrect number of parameters, must be exactly 1 number, instead got: ' +
filterParams;
} }
format += 'L'; format += 'L';
params.push(floatToFix(filterParams[0])); params.push(floatToFix(filterParams[0]));
@ -218,22 +226,15 @@ exports.requireExt = function(display, callback) {
X.seq_num++; X.seq_num++;
}; };
ext.CreateSolidFill = function(pid, r, g, b, a) { ext.CreateSolidFill = function(pid, r, g, b, a)
X.pack_stream.pack('CCSLSSSS', [ {
ext.majorOpcode, X.pack_stream.pack('CCSLSSSS', [ext.majorOpcode, 33, 4, pid, colorToFix(r), colorToFix(g), colorToFix(b), colorToFix(a)]);
33,
4,
pid,
colorToFix(r),
colorToFix(g),
colorToFix(b),
colorToFix(a)
]);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; };
ext.RadialGradient = function(pid, p1, p2, r1, r2, stops) { ext.RadialGradient = function(pid, p1, p2, r1, r2, stops)
{
var reqLen = 9+stops.length*3; //header + params + 1xStopfix+2xColors var reqLen = 9+stops.length*3; //header + params + 1xStopfix+2xColors
var format = 'CCSLLLLLLLL'; var format = 'CCSLLLLLLLL';
var params = [ext.majorOpcode, 35, reqLen, pid]; var params = [ext.majorOpcode, 35, reqLen, pid];
@ -247,22 +248,26 @@ exports.requireExt = function(display, callback) {
// [ [float stopDist, [float r, g, b, a] ], ...] // [ [float stopDist, [float r, g, b, a] ], ...]
// stop distances // stop distances
for (var i = 0; i < stops.length; ++i) { for (var i=0; i < stops.length; ++i)
{
format += 'L'; format += 'L';
// TODO: we know total params length in advance. ? params[index] = // TODO: we know total params length in advance. ? params[index] =
params.push(floatToFix(stops[i][0])); params.push(floatToFix(stops[i][0]))
} }
// colors // colors
for (var i = 0; i < stops.length; ++i) { for (var i=0; i < stops.length; ++i)
{
format += 'SSSS'; format += 'SSSS';
for (var j = 0; j < 4; ++j) params.push(colorToFix(stops[i][1][j])); for (var j=0; j < 4; ++j)
params.push(colorToFix(stops[i][1][j]));
} }
X.pack_stream.pack(format, params); X.pack_stream.pack(format, params);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; };
ext.LinearGradient = function(pid, p1, p2, stops) { ext.LinearGradient = function(pid, p1, p2, stops)
{
var reqLen = 7+stops.length*3; //header + params + 1xStopfix+2xColors var reqLen = 7+stops.length*3; //header + params + 1xStopfix+2xColors
var format = 'CCSLLLLLL'; var format = 'CCSLLLLLL';
var params = [ext.majorOpcode, 34, reqLen, pid]; var params = [ext.majorOpcode, 34, reqLen, pid];
@ -275,22 +280,26 @@ exports.requireExt = function(display, callback) {
// [ [float stopDist, [float r, g, b, a] ], ...] // [ [float stopDist, [float r, g, b, a] ], ...]
// stop distances // stop distances
for (var i = 0; i < stops.length; ++i) { for (var i=0; i < stops.length; ++i)
{
format += 'L'; format += 'L';
// TODO: we know total params length in advance. ? params[index] = // TODO: we know total params length in advance. ? params[index] =
params.push(floatToFix(stops[i][0])); params.push(floatToFix(stops[i][0]))
} }
// colors // colors
for (var i = 0; i < stops.length; ++i) { for (var i=0; i < stops.length; ++i)
{
format += 'SSSS'; format += 'SSSS';
for (var j = 0; j < 4; ++j) params.push(colorToFix(stops[i][1][j])); for (var j=0; j < 4; ++j)
params.push(colorToFix(stops[i][1][j]));
} }
X.pack_stream.pack(format, params); X.pack_stream.pack(format, params);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; }
ext.ConicalGradient = function(pid, center, angle, stops) { ext.ConicalGradient = function(pid, center, angle, stops)
{
var reqLen = 6+stops.length*3; //header + params + 1xStopfix+2xColors var reqLen = 6+stops.length*3; //header + params + 1xStopfix+2xColors
var format = 'CCSLLLLL'; var format = 'CCSLLLLL';
var params = [ext.majorOpcode, 36, reqLen, pid]; var params = [ext.majorOpcode, 36, reqLen, pid];
@ -302,27 +311,33 @@ exports.requireExt = function(display, callback) {
// [ [float stopDist, [float r, g, b, a] ], ...] // [ [float stopDist, [float r, g, b, a] ], ...]
// stop distances // stop distances
for (var i = 0; i < stops.length; ++i) { for (var i=0; i < stops.length; ++i)
{
format += 'L'; format += 'L';
// TODO: we know total params length in advance. ? params[index] = // TODO: we know total params length in advance. ? params[index] =
params.push(floatToFix(stops[i][0])); params.push(floatToFix(stops[i][0]))
} }
// colors // colors
for (var i = 0; i < stops.length; ++i) { for (var i=0; i < stops.length; ++i)
{
format += 'SSSS'; format += 'SSSS';
for (var j = 0; j < 4; ++j) params.push(colorToFix(stops[i][1][j])); for (var j=0; j < 4; ++j)
params.push(colorToFix(stops[i][1][j]));
} }
X.pack_stream.pack(format, params); X.pack_stream.pack(format, params);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; }
ext.FillRectangles = function(op, pid, color, rects) { ext.FillRectangles = function(op, pid, color, rects)
{
var reqLen = 5+rects.length/2; var reqLen = 5+rects.length/2;
var format = 'CCSCxxxLSSSS'; var format = 'CCSCxxxLSSSS';
var params = [ext.majorOpcode, 26, reqLen, op, pid]; var params = [ext.majorOpcode, 26, reqLen, op, pid];
for (var j = 0; j < 4; ++j) params.push(colorToFix(color[j])); for (var j=0; j < 4; ++j)
for (var i = 0; i < rects.length; i += 4) { params.push(colorToFix(color[j]));
for (var i=0; i < rects.length; i+=4)
{
format += 'ssSS'; format += 'ssSS';
params.push(rects[i*4]); params.push(rects[i*4]);
params.push(rects[i*4 + 1]); params.push(rects[i*4 + 1]);
@ -332,51 +347,28 @@ exports.requireExt = function(display, callback) {
X.pack_stream.pack(format, params); X.pack_stream.pack(format, params);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; }
ext.Composite = function( ext.Composite = function(op, src, mask, dst, srcX, srcY, maskX, maskY, dstX, dstY, width, height)
op, {
src, X.pack_stream.pack(
mask, 'CCSCxxxLLLssssssSS',
dst, [ext.majorOpcode, 8, 9, op, src, mask, dst, srcX, srcY, maskX, maskY, dstX, dstY, width, height]
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(); .flush();
X.seq_num++; X.seq_num++;
}; }
// note that Trapezoids is considered deprecated by Render extension // note that Trapezoids is considered deprecated by Render extension
ext.Trapezoids = function(op, src, srcX, srcY, dst, maskFormat, trapz) { ext.Trapezoids = function(op, src, srcX, srcY, dst, maskFormat, trapz)
{
var format = 'CCSCxxxLLLss'; var format = 'CCSCxxxLLLss';
var params = [ext.majorOpcode, 10, 6+trapz.length, op, src, dst, maskFormat, srcX, srcY]; var params = [ext.majorOpcode, 10, 6+trapz.length, op, src, dst, maskFormat, srcX, srcY];
for (var i = 0; i < trapz.length; i++) { for (var i=0; i < trapz.length; i++)
{
format += 'llllllllll'; format += 'llllllllll';
for (var j = 0; j < 10; ++j) params.push(floatToFix(trapz[i * 10 + j])); for (var j=0; j < 10; ++j)
params.push(floatToFix(trapz[i*10 + j]));
} }
X.pack_stream.pack(format, params); X.pack_stream.pack(format, params);
X.pack_stream.flush(); X.pack_stream.flush();
@ -386,7 +378,8 @@ exports.requireExt = function(display, callback) {
ext.AddTraps = function(pic, offX, offY, trapList) { ext.AddTraps = function(pic, offX, offY, trapList) {
var format = 'CCSLss'; var format = 'CCSLss';
var params = [ext.majorOpcode, 32, 3+trapList.length, pic, offX, offY]; var params = [ext.majorOpcode, 32, 3+trapList.length, pic, offX, offY];
for (var i = 0; i < trapList.length; i++) { for (var i=0; i < trapList.length; i++)
{
format += 'l'; format += 'l';
params.push(floatToFix(trapList[i])); params.push(floatToFix(trapList[i]));
} }
@ -395,10 +388,12 @@ exports.requireExt = function(display, callback) {
X.seq_num++; X.seq_num++;
}; };
ext.Triangles = function(op, src, srcX, srcY, dst, maskFormat, tris) { ext.Triangles = function(op, src, srcX, srcY, dst, maskFormat, tris)
{
var format = 'CCSCxxxLLLss'; var format = 'CCSCxxxLLLss';
var params = [ext.majorOpcode, 11, 6+tris.length, op, src, dst, maskFormat, srcX, srcY]; var params = [ext.majorOpcode, 11, 6+tris.length, op, src, dst, maskFormat, srcX, srcY];
for (var i = 0; i < tris.length; i += 6) { for (var i=0; i < tris.length; i+=6)
{
format += 'llllll'; format += 'llllll';
//TODO: Array.copy //TODO: Array.copy
params.push(floatToFix(tris[i + 0])); // x1 params.push(floatToFix(tris[i + 0])); // x1
@ -411,25 +406,25 @@ exports.requireExt = function(display, callback) {
X.pack_stream.pack(format, params); X.pack_stream.pack(format, params);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; }
ext.CreateGlyphSet = function(gsid, format) { ext.CreateGlyphSet = function(gsid, format) {
X.pack_stream.pack('CCSLL', [ext.majorOpcode, 17, 3, gsid, format]); X.pack_stream.pack('CCSLL', [ext.majorOpcode, 17, 3, gsid, format]);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; }
ext.ReferenceGlyphSet = function(gsid, existing) { ext.ReferenceGlyphSet = function(gsid, existing) {
X.pack_stream.pack('CCSLL', [ext.majorOpcode, 18, 3, gsid, existing]); X.pack_stream.pack('CCSLL', [ext.majorOpcode, 18, 3, gsid, existing]);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; }
ext.FreeGlyphSet = function(gsid) { ext.FreeGlyphSet = function(gsid) {
X.pack_stream.pack('CCSL', [ext.majorOpcode, 19, 2, gsid]); X.pack_stream.pack('CCSL', [ext.majorOpcode, 19, 2, gsid]);
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; }
ext.AddGlyphs = function(gsid, glyphs) { ext.AddGlyphs = function(gsid, glyphs) {
var numGlyphs = glyphs.length; var numGlyphs = glyphs.length;
@ -442,7 +437,7 @@ exports.requireExt = function(display, callback) {
glyph = glyphs[i]; glyph = glyphs[i];
if (glyph.width % 4 !== 0) { if (glyph.width % 4 !== 0) {
var stride = (glyph.width+3)&~3; var stride = (glyph.width+3)&~3;
var res = Buffer.alloc(glyph.height * stride); var res = new Buffer(glyph.height*stride);
res.fill(0); res.fill(0);
for (var y=0; y < glyph.height; ++y) { for (var y=0; y < glyph.height; ++y) {
glyph.image.copy(res, y*stride, y*glyph.width, y*glyph.width + glyph.width); glyph.image.copy(res, y*stride, y*glyph.width, y*glyph.width + glyph.width);
@ -468,14 +463,7 @@ exports.requireExt = function(display, callback) {
} }
// width + heiht + origin xy + advance xy // width + heiht + origin xy + advance xy
for (i = 0; i < numGlyphs; i++) { for (i = 0; i < numGlyphs; i++) {
X.pack_stream.pack('SSssss', [ X.pack_stream.pack('SSssss', [glyphs[i].width, glyphs[i].height, -glyphs[i].x, glyphs[i].y, glyphs[i].offX, glyphs[i].offY]);
glyphs[i].width,
glyphs[i].height,
-glyphs[i].x,
glyphs[i].y,
glyphs[i].offX,
glyphs[i].offY
]);
} }
// image // image
for (i = 0; i < numGlyphs; i++) { for (i = 0; i < numGlyphs; i++) {
@ -483,29 +471,12 @@ exports.requireExt = function(display, callback) {
} }
X.pack_stream.flush(); X.pack_stream.flush();
X.seq_num++; X.seq_num++;
}; }
// As far as I know this is not implemented in any X server and always retuen "Bad implementation" //AddGlyphsFromPicture, opcode=21 (not in spec)
// Also documentation looks misleading as it's not mention glyph ids. // FreeGlyps - opcode 22
ext.AddGlyphsFromPicture = function(gsid, src, glyphs) { // gsid(L) , glyphs.length (L) + each glyph id (L)
var len = 3 + glyphs.length * 5; //
X.pack_stream.pack('CCSLLL', [ext.majorOpcode, 21, 0, len + 1, gsid, src]);
for (i = 0; i < glyphs.length; i++) {
X.pack_stream.pack('L', [glyphs[i].id]);
}
for (i = 0; i < glyphs.length; i++) {
X.pack_stream.pack('SSssssss', [
glyphs[i].width,
glyphs[i].height,
-glyphs[i].x,
glyphs[i].y,
glyphs[i].offX,
glyphs[i].offY,
glyphs[i].srcX,
glyphs[i].srcY
]);
}
};
// each GlyphEle: // each GlyphEle:
// 1 byte - number of glyphs // 1 byte - number of glyphs
@ -521,128 +492,29 @@ exports.requireExt = function(display, callback) {
// glyphs as input: // 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 // [ "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 // TODO: pre-process input so strings larger than 254 chars are supported
// (split them into multiple entries with 0,0 offset) // (split them into multiple entries with 0,0 offset)
var formatFromBits = [ var formatFromBits = [,,,,,,,,'C',,,,,,,,'S',,,,,,,,,,,,,,,,'L'];
, var bufferWriteBits = [,,,,,,,,'writeUInt8',,,,,,,,'writeUInt16LE',,,,,,,,,,,,,,,,'writeUInt32LE'];
,
,
,
,
,
,
,
'C',
,
,
,
,
,
,
,
'S',
,
,
,
,
,
,
,
,
,
,
,
,
,
,
,
'L'
];
var bufferWriteBits = [
,
,
,
,
,
,
,
,
'writeUInt8',
,
,
,
,
,
,
,
'writeUInt16LE',
,
,
,
,
,
,
,
,
,
,
,
,
,
,
,
'writeUInt32LE'
];
// 8/16/32 bit string + 4-byte pad // 8/16/32 bit string + 4-byte pad
function wstring(bits, s) { function wstring(bits, s) {
var charLength = bits / 8; var charLength = bits / 8;
var dataLength = s.length*charLength; var dataLength = s.length*charLength;
var res = Buffer.alloc(xutil.padded_length(dataLength)); var res = new Buffer(xutil.padded_length(dataLength));
debugger; debugger;
var write = res[bufferWriteBits[bits]]; var write = res[bufferWriteBits[bits]]
res.fill(0); res.fill(0);
for (var i = 0; i < s.length; i++) write.call(res, s.charCodeAt(i), i * charLength); for(var i=0; i < s.length; i++)
write.call(res, s.charCodeAt(i), i*charLength);
return res; return res;
} }
var compositeGlyphsOpcodeFromBits = [ var compositeGlyphsOpcodeFromBits = [,,,,,,,,23,,,,,,,,24,,,,,,,,,,,,,,,,25];
, ext.CompositeGlyphs = function(glyphBits, op, src, dst, maskFormat, gsid, srcX, srcY, glyphs)
, {
,
,
,
,
,
,
23,
,
,
,
,
,
,
,
24,
,
,
,
,
,
,
,
,
,
,
,
,
,
,
,
25
];
ext.CompositeGlyphs = function(glyphBits, op, src, dst, maskFormat, gsid, srcX, srcY, glyphs) {
var opcode = compositeGlyphsOpcodeFromBits[glyphBits]; var opcode = compositeGlyphsOpcodeFromBits[glyphBits];
var charFormat = formatFromBits[glyphBits]; var charFormat = formatFromBits[glyphBits];
var charLength = glyphBits / 8; var charLength = glyphBits / 8;
@ -662,18 +534,10 @@ exports.requireExt = function(display, callback) {
break; break;
} }
} }
X.pack_stream.pack('CCSCxxxLLLLss', [ X.pack_stream.pack(
ext.majorOpcode, 'CCSCxxxLLLLss',
opcode, [ext.majorOpcode, opcode, length, op, src, dst, maskFormat, gsid, srcX, srcY]
length, );
op,
src,
dst,
maskFormat,
gsid,
srcX,
srcY
]);
for (var i=0; i < glyphs.length; ++i) { for (var i=0; i < glyphs.length; ++i) {
var g = glyphs[i]; var g = glyphs[i];
switch (typeof g) { switch (typeof g) {
@ -692,15 +556,18 @@ exports.requireExt = function(display, callback) {
X.seq_num++; X.seq_num++;
}; };
ext.CompositeGlyphs8 = function(op, src, dst, maskFormat, gsid, srcX, srcY, glyphs) { ext.CompositeGlyphs8 = function(op, src, dst, maskFormat, gsid, srcX, srcY, glyphs)
{
return ext.CompositeGlyphs(8, 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) { ext.CompositeGlyphs16 = function(op, src, dst, maskFormat, gsid, srcX, srcY, glyphs)
{
return ext.CompositeGlyphs(16, 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) { ext.CompositeGlyphs32 = function(op, src, dst, maskFormat, gsid, srcX, srcY, glyphs)
{
return ext.CompositeGlyphs(32, op, src, dst, maskFormat, gsid, srcX, srcY, glyphs); return ext.CompositeGlyphs(32, op, src, dst, maskFormat, gsid, srcX, srcY, glyphs);
}; };
@ -723,36 +590,32 @@ exports.requireExt = function(display, callback) {
// 11 - colormap or none // 11 - colormap or none
ext.QueryPictFormat(function(err, formats) { ext.QueryPictFormat(function(err, formats) {
if (err) return callback(err); if (err)
return callback(err);
for (var i=0; i < formats.formats.length; ++i) { for (var i=0; i < formats.formats.length; ++i) {
var f = formats.formats[i]; var f = formats.formats[i];
if (f[2] == 1 && f[10] == 1) ext.mono1 = f[0]; if (f[2] == 1 && f[10] == 1)
if (f[2] == 24 && f[3] == 16 && f[5] == 8 && f[7] == 0) ext.rgb24 = f[0]; 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 // 1, 32, 16, 255, 8, 255, 0, 255, 24, 255, 0
if ( if (f[2] == 32 && f[3] == 16 && f[4] == 255 && f[5] == 8 && f[6] == 255 && f[7] == 0 && f[9] == 24)
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] ; ext.rgba32 = f[0] ;
if (f[2] == 8 && f[10] == 255) ext.a8 = f[0]; if (f[2] == 8 && f[10] == 255)
ext.a8 = f[0];
} }
callback(null, ext); callback(null, ext);
}); });
[ [
'PICTFORMAT argument does not name a defined PICTFORMAT', "PICTFORMAT argument does not name a defined PICTFORMAT",
'PICTURE argument does not name a defined PICTURE', "PICTURE argument does not name a defined PICTURE",
'PICTOP argument does not name a defined PICTOP', "PICTOP argument does not name a defined PICTOP",
'GLYPHSET argument does not name a defined GLYPHSET', "GLYPHSET argument does not name a defined GLYPHSET",
'GLYPH argument does not name a defined GLYPH in the glyphset' "GLYPH argument does not name a defined GLYPH in the glyphset"
].forEach(function(desc, code) { ].forEach(function(desc, code) {
X.errorParsers[ext.firstError + code] = function(err) { X.errorParsers[ext.firstError + code] = function(err) {
err.message = 'XRender: a value for a ' + desc; err.message = "XRender: a value for a " + desc;
}; };
}); });
@ -864,4 +727,4 @@ exports.requireExt = function(display, callback) {
Best: 'best' Best: 'best'
}; };
}); });
}; }

View file

@ -112,31 +112,5 @@ exports.requireExt = function(display, callback)
callback(null, ext); callback(null, ext);
}); });
*/ */
ext.events = {
ShapeNotify: 0
}
X.eventParsers[ext.firstEvent + ext.events.ShapeNotify] = function(type, seq, extra, code, raw)
{
var event = {};
event.type = type;
event.kind = code;
event.seq = seq;
event.window = extra;
var values = raw.unpack('ssSSLC');
event.x = values[0];
event.y = values[1];
event.width = values[2];
event.height = values[3];
event.time = values[4];
event.shaped = values[5];
event.name = 'ShapeNotify';
return event;
};
}); });
} }

View file

@ -1,22 +0,0 @@
/*
* GCFunction named shortcuts
*/
module.exports = {
GXclear : 0x0,
GXand : 0x1,
GXandReverse : 0x2,
GXcopy : 0x3,
GXandInverted : 0x4,
GXnoop : 0x5,
GXxor : 0x6,
GXor : 0x7,
GXnor : 0x8,
GXequiv : 0x9,
GXinvert : 0xa,
GXorReverse : 0xb,
GXcopyInverted : 0xc,
GXorInverted : 0xd,
GXnand : 0xe,
GXset : 0xf
};

View file

@ -32,10 +32,7 @@ function readVisuals(bl, visuals, n_visuals, cb)
}); });
} }
function readScreens(bl, display, cbDisplayReady) function readDepths(bl, display, depths, n_depths, cb)
{
var numParsedDepths = 0;
var readDepths = function(bl, display, depths, n_depths, cb)
{ {
if (n_depths == 0) if (n_depths == 0)
{ {
@ -49,15 +46,8 @@ function readScreens(bl, display, cbDisplayReady)
var visuals = {}; var visuals = {};
readVisuals(bl, visuals, n_visuals, function() readVisuals(bl, visuals, n_visuals, function()
{ {
if (dep in depths) {
for (var visual in visuals) {
depths[dep][visual] = visuals[visual];
}
} else {
depths[dep] = visuals; depths[dep] = visuals;
} if (Object.keys(depths).length == n_depths)
numParsedDepths++;
if (numParsedDepths == n_depths)
cb(); cb();
else else
readDepths(bl, display, depths, n_depths, cb); readDepths(bl, display, depths, n_depths, cb);
@ -65,6 +55,8 @@ function readScreens(bl, display, cbDisplayReady)
}); });
} }
function readScreens(bl, display, cbDisplayReady)
{
// for (i=0; i < display.screen_num; ++i) // for (i=0; i < display.screen_num; ++i)
{ {
var scr = {}; var scr = {};
@ -98,7 +90,7 @@ function readScreens(bl, display, cbDisplayReady)
if (display.screen.length == display.screen_num) if (display.screen.length == display.screen_num)
{ {
delete display.screen_num; delete display.screen_num;
cbDisplayReady(null, display); cbDisplayReady(display);
return; return;
} else { } else {
readScreens(bl, display, cbDisplayReady); readScreens(bl, display, cbDisplayReady);
@ -116,14 +108,9 @@ bl.unpack('C', function(res) {
if (res[0] == 0) if (res[0] == 0)
{ {
// conection time error // conection time error
// unpack error // unpack error (? TODO)
bl.unpack('Cxxxxxx', function (rlen) {
bl.get(rlen[0], function (reason) {
var err = new Error; var err = new Error;
err.message = 'X server connection failed: ' + reason.toString(); cb(err); // TODO: detect that this is error on xcore side
cb(err);
});
});
// TODO: do we need to close stream from our side? // TODO: do we need to close stream from our side?
// TODO: api to close source stream via attached unpackstream // TODO: api to close source stream via attached unpackstream
return; return;
@ -202,12 +189,10 @@ function getByteOrder() {
} }
} }
function writeClientHello(stream, displayNum, authHost, authFamily) function writeClientHello(stream, displayNum, authHost)
{ {
getAuthString( displayNum, authHost, authFamily, function( err, cookie ) { getAuthString( displayNum, authHost, function( err, cookie ) {
if (err) { debugger;
throw err;
}
var byte_order = getByteOrder(); var byte_order = getByteOrder();
var protocol_major = 11; // TODO: config? env? var protocol_major = 11; // TODO: config? env?
var protocol_minor = 0; var protocol_minor = 0;

View file

@ -7,7 +7,7 @@
// It should create a pleasant looking hex dumb by default: // It should create a pleasant looking hex dumb by default:
// //
// var hexy = require('hexy.js'), // var hexy = require('hexy.js'),
// b = Buffer.alloc("\000\001\003\005\037\012\011bcdefghijklmnopqrstuvwxyz0123456789") // b = new Buffer("\000\001\003\005\037\012\011bcdefghijklmnopqrstuvwxyz0123456789")
// //
// console.log(hexy.hexy(b)) // console.log(hexy.hexy(b))
// //

View file

@ -11,11 +11,6 @@ Object.defineProperty(module.exports, 'keySyms', {
get: function() { return require('./keysyms'); } get: function() { return require('./keysyms'); }
}); });
Object.defineProperty(module.exports, 'gcFunction', {
enumerable: true,
get: function() { return require('./gcfunction'); }
});
//TODO: //TODO:
// keepe everything in namespace for consistensy (eventMask, keySyms, class, destination ... // keepe everything in namespace for consistensy (eventMask, keySyms, class, destination ...
// or put most used constants to top namespace? (currently class and destination in top) // or put most used constants to top namespace? (currently class and destination in top)

View file

@ -23,7 +23,7 @@ function ReadFixedRequest(length, callback)
{ {
this.length = length; this.length = length;
this.callback = callback; this.callback = callback;
this.data = Buffer.alloc(length); this.data = new Buffer(length);
this.received_bytes = 0; this.received_bytes = 0;
} }
@ -201,7 +201,7 @@ UnpackStream.prototype.pstr = function(str)
var len = xutil.padded_length(str.length); var len = xutil.padded_length(str.length);
if (len == 0) if (len == 0)
return; // nothing to write return; // nothing to write
var buf = Buffer.alloc(len); var buf = new Buffer(len);
buf.write(str, 'binary'); buf.write(str, 'binary');
this.write_queue.push(buf); this.write_queue.push(buf);
} }
@ -231,7 +231,7 @@ UnpackStream.prototype.pack = function(format, args)
} }
} }
var buf = Buffer.alloc(packetlength); var buf = new Buffer(packetlength);
var offset = 0; var offset = 0;
var arg = 0; var arg = 0;
for (var i = 0; i < format.length; ++i) for (var i = 0; i < format.length; ++i)
@ -269,14 +269,12 @@ UnpackStream.prototype.pack = function(format, args)
buf[offset++] = (n >> 16) & 0xff; buf[offset++] = (n >> 16) & 0xff;
buf[offset++] = (n >> 24) & 0xff; buf[offset++] = (n >> 24) & 0xff;
break; break;
case 'a': // string, buffer, or array case 'a': // string or buffer
var str = args[arg++]; var str = args[arg++];
if (Buffer.isBuffer(str)) if (Buffer.isBuffer(str))
{ {
str.copy(buf, offset); str.copy(buf, offset);
offset += str.length; offset += str.length;
} else if(Array.isArray(str)) {
for(var item of str) buf[offset++] = item;
} else { } else {
// TODO: buffer.write could be faster // TODO: buffer.write could be faster
for (var c = 0; c < str.length; ++c) for (var c = 0; c < str.length; ++c)

View file

@ -6,6 +6,7 @@ var handshake = require('./handshake');
var EventEmitter = require('events').EventEmitter; var EventEmitter = require('events').EventEmitter;
var PackStream = require('./unpackstream'); var PackStream = require('./unpackstream');
var coreRequestsTemplate = require('./corereqs');
var hexy = require('./hexy').hexy; var hexy = require('./hexy').hexy;
var Buffer = require('buffer').Buffer; var Buffer = require('buffer').Buffer;
@ -19,25 +20,6 @@ var coreRequests = require('./corereqs');
var stdatoms = require('./stdatoms'); var stdatoms = require('./stdatoms');
var em = require('./eventmask').eventMask; var em = require('./eventmask').eventMask;
function stash ()
{
require('./ext/apple-wm');
require('./ext/big-requests');
require('./ext/composite');
require('./ext/damage');
require('./ext/dpms');
require('./ext/fixes');
require('./ext/glxconstants');
require('./ext/glx');
require('./ext/glxrender');
require('./ext/randr');
require('./ext/render');
require('./ext/screen-saver');
require('./ext/shape');
require('./ext/xc-misc');
require('./ext/xtest');
}
function XClient(displayNum, screenNum, options) function XClient(displayNum, screenNum, options)
{ {
EventEmitter.call(this); EventEmitter.call(this);
@ -49,6 +31,7 @@ function XClient(displayNum, screenNum, options)
this.displayNum = displayNum; this.displayNum = displayNum;
this.screenNum = screenNum; this.screenNum = screenNum;
this.authHost = os.hostname();
} }
util.inherits(XClient, EventEmitter); util.inherits(XClient, EventEmitter);
@ -56,14 +39,6 @@ XClient.prototype.init = function(stream)
{ {
this.stream = stream; this.stream = stream;
this.authHost = stream.remoteAddress;
// Node v0.10.x does not have stream.remoteFamily, so dig in to find it
this.authFamily = stream._getpeername ? stream._getpeername().family : stream.remoteFamily;
if (!this.authHost || this.authHost === '127.0.0.1' || this.authHost === '::1') {
this.authHost = os.hostname();
this.authFamily = null;
}
var pack_stream = new PackStream(); var pack_stream = new PackStream();
// data received from stream is dispached to // data received from stream is dispached to
@ -132,13 +107,11 @@ XClient.prototype.init = function(stream)
this.event_consumers = {}; // maps window id to eventemitter TODO: bad name this.event_consumers = {}; // maps window id to eventemitter TODO: bad name
this.eventParsers = {}; this.eventParsers = {};
this.errorParsers = {}; this.errorParsers = {};
this._extensions = {};
this.importRequestsFromTemplates(this, coreRequests); this.importRequestsFromTemplates(this, coreRequests);
this.startHandshake(); this.startHandshake();
this._closing = false; this._closing = false;
this._unusedIds = [];
} }
// TODO: close() = set 'closing' flag, watch it in replies and writeQueue, terminate if empty // TODO: close() = set 'closing' flag, watch it in replies and writeQueue, terminate if empty
@ -267,17 +240,12 @@ XClient.prototype.importRequestsFromTemplates = function(target, reqs)
XClient.prototype.AllocID = function() XClient.prototype.AllocID = function()
{ {
if (this._unusedIds.length > 0) {
return this._unusedIds.pop();
}
// TODO: handle overflow (XCMiscGetXIDRange from XC_MISC ext) // TODO: handle overflow (XCMiscGetXIDRange from XC_MISC ext)
// TODO: unused id buffer
this.display.rsrc_id++; this.display.rsrc_id++;
return (this.display.rsrc_id << this.display.rsrc_shift) + this.display.resource_base; return (this.display.rsrc_id << this.display.rsrc_shift) + this.display.resource_base;
}; }
XClient.prototype.ReleaseID = function(id) {
this._unusedIds.push(id);
};
// TODO: move core events unpackers to corereqs.js // TODO: move core events unpackers to corereqs.js
XClient.prototype.unpackEvent = function(type, seq, extra, code, raw, headerBuf) XClient.prototype.unpackEvent = function(type, seq, extra, code, raw, headerBuf)
@ -322,14 +290,7 @@ XClient.prototype.unpackEvent = function(type, seq, extra, code, raw, headerBuf)
event.x = values[5]; event.x = values[5];
event.y = values[6]; event.y = values[6];
event.values = values event.values = values
} else if (type == 9) { // FocusIn
event.name = "FocusIn";
event.mode = raw.unpack('C')[0];
event.wid = extra;
} else if (type == 10) { // FocusOut
event.name = "FocusOut";
event.mode = raw.unpack('C')[0];
event.wid = extra;
} else if (type == 12) { // Expose } else if (type == 12) { // Expose
var values = raw.unpack('SSSSS'); var values = raw.unpack('SSSSS');
event.name = 'Expose' event.name = 'Expose'
@ -443,7 +404,6 @@ XClient.prototype.unpackEvent = function(type, seq, extra, code, raw, headerBuf)
event.name = 'ClientMessage'; event.name = 'ClientMessage';
event.format = code; event.format = code;
event.wid = extra; event.wid = extra;
event.message_type = raw.unpack('L')[0];
var format = (code === 32) ? 'LLLLL' : (code === 16) ? 'SSSSSSSSSS' : 'CCCCCCCCCCCCCCCCCCCC'; var format = (code === 32) ? 'LLLLL' : (code === 16) ? 'SSSSSSSSSS' : 'CCCCCCCCCCCCCCCCCCCC';
event.data = raw.unpack(format, 4); event.data = raw.unpack(format, 4);
} else if (type == 34) { } else if (type == 34) {
@ -515,7 +475,7 @@ XClient.prototype.expectReplyHeader = function()
// raw event 32-bytes packet (primarily for use in SendEvent); // raw event 32-bytes packet (primarily for use in SendEvent);
// TODO: Event::pack based on event parameters, inverse to unpackEvent // TODO: Event::pack based on event parameters, inverse to unpackEvent
ev.rawData = Buffer.alloc(32); ev.rawData = new Buffer(32);
headerBuf.copy(ev.rawData); headerBuf.copy(ev.rawData);
buf.copy(ev.rawData, 8); buf.copy(ev.rawData, 8);
@ -563,13 +523,11 @@ XClient.prototype.expectReplyHeader = function()
XClient.prototype.startHandshake = function() { XClient.prototype.startHandshake = function() {
var client = this; var client = this;
handshake.writeClientHello(this.pack_stream, this.displayNum, this.authHost, this.authFamily); handshake.writeClientHello(this.pack_stream, this.displayNum, this.authHost);
handshake.readServerHello(this.pack_stream, function(err, display) handshake.readServerHello(this.pack_stream, function(display)
{ {
if (err) { // TODO: readServerHello can set error state in display
client.emit('error', err); // emit error in that case
return;
}
client.expectReplyHeader(); client.expectReplyHeader();
client.display = display; client.display = display;
display.client = client; display.client = client;
@ -579,25 +537,10 @@ XClient.prototype.startHandshake = function() {
XClient.prototype.require = function(extName, callback) XClient.prototype.require = function(extName, callback)
{ {
var self = this; var ext = require('./ext/' + extName);
var ext = this._extensions[extName]; ext.requireExt(this.display, callback);
if (ext) {
return process.nextTick(function() {
callback(null, ext);
});
} }
ext = require('./ext/' + extName);
ext.requireExt(this.display, function(err, _ext) {
if (err) {
return callback(err);
}
self._extensions[extName] = _ext;
callback(null, _ext);
});
};
module.exports.createClient = function(options, initCb) module.exports.createClient = function(options, initCb)
{ {
if (typeof options === 'function') { if (typeof options === 'function') {
@ -616,6 +559,8 @@ module.exports.createClient = function(options, initCb)
throw new Error("Cannot parse display"); throw new Error("Cannot parse display");
var host = displayMatch[1]; var host = displayMatch[1];
if (!host)
host = '127.0.0.1';
var displayNum = displayMatch[2]; var displayNum = displayMatch[2];
if (!displayNum) if (!displayNum)
@ -640,9 +585,10 @@ module.exports.createClient = function(options, initCb)
{ {
socketPath = display; socketPath = display;
} }
} else if(!host) } else if(host == '127.0.0.1') //TODO check if it's consistent with xlib (DISPLAY=127.0.0.1:0 -> local unix socket or port 6000?)
socketPath = '/tmp/.X11-unix/X' + displayNum; socketPath = '/tmp/.X11-unix/X' + displayNum;
} }
//socketPath = '/tmp/.X11-unix/X' + displayNum;
var client = new XClient(displayNum, screenNum, options); var client = new XClient(displayNum, screenNum, options);
var connectStream = function() { var connectStream = function() {

View file

@ -12,7 +12,7 @@
"X" "X"
], ],
"homepage": "https://github.com/sidorares/node-x11", "homepage": "https://github.com/sidorares/node-x11",
"version": "2.3.0", "version": "2.0.5",
"maintainers": [ "maintainers": [
{ {
"name": "Andrey Sidorov", "name": "Andrey Sidorov",
@ -40,15 +40,14 @@
"node": "*" "node": "*"
}, },
"devDependencies": { "devDependencies": {
"async": "^3.0.1", "mocha": "*",
"mocha": "^7.1.2", "should": "*",
"sax": "^1.2.4", "sax": "*",
"should": "^13.2.1", "async": "*",
"sinon": "^7.2.5" "sinon": "*"
}, },
"scripts": { "scripts": {
"test": "node test-runner.js", "test": "node test-runner.js",
"prepublish": "npm prune" "prepublish": "npm prune"
}, }
"dependencies": {}
} }

View file

@ -63,7 +63,7 @@ describe('Atoms and atom names cache', function() {
}); });
}); });
xit('should be used after the first request for non-std atom_names', function(done) { it('should be used after the first request for non-std atom_names', function(done) {
var self = this; var self = this;
var my_name; var my_name;
/* /*
@ -94,7 +94,7 @@ describe('Atoms and atom names cache', function() {
function(err) { function(err) {
should.not.exist(err); should.not.exist(err);
should.exist(my_name); should.exist(my_name);
self.spy.resetHistory(); self.spy.reset();
self.X.InternAtom(true, my_name, function(err, atom) { self.X.InternAtom(true, my_name, function(err, atom) {
should.not.exist(err); should.not.exist(err);
my_atom.should.equal(atom); my_atom.should.equal(atom);

View file

@ -1,34 +0,0 @@
var x11 = require('../lib');
var should = require('should');
describe('requiring an X11 extension on same connection', function() {
before(function(done) {
var self = this;
var client = x11.createClient(function(err, dpy) {
should.not.exist(err);
self.X = dpy.client;
done();
});
client.on('error', function (err) {
console.error('Error : ', err);
});
});
it('should be cached', function(done) {
var self = this;
this.X.require('xtest', function(err, randr) {
should.not.exist(err);
self.X.require('xtest', function(err, randr1) {
should.not.exist(err);
randr.should.equal(randr1);
done();
});
});
});
after(function(done) {
this.X.terminate();
this.X.on('end', done);
});
});

View file

@ -27,7 +27,7 @@ describe('ChangeProperty', function() {
var self = this; var self = this;
this.X.InternAtom(false, TEST_PROPERTY, function(err, atom) { this.X.InternAtom(false, TEST_PROPERTY, function(err, atom) {
should.not.exist(err); should.not.exist(err);
var raw = Buffer.alloc(4); var raw = new Buffer(4);
raw.writeUInt32LE(self.wid, 0); raw.writeUInt32LE(self.wid, 0);
self.X.ChangeProperty(0, self.wid, atom, self.X.atoms.WINDOW, 32, raw); self.X.ChangeProperty(0, self.wid, atom, self.X.atoms.WINDOW, 32, raw);
self.X.once('event', function(ev) { self.X.once('event', function(ev) {
@ -47,7 +47,7 @@ describe('ChangeProperty', function() {
var self = this; var self = this;
this.X.InternAtom(false, TEST_PROPERTY, function(err, atom) { this.X.InternAtom(false, TEST_PROPERTY, function(err, atom) {
should.not.exist(err); should.not.exist(err);
var raw = Buffer.from(new Array(8)); var raw = new Buffer(new Array(8));
raw.writeUInt32LE(self.wid, 0); raw.writeUInt32LE(self.wid, 0);
raw.writeUInt32LE(self.wid_helper, 4); raw.writeUInt32LE(self.wid_helper, 4);
self.X.ChangeProperty(0, self.wid, atom, self.X.atoms.ATOM, 32, raw); self.X.ChangeProperty(0, self.wid, atom, self.X.atoms.ATOM, 32, raw);
@ -69,7 +69,7 @@ describe('ChangeProperty', function() {
var self = this; var self = this;
this.X.InternAtom(false, TEST_PROPERTY, function(err, atom) { this.X.InternAtom(false, TEST_PROPERTY, function(err, atom) {
should.not.exist(err); should.not.exist(err);
var raw = Buffer.alloc(0); var raw = new Buffer(0);
self.X.ChangeProperty(0, self.wid, atom, self.X.atoms.WINDOW, 32, raw); self.X.ChangeProperty(0, self.wid, atom, self.X.atoms.WINDOW, 32, raw);
self.X.once('event', function(ev) { self.X.once('event', function(ev) {
ev.type.should.equal(28); ev.type.should.equal(28);

View file

@ -1,54 +0,0 @@
var x11 = require('../lib');
var should = require('should');
describe('CreateGC', function() {
before(function(done) {
var self = this;
this.client = x11.createClient(function(err, dpy) {
should.not.exist(err);
self.X = dpy.client;
self.root = dpy.screen[0].root;
self.white = dpy.screen[0].white_pixel;
self.black = dpy.screen[0].black_pixel;
self.wid = self.X.AllocID();
self.X.CreateWindow(self.wid, self.root, 0, 0, 1, 1); // 1x1 pixel window
self.X.MapWindow(self.wid);
self.X.QueryTree(self.root, function(err, list) {
should.not.exist(err);
list.children.indexOf(self.wid).should.not.equal(-1);
done();
});
});
});
it('should create a Graphic Context correctly', function() {
var self = this;
this.client.on('error', function(err) {
should.not.exist(err);
});
this.gc = this.X.AllocID();
this.X.CreateGC(this.gc,
this.wid,
{
foreground: this.black,
background: this.white,
lineStyle : 0
}
);
this.X.ChangeGC(this.gc,
{
foreground: 0xffff00,
background: 0x0000ff,
lineStyle : 2
}
);
});
after(function(done) {
this.X.DestroyWindow(this.wid);
this.X.on('end', done);
this.X.terminate();
});
});

View file

@ -1,114 +0,0 @@
var x11 = require('../lib');
var UnpackStream = require('../lib/unpackstream.js');
var should = require('should');
//Used Atoms
var ATOM = {};
describe('ClientMessage', function() {
before(function(done) {
var self = this;
var client = x11.createClient(function(err, dpy) {
should.not.exist(err);
self.X = dpy.client;
self.wid = self.X.AllocID();
self.X.CreateWindow(self.wid, dpy.screen[0].root, 0, 0, 1, 1); // 1x1 pixel window
self.X.InternAtom(false, 'TEST_ATOM_1', function(err, atom) {
should.not.exist(err);
ATOM['TEST_ATOM_1'] = atom;
done();
});
});
client.on('error', done);
});
it('should receive client message with format=8', function(done) {
var self = this;
var client = x11.createClient(function(err, dpy) {
should.not.exist(err);
self.X.once('event', function(ev) {
ev.name.should.equal('ClientMessage');
ev.wid.should.equal(self.wid);
ev.message_type.should.equal(ATOM.TEST_ATOM_1);
ev.data.should.be.an.Array();
ev.data.length.should.equal(20);
done();
});
var X = dpy.client;
var eventData = Buffer.alloc(32);
eventData.writeInt8(33, 0); //Event Type 33 = ClientMessage
eventData.writeInt8(8, 1); //Format
eventData.writeInt32LE(self.wid, 4); //Window ID
eventData.writeInt32LE(ATOM.TEST_ATOM_1, 8); //Message Type
X.SendEvent(self.wid, false, 0, eventData);
});
client.on('error', done);
});
it('should receive client message with format=16', function(done) {
var self = this;
var client = x11.createClient(function(err, dpy) {
should.not.exist(err);
self.X.once('event', function(ev) {
ev.name.should.equal('ClientMessage');
ev.wid.should.equal(self.wid);
ev.message_type.should.equal(ATOM.TEST_ATOM_1);
ev.data.should.be.an.Array();
ev.data.length.should.equal(10);
done();
});
var X = dpy.client;
var eventData = Buffer.alloc(32);
eventData.writeInt8(33, 0); //Event Type 33 = ClientMessage
eventData.writeInt8(16, 1); //Format
eventData.writeInt32LE(self.wid, 4); //Window ID
eventData.writeInt32LE(ATOM.TEST_ATOM_1, 8); //Message Type
X.SendEvent(self.wid, false, 0, eventData);
});
client.on('error', done);
});
it('should receive client message with format=32', function(done) {
var self = this;
var client = x11.createClient(function(err, dpy) {
should.not.exist(err);
self.X.once('event', function(ev) {
ev.name.should.equal('ClientMessage');
ev.wid.should.equal(self.wid);
ev.message_type.should.equal(ATOM.TEST_ATOM_1);
ev.data.should.be.an.Array();
ev.data.length.should.equal(5);
done();
});
var X = dpy.client;
var eventData = Buffer.alloc(32);
eventData.writeInt8(33, 0); //Event Type 33 = ClientMessage
eventData.writeInt8(32, 1); //Format
eventData.writeInt32LE(self.wid, 4); //Window ID
eventData.writeInt32LE(ATOM.TEST_ATOM_1, 8); //Message Type
X.SendEvent(self.wid, false, 0, eventData);
});
client.on('error', done);
});
after(function(done) {
this.X.DestroyWindow(this.wid);
this.X.on('end', done);
this.X.terminate();
});
});

View file

@ -8,41 +8,43 @@ describe('KillKlient request', function() {
var X; var X;
beforeEach(function(done) { beforeEach(function(done) {
var client = x11.createClient(function(err, dpy) { var client = x11.createClient(function(err, dpy) {
should.not.exist(err); if (!err) {
display = dpy; display = dpy;
X = display.client; X = display.client;
var root = display.screen[0].root; }
var eventMask = x11.eventMask.SubstructureNotify;
X.ChangeWindowAttributes(root, { eventMask: eventMask });
done();
});
client.on('error', done); done(err);
});
client.on('error', function(err) {
done(err);
});
}); });
afterEach(function(done) { afterEach(function(done) {
X.on('end', done);
X.terminate(); X.terminate();
X.on('end', done);
X = null;
display = null;
}); });
it('should exist as client member', function() { it('should exist as client member', function(done) {
should.exist(X.KillKlient); should.exist(X.KillKlient);
assert.equal(typeof X.KillKlient, 'function'); assert.equal(typeof X.KillKlient, 'function');
done();
}); });
it('should terminate other client connection', function(done) { it('should terminate other client connection', function(done) {
x11.createClient(function(err, dpy) { x11.createClient(function(err, dpy) {
should.not.exist(err); if (!err) {
var otherclient = dpy.client; var otherclient = dpy.client;
var wnd = otherclient.AllocID(); var wnd = otherclient.AllocID();
X.once('event', function(ev) {
ev.name.should.equal('CreateNotify');
ev.wid.should.equal(wnd);
X.KillKlient(wnd);
});
otherclient.CreateWindow(wnd, dpy.screen[0].root, 0, 0, 1, 1); otherclient.CreateWindow(wnd, dpy.screen[0].root, 0, 0, 1, 1);
otherclient.on('end', done); otherclient.on('end', done);
X.KillKlient(wnd);
} else {
done(err);
}
}); });
}); });
}); });

View file

@ -1,5 +1,4 @@
var x11 = require('../lib'); var x11 = require('../lib');
var async = require('async');
var should = require('should'); var should = require('should');
var assert = require('assert'); var assert = require('assert');
var util = require('util'); var util = require('util');
@ -36,26 +35,6 @@ describe('RANDR extension', function() {
}); });
}); });
it('GetScreenResources && GetOutputInfo', function(done) {
var self = this;
this.randr.GetScreenResources(this.root, function(err, resources) {
should.not.exist(err);
should.exist(resources);
async.each(
resources.outputs,
function(output, cb) {
self.randr.GetOutputInfo(output, 0, function(err, info) {
should.not.exist(err);
should.exist(info);
cb();
});
},
done
);
});
});
after(function(done) { after(function(done) {
this.X.terminate(); this.X.terminate();
this.X.on('end', done); this.X.on('end', done);