--- text Define shapes based upon true type fonts and a string Parameters, associated elements and modifiers must be within braces. text {...} The text definition is assignable to a scene language ID. #declare ID00 = text {...} #local ID00 = text {...} Identifiers (IDs) may be referenced within blocks: object{}, looks_like{}, projected_through{}, bounded_by{} and clipped_by{}. IDs may also be used as objects in the *_object collection of patterns. Note. Normally one would expect to be able to use a text identifier within a text{} block - and one still can, but it is strongly discouraged. The text{} object creates a union{} or merge{} of text{} objects internally. A text{} wrap of such IDs is incorrect, but usually works. As of release R19 of the yuqk fork, the debug compile will flag text{ TextID } syntax as a parse time error because the TextID has is not of type 'text{}'. The yuqk debug compile - as a trial - has added identifier type checking via specific wrapping block types for some object types, including text{}. For example, when running the yuqk debug compile: #declare Light00 = light_source ... ... light_source { Light00 } sphere { Light00 } // <-- Will die for a parse error here. Normally the // parser would simply create a second copy of the // Light00 light_source{}... Raw text{} definitions may be used within blocks: object{}, looks_like{}, projected_through{}, bounded_by{}, clipped_by{}, difference{}, intersection{}, merge{} and union{}. The raw / immediate text{} definitions may also be used as objects in the *_object{} pattern collection. Options: -------- The syntax is: text { ttf "fontname.ttf" "String_of_Text" Thickness, <3D offset vector> [merge | (merge on | off)] [caps | (caps on | caps off)] [open | (open on | open off)] [OBJECT_MODIFIERS...] } --- Font type The 'ttf' string must always be specified and it is currently the only font format option. This true even where the file name is a true type collection and has the .tto suffix. Note. Other .tto files which are, for example, open type font collections are not supported. --- Font filename The '"fontname.ttf"' string is the font filename. It can be a string ID, or function returning a string. Arrays and dictionaries may wrap the string representation. Where the filename is not fully specified - including the directory path to the file, POV-Ray (yuqk) will search the current POV-Ray library paths. Note. In the yuqk fork NO fonts are inbuilt. This reversion to pre v3.7 behavior was taken to make the compiled executables and core install very small. --- Extrusion thickness The 'Thickness' value is usually >0 and it specifies the extrusion depth in the positive z direction. It, however, can be 0.0 or negative. When zero, a single, 2D, top surface is returned. The normal for that 2D surface is as if the thickness was positive, but only the top, cap surface is returned. Where the thickness is negative, the extrusion is in the 0 to -z direction and the normal direction for the caps is the opposite that for positive thickness extrusion. --- Offsets character to character in x,y and z The '<3D offset vector>' vector is often specified as 0 or <0,0,0>. At zero it does nothing. It can add a little space in x between characters, say by using <0.1,0,0>. The more essential feature is tiny offsets in z (and sometimes y) are needed to avoid coincident surfaces where adjacent characters in a font have portions which overlap. With mono fonts (set pitch fonts), the vector can be used to place characters vertically or right to left from the default. See examples below. --- Having text{} create a union{} or merge{} of single characters The text implementation has long, created a union{} of individual text{} objects from each character in the passed text string. The yuqk fork has added the option to create a merge{} of the individual text{} objects. The internal merge{} is created by specifying: merge or merge on Using: merge off creates again a union{} and is equivalent to not specifying a merge setting at all. Note. The keyword 'merge' must follow the offset vector specification. The merge capability is necessary any time the font has adjacent characters which overlap and the texture is partly transparent. Otherwise the internal surfaces often cause unwanted artefacts. --- Ability to keep only the caps (top and bottom) or only sides of characters. The yuqk fork adds two new features to the text{} object in 'open' and 'caps'. These allow users to keep only the character sides or only the character caps (ends / top and bottom surfaces). These features allow for different texturing to be applied to the ends versus the sides, for example. The caps option may be: caps or caps on or caps off The open option may be: open or open on or open off Though it is more common to use only one of the features: 'open' or 'caps', both can be specified. In such cases the last specified wins where it changes the prior selection of surface intersections returned. - If both are set off or on, all surface intersections are returned. - open off -> caps on (caps wins; only top and bottom intersections) - open on -> caps off (caps wins; all surface intersections) - caps off -> open on (open wins; only side intersections) - caps on -> open off (open wins; all surface intersections) Note. The keywords 'caps' and/or 'open' must follow the offset vector specification. Defaults: --------- The default settings: merge - Not specified. The text{} object creates and internal union{}. This is equivalent to specifying 'merge off'. open - Not specified. Return the caps / ends / top and bottom surface intersections and the sides. This is equivalent to 'open off'. caps - Not specified. Return the side surface intersections along with the cap / end intersections. This is equivalent to 'caps off'. Description: ------------ The following description partly taken from: https://wiki.povray.org/content/Reference:Text and https://wiki.povray.org/content/Documentation:Tutorial_Section_3#Index_Entry_text_tutorial A text{} object creates 3-D text as extruded set characters, or 2D text as a set of surfaces, using a font file and text-string inputs. The 3D (or 2D) object created can be used along with shapes like spheres and cylinders as full fledged objects. The 3D output has a defined inside. As such they may be used with Constructive Solid Geometry (CSG) features, be transformed and general object modifiers like 'hollow' and 'no_shadow' may be used. Currently only TrueType Fonts are supported whether found in .ttf or .ttc files. Characters are generally sized so that 1 unit of vertical spacing is correct. The characters usually consume 0.5 to 0.75 units horizontally in western fonts with varying horizontal space - except in mono fonts. More generally, fonts are usually smaller horizontally than vertically or at most square, but the font defines the relative character sizes and spacings. Normally, the text will start with the origin at the lower left, the front of the first character and will extend in the +x-direction. The baseline of the text follows the x-axis and descender drop into the -y-direction. The front of the character sits in the x-y-plane and the text is extruded in the +z-direction. The front-to-back (or back to front) thickness is specified by the required thickness value. The thickness value is normally positive. It can also be negative or zero. When negative the text is extruded in the -z direction and the normals of the cap surfaces is inverted. When the thickness is 0 the inside test will always return false. Further, the code set up expects the user will normally be using the - new to yuqk - 'caps' or 'caps on' option which returns only the top cap when the thickness is zero. In other words, users can now get true 2D surfaces from the text{} object. The horizontal spacing is handled by POV-Ray internally including most kerning information stored in the font. The required 3D vector <Offset> defines any translation between each character. With mono fonts it can be used to arrange the text vertically or so the characters run right to left (backwards). Normally you should specify a zero for this 3D vector value. Specifying 0.1*x, for example, would put additional 0.1 units of space between each character. Specifying 0.0001*z, might be used to prevent coincident surfaces where a font overlaps adjacent characters. Here's an example: text { ttf "timrom.ttf" "POV-Ray (yuqk)" 1, 0 pigment { rgb <1,0,0> } } Note. Currently POV-Ray (yuqk) supports Unicode characters in the Basic Multilingual Plane (BMP). In other words characters with decimal values of 0 to 65535 (Hexadecimal #0000 to #FFFF). TrueType Font (.ttf) files vs TrueType Collection (.ttc) files -------------------------------------------------------------- Input files can be stand alone TrueType Font files (.ttf) consisting of one style of the font. Or input files may be TrueType Collections (.ttc) containing multiple styles of the same font, or multiple derivative fonts of the same style or combinations thereof. However, with collections POV-Ray always uses the first TrueType font in the collection(*). (*) - Which can introduce instability over time should the collection ship later with a changed ordering of internal TrueType fonts. Using the LiberationMono font as an example, the styles of the font can be provided as individual .ttf files: /home/pokorny/Fonts/LiberationMono-Regular.ttf /home/pokorny/Fonts/LiberationMono-Italic.ttf /home/pokorny/Fonts/LiberationMono-Bold.ttf /home/pokorny/Fonts/LiberationMono-BoldItalic.ttf Or those four styles could be packed into a single .ttc collection: /home/pokorny/Fonts/LiberationMono.ttc Note! The .ttc suffix is used for collections which are NOT TrueType fonts and so not currently supported by POV-Ray's text{} object. For example, many linux distribution have CJK fonts like: /usr/share/fonts/opentype/noto/NotoSansCJK-Regular.ttc Tip. If running on a Linux or Unix system and you have copied or downloaded a font collection file locally and you want to know whether it contains TrueType fonts, you can use the command: file NotoSansCJK-Regular.ttc ------------------------------- General yuqk fork modifications ------------------------------- Over the years the yuqk fork has made a number of internal modifications to the code to make it faster and more robust. See the issues section below for more detail about bug fixes and remaining issues. a) The quadratic solver was changed to a version from CERN which is usually faster and always more accurate. b) Two changes which improve inside test performance. One of which more generally applies to unions{} which are created by the text{} object. ---- Tips ---- Quick hack for somewhat border characters ----------------------------------------- It's a hack, but you can get somewhat of a bordered character effect in POV-Ray where you are using a font which offers different styles like regular, bold, italic and bolditalic. You can quickly place additional text{} objects with differing fonts slightly behind the regular style font. For example, placing a text{} object using the bold style just behind the text{} object using the regular style. Because of how fonts typically work they are often constrained vertically. This means for 3D shape applications over 2D there may well be coincident surfaces which could cause problems. It's a hack - but a quick one. More robust bordered characters ------------------------------- There is also the dens_object{} pattern in the yuqk fork which can in many cases implement a good bordered character - or bordered any object with a defined inside. See: https://news.povray.org/povray.binaries.images/thread/%3C64f53f14%241%40news.povray.org%3E/ Names and data for Unicode characters ------------------------------------- See: https://www.unicode.org/Public/UCD/latest/ucd/NamesList.txt The Unicode indexing is usually by hexadecimal values rather than the decimal value we'd pass to chr() in a scene. See conversion and specification methods just below. --------- Examples: --------- Does your font file support a particular Unicode character? For example, the euro symbol (€). It's decimal value is 8364 (The hexadecimal value is 20AC). //--- // yuqk isFontCharSupported.pov +mv3.8 +w400 +h400 +a +p text { ttf "timrom.ttf" chr(8364) // Or chr(val("0x20AC")) //chr(val(S)) // On command line add declare=S=\"0x20AC\" //"€" // OK. //"\u20AC" // OK. //chr(asc("€")) // OK. 0.01, 0 pigment { rgb <1,0,0> } finish { emission 1 } scale 0.2 translate <0,0,2> } // timrom.ttf (No) // LiberationMono-Regular.ttf (Yes) //--- --- // Sample scene. // yuqk monoBoxChars.pov +w800 +h800 +p // As animation: // yuqk monoBoxChars.pov +w800 +h800 +p +kff10 #version unofficial 3.8; // yuqk #if (file_exists("version.inc")) #include "version.inc" #end #if (!defined(Fork_yuqk)) #error "This POV-Ray SDL code requires the yuqk fork." #end global_settings { assumed_gamma 1.0 } #declare VarOrthoMult = 3.0/max(image_width/image_height,image_height/image_width); #declare Camera01z = camera { orthographic location <0,0,-2> direction z right VarOrthoMult*x*max(1,image_width/image_height) up VarOrthoMult*y*max(1,image_height/image_width) } #include "functions.inc" // Liberation fonts usually ship with Linux distributions. // Use command: 'fc-list | grep LiberationMono' to find // the location on your computer. #declare // FontFile = "/home/pokorny/Fonts/LiberationMono-Regular.ttf" FontFile = "/home/pokorny/Fonts/UbuntuMono-R.ttf" #declare BoxChars = "─│┌┐└┘├┤┬┴┼┼┼┼┼─│" #declare RndSd = seed(f_clock(0,0,0)*1e2); #declare Rows = 12; #declare Cols = 20; #declare Ary = array[Rows] #for (R,0,Rows-1,1) #declare Ary[R] = concat( #for (C,0,Cols-1,1) substr(BoxChars,int(1+rand(RndSd)*(10-1e-6)),1), #end "") #end plane { -z 0 pigment { bool_object { #for (R,0,Rows-1,1) text { ttf FontFile Ary[R] 1,0 translate <0,6-(1*R),0> } #end pigment { rgb 0 } pigment { rgb <1,1,0> } } scale <1/5,1/5,1> translate <-1,-0.1,-0.5> } finish { emission 1 } } camera { Camera01z } //--- --- // Sample scene. // 'yuqk textDirections.pov +w800 +h800 +p' // With mono fonts right to left and vertical text{}s possible #version unofficial 3.8; // yuqk #if (file_exists("version.inc")) #include "version.inc" #end #if (!defined(Fork_yuqk)) #error "This POV-Ray SDL code requires the yuqk fork." #end global_settings{ assumed_gamma 1 } #declare VarOrthoMult = 12.0/max(image_width/image_height,image_height/image_width); #declare Camera01z = camera { orthographic location <0,0,-2> direction z right VarOrthoMult*x*max(1,image_width/image_height) up VarOrthoMult*y*max(1,image_height/image_width) } // Liberation fonts usually ship with Linux distributions. // Use command: 'fc-list | grep LiberationMono' to find // the location on your computer. #declare FontFile = "/home/pokorny/Fonts/LiberationMono-Regular.ttf" // "/home/pokorny/Fonts/LiberationMono-BoldItalic.ttf" // "/home/pokorny/Fonts/UbuntuMono-R.ttf" // Use Chris R method to determine character pitches #local _T1 = text { ttf FontFile "A" 1.0, 0 } #local _T2 = text { ttf FontFile "AA" 1.0, 0 } #local _T3 = text { ttf FontFile "│" 1.0, 0 } #local _T1sz = max_extent(_T1) - min_extent(_T1); #local _T2sz = max_extent(_T2) - min_extent(_T2); #local _T3sz = max_extent(_T3) - min_extent(_T3); #declare _Cwidth = (_T2sz - _T1sz).x; #declare _Cheight = _T3sz.y; #declare TxtRight = text { ttf FontFile "ABCDEFGHIJ", 2e-6, <0,0,0> //rotate <0,0,1e-6> // User fixes axis aligned artefacts translate <-5*_Cwidth,+_Cheight+(_Cheight/4),-1e-6> } #declare TxtLeft = text { ttf FontFile "ABCDEFGHIJ", 2e-6, <-2*_Cwidth,0,0> //rotate <0,0,1e-6> // User fixes axis aligned artefacts translate <+4*_Cwidth,-2*_Cheight+(_Cheight/4),-1e-6> } #declare TxtDown = text { ttf FontFile "ABCDEFGHIJ", 2e-6, <-_Cwidth,-_Cheight,0> //rotate <0,0,1e-6> // User fixes axis aligned artefacts translate <-7*_Cwidth,+4*_Cheight+(_Cheight/4),-1e-6> } #declare TxtUp = text { ttf FontFile "ABCDEFGHIJ",2e-6, <-_Cwidth,+_Cheight,0> //rotate <0,0,1e-6> // User fixes axis aligned artefacts translate <+6*_Cwidth,-5*_Cheight+(_Cheight/4),-1e-6> } #declare Fn = function { pattern { bool_object { TxtRight TxtLeft TxtDown TxtUp } warp { it_amount <0.05,0.05,0> it_lambda 4 it_scale 0.9} } } #include "functions.inc" plane { -z 0 pigment { #if (0) bool_object { TxtRight TxtLeft TxtDown TxtUp } warp { it_amount <0.05,0.05,0> it_octaves 3 it_omega 0.9 it_lambda 7 it_scale 1.7 } #elseif (0) function { Fn(x,y,z) } #else function { Fn(x,y,z)*(f_popnrm_rnoise(x,y,z,now,1)+1.0) } inverse #end } finish { emission 1 } } camera { Camera01z } sphere { 0, 0.5 pigment { rgbt <1,0,0,0.7> } finish { emission 1 } } --- // Sample scene. // yuqk outLineFontMethod.pov +w800 +h800 +p // Using negative area light projected through text cap to turn // any font into an apparent outline font. #version unofficial 3.8; // yuqk #if (file_exists("version.inc")) #include "version.inc" #end #if (!defined(Fork_yuqk)) #error "This POV-Ray SDL code requires the yuqk fork." #end global_settings { assumed_gamma 1.0 } #declare VarOrthoMult = 5.0/max(image_width/image_height,image_height/image_width); #declare Camera01z = camera { orthographic location <0,0,-2> direction z right VarOrthoMult*x*max(1,image_width/image_height) up VarOrthoMult*y*max(1,image_height/image_width) } // Liberation fonts usually ship with Linux distributions. // Use command: 'fc-list | grep LiberationMono' to find // the location on your computer. #declare // FontFile = "/home/pokorny/Fonts/LiberationMono-Regular.ttf" FontFile = "/home/pokorny/Fonts/UbuntuMono-R.ttf" #declare String = "outline" // Use Chris R method to determine character pitches #local _T1 = text { ttf FontFile "A" 1.0, 0 } #local _T2 = text { ttf FontFile "AA" 1.0, 0 } #local _T3 = text { ttf FontFile "│" 1.0, 0 } #local _T1sz = max_extent(_T1) - min_extent(_T1); #local _T2sz = max_extent(_T2) - min_extent(_T2); #local _T3sz = max_extent(_T3) - min_extent(_T3); #declare _Cwidth = (_T2sz - _T1sz).x; #declare _Cheight = _T3sz.y; #declare TxtRight = text { ttf FontFile String, 0, <0,0,0> caps on // At depth of 0 get 2D top cap only translate <-3.5*_Cwidth,-_Cheight/4,0> } #declare Text00 = object { TxtRight pigment { rgb <1,1,0> } finish { emission 1 } interior_texture { pigment { rgb <0,1,1> } finish { emission 1 } } translate <0.0,0.0,-0.5> } #declare Light00 = light_source { <0,0,-10>, rgb -5 parallel point_at <0,0,0> projected_through { union { object { TxtRight translate <0,+1,-0.5> } object { TxtRight translate <0,+0,-0.5> } object { TxtRight translate <0,-1,-0.5> } } } area_light 3*x,5*y,33,55 adaptive 1 jitter circular orient } //--- camera { Camera01z } light_source { Light00 } plane { -z 0 pigment { rgb 1 } finish { emission 1 diffuse 1} } union { object { Text00 translate <0,+2,0> } object { Text00 translate <0,+1,0> } object { Text00 translate <0,+0,0> } object { Text00 translate <0,-1,0> } object { Text00 translate <0,-2,0> } } --- // Sample scene. // yuqk caps_open_features.pov +w800 +h800 +p #version unofficial 3.8; // yuqk #if (file_exists("version.inc")) #include "version.inc" #end #if (!defined(Fork_yuqk)) #error "This POV-Ray SDL code requires the yuqk fork." #end global_settings { assumed_gamma 1.0 } #declare Camera00 = camera { perspective location <1,0.3,-3.001>*0.7 sky y angle 35 right x*(image_width/image_height) look_at <0,0,0> } #declare VarOrthoMult = 5.0/max(image_width/image_height,image_height/image_width); #declare Camera01z = camera { orthographic location <0,0,-2> direction z right VarOrthoMult*x*max(1,image_width/image_height) up VarOrthoMult*y*max(1,image_height/image_width) } #declare White = srgb <1,1,1>; #declare Light00 = light_source { <50,150,-250>, White } #declare Red = srgb <1,0,0>; #declare CylinderX = cylinder { -1*x, 1*x, 0.01 pigment { Red } } #declare Green = srgb <0,1,0>; #declare CylinderY = cylinder { -1*y, 1*y, 0.01 pigment { Green } } #declare Blue = srgb <0,0,1>; #declare CylinderZ = cylinder { -1*z, 1*z, 0.01 pigment { Blue } } // Liberation fonts usually ship with Linux distributions. // Use command: 'fc-list | grep LiberationMono' to find // the location on your computer. #declare // FontFile = "/home/pokorny/Fonts/LiberationMono-Regular.ttf" FontFile = "/home/pokorny/Fonts/UbuntuMono-R.ttf" #declare String = "XYZ 123" // Use Chris R method to determine character pitches #local _T1 = text { ttf FontFile "A" 1.0, 0 } #local _T2 = text { ttf FontFile "AA" 1.0, 0 } #local _T3 = text { ttf FontFile "│" 1.0, 0 } #local _T1sz = max_extent(_T1) - min_extent(_T1); #local _T2sz = max_extent(_T2) - min_extent(_T2); #local _T3sz = max_extent(_T3) - min_extent(_T3); #declare _Cwidth = (_T2sz - _T1sz).x; #declare _Cheight = _T3sz.y; #declare TxtrO = texture { pigment { rgb <0.5,0.5,0> } finish { emission <0.1,0,0> } } #declare TxtrI = texture { pigment { rgb <0,0.5,0.5> } finish { emission <0,0.1,0> } } #declare Depth = 1; #declare TxtPosThickness = text { ttf FontFile String, +Depth, <0,0,0> texture { TxtrO } interior_texture { TxtrI } translate <-3.5*_Cwidth,-_Cheight/4,0> } #declare TxtNegThickness = text { ttf FontFile String, -Depth, <0,0,0> texture { TxtrO } interior_texture { TxtrI } translate <-3.5*_Cwidth,-_Cheight/4,0> } #declare TxtPosThickness_caps = text { ttf FontFile String, +Depth, <0,0,0> caps on texture { TxtrO } interior_texture { TxtrI } translate <-3.5*_Cwidth,-_Cheight/4,0> } #declare TxtNegThickness_caps = text { ttf FontFile String, -Depth, <0,0,0> caps on texture { TxtrO } interior_texture { TxtrI } translate <-3.5*_Cwidth,-_Cheight/4,0> } #declare TxtPosThickness_open = text { ttf FontFile String, +Depth, <0,0,0> open on texture { TxtrO } interior_texture { TxtrI } translate <-3.5*_Cwidth,-_Cheight/4,0> } #declare TxtNegThickness_open = text { ttf FontFile String, -Depth, <0,0,0> open on texture { TxtrO } interior_texture { TxtrI } translate <-3.5*_Cwidth,-_Cheight/4,0> } #declare U00 = union { object { TxtPosThickness translate <0,+5*_Cheight,0> } object { TxtNegThickness translate <0,+4*_Cheight,0> } object { TxtPosThickness_caps translate <0,+3*_Cheight,0> } object { TxtNegThickness_caps translate <0,+2*_Cheight,0> } object { TxtPosThickness_open translate <0,+1*_Cheight,0> } object { TxtNegThickness_open translate <0,+0*_Cheight,0> } translate <0,-2.5*_Cheight,0> scale 0.2 } //--- camera { Camera00 } // camera { Camera01z } light_source { Light00 } #if (1) object { U00 } #elseif (0) intersection { // negative thickness text{} object { U00 } box { <-1,-1,-0.10>,<+1,+1,-0.05> } } #elseif (0) intersection { // positive thickness text{} object { U00 } box { <-1,-1,+0.05>,<+1,+1,+0.10> } } #end sphere { 0, 0.005 pigment { rgb 0.2 } } --- //--- // Sample scene. // yuqk monoFontFileMovie.pov +w900 +h450 +p // Final frames: // yuqk monoFontFileMovie.pov +w900 +h450 -p -d +kff1311 \ // +a0.002 +am3 +rd3 +rd5 +j1.1 // Creating an mp4 movie of all the characters defined in the Basic // Multilingual Plane (BMP) for a mono font file. // // Note. Character image compares are done here with POV-Ray (yuqk) // SDL during parsing. Expect the entire 1311 frames to take hours. // To render, say only the first 20 frames (1000 characters) add the // flags: +sf1 +ef20 to the final frames command above. #version unofficial 3.8; // yuqk #if (file_exists("version.inc")) #include "version.inc" #end #if (!defined(Fork_yuqk)) #error "This POV-Ray SDL code requires the yuqk fork." #end global_settings { assumed_gamma 1.0 } #declare VarOrthoMult = 4.0/max(image_width/image_height,image_height/image_width); #declare Camera01z = camera { orthographic location <0,0,-2> direction z right VarOrthoMult*x*max(1,image_width/image_height) up VarOrthoMult*y*max(1,image_height/image_width) } // Liberation fonts usually ship with Linux distributions. // Use command: 'fc-list | grep LiberationMono' to find // the location on your computer. #declare // FontFile = "/home/pokorny/Fonts/LiberationMono-Regular.ttf" FontFile = "/home/pokorny/Fonts/UbuntuMono-R.ttf" // FontFile = "/home/pokorny/Fonts/Hack-Regular.ttf" // Use Chris R method to determine character pitches #local _T1 = text { ttf FontFile "A" 1.0, 0 } #local _T2 = text { ttf FontFile "AA" 1.0, 0 } #local _T3 = text { ttf FontFile "│" 1.0, 0 } #local _T1sz = max_extent(_T1) - min_extent(_T1); #local _T2sz = max_extent(_T2) - min_extent(_T2); #local _T3sz = max_extent(_T3) - min_extent(_T3); #declare _Cwidth = (_T2sz - _T1sz).x; #declare _Cheight = _T3sz.y; #local CharsPerPage = 50; #local I = max(0,frame_number-1)*CharsPerPage; #local I = min(65535-CharsPerPage+1,I); #local Str00n = concat( "|",str(I+ 0,-5,0),"|",str(I+ 1,-5,0),"|",str(I+ 2,-5,0),"|",str(I+ 3,-5,0),"|",str(I+ 4,-5,0),"|",str(I+ 5,-5,0),"|",str(I+ 6,-5,0),"|",str(I+ 7,-5,0),"|",str(I+ 8,-5,0),"|",str(I+ 9,-5,0),"|") #local Str01n = concat( "|",str(I+10,-5,0),"|",str(I+11,-5,0),"|",str(I+12,-5,0),"|",str(I+13,-5,0),"|",str(I+14,-5,0),"|",str(I+15,-5,0),"|",str(I+16,-5,0),"|",str(I+17,-5,0),"|",str(I+18,-5,0),"|",str(I+19,-5,0),"|") #local Str02n = concat( "|",str(I+20,-5,0),"|",str(I+21,-5,0),"|",str(I+22,-5,0),"|",str(I+23,-5,0),"|",str(I+24,-5,0),"|",str(I+25,-5,0),"|",str(I+26,-5,0),"|",str(I+27,-5,0),"|",str(I+28,-5,0),"|",str(I+29,-5,0),"|") #local Str03n = concat( "|",str(I+30,-5,0),"|",str(I+31,-5,0),"|",str(I+32,-5,0),"|",str(I+33,-5,0),"|",str(I+34,-5,0),"|",str(I+35,-5,0),"|",str(I+36,-5,0),"|",str(I+37,-5,0),"|",str(I+38,-5,0),"|",str(I+39,-5,0),"|") #local Str04n = concat( "|",str(I+40,-5,0),"|",str(I+41,-5,0),"|",str(I+42,-5,0),"|",str(I+43,-5,0),"|",str(I+44,-5,0),"|",str(I+45,-5,0),"|",str(I+46,-5,0),"|",str(I+47,-5,0),"|",str(I+48,-5,0),"|",str(I+49,-5,0),"|") #if (strlen(chr(I+0))!=1) // Special handling for null character #local Chr00 = " " // Also useful for image character comparisons #else #local Chr00 = chr(I+0) #end #local Str00c = concat( "| ", Chr00," | ",chr(I+ 1)," | ",chr(I+ 2)," | ",chr(I+ 3)," | ",chr(I+ 4)," | ",chr(I+ 5)," | ",chr(I+ 6)," | ",chr(I+ 7)," | ",chr(I+ 8)," | ",chr(I+ 9)," |") #local Str01c = concat( "| ",chr(I+10)," | ",chr(I+11)," | ",chr(I+12)," | ",chr(I+13)," | ",chr(I+14)," | ",chr(I+15)," | ",chr(I+16)," | ",chr(I+17)," | ",chr(I+18)," | ",chr(I+19)," |") #local Str02c = concat( "| ",chr(I+20)," | ",chr(I+21)," | ",chr(I+22)," | ",chr(I+23)," | ",chr(I+24)," | ",chr(I+25)," | ",chr(I+26)," | ",chr(I+27)," | ",chr(I+28)," | ",chr(I+29)," |") #local Str03c = concat( "| ",chr(I+30)," | ",chr(I+31)," | ",chr(I+32)," | ",chr(I+33)," | ",chr(I+34)," | ",chr(I+35)," | ",chr(I+36)," | ",chr(I+37)," | ",chr(I+38)," | ",chr(I+39)," |") #local Str04c = concat( "| ",chr(I+40)," | ",chr(I+41)," | ",chr(I+42)," | ",chr(I+43)," | ",chr(I+44)," | ",chr(I+45)," | ",chr(I+46)," | ",chr(I+47)," | ",chr(I+48)," | ",chr(I+49)," |") #declare Text00n = text { ttf FontFile Str00n, 1.0, 0 translate <0,+0*_Cheight,-0.5> } #declare Text00c = text { ttf FontFile Str00c, 1.0, 0 translate <0,-1*_Cheight,-0.5> } #declare Text01n = text { ttf FontFile Str01n, 1.0, 0 translate <0,-2*_Cheight,-0.5> } #declare Text01c = text { ttf FontFile Str01c, 1.0, 0 translate <0,-3*_Cheight,-0.5> } #declare Text02n = text { ttf FontFile Str02n, 1.0, 0 translate <0,-4*_Cheight,-0.5> } #declare Text02c = text { ttf FontFile Str02c, 1.0, 0 translate <0,-5*_Cheight,-0.5> } #declare Text03n = text { ttf FontFile Str03n, 1.0, 0 translate <0,-6*_Cheight,-0.5> } #declare Text03c = text { ttf FontFile Str03c, 1.0, 0 translate <0,-7*_Cheight,-0.5> } #declare Text04n = text { ttf FontFile Str04n, 1.0, 0 translate <0,-8*_Cheight,-0.5> } #declare Text04c = text { ttf FontFile Str04c, 1.0, 0 translate <0,-9*_Cheight,-0.5> } // Because almost all fonts are sparsely implemented in the BMP, do some work to // create completely black images where characters are defaulting to some common glyph. #local _TCmp1 = text { ttf FontFile Chr00 1.0, 0 translate -0.5*z } #declare FnBaseImg = function { pattern { bool_object {_TCmp1} } } #local Resolution = 199; #local InvDelta = 1.0/(Resolution-30); #local Matched = true; #for (KK,I+1,I+CharsPerPage-1,1) #local KK = min(65535,KK); #if (strlen(chr(KK))!=1) #error concat("ERROR! Saw strlen() other than 1 at character ",str(KK,5,0),"\n") #end #local _TCmp2 = text { ttf FontFile chr(KK) 1.0, 0 translate -0.5*z } #if (defined(FnTestImg)) #undef FnTestImg #end #declare FnTestImg = function { pattern { bool_object {_TCmp2} } } #for (YY,-50,Resolution,1) #for (XX,-50,Resolution,1) #if ( FnBaseImg(XX*InvDelta,YY*InvDelta,0) != FnTestImg(XX*InvDelta,YY*InvDelta,0)) #local Matched = false; //#debug concat("Failed match at XX = ",str(XX,4,0)," YY = ",str(YY,4,0),"\n") #end #if (!Matched) #break #end #end //#debug concat("YY = ",str(YY,4,0),"\n") #if (!Matched) #break #end #end #end #undef KK #if (!Matched) plane { -z 0 pigment { bool_object { Text00n Text00c Text01n Text01c Text02n Text02c Text03n Text03c Text04n Text04c pigment { rgb <0,0,0> } pigment { rgb <0,1,0> } } translate <-(max_extent(Text00n)-min_extent(Text00n)).x/2.0,5*_Cheight-0.5,0> scale 0.1 } finish { emission 1 } } #end camera { Camera01z } //--- // The following ReduceFontFrames.tcl script is one way to extract // non-black frames for the final animation. // #!/usr/bin/tclsh // # ReduceFontFrames.tcl // # This TCL script set up to run after 1311 frames of 50 characters each have // # been rendered to images covering the Basic Multilingual Plane (BMP) for // # a given mono-pitch font. Frames for which no character glyphs exist are // # rendered black and have a small size. Frame images with at least one // # character having glyphs are rendered as a table and have a larger size // # // # We use this size difference to create a subset of frames with indexes // # from 1 to N of the table versions. Only these frames become part of an // # mp4 animation encoded with ffmpeg. // # // # The mp4 animation can be viewed to review the characters supported by // # the particular font file. // # // # After this script runs (./ReduceFontFrames.tcl) there will be files of // # the form: monoFontFileMovie__001.png, monoFontFileMovie__002.png, ... // # // # Use the ffmpeg program to turn the reduced collection of frames into a // # mp4 movie at a frame per second: // # // # fmpeg -y -r 1 -i monoFontFileMovie__%3d.png -pix_fmt yuv420p -strict -2 UbuntuMonoR.mp4 // # // # View with: 'ffplay UbuntuMonoR.mp4', your web browser or other mp4 viewing // # program. // # // set tmpdirVal [pwd] // set tmpList [glob -directory $tmpdirVal -nocomplain monoFontFileMovie????.png] // puts "Number of frames found: [llength $tmpList]" // set keepList [list] // set ignoreList [list] // foreach filename $tmpList { // if { [file size $filename] < 5000 } { // lappend ignoreList $filename // } else { // lappend keepList $filename // } // } // puts "Keeping [llength $keepList] frames images" // set cnt 1 // foreach filename $keepList { // set rootName [file rootname $filename] // set rootName [string range $rootName 0 [expr {[string length $rootName]-5}] ] // set newFileName [format {%s__%03d.png} $rootName $cnt] // incr cnt // puts "copy $filename to $newFileName" // file copy -force $filename $newFileName // } // exit 0; --- Issues: ------- While an attempt has been made to list bugs and fixes first, the following list of issues is in no particular order of importance. --- a) The yuqk fork fixed a forever standing bug with diacritical marks as implemented (correctly) by many fonts. This means the yuqk fork renders these characters correctly where official POV-Ray releases often do not. See: https://news.povray.org/povray.binaries.images/thread/%3C63cd6b12%241%40news.povray.org%3E/ https://news.povray.org/povray.beta-test/thread/%3C642f33e8%241%40news.povray.org%3E/ b) The z component of the offset vector was not implemented correctly in any relatively recent official release of POV-Ray. A tiny z offset is necessary where fonts are set up such that adjacent characters slightly overlap after kerning and placement. We end up with coincident front and back surfaces without the z offset capability. This is fixed in the yuqk fork. Note. Occasionally a small y offset is necessary too - which does work in official POV-Ray releases. c) Official releases of POV-Ray do not correctly implement negative thickness. It only partly works for the caps / ends / (top and bottom) surface intersections. The yuqk fork supports negative thickness values. d) The yuqk fork has fixed an occasional segmentation fault / memory addressing bug during parsing which exists in some v3.8 and v4.0 versions of POV-Ray. The failure rate was on the order of maybe 1/1000 parse time interpretations of text{} objects. See: https://news.povray.org/povray.bugreports/thread/%3C67cc705e%40news.povray.org%3E/ e) The yuqk fork has used v4.0 changes, backed out of the v3.8 beta releases, which allow more utf8 / Unicode characters to render correctly. This true especially on Linux / Unix machines. These v4.0 changes to text{} were part of a larger group fixes made by Christoph Lipka (aka clipka) during v3.8 development in an effort to improve Unicode support for the Basic Multilingual Plane (BMP) within POV-ray. I do not know the complete details of these changes. For example, using the isFontCharSupported.pov example scene above and the "LiberationMono-Regular.ttf" font file shipped with my Ubuntu 24.04 Linux machine, the yuqk fork correctly renders the "│" / char(9474) / hexadecimal 0x2502 Unicode character - where the official v3.8 beta 2 release does not. f) There is a tiny rotation applied always to text{} objects of <0.001,0.001,0.001> with all official releases of POV-Ray. The rotation was apparently added to eliminate axis aligned numerical seams creating artefacts. The hidden rotation, of course, introduces placement and positioning noise. Noise which is magnified as the text{} object's string length grows. This another reason in official POV-Ray releases that breaking long strings up into separate single character text{} objects can be useful. Reference: https://news.povray.org/povray.pov4.discussion.general/thread/%3C67d31e6b%241%40news.povray.org%3E/ As of release R19 of yuqk, this rotation has been removed as a trial. In testing no numerical artefacts were seen. The thinking is the user should always have been left with the fixes here as is done for other numerical situations like avoiding coincident surfaces. Should numerical artefacts appear the user can always add the same rotation themselves - but it's very likely they can find some more minimal transform to achieve the same result. g) The yuqk fork moved to the same CERN quadratic server code for ttf fonts as is used for many shapes. It's generally faster - and in being more accurate will have addressed / fixed some numerical artefacts. The text{} truetype shape code maintained it's own quadratic solver with a very aggressive skip of second possible root method - if the first of dual roots was outside the [0..1] range. The shortcut helped almost exclusively 'orthographic rays to text character face' scenes. In testing there were very few missed roots, but missed they were... It's a cool trick, but both one not available with the CERN version and it carries risk as it does sometimes miss valid roots. The new solver is faster and more accurate in testing - excepting slower for that particular orthographic ray to text character case where it is 4-5% slower. h) The inside test as it relates to the winding test used by POV-Ray differs from that defined for ttf fonts. The winding test checks for the number of crossing of a splines path from a evaluation point in space direction. Where the crossing count is even the evaluation point is seen as outside the overall region as defined by the splines. This method works well and specifically allows holes (or bites) to be cut in larger spline loops by loops wholly contained or partly overlapping. TrueType fonts allow curve loops to overlap in an additive way. Though most font implementations don't overlap loops - probably so they work with programs like POV-Ray too. Where fonts contain character glyphs which have loops that overlap, these will not be correctly rendered correctly in total. Often the overlaps are quite small and not very noticeable. There is no fix or work around in these cases - other than using another font implementation. i) Bounding. The POV-Ray code creates an internal union{} object with a single bounding box from all the individual characters in the passed text{}'s string. The yuqk fork can optionally create an internal merge{} instead. For most ray tracing aspects, the overall union{}s containing individual characters are flattened and bounded individually - leading to better performance. The merge{}s are treated whole and as merges have some unresolved bounding issues I've not enumerated herein - use merge{}s only when needed. For inside testing(*) the union{} hierarchy is not flattened and this testing can be slow - especially for something like the text{} object. The yuqk fork has a couple of internal enhancements which speed up the text{}'s union{} inside testing - and one of them should help across all union{}s of objects. (*) - Note. The bounding tests part of inside testing do not get recorded or reported in POV-Ray's statistics. Note! Even with the yuqk inside test enhancements, it can be a render time win to break long strings in single character text{} objects (which still end up as union{}s or merge{}s internally). Of course, this leaves the character to character placement to the user, and for complicated overlapping character fonts it might not be practical. j) There are small fudge factors due bounding which can cause differences when trying to align, say, box drawing characters by transform with respect to portions 'placed' by inclusion in a longer passed string. It might be in some cases ones needs to do all placements by explicit transform or by using all longer text{} strings (this last works better in yuqk do the elimination of the hidden rotation). k) There are small differences in the specified font accuracy coming into POV-Ray too. In all fonts there is in the Font Header section a setting of: unitsPerEm: <number> Google AI says: "In TrueType fonts, "units per em" (UPM) refers to the number of font units that define the height of the "em square," a fundamental metric for font design and scaling, often set to 2048." I don't routinely pay attention to this value when using fonts, but I've noticed settings of 1000, 1024 and 2048. The value matters because it is used within POV-Ray to normalize each all the character glyph contours (Bezier quadratic piecemeal loops in the ttf fonts) to fit roughly within a +x+y unit square (though the characters are rarely square in western fonts). The lower that unitsPerEm value and the lower the accuracy of POV-Ray's normalization. Further, fonts often have differing height to width aspect ratios - which itself can affect accuracy of the internal glyphs on POV-Ray's normalization. l) The text{} object today only handles TrueType Fonts (ttf / <filename>.ttf). m) The official documentation for POV-Ray claims it can read TrueType Collection (.ttc) files containing multiple fonts. It in fact has never been able to work with more than the first font in the collection. Not being able to set which font to use with .ttc files makes their use iffy over time. Note! Not all .ttc files contain true type coded fonts. Some, for example, contain open type fonts which are not today supported. n) Even at character screen sizes which might normally be read, some characters (or some entire fonts - such as 'hairline' fonts) do not render cleanly. The shading characters such as: chr(9608),chr(9617),chr(9618) and chr(9619) are made up of very small shapes or holes, as another example. Where the detail is too small to be reliably picked up; The yuqk fork's +rd n / anti-alias_min_depth=n option can render these cleanly. The jitter option can sometimes help too, but it can be problematic too in that statistically it can still miss small features - and where it doesn't - it plays slightly with final intensity which results in larger stored images and animations (they compress less well). In POV-Ray proper, the only clean option in these cases is to render at a larger image size that is scaled down to a final presentation size. o) Though some horizontal kerning is supported. Good fonts often have additional kerning features, hinting / auto-hinting with tiny programming kernels which are not supported by the POV-Ray's true type shape implementation. This means for some fonts, and some characters, the appearance in a good word processor or type setting program may be better / differ from what is seen for POV-Ray results. p) An ability to create an internal merge{} of character objects, rather than only a union{} of character objects from the text{} string has also long been missing. With the yuqk fork, tfhe keyword 'merge' or 'merge on|off' can be specified within the text{} block. Reference: https://news.povray.org/povray.binaries.animations/thread/%3C67dea582%241%40news.povray.org%3E/ q) Listing this as an issue, but it's happening by original design and I'm unsure what to do otherwise(*). The first character at x=0.0 gets special treatment in that no extra left buffer is added as it is to characters with x!=0.0. I think it was done this way to make it easier to position the start of the string - and it does this. However, it also introduces an 'x' direction discontinuity in placement which can show up if, for example, the user is creating an animation using changes in the x value of the offset vector. (*) Had the thought another keyword could be added to turn the special handling on and off, but maybe being aware of the issue is enough for the users to avoid it. r) Font files contain mistakes! This is especially true for the huge number of amateur created fonts which can be found on the web. Some of these are auto-created by tracing someone's hand writing or drawings and the glyph contours for these are often 'chaotic'. Fonts are often rendered on the web by the FreeType library (https://freetype.org/) its target output is creating (rasterized) glyph images. FreeType can handle badly implemented font packages much better than POV-Ray because of the huge difference in development effort - and that it is often the gate keeper / filter during a font's implementation. POV-Ray needs clean character glyphs due the 3D extrusion more than if it were generating rasterized output. All, a long winded way of saying a font which looks good on the web, might not look good in POV-Ray. Professionally created fonts will likely be best on average. s) POV-Ray (yuqk) ships with the true type font files: crystal.ttf, cyrvetic.ttf, povlogo.ttf and timrom.ttf. These are also available for package installation with many linux distributions today. They are, as fonts, limited overall - especially with respect to character support. t) Jérôme Grimbert fixed long standing issues with cap normals during v3.8 development. Both v3.8 versions and so the yuqk fork will differ some from earlier versions of POV-Ray with respect to top and bottom normals. u) With orthogonal rays as when using an orthographic camera, where substantial parts of a character's glyph runs horizontally or vertically there is an exposure to position bias. The apparent 'width' of parts of the character can change depending upon its position relative to the incoming rays. With an orthographic camera this is due how the rendered pixels happen to align with the underlying character glyph. The only solution / amelioration is to use yuqk's +rdn / anti-alias_min_depth=n option, AA jitter - or to render larger images which are later scaled down. This bias issue tangled in the hinting aspect of font files not being supported by the POV-Ray code. To a degree this position bias relative to incoming rays happens with objects in general, but it's much more noticeable with text{} objects. v) From work Christoph Lipka (clipka) did during v3.8 development that is still present in POV-Ray's v4.0 master branch and the yuqk v3.8 based fork, there exists a cmap{} sub feature for the text{} object which is only sort of implemented. In limited testing it sometimes works and sometimes not, but it could well be I'm not using it correctly too. The 'cmap{}' capability - or some derivative form of it - likely of use. Further work is required. The feature is not documented in the yuqk documentation. References: https://news.povray.org/povray.general/thread/%3C5c38071f%241%40news.povray.org%3E/ https://news.povray.org/povray.beta-test/thread/%3C5c323613%241%40news.povray.org%3E/ https://news.povray.org/povray.beta-test/thread/%3Cweb.5b1819544a827126535efa580%40news.povray.org%3E/ https://news.povray.org/povray.pov4.discussion.general/thread/%3C6427e66f%241%40news.povray.org%3E/ The yuqk fork state: -------------------- 'text' was modified. Where useable: -------------- Only within Scene Description Language (SDL) during parsing. // SPDX-License-Identifier: CC-BY-SA-4.0