2019-04-05 17:00:01 +00:00
|
|
|
describe('Core canvas behavior', function() {
|
|
|
|
let container = document.createElement('div');
|
|
|
|
document.body.appendChild(container);
|
|
|
|
const CANVAS_WIDTH = 600;
|
|
|
|
const CANVAS_HEIGHT = 600;
|
|
|
|
|
|
|
|
beforeEach(function() {
|
|
|
|
container.innerHTML = `
|
|
|
|
<canvas width=600 height=600 id=test></canvas>
|
|
|
|
<canvas width=600 height=600 id=report></canvas>`;
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(function() {
|
|
|
|
container.innerHTML = '';
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can draw an SkPicture', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
// This is taken from example.html
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
2019-09-16 17:23:26 +00:00
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
2019-04-05 17:00:01 +00:00
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const spr = new CanvasKit.SkPictureRecorder();
|
|
|
|
const rcanvas = spr.beginRecording(
|
|
|
|
CanvasKit.LTRBRect(0, 0, surface.width(), surface.height()));
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
paint.setStrokeWidth(2.0);
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setColor(CanvasKit.Color(0, 0, 0, 1.0));
|
|
|
|
paint.setStyle(CanvasKit.PaintStyle.Stroke);
|
|
|
|
|
|
|
|
rcanvas.drawRoundRect(CanvasKit.LTRBRect(5, 35, 45, 80), 15, 10, paint);
|
|
|
|
|
|
|
|
const font = new CanvasKit.SkFont(null, 20);
|
|
|
|
rcanvas.drawText('this picture has a round rect', 5, 100, paint, font);
|
|
|
|
const pic = spr.finishRecordingAsPicture();
|
|
|
|
spr.delete();
|
|
|
|
|
|
|
|
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
canvas.drawPicture(pic);
|
|
|
|
|
|
|
|
reportSurface(surface, 'picture_test', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2020-03-26 13:27:48 +00:00
|
|
|
function uIntColorToCanvasKitColor(c) {
|
|
|
|
return CanvasKit.Color(
|
|
|
|
(c >> 16) & 0xFF,
|
|
|
|
(c >> 8) & 0xFF,
|
|
|
|
(c >> 0) & 0xFF,
|
|
|
|
((c >> 24) & 0xFF) / 255
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-09-11 19:23:00 +00:00
|
|
|
it('can compute tonal colors', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const input = {
|
|
|
|
ambient: CanvasKit.BLUE,
|
|
|
|
spot: CanvasKit.RED,
|
|
|
|
};
|
|
|
|
const out = CanvasKit.computeTonalColors(input);
|
2020-03-26 13:27:48 +00:00
|
|
|
expect(new Float32Array(out.ambient)).toEqual(CanvasKit.BLACK);
|
|
|
|
const expectedSpot = [0.173, 0, 0, 0.969];
|
|
|
|
expect(out.spot[0]).toBeCloseTo(expectedSpot[0], 3);
|
|
|
|
expect(out.spot[1]).toBeCloseTo(expectedSpot[1], 3);
|
|
|
|
expect(out.spot[2]).toBeCloseTo(expectedSpot[2], 3);
|
|
|
|
expect(out.spot[3]).toBeCloseTo(expectedSpot[3], 3);
|
2019-09-11 19:23:00 +00:00
|
|
|
done();
|
|
|
|
}));
|
2019-09-12 15:11:25 +00:00
|
|
|
});
|
2019-09-11 19:23:00 +00:00
|
|
|
|
2020-01-08 13:02:37 +00:00
|
|
|
// This helper is used for all the MakeImageFromEncoded tests.
|
|
|
|
function decodeAndDrawSingleFrameImage(imgName, goldName, done) {
|
|
|
|
const imgPromise = fetch(imgName)
|
2019-09-18 20:18:17 +00:00
|
|
|
.then((response) => response.arrayBuffer());
|
|
|
|
Promise.all([imgPromise, LoadCanvasKit]).then((values) => {
|
2020-01-08 13:02:37 +00:00
|
|
|
const imgData = values[0];
|
|
|
|
expect(imgData).toBeTruthy();
|
2019-09-18 20:18:17 +00:00
|
|
|
catchException(done, () => {
|
2020-01-08 13:02:37 +00:00
|
|
|
let img = CanvasKit.MakeImageFromEncoded(imgData);
|
2019-09-18 20:18:17 +00:00
|
|
|
expect(img).toBeTruthy();
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
2019-12-02 13:26:48 +00:00
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
2019-09-18 20:18:17 +00:00
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
let paint = new CanvasKit.SkPaint();
|
|
|
|
canvas.drawImage(img, 0, 0, paint);
|
|
|
|
|
|
|
|
paint.delete();
|
|
|
|
img.delete();
|
|
|
|
|
2020-01-08 13:02:37 +00:00
|
|
|
reportSurface(surface, goldName, done);
|
2019-09-18 20:18:17 +00:00
|
|
|
})();
|
|
|
|
});
|
2020-01-08 13:02:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
it('can decode and draw a png', function(done) {
|
|
|
|
decodeAndDrawSingleFrameImage('/assets/mandrill_512.png', 'drawImage_png', done);
|
2019-09-18 20:18:17 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('can decode and draw a jpg', function(done) {
|
2020-01-08 13:02:37 +00:00
|
|
|
decodeAndDrawSingleFrameImage('/assets/mandrill_h1v1.jpg', 'drawImage_jpg', done);
|
2019-09-18 20:18:17 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('can decode and draw a (still) gif', function(done) {
|
2020-01-08 13:02:37 +00:00
|
|
|
decodeAndDrawSingleFrameImage('/assets/flightAnim.gif', 'drawImage_gif', done);
|
|
|
|
});
|
2019-09-18 20:18:17 +00:00
|
|
|
|
2020-01-08 13:02:37 +00:00
|
|
|
it('can decode and draw a still webp', function(done) {
|
|
|
|
decodeAndDrawSingleFrameImage('/assets/color_wheel.webp', 'drawImage_webp', done);
|
2019-09-18 20:18:17 +00:00
|
|
|
});
|
|
|
|
|
2020-01-22 20:29:14 +00:00
|
|
|
it('can readPixels from an SkImage', function(done) {
|
|
|
|
const imgPromise = fetch('/assets/mandrill_512.png')
|
|
|
|
.then((response) => response.arrayBuffer());
|
|
|
|
Promise.all([imgPromise, LoadCanvasKit]).then((values) => {
|
|
|
|
const imgData = values[0];
|
|
|
|
expect(imgData).toBeTruthy();
|
|
|
|
catchException(done, () => {
|
|
|
|
let img = CanvasKit.MakeImageFromEncoded(imgData);
|
|
|
|
expect(img).toBeTruthy();
|
|
|
|
const imageInfo = {
|
|
|
|
alphaType: CanvasKit.AlphaType.Unpremul,
|
|
|
|
colorType: CanvasKit.ColorType.RGBA_8888,
|
|
|
|
width: img.width(),
|
|
|
|
height: img.height(),
|
|
|
|
};
|
|
|
|
|
|
|
|
const pixels = img.readPixels(imageInfo, 0, 0);
|
|
|
|
// We know the image is 512 by 512 pixels in size, each pixel
|
|
|
|
// requires 4 bytes (R, G, B, A).
|
|
|
|
expect(pixels.length).toEqual(512 * 512 * 4);
|
|
|
|
|
|
|
|
img.delete();
|
|
|
|
done();
|
|
|
|
})();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2019-09-18 20:18:17 +00:00
|
|
|
it('can decode and draw an animated gif', function(done) {
|
|
|
|
const imgPromise = fetch('/assets/flightAnim.gif')
|
|
|
|
.then((response) => response.arrayBuffer());
|
|
|
|
Promise.all([imgPromise, LoadCanvasKit]).then((values) => {
|
|
|
|
const gifData = values[0];
|
|
|
|
expect(gifData).toBeTruthy();
|
|
|
|
catchException(done, () => {
|
|
|
|
let aImg = CanvasKit.MakeAnimatedImageFromEncoded(gifData);
|
|
|
|
expect(aImg).toBeTruthy();
|
|
|
|
expect(aImg.getRepetitionCount()).toEqual(-1); // infinite loop
|
2019-11-08 14:55:15 +00:00
|
|
|
expect(aImg.width()).toEqual(320);
|
|
|
|
expect(aImg.height()).toEqual(240);
|
|
|
|
expect(aImg.getFrameCount()).toEqual(60);
|
2019-09-18 20:18:17 +00:00
|
|
|
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
2019-12-02 13:26:48 +00:00
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
2019-09-18 20:18:17 +00:00
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
canvas.drawAnimatedImage(aImg, 0, 0);
|
|
|
|
|
|
|
|
let c = aImg.decodeNextFrame();
|
|
|
|
expect(c).not.toEqual(-1);
|
|
|
|
canvas.drawAnimatedImage(aImg, 300, 0);
|
|
|
|
for(let i = 0; i < 10; i++) {
|
|
|
|
c = aImg.decodeNextFrame();
|
|
|
|
expect(c).not.toEqual(-1);
|
|
|
|
}
|
|
|
|
canvas.drawAnimatedImage(aImg, 0, 300);
|
|
|
|
for(let i = 0; i < 10; i++) {
|
|
|
|
c = aImg.decodeNextFrame();
|
|
|
|
expect(c).not.toEqual(-1);
|
|
|
|
}
|
2019-12-02 13:26:48 +00:00
|
|
|
canvas.drawAnimatedImage(aImg, 300, 300);
|
2019-09-18 20:18:17 +00:00
|
|
|
|
|
|
|
aImg.delete();
|
|
|
|
|
|
|
|
reportSurface(surface, 'drawDrawable_animated_gif', done);
|
|
|
|
})();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2019-12-02 13:26:48 +00:00
|
|
|
it('can create an image "from scratch" by specifying pixels/colorInfo manually', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
canvas.clear(CanvasKit.WHITE);
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
|
|
|
|
// This creates and draws an SkImage that is 1 pixel wide, 4 pixels tall with
|
|
|
|
// the colors listed below.
|
|
|
|
const pixels = Uint8Array.from([
|
|
|
|
255, 0, 0, 255, // opaque red
|
|
|
|
0, 255, 0, 255, // opaque green
|
|
|
|
0, 0, 255, 255, // opaque blue
|
|
|
|
255, 0, 255, 100, // transparent purple
|
|
|
|
]);
|
|
|
|
const img = CanvasKit.MakeImage(pixels, 1, 4, CanvasKit.AlphaType.Unpremul, CanvasKit.ColorType.RGBA_8888);
|
|
|
|
canvas.drawImage(img, 1, 1, paint);
|
|
|
|
|
|
|
|
reportSurface(surface, '1x4_from_scratch', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2020-03-16 16:17:30 +00:00
|
|
|
it('can draw a sweep gradient', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
const shader = CanvasKit.SkShader.MakeSweepGradient(
|
2020-04-01 17:42:15 +00:00
|
|
|
100, 100, // X, Y coordinates
|
2020-03-26 13:27:48 +00:00
|
|
|
[CanvasKit.GREEN, CanvasKit.BLUE],
|
2020-03-16 16:17:30 +00:00
|
|
|
[0.0, 1.0],
|
|
|
|
CanvasKit.TileMode.Clamp,
|
|
|
|
);
|
|
|
|
expect(shader).toBeTruthy('Could not make shader');
|
|
|
|
if (!shader) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
paint.setShader(shader);
|
|
|
|
canvas.drawPaint(paint);
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
paint.delete();
|
|
|
|
shader.delete();
|
|
|
|
|
|
|
|
reportSurface(surface, 'sweep_gradient', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2020-03-18 18:16:12 +00:00
|
|
|
// Inspired by https://fiddle.skia.org/c/b29ce50a341510784ac7d5281586d076
|
|
|
|
it('draws linear gradients', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface')
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
canvas.clear(CanvasKit.WHITE);
|
|
|
|
canvas.scale(2, 2);
|
|
|
|
const strokePaint = new CanvasKit.SkPaint();
|
|
|
|
strokePaint.setStyle(CanvasKit.PaintStyle.Stroke);
|
|
|
|
strokePaint.setColor(CanvasKit.BLACK);
|
|
|
|
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
paint.setStyle(CanvasKit.PaintStyle.Fill);
|
|
|
|
const transparentGreen = CanvasKit.Color(0, 255, 255, 0);
|
|
|
|
|
|
|
|
const lgs = CanvasKit.SkShader.MakeLinearGradient(
|
|
|
|
[0, 0], [50, 100], // start and stop points
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror
|
|
|
|
);
|
|
|
|
paint.setShader(lgs);
|
|
|
|
let r = CanvasKit.LTRBRect(0, 0, 100, 100);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
const lgsPremul = CanvasKit.SkShader.MakeLinearGradient(
|
|
|
|
[100, 0], [150, 100], // start and stop points
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror,
|
|
|
|
null, // no local matrix
|
|
|
|
1 // interpolate colors in premul
|
|
|
|
);
|
|
|
|
paint.setShader(lgsPremul);
|
|
|
|
r = CanvasKit.LTRBRect(100, 0, 200, 100);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
const lgs45 = CanvasKit.SkShader.MakeLinearGradient(
|
|
|
|
[0, 100], [50, 200], // start and stop points
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror,
|
|
|
|
CanvasKit.SkMatrix.rotated(Math.PI/4, 0, 100),
|
|
|
|
);
|
|
|
|
paint.setShader(lgs45);
|
|
|
|
r = CanvasKit.LTRBRect(0, 100, 100, 200);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
const lgs45Premul = CanvasKit.SkShader.MakeLinearGradient(
|
|
|
|
[100, 100], [150, 200], // start and stop points
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror,
|
|
|
|
CanvasKit.SkMatrix.rotated(Math.PI/4, 100, 100),
|
|
|
|
1 // interpolate colors in premul
|
|
|
|
);
|
|
|
|
paint.setShader(lgs45Premul);
|
|
|
|
r = CanvasKit.LTRBRect(100, 100, 200, 200);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
lgs.delete();
|
|
|
|
lgs45.delete();
|
|
|
|
lgsPremul.delete();
|
|
|
|
lgs45Premul.delete();
|
|
|
|
strokePaint.delete();
|
|
|
|
paint.delete();
|
|
|
|
reportSurface(surface, 'linear_gradients', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('draws radial gradients', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface')
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
canvas.clear(CanvasKit.WHITE);
|
|
|
|
canvas.scale(2, 2);
|
|
|
|
const strokePaint = new CanvasKit.SkPaint();
|
|
|
|
strokePaint.setStyle(CanvasKit.PaintStyle.Stroke);
|
|
|
|
strokePaint.setColor(CanvasKit.BLACK);
|
|
|
|
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
paint.setStyle(CanvasKit.PaintStyle.Fill);
|
|
|
|
const transparentGreen = CanvasKit.Color(0, 255, 255, 0);
|
|
|
|
|
|
|
|
const rgs = CanvasKit.SkShader.MakeRadialGradient(
|
|
|
|
[50, 50], 50, // center, radius
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror
|
|
|
|
);
|
|
|
|
paint.setShader(rgs);
|
|
|
|
let r = CanvasKit.LTRBRect(0, 0, 100, 100);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
const rgsPremul = CanvasKit.SkShader.MakeRadialGradient(
|
|
|
|
[150, 50], 50, // center, radius
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror,
|
|
|
|
null, // no local matrix
|
|
|
|
1 // interpolate colors in premul
|
|
|
|
);
|
|
|
|
paint.setShader(rgsPremul);
|
|
|
|
r = CanvasKit.LTRBRect(100, 0, 200, 100);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
const rgsSkew = CanvasKit.SkShader.MakeRadialGradient(
|
|
|
|
[50, 150], 50, // center, radius
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror,
|
|
|
|
CanvasKit.SkMatrix.skewed(0.5, 0, 100, 100)
|
|
|
|
);
|
|
|
|
paint.setShader(rgsSkew);
|
|
|
|
r = CanvasKit.LTRBRect(0, 100, 100, 200);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
const rgsSkewPremul = CanvasKit.SkShader.MakeRadialGradient(
|
|
|
|
[150, 150], 50, // center, radius
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror,
|
|
|
|
CanvasKit.SkMatrix.skewed(0.5, 0, 100, 100),
|
|
|
|
1 // interpolate colors in premul
|
|
|
|
);
|
|
|
|
paint.setShader(rgsSkewPremul);
|
|
|
|
r = CanvasKit.LTRBRect(100, 100, 200, 200);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
rgs.delete();
|
|
|
|
rgsPremul.delete();
|
|
|
|
rgsSkew.delete();
|
|
|
|
rgsSkewPremul.delete();
|
|
|
|
strokePaint.delete();
|
|
|
|
paint.delete();
|
|
|
|
reportSurface(surface, 'radial_gradients', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('draws two-point conical gradients', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface')
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
canvas.clear(CanvasKit.WHITE);
|
|
|
|
canvas.scale(2, 2);
|
|
|
|
const strokePaint = new CanvasKit.SkPaint();
|
|
|
|
strokePaint.setStyle(CanvasKit.PaintStyle.Stroke);
|
|
|
|
strokePaint.setColor(CanvasKit.BLACK);
|
|
|
|
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
paint.setStyle(CanvasKit.PaintStyle.Fill);
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
const transparentGreen = CanvasKit.Color(0, 255, 255, 0);
|
|
|
|
|
|
|
|
const cgs = CanvasKit.SkShader.MakeTwoPointConicalGradient(
|
|
|
|
[80, 10], 15, // start, radius
|
|
|
|
[10, 110], 60, // end, radius
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror
|
|
|
|
);
|
|
|
|
paint.setShader(cgs);
|
|
|
|
let r = CanvasKit.LTRBRect(0, 0, 100, 100);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
const cgsPremul = CanvasKit.SkShader.MakeTwoPointConicalGradient(
|
|
|
|
[180, 10], 15, // start, radius
|
|
|
|
[110, 110], 60, // end, radius
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror,
|
|
|
|
null, // no local matrix
|
|
|
|
1, // interpolate colors in premul
|
|
|
|
);
|
|
|
|
paint.setShader(cgsPremul);
|
|
|
|
r = CanvasKit.LTRBRect(100, 0, 200, 100);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
const cgs45 = CanvasKit.SkShader.MakeTwoPointConicalGradient(
|
|
|
|
[80, 110], 15, // start, radius
|
|
|
|
[10, 210], 60, // end, radius
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror,
|
|
|
|
CanvasKit.SkMatrix.rotated(Math.PI/4, 0, 100)
|
|
|
|
);
|
|
|
|
paint.setShader(cgs45);
|
|
|
|
r = CanvasKit.LTRBRect(0, 100, 100, 200);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
const cgs45Premul = CanvasKit.SkShader.MakeTwoPointConicalGradient(
|
|
|
|
[180, 110], 15, // start, radius
|
|
|
|
[110, 210], 60, // end, radius
|
|
|
|
[transparentGreen, CanvasKit.BLUE, CanvasKit.RED],
|
|
|
|
[0, 0.65, 1.0],
|
|
|
|
CanvasKit.TileMode.Mirror,
|
|
|
|
CanvasKit.SkMatrix.rotated(Math.PI/4, 100, 100),
|
|
|
|
1 // interpolate colors in premul
|
|
|
|
);
|
|
|
|
paint.setShader(cgs45Premul);
|
|
|
|
r = CanvasKit.LTRBRect(100, 100, 200, 200);
|
|
|
|
canvas.drawRect(r, paint);
|
|
|
|
canvas.drawRect(r, strokePaint);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
cgs.delete();
|
|
|
|
cgsPremul.delete();
|
|
|
|
cgs45.delete();
|
|
|
|
strokePaint.delete();
|
|
|
|
paint.delete();
|
|
|
|
reportSurface(surface, 'conical_gradients', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2019-10-29 13:55:39 +00:00
|
|
|
it('can blur using ImageFilter or MaskFilter', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
2019-12-02 13:26:48 +00:00
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
2019-10-29 13:55:39 +00:00
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
const pathUL = starPath(CanvasKit, 100, 100, 80);
|
|
|
|
const pathBR = starPath(CanvasKit, 400, 300, 80);
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
const textFont = new CanvasKit.SkFont(null, 24);
|
|
|
|
|
|
|
|
canvas.drawText('Above: MaskFilter', 20, 220, paint, textFont);
|
|
|
|
canvas.drawText('Right: ImageFilter', 20, 260, paint, textFont);
|
|
|
|
|
|
|
|
paint.setColor(CanvasKit.BLUE);
|
|
|
|
|
|
|
|
const blurMask = CanvasKit.SkMaskFilter.MakeBlur(CanvasKit.BlurStyle.Normal, 5, true);
|
|
|
|
paint.setMaskFilter(blurMask);
|
|
|
|
canvas.drawPath(pathUL, paint);
|
|
|
|
|
|
|
|
const blurIF = CanvasKit.SkImageFilter.MakeBlur(8, 1, CanvasKit.TileMode.Decal, null);
|
|
|
|
paint.setImageFilter(blurIF);
|
|
|
|
canvas.drawPath(pathBR, paint);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
pathUL.delete();
|
|
|
|
pathBR.delete();
|
|
|
|
paint.delete();
|
|
|
|
blurMask.delete();
|
|
|
|
blurIF.delete();
|
|
|
|
textFont.delete();
|
|
|
|
|
|
|
|
reportSurface(surface, 'blur_filters', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can use various image filters', function(done) {
|
|
|
|
const imgPromise = fetch('/assets/mandrill_512.png')
|
|
|
|
.then((response) => response.arrayBuffer());
|
|
|
|
Promise.all([imgPromise, LoadCanvasKit]).then((values) => {
|
|
|
|
const pngData = values[0];
|
|
|
|
expect(pngData).toBeTruthy();
|
|
|
|
catchException(done, () => {
|
|
|
|
let img = CanvasKit.MakeImageFromEncoded(pngData);
|
|
|
|
expect(img).toBeTruthy();
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
2019-12-02 13:26:48 +00:00
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
2019-10-29 13:55:39 +00:00
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
canvas.clear(CanvasKit.WHITE);
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setColor(CanvasKit.Color(0, 255, 0, 1.0));
|
|
|
|
const redCF = CanvasKit.SkColorFilter.MakeBlend(
|
|
|
|
CanvasKit.Color(255, 0, 0, 0.1), CanvasKit.BlendMode.SrcOver);
|
|
|
|
const redIF = CanvasKit.SkImageFilter.MakeColorFilter(redCF, null);
|
|
|
|
const blurIF = CanvasKit.SkImageFilter.MakeBlur(8, 0.2, CanvasKit.TileMode.Decal, null);
|
|
|
|
const combined = CanvasKit.SkImageFilter.MakeCompose(redIF, blurIF);
|
|
|
|
|
2019-11-15 19:13:20 +00:00
|
|
|
// rotate 10 degrees centered on 200, 200
|
|
|
|
const m = CanvasKit.SkMatrix.rotated(Math.PI/18, 200, 200);
|
|
|
|
const rotated = CanvasKit.SkImageFilter.MakeMatrixTransform(m, CanvasKit.FilterQuality.Medium, combined);
|
|
|
|
paint.setImageFilter(rotated);
|
2019-10-29 13:55:39 +00:00
|
|
|
|
2019-11-15 19:13:20 +00:00
|
|
|
//canvas.rotate(10, 200, 200);
|
2019-10-29 13:55:39 +00:00
|
|
|
canvas.drawImage(img, 0, 0, paint);
|
|
|
|
canvas.drawRect(CanvasKit.LTRBRect(5, 35, 45, 80), paint);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
paint.delete();
|
|
|
|
redIF.delete();
|
|
|
|
redCF.delete();
|
|
|
|
blurIF.delete();
|
|
|
|
combined.delete();
|
2019-11-15 19:13:20 +00:00
|
|
|
rotated.delete();
|
2019-11-11 15:06:08 +00:00
|
|
|
img.delete();
|
2019-10-29 13:55:39 +00:00
|
|
|
|
|
|
|
reportSurface(surface, 'combined_filters', done);
|
|
|
|
})();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-01-08 18:29:31 +00:00
|
|
|
it('can use various image filters on animated images', function(done) {
|
|
|
|
const imgPromise = fetch('/assets/flightAnim.gif')
|
|
|
|
.then((response) => response.arrayBuffer());
|
|
|
|
Promise.all([imgPromise, LoadCanvasKit]).then((values) => {
|
|
|
|
const gifData = values[0];
|
|
|
|
expect(gifData).toBeTruthy();
|
|
|
|
catchException(done, () => {
|
|
|
|
let img = CanvasKit.MakeAnimatedImageFromEncoded(gifData);
|
|
|
|
expect(img).toBeTruthy();
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
img.decodeNextFrame();
|
|
|
|
img.decodeNextFrame();
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
canvas.clear(CanvasKit.WHITE);
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setColor(CanvasKit.Color(0, 255, 0, 1.0));
|
|
|
|
const redCF = CanvasKit.SkColorFilter.MakeBlend(
|
|
|
|
CanvasKit.Color(255, 0, 0, 0.1), CanvasKit.BlendMode.SrcOver);
|
|
|
|
const redIF = CanvasKit.SkImageFilter.MakeColorFilter(redCF, null);
|
|
|
|
const blurIF = CanvasKit.SkImageFilter.MakeBlur(8, 0.2, CanvasKit.TileMode.Decal, null);
|
|
|
|
const combined = CanvasKit.SkImageFilter.MakeCompose(redIF, blurIF);
|
|
|
|
paint.setImageFilter(combined);
|
|
|
|
|
|
|
|
const frame = img.getCurrentFrame();
|
|
|
|
canvas.drawImage(frame, 100, 50, paint);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
paint.delete();
|
|
|
|
redIF.delete();
|
|
|
|
redCF.delete();
|
|
|
|
blurIF.delete();
|
|
|
|
combined.delete();
|
|
|
|
frame.delete();
|
|
|
|
img.delete();
|
|
|
|
|
|
|
|
reportSurface(surface, 'animated_filters', done);
|
|
|
|
})();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-01-14 13:39:09 +00:00
|
|
|
it('can decode and draw an skp', function(done) {
|
|
|
|
const skpPromise = fetch('/assets/red_line.skp')
|
|
|
|
.then((response) => response.arrayBuffer());
|
|
|
|
Promise.all([skpPromise, LoadCanvasKit]).then((values) => {
|
|
|
|
const skpData = values[0];
|
|
|
|
expect(skpData).toBeTruthy();
|
|
|
|
catchException(done, () => {
|
|
|
|
let pic = CanvasKit.MakeSkPicture(skpData);
|
|
|
|
expect(pic).toBeTruthy();
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
canvas.clear(CanvasKit.TRANSPARENT);
|
|
|
|
canvas.drawPicture(pic);
|
|
|
|
|
|
|
|
reportSurface(surface, 'drawImage_skp', done);
|
|
|
|
})();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-01-22 21:49:41 +00:00
|
|
|
it('can draw once using drawOnce utility method', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
function drawFrame(canvas) {
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
paint.setStrokeWidth(1.0);
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setColor(CanvasKit.Color(0, 0, 0, 1.0));
|
|
|
|
paint.setStyle(CanvasKit.PaintStyle.Stroke);
|
|
|
|
const path = new CanvasKit.SkPath();
|
|
|
|
path.moveTo(20, 5);
|
|
|
|
path.lineTo(30, 20);
|
|
|
|
path.lineTo(40, 10);
|
|
|
|
canvas.drawPath(path, paint);
|
|
|
|
path.delete();
|
|
|
|
paint.delete();
|
2020-01-27 15:01:25 +00:00
|
|
|
// surface hasn't been flushed yet (nor should we call flush
|
|
|
|
// ourselves), so reportSurface would likely be blank if we
|
|
|
|
// were to call it.
|
|
|
|
done();
|
2020-01-22 21:49:41 +00:00
|
|
|
}
|
|
|
|
surface.drawOnce(drawFrame);
|
2020-01-27 15:01:25 +00:00
|
|
|
// Reminder: drawOnce is async. In this test, we are just making
|
|
|
|
// sure the drawOnce function is there and doesn't crash, so we can
|
|
|
|
// just call done() when the frame is rendered.
|
2020-01-22 21:49:41 +00:00
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2019-11-25 18:45:28 +00:00
|
|
|
it('can use DecodeCache APIs', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const initialLimit = CanvasKit.getDecodeCacheLimitBytes();
|
|
|
|
expect(initialLimit).toBeGreaterThan(1024 * 1024);
|
|
|
|
|
|
|
|
const newLimit = 42 * 1024 * 1024;
|
|
|
|
CanvasKit.setDecodeCacheLimitBytes(newLimit);
|
|
|
|
expect(CanvasKit.getDecodeCacheLimitBytes()).toEqual(newLimit);
|
|
|
|
|
|
|
|
// We cannot make any assumptions about this value,
|
|
|
|
// so we just make sure it doesn't crash.
|
|
|
|
CanvasKit.getDecodeCacheUsedBytes();
|
|
|
|
|
|
|
|
done();
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2019-12-09 14:04:26 +00:00
|
|
|
it('can combine shaders', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
|
|
|
|
const rShader = CanvasKit.SkShader.Color(CanvasKit.Color(255, 0, 0, 1.0));
|
|
|
|
const gShader = CanvasKit.SkShader.Color(CanvasKit.Color(0, 255, 0, 0.6));
|
|
|
|
const bShader = CanvasKit.SkShader.Color(CanvasKit.Color(0, 0, 255, 1.0));
|
|
|
|
|
|
|
|
const rgShader = CanvasKit.SkShader.Blend(CanvasKit.BlendMode.SrcOver, rShader, gShader);
|
|
|
|
|
|
|
|
const p = new CanvasKit.SkPaint();
|
|
|
|
p.setShader(rgShader);
|
|
|
|
canvas.drawPaint(p);
|
|
|
|
|
|
|
|
const gbShader = CanvasKit.SkShader.Lerp(0.5, gShader, bShader);
|
|
|
|
|
|
|
|
p.setShader(gbShader);
|
|
|
|
canvas.drawRect(CanvasKit.LTRBRect(5, 100, 300, 400), p);
|
|
|
|
|
|
|
|
|
|
|
|
reportSurface(surface, 'combined_shaders', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2019-12-17 21:05:02 +00:00
|
|
|
it('exports consts correctly', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
2020-03-26 13:27:48 +00:00
|
|
|
expect(CanvasKit.TRANSPARENT).toEqual(Float32Array.of(0, 0, 0, 0));
|
|
|
|
expect(CanvasKit.RED).toEqual(Float32Array.of(1, 0, 0, 1));
|
2019-12-17 21:05:02 +00:00
|
|
|
|
|
|
|
expect(CanvasKit.QUAD_VERB).toEqual(2);
|
|
|
|
expect(CanvasKit.CONIC_VERB).toEqual(3);
|
|
|
|
|
|
|
|
expect(CanvasKit.SaveLayerInitWithPrevious).toEqual(4);
|
|
|
|
expect(CanvasKit.SaveLayerF16ColorType).toEqual(16);
|
|
|
|
|
|
|
|
done();
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2020-03-26 13:27:48 +00:00
|
|
|
it('can set and get a 4f color on a paint', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
paint.setColor(CanvasKit.Color4f(3.3, 2.2, 1.1, 0.5));
|
|
|
|
expect(paint.getColor()).toEqual(Float32Array.of(3.3, 2.2, 1.1, 0.5));
|
|
|
|
|
|
|
|
done();
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2020-04-01 17:42:15 +00:00
|
|
|
describe('DOMMatrix support', function() {
|
|
|
|
it('can supply a DOMMatrix to make a shader', function(done) {
|
|
|
|
LoadCanvasKit.then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface');
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
const shader = CanvasKit.SkShader.MakeSweepGradient(
|
|
|
|
100, 100, // x y coordinates
|
|
|
|
[CanvasKit.GREEN, CanvasKit.BLUE],
|
|
|
|
[0.0, 1.0],
|
|
|
|
CanvasKit.TileMode.Clamp,
|
|
|
|
new DOMMatrix().translate(-10, 100),
|
|
|
|
);
|
|
|
|
expect(shader).toBeTruthy('Could not make shader');
|
|
|
|
if (!shader) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
paint.setShader(shader);
|
|
|
|
canvas.drawPaint(paint);
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
paint.delete();
|
|
|
|
shader.delete();
|
|
|
|
|
|
|
|
reportSurface(surface, 'sweep_gradient_dommatrix', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
})
|
|
|
|
|
2019-04-05 17:00:01 +00:00
|
|
|
});
|