2011-10-25 18:18:22 -07:00
|
|
|
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
|
2012-08-31 15:48:21 -07:00
|
|
|
/* Copyright 2012 Mozilla Foundation
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2013-02-02 16:49:19 -06:00
|
|
|
/* globals CanvasGraphics, ColorSpace, createScratchCanvas, DeviceRgbCS, error,
|
|
|
|
info, isArray, isPDFFunction, isStream, PDFFunction, TODO, Util,
|
2013-05-30 19:42:26 -05:00
|
|
|
warn, CachedCanvases */
|
2011-10-25 18:18:22 -07:00
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
2011-12-05 00:38:45 +02:00
|
|
|
var PatternType = {
|
|
|
|
AXIAL: 2,
|
|
|
|
RADIAL: 3
|
|
|
|
};
|
2011-12-04 22:22:39 +02:00
|
|
|
|
2011-12-09 00:18:43 +02:00
|
|
|
var Pattern = (function PatternClosure() {
|
2011-10-24 16:55:23 -07:00
|
|
|
// Constructor should define this.getPattern
|
2011-12-09 00:18:43 +02:00
|
|
|
function Pattern() {
|
2011-10-24 16:55:23 -07:00
|
|
|
error('should not call Pattern constructor');
|
|
|
|
}
|
|
|
|
|
2011-12-09 00:18:43 +02:00
|
|
|
Pattern.prototype = {
|
2011-10-24 16:55:23 -07:00
|
|
|
// Input: current Canvas context
|
|
|
|
// Output: the appropriate fillStyle or strokeStyle
|
2012-04-04 23:43:26 +03:00
|
|
|
getPattern: function Pattern_getPattern(ctx) {
|
2011-10-24 16:55:23 -07:00
|
|
|
error('Should not call Pattern.getStyle: ' + ctx);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-04-05 12:46:23 -07:00
|
|
|
Pattern.shadingFromIR = function Pattern_shadingFromIR(raw) {
|
2012-03-29 08:53:51 -07:00
|
|
|
return Shadings[raw[0]].fromIR(raw);
|
2011-11-10 21:09:05 +02:00
|
|
|
};
|
2011-10-24 16:55:23 -07:00
|
|
|
|
2012-04-05 15:33:00 -07:00
|
|
|
Pattern.parseShading = function Pattern_parseShading(shading, matrix, xref,
|
|
|
|
res) {
|
2011-10-24 16:55:23 -07:00
|
|
|
|
|
|
|
var dict = isStream(shading) ? shading.dict : shading;
|
|
|
|
var type = dict.get('ShadingType');
|
|
|
|
|
|
|
|
switch (type) {
|
2011-12-05 00:38:45 +02:00
|
|
|
case PatternType.AXIAL:
|
|
|
|
case PatternType.RADIAL:
|
2011-12-04 22:22:39 +02:00
|
|
|
// Both radial and axial shadings are handled by RadialAxial shading.
|
2012-03-29 08:53:51 -07:00
|
|
|
return new Shadings.RadialAxial(dict, matrix, xref, res);
|
2011-10-24 16:55:23 -07:00
|
|
|
default:
|
2012-08-01 10:59:21 -05:00
|
|
|
TODO('Unsupported shading type: ' + type);
|
2011-10-25 09:10:56 -07:00
|
|
|
return new Shadings.Dummy();
|
2011-10-24 16:55:23 -07:00
|
|
|
}
|
|
|
|
};
|
2011-12-09 00:18:43 +02:00
|
|
|
return Pattern;
|
2011-10-24 16:55:23 -07:00
|
|
|
})();
|
|
|
|
|
2011-10-25 09:10:56 -07:00
|
|
|
var Shadings = {};
|
2011-10-24 16:55:23 -07:00
|
|
|
|
2012-06-07 16:00:07 -07:00
|
|
|
// A small number to offset the first/last color stops so we can insert ones to
|
|
|
|
// support extend. Number.MIN_VALUE appears to be too small and breaks the
|
2012-09-10 13:23:07 -07:00
|
|
|
// extend. 1e-7 works in FF but chrome seems to use an even smaller sized number
|
|
|
|
// internally so we have to go bigger.
|
|
|
|
Shadings.SMALL_NUMBER = 1e-2;
|
2012-06-07 16:00:07 -07:00
|
|
|
|
2011-10-24 16:55:23 -07:00
|
|
|
// Radial and axial shading have very similar implementations
|
|
|
|
// If needed, the implementations can be broken into two classes
|
2011-12-09 00:18:43 +02:00
|
|
|
Shadings.RadialAxial = (function RadialAxialClosure() {
|
|
|
|
function RadialAxial(dict, matrix, xref, res, ctx) {
|
2011-10-24 16:55:23 -07:00
|
|
|
this.matrix = matrix;
|
|
|
|
this.coordsArr = dict.get('Coords');
|
|
|
|
this.shadingType = dict.get('ShadingType');
|
|
|
|
this.type = 'Pattern';
|
|
|
|
this.ctx = ctx;
|
|
|
|
var cs = dict.get('ColorSpace', 'CS');
|
|
|
|
cs = ColorSpace.parse(cs, xref, res);
|
|
|
|
this.cs = cs;
|
|
|
|
|
|
|
|
var t0 = 0.0, t1 = 1.0;
|
|
|
|
if (dict.has('Domain')) {
|
|
|
|
var domainArr = dict.get('Domain');
|
|
|
|
t0 = domainArr[0];
|
|
|
|
t1 = domainArr[1];
|
|
|
|
}
|
|
|
|
|
|
|
|
var extendStart = false, extendEnd = false;
|
|
|
|
if (dict.has('Extend')) {
|
|
|
|
var extendArr = dict.get('Extend');
|
|
|
|
extendStart = extendArr[0];
|
|
|
|
extendEnd = extendArr[1];
|
|
|
|
}
|
|
|
|
|
2012-09-10 13:23:07 -07:00
|
|
|
if (this.shadingType === PatternType.RADIAL &&
|
|
|
|
(!extendStart || !extendEnd)) {
|
|
|
|
// Radial gradient only currently works if either circle is fully within
|
|
|
|
// the other circle.
|
|
|
|
var x1 = this.coordsArr[0];
|
|
|
|
var y1 = this.coordsArr[1];
|
|
|
|
var r1 = this.coordsArr[2];
|
|
|
|
var x2 = this.coordsArr[3];
|
|
|
|
var y2 = this.coordsArr[4];
|
|
|
|
var r2 = this.coordsArr[5];
|
|
|
|
var distance = Math.sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2));
|
|
|
|
if (r1 <= r2 + distance &&
|
|
|
|
r2 <= r1 + distance) {
|
|
|
|
warn('Unsupported radial gradient.');
|
|
|
|
}
|
2011-10-24 16:55:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
this.extendStart = extendStart;
|
|
|
|
this.extendEnd = extendEnd;
|
|
|
|
|
|
|
|
var fnObj = dict.get('Function');
|
2012-11-24 15:13:13 -06:00
|
|
|
var fn;
|
|
|
|
if (isArray(fnObj)) {
|
|
|
|
var fnArray = [];
|
|
|
|
for (var j = 0, jj = fnObj.length; j < jj; j++) {
|
|
|
|
var obj = xref.fetchIfRef(fnObj[j]);
|
|
|
|
if (!isPDFFunction(obj)) {
|
|
|
|
error('Invalid function');
|
|
|
|
}
|
|
|
|
fnArray.push(PDFFunction.parse(xref, obj));
|
|
|
|
}
|
|
|
|
fn = function radialAxialColorFunction(arg) {
|
|
|
|
var out = [];
|
|
|
|
for (var i = 0, ii = fnArray.length; i < ii; i++) {
|
|
|
|
out.push(fnArray[i](arg)[0]);
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
if (!isPDFFunction(fnObj)) {
|
|
|
|
error('Invalid function');
|
|
|
|
}
|
|
|
|
fn = PDFFunction.parse(xref, fnObj);
|
|
|
|
}
|
2011-10-24 16:55:23 -07:00
|
|
|
|
|
|
|
// 10 samples seems good enough for now, but probably won't work
|
|
|
|
// if there are sharp color changes. Ideally, we would implement
|
|
|
|
// the spec faithfully and add lossless optimizations.
|
|
|
|
var diff = t1 - t0;
|
2012-06-07 16:00:07 -07:00
|
|
|
var step = diff / 10;
|
|
|
|
|
|
|
|
var colorStops = this.colorStops = [];
|
|
|
|
|
|
|
|
// Protect against bad domains so we don't end up in an infinte loop below.
|
|
|
|
if (t0 >= t1 || step <= 0) {
|
|
|
|
// Acrobat doesn't seem to handle these cases so we'll ignore for
|
|
|
|
// now.
|
|
|
|
info('Bad shading domain.');
|
|
|
|
return;
|
|
|
|
}
|
2011-10-24 16:55:23 -07:00
|
|
|
|
|
|
|
for (var i = t0; i <= t1; i += step) {
|
2012-11-28 19:32:27 -06:00
|
|
|
var rgbColor = cs.getRgb(fn([i]), 0);
|
|
|
|
var cssColor = Util.makeCssRgb(rgbColor);
|
2012-01-15 14:01:36 -06:00
|
|
|
colorStops.push([(i - t0) / diff, cssColor]);
|
2011-10-24 16:55:23 -07:00
|
|
|
}
|
|
|
|
|
2012-09-10 13:23:07 -07:00
|
|
|
var background = 'transparent';
|
|
|
|
if (dict.has('Background')) {
|
2012-11-28 19:32:27 -06:00
|
|
|
var rgbColor = cs.getRgb(dict.get('Background'), 0);
|
|
|
|
background = Util.makeCssRgb(rgbColor);
|
2012-09-10 13:23:07 -07:00
|
|
|
}
|
|
|
|
|
2012-06-07 16:00:07 -07:00
|
|
|
if (!extendStart) {
|
|
|
|
// Insert a color stop at the front and offset the first real color stop
|
|
|
|
// so it doesn't conflict with the one we insert.
|
2012-09-10 13:23:07 -07:00
|
|
|
colorStops.unshift([0, background]);
|
2012-06-07 16:00:07 -07:00
|
|
|
colorStops[1][0] += Shadings.SMALL_NUMBER;
|
|
|
|
}
|
|
|
|
if (!extendEnd) {
|
|
|
|
// Same idea as above in extendStart but for the end.
|
|
|
|
colorStops[colorStops.length - 1][0] -= Shadings.SMALL_NUMBER;
|
2012-09-10 13:23:07 -07:00
|
|
|
colorStops.push([1, background]);
|
2012-06-07 16:00:07 -07:00
|
|
|
}
|
|
|
|
|
2011-10-24 16:55:23 -07:00
|
|
|
this.colorStops = colorStops;
|
|
|
|
}
|
|
|
|
|
2012-04-05 12:46:23 -07:00
|
|
|
RadialAxial.fromIR = function RadialAxial_fromIR(raw) {
|
2011-10-24 16:55:23 -07:00
|
|
|
var type = raw[1];
|
|
|
|
var colorStops = raw[2];
|
|
|
|
var p0 = raw[3];
|
|
|
|
var p1 = raw[4];
|
|
|
|
var r0 = raw[5];
|
|
|
|
var r1 = raw[6];
|
2012-03-29 08:53:51 -07:00
|
|
|
return {
|
|
|
|
type: 'Pattern',
|
2012-05-01 20:48:07 +03:00
|
|
|
getPattern: function RadialAxial_getPattern(ctx) {
|
2012-03-29 08:53:51 -07:00
|
|
|
var grad;
|
|
|
|
if (type == PatternType.AXIAL)
|
|
|
|
grad = ctx.createLinearGradient(p0[0], p0[1], p1[0], p1[1]);
|
|
|
|
else if (type == PatternType.RADIAL)
|
|
|
|
grad = ctx.createRadialGradient(p0[0], p0[1], r0, p1[0], p1[1], r1);
|
|
|
|
|
|
|
|
for (var i = 0, ii = colorStops.length; i < ii; ++i) {
|
|
|
|
var c = colorStops[i];
|
|
|
|
grad.addColorStop(c[0], c[1]);
|
|
|
|
}
|
|
|
|
return grad;
|
|
|
|
}
|
|
|
|
};
|
2011-11-10 21:09:05 +02:00
|
|
|
};
|
2011-10-24 16:55:23 -07:00
|
|
|
|
2011-12-09 00:18:43 +02:00
|
|
|
RadialAxial.prototype = {
|
2012-04-04 23:43:26 +03:00
|
|
|
getIR: function RadialAxial_getIR() {
|
2011-10-24 16:55:23 -07:00
|
|
|
var coordsArr = this.coordsArr;
|
|
|
|
var type = this.shadingType;
|
2011-12-05 00:38:45 +02:00
|
|
|
if (type == PatternType.AXIAL) {
|
2011-10-24 16:55:23 -07:00
|
|
|
var p0 = [coordsArr[0], coordsArr[1]];
|
|
|
|
var p1 = [coordsArr[2], coordsArr[3]];
|
|
|
|
var r0 = null;
|
|
|
|
var r1 = null;
|
2011-12-05 00:38:45 +02:00
|
|
|
} else if (type == PatternType.RADIAL) {
|
2011-10-24 16:55:23 -07:00
|
|
|
var p0 = [coordsArr[0], coordsArr[1]];
|
|
|
|
var p1 = [coordsArr[3], coordsArr[4]];
|
|
|
|
var r0 = coordsArr[2];
|
|
|
|
var r1 = coordsArr[5];
|
|
|
|
} else {
|
|
|
|
error('getPattern type unknown: ' + type);
|
|
|
|
}
|
|
|
|
|
|
|
|
var matrix = this.matrix;
|
|
|
|
if (matrix) {
|
|
|
|
p0 = Util.applyTransform(p0, matrix);
|
|
|
|
p1 = Util.applyTransform(p1, matrix);
|
|
|
|
}
|
|
|
|
|
2011-10-25 09:10:56 -07:00
|
|
|
return ['RadialAxial', type, this.colorStops, p0, p1, r0, r1];
|
2011-10-24 16:55:23 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2011-12-09 00:18:43 +02:00
|
|
|
return RadialAxial;
|
2011-10-24 16:55:23 -07:00
|
|
|
})();
|
|
|
|
|
2011-12-09 00:18:43 +02:00
|
|
|
Shadings.Dummy = (function DummyClosure() {
|
|
|
|
function Dummy() {
|
2011-10-25 09:10:56 -07:00
|
|
|
this.type = 'Pattern';
|
|
|
|
}
|
|
|
|
|
2012-04-04 23:43:26 +03:00
|
|
|
Dummy.fromIR = function Dummy_fromIR() {
|
2012-08-01 10:59:21 -05:00
|
|
|
return {
|
|
|
|
type: 'Pattern',
|
|
|
|
getPattern: function Dummy_fromIR_getPattern() {
|
|
|
|
return 'hotpink';
|
|
|
|
}
|
|
|
|
};
|
2011-11-10 21:09:05 +02:00
|
|
|
};
|
2011-10-25 09:10:56 -07:00
|
|
|
|
2011-12-09 00:18:43 +02:00
|
|
|
Dummy.prototype = {
|
2012-04-04 23:43:26 +03:00
|
|
|
getIR: function Dummy_getIR() {
|
2011-10-25 09:10:56 -07:00
|
|
|
return ['Dummy'];
|
|
|
|
}
|
|
|
|
};
|
2011-12-09 00:18:43 +02:00
|
|
|
return Dummy;
|
2011-10-25 09:10:56 -07:00
|
|
|
})();
|
|
|
|
|
2011-12-09 00:18:43 +02:00
|
|
|
var TilingPattern = (function TilingPatternClosure() {
|
2011-12-06 22:07:35 +02:00
|
|
|
var PaintType = {
|
|
|
|
COLORED: 1,
|
|
|
|
UNCOLORED: 2
|
|
|
|
};
|
2013-04-25 23:21:05 +03:00
|
|
|
|
|
|
|
var MAX_PATTERN_SIZE = 8192;
|
2011-10-24 16:55:23 -07:00
|
|
|
|
2013-03-03 21:36:44 +09:00
|
|
|
function TilingPattern(IR, color, ctx, objs, commonObjs) {
|
2013-04-25 23:21:05 +03:00
|
|
|
this.name = IR[1][0].name;
|
|
|
|
this.operatorList = IR[2];
|
2012-12-27 09:35:25 +02:00
|
|
|
this.matrix = IR[3] || [1, 0, 0, 1, 0, 0];
|
2013-04-25 23:21:05 +03:00
|
|
|
this.bbox = IR[4];
|
|
|
|
this.xstep = IR[5];
|
|
|
|
this.ystep = IR[6];
|
|
|
|
this.paintType = IR[7];
|
|
|
|
this.tilingType = IR[8];
|
|
|
|
this.color = color;
|
|
|
|
this.objs = objs;
|
|
|
|
this.commonObjs = commonObjs;
|
2011-10-24 16:55:23 -07:00
|
|
|
this.curMatrix = ctx.mozCurrentTransform;
|
|
|
|
this.type = 'Pattern';
|
2013-04-25 23:21:05 +03:00
|
|
|
this.ctx = ctx;
|
2011-10-24 16:55:23 -07:00
|
|
|
}
|
|
|
|
|
2012-04-04 23:43:26 +03:00
|
|
|
TilingPattern.getIR = function TilingPattern_getIR(operatorList, dict, args) {
|
2011-10-24 16:55:23 -07:00
|
|
|
var matrix = dict.get('Matrix');
|
|
|
|
var bbox = dict.get('BBox');
|
|
|
|
var xstep = dict.get('XStep');
|
|
|
|
var ystep = dict.get('YStep');
|
|
|
|
var paintType = dict.get('PaintType');
|
2012-09-05 22:36:20 +03:00
|
|
|
var tilingType = dict.get('TilingType');
|
2011-10-24 16:55:23 -07:00
|
|
|
|
|
|
|
return [
|
2012-09-05 22:36:20 +03:00
|
|
|
'TilingPattern', args, operatorList, matrix, bbox, xstep, ystep,
|
|
|
|
paintType, tilingType
|
2011-10-24 16:55:23 -07:00
|
|
|
];
|
2011-11-10 21:09:05 +02:00
|
|
|
};
|
2011-10-24 16:55:23 -07:00
|
|
|
|
|
|
|
TilingPattern.prototype = {
|
2013-04-25 23:21:05 +03:00
|
|
|
createPatternCanvas: function TilinPattern_createPatternCanvas(tmpCanvas) {
|
|
|
|
var operatorList = this.operatorList;
|
|
|
|
var bbox = this.bbox;
|
|
|
|
var xstep = this.xstep;
|
|
|
|
var ystep = this.ystep;
|
|
|
|
var paintType = this.paintType;
|
|
|
|
var tilingType = this.tilingType;
|
|
|
|
var color = this.color;
|
|
|
|
var objs = this.objs;
|
|
|
|
var commonObjs = this.commonObjs;
|
|
|
|
var ctx = this.ctx;
|
|
|
|
|
|
|
|
TODO('TilingType: ' + tilingType);
|
|
|
|
|
|
|
|
var x0 = bbox[0], y0 = bbox[1], x1 = bbox[2], y1 = bbox[3];
|
|
|
|
|
|
|
|
var topLeft = [x0, y0];
|
|
|
|
// we want the canvas to be as large as the step size
|
|
|
|
var botRight = [x0 + xstep, y0 + ystep];
|
|
|
|
|
|
|
|
var width = botRight[0] - topLeft[0];
|
|
|
|
var height = botRight[1] - topLeft[1];
|
|
|
|
|
|
|
|
// Obtain scale from matrix and current transformation matrix.
|
|
|
|
var matrixScale = Util.singularValueDecompose2dScale(this.matrix);
|
|
|
|
var curMatrixScale = Util.singularValueDecompose2dScale(this.curMatrix);
|
|
|
|
var combinedScale = [matrixScale[0] * curMatrixScale[0],
|
|
|
|
matrixScale[1] * curMatrixScale[1]];
|
|
|
|
|
|
|
|
// MAX_PATTERN_SIZE is used to avoid OOM situation.
|
|
|
|
// Use width and height values that are as close as possible to the end
|
|
|
|
// result when the pattern is used. Too low value makes the pattern look
|
|
|
|
// blurry. Too large value makes it look too crispy.
|
|
|
|
width = Math.min(Math.ceil(Math.abs(width * combinedScale[0])),
|
|
|
|
MAX_PATTERN_SIZE);
|
|
|
|
|
|
|
|
height = Math.min(Math.ceil(Math.abs(height * combinedScale[1])),
|
|
|
|
MAX_PATTERN_SIZE);
|
|
|
|
|
|
|
|
tmpCanvas.width = width;
|
|
|
|
tmpCanvas.height = height;
|
|
|
|
|
|
|
|
// set the new canvas element context as the graphics context
|
|
|
|
var tmpCtx = tmpCanvas.getContext('2d');
|
|
|
|
var graphics = new CanvasGraphics(tmpCtx, commonObjs, objs);
|
|
|
|
|
|
|
|
this.setFillAndStrokeStyleToContext(tmpCtx, paintType, color);
|
|
|
|
|
|
|
|
this.setScale(width, height, xstep, ystep);
|
|
|
|
this.transformToScale(graphics);
|
|
|
|
|
|
|
|
// transform coordinates to pattern space
|
|
|
|
var tmpTranslate = [1, 0, 0, 1, -topLeft[0], -topLeft[1]];
|
|
|
|
graphics.transform.apply(graphics, tmpTranslate);
|
|
|
|
|
|
|
|
this.clipBbox(graphics, bbox, x0, y0, x1, y1);
|
|
|
|
|
|
|
|
graphics.executeOperatorList(operatorList);
|
|
|
|
},
|
|
|
|
|
2012-12-27 09:35:25 +02:00
|
|
|
setScale: function TilingPattern_setScale(width, height, xstep, ystep) {
|
|
|
|
this.scale = [width / xstep, height / ystep];
|
|
|
|
},
|
|
|
|
|
|
|
|
transformToScale: function TilingPattern_transformToScale(graphics) {
|
|
|
|
var scale = this.scale;
|
|
|
|
var tmpScale = [scale[0], 0, 0, scale[1], 0, 0];
|
|
|
|
graphics.transform.apply(graphics, tmpScale);
|
|
|
|
},
|
|
|
|
|
|
|
|
scaleToContext: function TilingPattern_scaleToContext() {
|
|
|
|
var scale = this.scale;
|
|
|
|
this.ctx.scale(1 / scale[0], 1 / scale[1]);
|
|
|
|
},
|
|
|
|
|
|
|
|
clipBbox: function clipBbox(graphics, bbox, x0, y0, x1, y1) {
|
|
|
|
if (bbox && isArray(bbox) && 4 == bbox.length) {
|
|
|
|
var bboxWidth = x1 - x0;
|
|
|
|
var bboxHeight = y1 - y0;
|
|
|
|
graphics.rectangle(x0, y0, bboxWidth, bboxHeight);
|
|
|
|
graphics.clip();
|
|
|
|
graphics.endPath();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
setFillAndStrokeStyleToContext:
|
|
|
|
function setFillAndStrokeStyleToContext(context, paintType, color) {
|
|
|
|
switch (paintType) {
|
|
|
|
case PaintType.COLORED:
|
|
|
|
var ctx = this.ctx;
|
|
|
|
context.fillStyle = ctx.fillStyle;
|
|
|
|
context.strokeStyle = ctx.strokeStyle;
|
|
|
|
break;
|
|
|
|
case PaintType.UNCOLORED:
|
|
|
|
var rgbColor = new DeviceRgbCS().getRgb(color, 0);
|
|
|
|
var cssColor = Util.makeCssRgb(rgbColor);
|
|
|
|
context.fillStyle = cssColor;
|
|
|
|
context.strokeStyle = cssColor;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
error('Unsupported paint type: ' + paintType);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-04-04 23:43:26 +03:00
|
|
|
getPattern: function TilingPattern_getPattern() {
|
2013-05-30 19:42:26 -05:00
|
|
|
var temporaryPatternCanvas = CachedCanvases.getCanvas('pattern');
|
2013-04-25 23:21:05 +03:00
|
|
|
this.createPatternCanvas(temporaryPatternCanvas);
|
2011-10-24 16:55:23 -07:00
|
|
|
|
2013-04-25 23:21:05 +03:00
|
|
|
var ctx = this.ctx;
|
|
|
|
ctx.setTransform.apply(ctx, this.curMatrix);
|
|
|
|
ctx.transform.apply(ctx, this.matrix);
|
2012-12-27 09:35:25 +02:00
|
|
|
this.scaleToContext();
|
2011-10-24 16:55:23 -07:00
|
|
|
|
2013-04-25 23:21:05 +03:00
|
|
|
return ctx.createPattern(temporaryPatternCanvas, 'repeat');
|
2011-10-24 16:55:23 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return TilingPattern;
|
|
|
|
})();
|
2011-10-27 21:51:10 +03:00
|
|
|
|