/*
* Copyright 2003-2006, 2009, 2017, 2020 United States Government, as represented
* by the Administrator of the National Aeronautics and Space Administration.
* All rights reserved.
*
* The NASAWorldWind/WebWorldWind platform is 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.
*
* NASAWorldWind/WebWorldWind also contains the following 3rd party Open Source
* software:
*
* ES6-Promise – under MIT License
* libtess.js – SGI Free Software License B
* Proj4 – under MIT License
* JSZip – under MIT License
*
* A complete listing of 3rd Party software notices and licenses included in
* WebWorldWind can be found in the WebWorldWind 3rd-party notices and licenses
* PDF found in code directory.
*/
/**
* @exports TextRenderer
*/
define([
'../error/ArgumentError',
'../shaders/BasicTextureProgram',
'../util/Color',
'../util/Font',
'../util/Logger',
'../geom/Matrix',
'../render/Texture',
'../geom/Vec2'
],
function (ArgumentError,
BasicTextureProgram,
Color,
Font,
Logger,
Matrix,
Texture,
Vec2) {
"use strict";
/**
* Constructs a TextRenderer instance.
* @alias TextRenderer
* @constructor
* @classdesc Provides methods useful for displaying text. An instance of this class is attached to the
* WorldWindow {@link DrawContext} and is not intended to be used independently of that. Applications typically do
* not create instances of this class.
* @param {drawContext} drawContext The current draw context. Typically the same draw context that TextRenderer
* is attached to.
* @throws {ArgumentError} If the specified draw context is null or undefined.
*/
var TextRenderer = function (drawContext) {
if (!drawContext) {
throw new ArgumentError(Logger.logMessage(Logger.LEVEL_SEVERE, "TextRenderer", "constructor",
"missingDc"));
}
// Internal use only. Intentionally not documented.
this.canvas2D = document.createElement("canvas");
// Internal use only. Intentionally not documented.
this.ctx2D = this.canvas2D.getContext("2d");
// Internal use only. Intentionally not documented.
this.dc = drawContext;
/**
* Indicates if the text will feature an outline around its characters.
* @type {boolean}
*/
this.enableOutline = true;
// Internal use only. Intentionally not documented.
this.lineSpacing = 0.15; // fraction of font size
/**
* The color for the Text outline.
* Its default has half transparency to avoid visual artifacts that appear while fully opaque.
* @type {Color}
*/
this.outlineColor = new Color(0, 0, 0, 0.5);
/**
* Indicates the text outline width (or thickness) in pixels.
* @type {number}
*/
this.outlineWidth = 4;
/**
* The text color.
* @type {Color}
*/
this.textColor = new Color(1, 1, 1, 1);
/**
* The text size, face and other characteristics, as described in [Font]{@link Font}.
* @type {Font}
*/
this.typeFace = new Font(14);
};
/**
* Returns the width and height of a specified text string considering the current typeFace and outline usage.
* @param {string} text The text string.
* @returns {Vec2} A vector indicating the text's width and height, respectively, in pixels.
*/
TextRenderer.prototype.textSize = function (text) {
if (text.length === 0) {
return new Vec2(0, 0);
}
this.ctx2D.font = this.typeFace.fontString;
var lines = text.split("\n"),
height = lines.length * (this.typeFace.size * (1 + this.lineSpacing)),
maxWidth = 0;
for (var i = 0; i < lines.length; i++) {
maxWidth = Math.max(maxWidth, this.ctx2D.measureText(lines[i]).width);
}
if (this.enableOutline) {
maxWidth += this.outlineWidth;
height += this.outlineWidth;
}
return new Vec2(maxWidth, height);
};
/**
* Creates a texture for a specified text string and current TextRenderer state.
* @param {String} text The text string.
* @returns {Texture} A texture for the specified text string.
*/
TextRenderer.prototype.renderText = function (text) {
if (text && text.length > 0) {
var canvas2D = this.drawText(text);
return new Texture(this.dc.currentGlContext, canvas2D);
} else {
return null;
}
};
/**
* Creates a 2D Canvas for a specified text string while considering current TextRenderer state in
* regards to outline usage and color, text color, typeface, and outline width.
* @param {String} text The text string.
* @returns {canvas2D} A 2D Canvas for the specified text string.
*/
TextRenderer.prototype.drawText = function (text) {
var ctx2D = this.ctx2D,
canvas2D = this.canvas2D,
textSize = this.textSize(text),
lines = text.split("\n"),
strokeOffset = this.enableOutline ? this.outlineWidth / 2 : 0,
pixelScale = this.dc.pixelScale;
canvas2D.width = Math.ceil(textSize[0]) * pixelScale;
canvas2D.height = Math.ceil(textSize[1]) * pixelScale;
ctx2D.scale(pixelScale, pixelScale);
ctx2D.font = this.typeFace.fontString;
ctx2D.textBaseline = "bottom";
ctx2D.textAlign = this.typeFace.horizontalAlignment;
ctx2D.fillStyle = this.textColor.toCssColorString();
ctx2D.strokeStyle = this.outlineColor.toCssColorString();
ctx2D.lineWidth = this.outlineWidth;
ctx2D.lineCap = "round";
ctx2D.lineJoin = "round";
if (this.typeFace.horizontalAlignment === "left") {
ctx2D.translate(strokeOffset, 0);
} else if (this.typeFace.horizontalAlignment === "right") {
ctx2D.translate(textSize[0] - strokeOffset, 0);
} else {
ctx2D.translate(textSize[0] / 2, 0);
}
for (var i = 0; i < lines.length; i++) {
ctx2D.translate(0, this.typeFace.size * (1 + this.lineSpacing) + strokeOffset);
if (this.enableOutline) {
ctx2D.strokeText(lines[i], 0, 0);
}
ctx2D.fillText(lines[i], 0, 0);
}
return canvas2D;
};
/**
* Calculates maximum line height based on the current typeFace and outline usage of TextRenderer.
* @returns {Vec2} A vector indicating the text's width and height, respectively, in pixels.
*/
TextRenderer.prototype.getMaxLineHeight = function () {
// Check underscore + capital E with acute accent
return this.textSize("_\u00c9")[1];
};
/**
* Wraps the text based on width and height using new line delimiter
* @param {String} text The text to wrap.
* @param {Number} width The width in pixels.
* @param {Number} height The height in pixels.
* @returns {String} The wrapped text.
*/
TextRenderer.prototype.wrap = function (text, width, height) {
if (!text) {
throw new ArgumentError(
Logger.logMessage(Logger.WARNING, "TextRenderer", "wrap", "missing text"));
}
var i;
var lines = text.split("\n");
var wrappedText = "";
// Wrap each line
for (i = 0; i < lines.length; i++) {
lines[i] = this.wrapLine(lines[i], width);
}
// Concatenate all lines in one string with new line separators
// between lines - not at the end
// Checks for height limit.
var currentHeight = 0;
var heightExceeded = false;
var maxLineHeight = this.getMaxLineHeight();
for (i = 0; i < lines.length && !heightExceeded; i++) {
var subLines = lines[i].split("\n");
for (var j = 0; j < subLines.length && !heightExceeded; j++) {
if (height <= 0 || currentHeight + maxLineHeight <= height) {
wrappedText += subLines[j];
currentHeight += maxLineHeight + this.lineSpacing;
if (j < subLines.length - 1) {
wrappedText += '\n';
}
}
else {
heightExceeded = true;
}
}
if (i < lines.length - 1 && !heightExceeded) {
wrappedText += '\n';
}
}
// Add continuation string if text truncated
if (heightExceeded) {
if (wrappedText.length > 0) {
wrappedText = wrappedText.substring(0, wrappedText.length - 1);
}
wrappedText += "...";
}
return wrappedText;
};
/**
* Wraps a line of text based on width and height
* @param {String} text The text to wrap.
* @param {Number} width The width in pixels.
* @returns {String} The wrapped text.
*/
TextRenderer.prototype.wrapLine = function (text, width) {
var wrappedText = "";
// Single line - trim leading and trailing spaces
var source = text.trim();
var lineBounds = this.textSize(source);
if (lineBounds[0] > width) {
// Split single line to fit preferred width
var line = "";
var start = 0;
var end = source.indexOf(' ', start + 1);
while (start < source.length) {
if (end === -1) {
end = source.length; // last word
}
// Extract a 'word' which is in fact a space and a word
var word = source.substring(start, end);
var linePlusWord = line + word;
if (this.textSize(linePlusWord)[0] <= width) {
// Keep adding to the current line
line += word;
}
else {
// Width exceeded
if (line.length !== 0) {
// Finish current line and start new one
wrappedText += line;
wrappedText += '\n';
line = "";
line += word.trim(); // get read of leading space(s)
}
else {
// Line is empty, force at least one word
line += word.trim();
}
}
// Move forward in source string
start = end;
if (start < source.length - 1) {
end = source.indexOf(' ', start + 1);
}
}
// Gather last line
wrappedText += line;
}
else {
// Line doesn't need to be wrapped
wrappedText += source;
}
return wrappedText;
};
return TextRenderer;
});