2020-01-15 18:49:53 +00:00
|
|
|
describe('CanvasKit\'s Font Behavior', function() {
|
2019-02-22 15:04:06 +00:00
|
|
|
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 = '';
|
|
|
|
});
|
|
|
|
|
2019-03-15 19:36:29 +00:00
|
|
|
let notSerifFontBuffer = null;
|
|
|
|
// This font is known to support kerning
|
|
|
|
const notoSerifFontLoaded = fetch('/assets/NotoSerif-Regular.ttf').then(
|
|
|
|
(response) => response.arrayBuffer()).then(
|
|
|
|
(buffer) => {
|
|
|
|
notSerifFontBuffer = buffer;
|
|
|
|
});
|
2019-02-22 15:04:06 +00:00
|
|
|
|
2019-09-26 17:25:26 +00:00
|
|
|
let bungeeFontBuffer = null;
|
|
|
|
// This font has tofu for incorrect null terminators
|
|
|
|
// see https://bugs.chromium.org/p/skia/issues/detail?id=9314
|
|
|
|
const bungeeFontLoaded = fetch('/assets/Bungee-Regular.ttf').then(
|
|
|
|
(response) => response.arrayBuffer()).then(
|
|
|
|
(buffer) => {
|
|
|
|
bungeeFontBuffer = buffer;
|
|
|
|
});
|
|
|
|
|
2019-03-15 19:36:29 +00:00
|
|
|
it('can draw shaped and unshaped text', function(done) {
|
2019-09-26 17:25:26 +00:00
|
|
|
Promise.all([LoadCanvasKit, notoSerifFontLoaded]).then(catchException(done, () => {
|
|
|
|
// This is taken from example.html
|
|
|
|
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();
|
|
|
|
|
|
|
|
paint.setColor(CanvasKit.BLUE);
|
|
|
|
paint.setStyle(CanvasKit.PaintStyle.Stroke);
|
|
|
|
|
|
|
|
const fontMgr = CanvasKit.SkFontMgr.RefDefault();
|
|
|
|
const notoSerif = fontMgr.MakeTypefaceFromData(notSerifFontBuffer);
|
|
|
|
|
|
|
|
const textPaint = new CanvasKit.SkPaint();
|
|
|
|
const textFont = new CanvasKit.SkFont(notoSerif, 20);
|
|
|
|
|
|
|
|
canvas.drawRect(CanvasKit.LTRBRect(30, 30, 200, 200), paint);
|
|
|
|
canvas.drawText('This text is not shaped, and overflows the boundary',
|
|
|
|
35, 50, textPaint, textFont);
|
|
|
|
|
|
|
|
const shapedText = new CanvasKit.ShapedText({
|
|
|
|
font: textFont,
|
|
|
|
leftToRight: true,
|
|
|
|
text: 'This text *is* shaped, and wraps to the right width.',
|
|
|
|
width: 160,
|
|
|
|
});
|
|
|
|
const textBoxX = 35;
|
|
|
|
const textBoxY = 55;
|
|
|
|
canvas.drawText(shapedText, textBoxX, textBoxY, textPaint);
|
|
|
|
const bounds = shapedText.getBounds();
|
|
|
|
|
|
|
|
bounds.fLeft += textBoxX;
|
|
|
|
bounds.fRight += textBoxX;
|
|
|
|
bounds.fTop += textBoxY;
|
|
|
|
bounds.fBottom += textBoxY
|
|
|
|
|
|
|
|
canvas.drawRect(bounds, paint);
|
|
|
|
const SHAPE_TEST_TEXT = 'VAVAVAVAVAFIfi';
|
|
|
|
const textFont2 = new CanvasKit.SkFont(notoSerif, 60);
|
|
|
|
const shapedText2 = new CanvasKit.ShapedText({
|
|
|
|
font: textFont2,
|
|
|
|
leftToRight: true,
|
|
|
|
text: SHAPE_TEST_TEXT,
|
|
|
|
width: 600,
|
2019-02-22 15:04:06 +00:00
|
|
|
});
|
2019-09-26 17:25:26 +00:00
|
|
|
|
|
|
|
canvas.drawText('no kerning ↓', 10, 240, textPaint, textFont);
|
|
|
|
canvas.drawText(SHAPE_TEST_TEXT, 10, 300, textPaint, textFont2);
|
|
|
|
canvas.drawText(shapedText2, 10, 300, textPaint);
|
|
|
|
canvas.drawText('kerning ↑', 10, 390, textPaint, textFont);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
paint.delete();
|
|
|
|
notoSerif.delete();
|
|
|
|
textPaint.delete();
|
|
|
|
textFont.delete();
|
|
|
|
shapedText.delete();
|
|
|
|
textFont2.delete();
|
|
|
|
shapedText2.delete();
|
|
|
|
fontMgr.delete();
|
|
|
|
reportSurface(surface, 'text_shaping', done);
|
2019-02-22 15:04:06 +00:00
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2019-03-15 19:36:29 +00:00
|
|
|
it('can draw text following a path', 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();
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setStyle(CanvasKit.PaintStyle.Stroke);
|
|
|
|
|
|
|
|
const font = new CanvasKit.SkFont(null, 24);
|
|
|
|
const fontPaint = new CanvasKit.SkPaint();
|
|
|
|
fontPaint.setAntiAlias(true);
|
|
|
|
fontPaint.setStyle(CanvasKit.PaintStyle.Fill);
|
|
|
|
|
|
|
|
|
|
|
|
const arc = new CanvasKit.SkPath();
|
|
|
|
arc.arcTo(CanvasKit.LTRBRect(20, 40, 280, 300), -160, 140, true);
|
|
|
|
arc.lineTo(210, 140);
|
|
|
|
arc.arcTo(CanvasKit.LTRBRect(20, 0, 280, 260), 160, -140, true);
|
|
|
|
|
|
|
|
// Only 1 dot should show up in the image, because we run out of path.
|
|
|
|
const str = 'This téxt should follow the curve across contours...';
|
|
|
|
const textBlob = CanvasKit.SkTextBlob.MakeOnPath(str, arc, font);
|
|
|
|
|
|
|
|
canvas.drawPath(arc, paint);
|
|
|
|
canvas.drawTextBlob(textBlob, 0, 0, fontPaint);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
textBlob.delete();
|
|
|
|
arc.delete();
|
|
|
|
paint.delete();
|
|
|
|
font.delete();
|
|
|
|
fontPaint.delete();
|
|
|
|
|
|
|
|
reportSurface(surface, 'monospace_text_on_path', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can draw text following a path with a non-serif font', function(done) {
|
2019-09-26 17:25:26 +00:00
|
|
|
Promise.all([LoadCanvasKit, notoSerifFontLoaded]).then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface')
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const fontMgr = CanvasKit.SkFontMgr.RefDefault();
|
|
|
|
const notoSerif = fontMgr.MakeTypefaceFromData(notSerifFontBuffer);
|
|
|
|
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
const paint = new CanvasKit.SkPaint();
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setStyle(CanvasKit.PaintStyle.Stroke);
|
|
|
|
|
|
|
|
const font = new CanvasKit.SkFont(notoSerif, 24);
|
|
|
|
const fontPaint = new CanvasKit.SkPaint();
|
|
|
|
fontPaint.setAntiAlias(true);
|
|
|
|
fontPaint.setStyle(CanvasKit.PaintStyle.Fill);
|
|
|
|
|
|
|
|
|
|
|
|
const arc = new CanvasKit.SkPath();
|
|
|
|
arc.arcTo(CanvasKit.LTRBRect(20, 40, 280, 300), -160, 140, true);
|
|
|
|
arc.lineTo(210, 140);
|
|
|
|
arc.arcTo(CanvasKit.LTRBRect(20, 0, 280, 260), 160, -140, true);
|
|
|
|
|
|
|
|
const str = 'This téxt should follow the curve across contours...';
|
|
|
|
const textBlob = CanvasKit.SkTextBlob.MakeOnPath(str, arc, font, 60.5);
|
|
|
|
|
|
|
|
canvas.drawPath(arc, paint);
|
|
|
|
canvas.drawTextBlob(textBlob, 0, 0, fontPaint);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
textBlob.delete();
|
|
|
|
arc.delete();
|
|
|
|
paint.delete();
|
|
|
|
notoSerif.delete();
|
|
|
|
font.delete();
|
|
|
|
fontPaint.delete();
|
|
|
|
fontMgr.delete();
|
|
|
|
reportSurface(surface, 'serif_text_on_path', done);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
|
|
|
// https://bugs.chromium.org/p/skia/issues/detail?id=9314
|
2019-09-26 20:12:41 +00:00
|
|
|
it('does not draw tofu for null terminators at end of text', function(done) {
|
2019-09-26 17:25:26 +00:00
|
|
|
Promise.all([LoadCanvasKit, bungeeFontLoaded]).then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface')
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const fontMgr = CanvasKit.SkFontMgr.RefDefault();
|
|
|
|
const bungee = fontMgr.MakeTypefaceFromData(bungeeFontBuffer);
|
|
|
|
|
|
|
|
const canvas = surface.getCanvas();
|
|
|
|
// yellow, to make sure tofu is plainly visible
|
|
|
|
canvas.clear(CanvasKit.Color(255, 255, 0, 1));
|
|
|
|
|
|
|
|
const font = new CanvasKit.SkFont(bungee, 24);
|
|
|
|
const fontPaint = new CanvasKit.SkPaint();
|
|
|
|
fontPaint.setAntiAlias(true);
|
|
|
|
fontPaint.setStyle(CanvasKit.PaintStyle.Fill);
|
|
|
|
|
|
|
|
|
|
|
|
const str = 'This is téxt';
|
|
|
|
const textBlob = CanvasKit.SkTextBlob.MakeFromText(str + ' text blob', font);
|
|
|
|
|
|
|
|
canvas.drawTextBlob(textBlob, 10, 50, fontPaint);
|
|
|
|
|
|
|
|
canvas.drawText(str + ' normal', 10, 100, fontPaint, font);
|
|
|
|
|
|
|
|
canvas.drawText('null terminator ->\u0000<- on purpose', 10, 150, fontPaint, font);
|
|
|
|
|
|
|
|
surface.flush();
|
|
|
|
|
|
|
|
textBlob.delete();
|
|
|
|
bungee.delete();
|
|
|
|
font.delete();
|
|
|
|
fontPaint.delete();
|
|
|
|
fontMgr.delete();
|
2019-09-26 20:12:41 +00:00
|
|
|
reportSurface(surface, 'nullterminators_skbug_9314', done);
|
2019-03-15 19:36:29 +00:00
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2019-09-26 17:20:50 +00:00
|
|
|
it('can make a font mgr with passed in fonts', function(done) {
|
|
|
|
Promise.all([LoadCanvasKit, bungeeFontLoaded, notoSerifFontLoaded]).then(catchException(done, () => {
|
|
|
|
const surface = CanvasKit.MakeCanvasSurface('test');
|
|
|
|
expect(surface).toBeTruthy('Could not make surface')
|
|
|
|
if (!surface) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// CanvasKit.SkFontMgr.FromData([bungeeFontBuffer, notSerifFontBuffer]) also works
|
|
|
|
const fontMgr = CanvasKit.SkFontMgr.FromData(bungeeFontBuffer, notSerifFontBuffer);
|
|
|
|
expect(fontMgr).toBeTruthy();
|
|
|
|
expect(fontMgr.countFamilies()).toBe(2);
|
|
|
|
// in debug mode, let's list them.
|
|
|
|
if (fontMgr.dumpFamilies) {
|
|
|
|
fontMgr.dumpFamilies();
|
|
|
|
}
|
|
|
|
fontMgr.delete();
|
|
|
|
done();
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
|
2019-02-22 15:04:06 +00:00
|
|
|
});
|