babylon.asciiArtPostProcess.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. /// <reference path="../../../dist/preview release/babylon.d.ts"/>
  2. var __extends = (this && this.__extends) || (function () {
  3. var extendStatics = Object.setPrototypeOf ||
  4. ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
  5. function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
  6. return function (d, b) {
  7. extendStatics(d, b);
  8. function __() { this.constructor = d; }
  9. d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
  10. };
  11. })();
  12. var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
  13. var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
  14. if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
  15. else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
  16. return c > 3 && r && Object.defineProperty(target, key, r), r;
  17. };
  18. var BABYLON;
  19. (function (BABYLON) {
  20. /**
  21. * AsciiArtFontTexture is the helper class used to easily create your ascii art font texture.
  22. *
  23. * It basically takes care rendering the font front the given font size to a texture.
  24. * This is used later on in the postprocess.
  25. */
  26. var AsciiArtFontTexture = /** @class */ (function (_super) {
  27. __extends(AsciiArtFontTexture, _super);
  28. /**
  29. * Create a new instance of the Ascii Art FontTexture class
  30. * @param name the name of the texture
  31. * @param font the font to use, use the W3C CSS notation
  32. * @param text the caracter set to use in the rendering.
  33. * @param scene the scene that owns the texture
  34. */
  35. function AsciiArtFontTexture(name, font, text, scene) {
  36. if (scene === void 0) { scene = null; }
  37. var _this = _super.call(this, scene) || this;
  38. scene = _this.getScene();
  39. if (!scene) {
  40. return _this;
  41. }
  42. _this.name = name;
  43. _this._text == text;
  44. _this._font == font;
  45. _this.wrapU = BABYLON.Texture.CLAMP_ADDRESSMODE;
  46. _this.wrapV = BABYLON.Texture.CLAMP_ADDRESSMODE;
  47. //this.anisotropicFilteringLevel = 1;
  48. // Get the font specific info.
  49. var maxCharHeight = _this.getFontHeight(font);
  50. var maxCharWidth = _this.getFontWidth(font);
  51. _this._charSize = Math.max(maxCharHeight.height, maxCharWidth);
  52. // This is an approximate size, but should always be able to fit at least the maxCharCount.
  53. var textureWidth = Math.ceil(_this._charSize * text.length);
  54. var textureHeight = _this._charSize;
  55. // Create the texture that will store the font characters.
  56. _this._texture = scene.getEngine().createDynamicTexture(textureWidth, textureHeight, false, BABYLON.Texture.NEAREST_SAMPLINGMODE);
  57. //scene.getEngine().setclamp
  58. var textureSize = _this.getSize();
  59. // Create a canvas with the final size: the one matching the texture.
  60. var canvas = document.createElement("canvas");
  61. canvas.width = textureSize.width;
  62. canvas.height = textureSize.height;
  63. var context = canvas.getContext("2d");
  64. context.textBaseline = "top";
  65. context.font = font;
  66. context.fillStyle = "white";
  67. context.imageSmoothingEnabled = false;
  68. // Sets the text in the texture.
  69. for (var i = 0; i < text.length; i++) {
  70. context.fillText(text[i], i * _this._charSize, -maxCharHeight.offset);
  71. }
  72. // Flush the text in the dynamic texture.
  73. scene.getEngine().updateDynamicTexture(_this._texture, canvas, false, true);
  74. return _this;
  75. }
  76. Object.defineProperty(AsciiArtFontTexture.prototype, "charSize", {
  77. /**
  78. * Gets the size of one char in the texture (each char fits in size * size space in the texture).
  79. */
  80. get: function () {
  81. return this._charSize;
  82. },
  83. enumerable: true,
  84. configurable: true
  85. });
  86. /**
  87. * Gets the max char width of a font.
  88. * @param font the font to use, use the W3C CSS notation
  89. * @return the max char width
  90. */
  91. AsciiArtFontTexture.prototype.getFontWidth = function (font) {
  92. var fontDraw = document.createElement("canvas");
  93. var ctx = fontDraw.getContext('2d');
  94. ctx.fillStyle = 'white';
  95. ctx.font = font;
  96. return ctx.measureText("W").width;
  97. };
  98. // More info here: https://videlais.com/2014/03/16/the-many-and-varied-problems-with-measuring-font-height-for-html5-canvas/
  99. /**
  100. * Gets the max char height of a font.
  101. * @param font the font to use, use the W3C CSS notation
  102. * @return the max char height
  103. */
  104. AsciiArtFontTexture.prototype.getFontHeight = function (font) {
  105. var fontDraw = document.createElement("canvas");
  106. var ctx = fontDraw.getContext('2d');
  107. ctx.fillRect(0, 0, fontDraw.width, fontDraw.height);
  108. ctx.textBaseline = 'top';
  109. ctx.fillStyle = 'white';
  110. ctx.font = font;
  111. ctx.fillText('jH|', 0, 0);
  112. var pixels = ctx.getImageData(0, 0, fontDraw.width, fontDraw.height).data;
  113. var start = -1;
  114. var end = -1;
  115. for (var row = 0; row < fontDraw.height; row++) {
  116. for (var column = 0; column < fontDraw.width; column++) {
  117. var index = (row * fontDraw.width + column) * 4;
  118. if (pixels[index] === 0) {
  119. if (column === fontDraw.width - 1 && start !== -1) {
  120. end = row;
  121. row = fontDraw.height;
  122. break;
  123. }
  124. continue;
  125. }
  126. else {
  127. if (start === -1) {
  128. start = row;
  129. }
  130. break;
  131. }
  132. }
  133. }
  134. return { height: (end - start) + 1, offset: start - 1 };
  135. };
  136. /**
  137. * Clones the current AsciiArtTexture.
  138. * @return the clone of the texture.
  139. */
  140. AsciiArtFontTexture.prototype.clone = function () {
  141. return new AsciiArtFontTexture(this.name, this._font, this._text, this.getScene());
  142. };
  143. /**
  144. * Parses a json object representing the texture and returns an instance of it.
  145. * @param source the source JSON representation
  146. * @param scene the scene to create the texture for
  147. * @return the parsed texture
  148. */
  149. AsciiArtFontTexture.Parse = function (source, scene) {
  150. var texture = BABYLON.SerializationHelper.Parse(function () { return new AsciiArtFontTexture(source.name, source.font, source.text, scene); }, source, scene, null);
  151. return texture;
  152. };
  153. __decorate([
  154. BABYLON.serialize("font")
  155. ], AsciiArtFontTexture.prototype, "_font", void 0);
  156. __decorate([
  157. BABYLON.serialize("text")
  158. ], AsciiArtFontTexture.prototype, "_text", void 0);
  159. return AsciiArtFontTexture;
  160. }(BABYLON.BaseTexture));
  161. BABYLON.AsciiArtFontTexture = AsciiArtFontTexture;
  162. /**
  163. * AsciiArtPostProcess helps rendering everithing in Ascii Art.
  164. *
  165. * Simmply add it to your scene and let the nerd that lives in you have fun.
  166. * Example usage: var pp = new AsciiArtPostProcess("myAscii", "20px Monospace", camera);
  167. */
  168. var AsciiArtPostProcess = /** @class */ (function (_super) {
  169. __extends(AsciiArtPostProcess, _super);
  170. /**
  171. * Instantiates a new Ascii Art Post Process.
  172. * @param name the name to give to the postprocess
  173. * @camera the camera to apply the post process to.
  174. * @param options can either be the font name or an option object following the IAsciiArtPostProcessOptions format
  175. */
  176. function AsciiArtPostProcess(name, camera, options) {
  177. var _this = _super.call(this, name, 'asciiart', ['asciiArtFontInfos', 'asciiArtOptions'], ['asciiArtFont'], {
  178. width: camera.getEngine().getRenderWidth(),
  179. height: camera.getEngine().getRenderHeight()
  180. }, camera, BABYLON.Texture.TRILINEAR_SAMPLINGMODE, camera.getEngine(), true) || this;
  181. /**
  182. * This defines the amount you want to mix the "tile" or caracter space colored in the ascii art.
  183. * This number is defined between 0 and 1;
  184. */
  185. _this.mixToTile = 0;
  186. /**
  187. * This defines the amount you want to mix the normal rendering pass in the ascii art.
  188. * This number is defined between 0 and 1;
  189. */
  190. _this.mixToNormal = 0;
  191. // Default values.
  192. var font = "40px Monospace";
  193. var characterSet = " `-.'_:,\"=^;<+!*?/cL\\zrs7TivJtC{3F)Il(xZfY5S2eajo14[nuyE]P6V9kXpKwGhqAUbOd8#HRDB0$mgMW&Q%N@";
  194. // Use options.
  195. if (options) {
  196. if (typeof (options) === "string") {
  197. font = options;
  198. }
  199. else {
  200. font = options.font || font;
  201. characterSet = options.characterSet || characterSet;
  202. _this.mixToTile = options.mixToTile || _this.mixToTile;
  203. _this.mixToNormal = options.mixToNormal || _this.mixToNormal;
  204. }
  205. }
  206. _this._asciiArtFontTexture = new AsciiArtFontTexture(name, font, characterSet, camera.getScene());
  207. var textureSize = _this._asciiArtFontTexture.getSize();
  208. _this.onApply = function (effect) {
  209. effect.setTexture("asciiArtFont", _this._asciiArtFontTexture);
  210. effect.setFloat4("asciiArtFontInfos", _this._asciiArtFontTexture.charSize, characterSet.length, textureSize.width, textureSize.height);
  211. effect.setFloat4("asciiArtOptions", _this.width, _this.height, _this.mixToNormal, _this.mixToTile);
  212. };
  213. return _this;
  214. }
  215. return AsciiArtPostProcess;
  216. }(BABYLON.PostProcess));
  217. BABYLON.AsciiArtPostProcess = AsciiArtPostProcess;
  218. })(BABYLON || (BABYLON = {}));
  219. //# sourceMappingURL=babylon.asciiArtPostProcess.js.map
  220. BABYLON.Effect.ShadersStore['asciiartPixelShader'] = "\nvarying vec2 vUV;\nuniform sampler2D textureSampler;\nuniform sampler2D asciiArtFont;\n\nuniform vec4 asciiArtFontInfos;\nuniform vec4 asciiArtOptions;\n\nfloat getLuminance(vec3 color)\n{\nreturn clamp(dot(color,vec3(0.2126,0.7152,0.0722)),0.,1.);\n}\n\nvoid main(void) \n{\nfloat caracterSize=asciiArtFontInfos.x;\nfloat numChar=asciiArtFontInfos.y-1.0;\nfloat fontx=asciiArtFontInfos.z;\nfloat fonty=asciiArtFontInfos.w;\nfloat screenx=asciiArtOptions.x;\nfloat screeny=asciiArtOptions.y;\nfloat tileX=float(floor((gl_FragCoord.x)/caracterSize))*caracterSize/screenx;\nfloat tileY=float(floor((gl_FragCoord.y)/caracterSize))*caracterSize/screeny;\nvec2 tileUV=vec2(tileX,tileY);\nvec4 tileColor=texture2D(textureSampler,tileUV);\nvec4 baseColor=texture2D(textureSampler,vUV);\nfloat tileLuminance=getLuminance(tileColor.rgb);\nfloat offsetx=(float(floor(tileLuminance*numChar)))*caracterSize/fontx;\nfloat offsety=0.0;\nfloat x=float(mod(gl_FragCoord.x,caracterSize))/fontx;\nfloat y=float(mod(gl_FragCoord.y,caracterSize))/fonty;\nvec4 finalColor=texture2D(asciiArtFont,vec2(offsetx+x,offsety+(caracterSize/fonty-y)));\nfinalColor.rgb*=tileColor.rgb;\nfinalColor.a=1.0;\nfinalColor=mix(finalColor,tileColor,asciiArtOptions.w);\nfinalColor=mix(finalColor,baseColor,asciiArtOptions.z);\ngl_FragColor=finalColor;\n}";