#region MIT License /*Copyright (c) 2012 Robert Rouhani SharpFont based on Tao.FreeType, Copyright (c) 2003-2007 Tao Framework Team Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.*/ #endregion using System; using System.Collections.Generic; using SharpFont.Cache; using SharpFont.Internal; using SharpFont.TrueType; namespace SharpFont { /// /// A handle to a FreeType library instance. Each ‘library’ is completely independent from the others; it is the /// ‘root’ of a set of objects like fonts, faces, sizes, etc. /// /// It also embeds a memory manager (see ), as well as a scan-line converter object (see /// ). /// /// For multi-threading applications each thread should have its own object. /// public sealed class Library : IDisposable { #region Fields private IntPtr reference; private bool customMemory; private bool disposed; private List childFaces; private List childGlyphs; private List childOutlines; private List childStrokers; private List childManagers; #endregion #region Constructors /// /// Initializes a new instance of the class. /// public Library() : this(false) { IntPtr libraryRef; Error err = FT.FT_Init_FreeType(out libraryRef); if (err != Error.Ok) throw new FreeTypeException(err); Reference = libraryRef; } /// /// Initializes a new instance of the class. /// /// A custom FreeType memory manager. public Library(Memory memory) : this(false) { IntPtr libraryRef; Error err = FT.FT_New_Library(memory.Reference, out libraryRef); if (err != Error.Ok) throw new FreeTypeException(err); Reference = libraryRef; customMemory = true; } private Library(bool duplicate) { childFaces = new List(); childGlyphs = new List(); childOutlines = new List(); childStrokers = new List(); childManagers = new List(); } /// /// Finalizes an instance of the class. /// ~Library() { Dispose(false); } #endregion #region Properties /// /// Gets a value indicating whether the object has been disposed. /// public bool IsDisposed { get { return disposed; } } /// /// Gets the version of the FreeType library being used. /// public Version Version { get { if (disposed) throw new ObjectDisposedException("Version", "Cannot access a disposed object."); int major, minor, patch; FT.FT_Library_Version(Reference, out major, out minor, out patch); return new Version(major, minor, patch); } } internal IntPtr Reference { get { if (disposed) throw new ObjectDisposedException("Reference", "Cannot access a disposed object."); return reference; } set { if (disposed) throw new ObjectDisposedException("Reference", "Cannot access a disposed object."); reference = value; } } #endregion #region Methods #region Base Interface /// /// This function calls to open a font by its pathname. /// /// A path to the font file. /// The index of the face within the font. The first face has index 0. /// /// A handle to a new face object. If ‘faceIndex’ is greater than or equal to zero, it must be non-NULL. /// /// public Face NewFace(string path, int faceIndex) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); return new Face(this, path, faceIndex); } /// /// This function calls to open a font which has been loaded into memory. /// /// /// You must not deallocate the memory before calling . /// /// A pointer to the beginning of the font data. /// The index of the face within the font. The first face has index 0. /// /// A handle to a new face object. If ‘faceIndex’ is greater than or equal to zero, it must be non-NULL. /// /// public Face NewMemoryFace(byte[] file, int faceIndex) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); return new Face(this, file, faceIndex); } /// /// Create a object from a given resource described by . /// /// /// Unlike FreeType 1.x, this function automatically creates a glyph slot for the face object which can be /// accessed directly through . /// /// OpenFace can be used to quickly check whether the font format of a given font resource is supported by /// FreeType. If the ‘faceIndex’ field is negative, the function's return value is 0 if the font format is /// recognized, or non-zero otherwise; the function returns a more or less empty face handle in ‘*aface’ (if /// ‘aface’ isn't NULL). The only useful field in this special case is which gives /// the number of faces within the font file. After examination, the returned structure /// should be deallocated with a call to . /// /// Each new face object created with this function also owns a default object, accessible /// as . /// /// See the discussion of reference counters in the description of FT_Reference_Face. /// /// /// A pointer to an structure which must be filled by the caller. /// /// The index of the face within the font. The first face has index 0. /// /// A handle to a new face object. If ‘faceIndex’ is greater than or equal to zero, it must be non-NULL. /// public Face OpenFace(OpenArgs args, int faceIndex) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); IntPtr faceRef; Error err = FT.FT_Open_Face(Reference, args.Reference, faceIndex, out faceRef); if (err != Error.Ok) throw new FreeTypeException(err); return new Face(faceRef, this); } #endregion #region Mac Specific Interface /// /// Create a new face object from a FOND resource. /// /// /// This function can be used to create objects from fonts that are installed in the system /// as follows. /// /// fond = GetResource( 'FOND', fontName ); /// error = FT_New_Face_From_FOND( library, fond, 0, &face ); /// /// /// A FOND resource. /// Only supported for the -1 ‘sanity check’ special case. /// A handle to a new face object. public Face NewFaceFromFond(IntPtr fond, int faceIndex) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); IntPtr faceRef; Error err = FT.FT_New_Face_From_FOND(Reference, fond, faceIndex, out faceRef); if (err != Error.Ok) throw new FreeTypeException(err); return new Face(faceRef, this); } /// /// Create a new face object from a given resource and typeface index using an FSSpec to the font file. /// /// /// is identical to except it accepts an FSSpec instead /// of a path. /// /// FSSpec to the font file. /// The index of the face within the resource. The first face has index 0. /// A handle to a new face object. public Face NewFaceFromFSSpec(IntPtr spec, int faceIndex) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); IntPtr faceRef; Error err = FT.FT_New_Face_From_FSSpec(Reference, spec, faceIndex, out faceRef); if (err != Error.Ok) throw new FreeTypeException(err); return new Face(faceRef, this); } /// /// Create a new face object from a given resource and typeface index using an FSRef to the font file. /// /// /// is identical to except it accepts an FSRef instead of /// a path. /// /// FSRef to the font file. /// The index of the face within the resource. The first face has index 0. /// A handle to a new face object. public Face NewFaceFromFSRef(IntPtr @ref, int faceIndex) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); IntPtr faceRef; Error err = FT.FT_New_Face_From_FSRef(Reference, @ref, faceIndex, out faceRef); if (err != Error.Ok) throw new FreeTypeException(err); return new Face(faceRef, this); } #endregion #region Module Management /// /// Add a new module to a given library instance. /// /// /// An error will be returned if a module already exists by that name, or if the module requires a version of /// FreeType that is too great. /// /// A pointer to class descriptor for the module. public void AddModule(ModuleClass clazz) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); Error err = FT.FT_Add_Module(Reference, clazz.Reference); if (err != Error.Ok) throw new FreeTypeException(err); } /// /// Find a module by its name. /// /// /// FreeType's internal modules aren't documented very well, and you should look up the source code for /// details. /// /// The module's name (as an ASCII string). /// A module handle. 0 if none was found. public Module GetModule(string moduleName) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); return new Module(FT.FT_Get_Module(Reference, moduleName)); } /// /// Remove a given module from a library instance. /// /// /// The module object is destroyed by the function in case of success. /// /// A handle to a module object. public void RemoveModule(Module module) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); if (module == null) throw new ArgumentNullException("module"); Error err = FT.FT_Remove_Module(Reference, module.Reference); if (err != Error.Ok) throw new FreeTypeException(err); } /// /// Set a debug hook function for debugging the interpreter of a font format. /// /// /// Currently, four debug hook slots are available, but only two (for the TrueType and the Type 1 interpreter) /// are defined. /// /// Since the internal headers of FreeType are no longer installed, the symbol ‘FT_DEBUG_HOOK_TRUETYPE’ isn't /// available publicly. This is a bug and will be fixed in a forthcoming release. /// /// The index of the debug hook. You should use the values defined in ‘ftobjs.h’, e.g., ‘FT_DEBUG_HOOK_TRUETYPE’. /// The function used to debug the interpreter. [CLSCompliant(false)] public void SetDebugHook(uint hookIndex, IntPtr debugHook) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); FT.FT_Set_Debug_Hook(Reference, hookIndex, debugHook); } /// /// Add the set of default drivers to a given library object. This is only useful when you create a library /// object with (usually to plug a custom memory manager). /// public void AddDefaultModules() { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); FT.FT_Add_Default_Modules(Reference); } /// /// Retrieve the current renderer for a given glyph format. /// /// /// An error will be returned if a module already exists by that name, or if the module requires a version of /// FreeType that is too great. /// /// To add a new renderer, simply use . To retrieve a renderer by its name, use /// . /// /// The glyph format. /// A renderer handle. 0 if none found. [CLSCompliant(false)] public Renderer GetRenderer(GlyphFormat format) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); return new Renderer(FT.FT_Get_Renderer(Reference, format)); } /// /// Set the current renderer to use, and set additional mode. /// /// /// In case of success, the renderer will be used to convert glyph images in the renderer's known format into /// bitmaps. /// /// This doesn't change the current renderer for other formats. /// /// Currently, only the B/W renderer, if compiled with FT_RASTER_OPTION_ANTI_ALIASING (providing a 5-levels /// anti-aliasing mode; this option must be set directly in ‘ftraster.c’ and is undefined by default) accepts a /// single tag ‘pal5’ to set its gray palette as a character string with 5 elements. Consequently, the third /// and fourth argument are zero normally. /// /// A handle to the renderer object. /// The number of additional parameters. /// Additional parameters. [CLSCompliant(false)] public unsafe void SetRenderer(Renderer renderer, uint numParams, Parameter[] parameters) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); if (renderer == null) throw new ArgumentNullException("renderer"); if (parameters == null) throw new ArgumentNullException("parameters"); ParameterRec[] paramRecs = Array.ConvertAll(parameters, p => p.Record); fixed (void* ptr = paramRecs) { Error err = FT.FT_Set_Renderer(Reference, renderer.Reference, numParams, (IntPtr)ptr); if (err != Error.Ok) throw new FreeTypeException(err); } } #endregion #region LCD Filtering /// /// This function is used to apply color filtering to LCD decimated bitmaps, like the ones used when calling /// with or /// . /// /// /// This feature is always disabled by default. Clients must make an explicit call to this function with a /// ‘filter’ value other than in order to enable it. /// /// Due to PATENTS covering subpixel rendering, this function doesn't do anything except returning /// if the configuration macro FT_CONFIG_OPTION_SUBPIXEL_RENDERING is /// not defined in your build of the library, which should correspond to all default builds of FreeType. /// /// The filter affects glyph bitmaps rendered through , /// , , and . /// /// It does not affect the output of and /// . /// /// If this feature is activated, the dimensions of LCD glyph bitmaps are either larger or taller than the /// dimensions of the corresponding outline with regards to the pixel grid. For example, for /// , the filter adds up to 3 pixels to the left, and up to 3 pixels to the right. /// /// The bitmap offset values are adjusted correctly, so clients shouldn't need to modify their layout and glyph /// positioning code when enabling the filter. /// /// /// The filter type. /// /// You can use here to disable this feature, or /// to use a default filter that should work well on most LCD screens. /// public void SetLcdFilter(LcdFilter filter) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); Error err = FT.FT_Library_SetLcdFilter(Reference, filter); if (err != Error.Ok) throw new FreeTypeException(err); } /// /// Use this function to override the filter weights selected by . By default, /// FreeType uses the quintuple (0x00, 0x55, 0x56, 0x55, 0x00) for , and (0x10, /// 0x40, 0x70, 0x40, 0x10) for and . /// /// /// Due to PATENTS covering subpixel rendering, this function doesn't do anything except returning /// if the configuration macro FT_CONFIG_OPTION_SUBPIXEL_RENDERING is /// not defined in your build of the library, which should correspond to all default builds of FreeType. /// /// This function must be called after to have any effect. /// /// /// A pointer to an array; the function copies the first five bytes and uses them to specify the filter /// weights. /// public void SetLcdFilterWeights(byte[] weights) { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); if (weights == null) throw new ArgumentNullException("weights"); Error err = FT.FT_Library_SetLcdFilterWeights(Reference, weights); if (err != Error.Ok) throw new FreeTypeException(err); } #endregion #region The TrueType Engine /// /// Return an value to indicate which level of the TrueType virtual machine a given /// library instance supports. /// /// A value indicating which level is supported. public EngineType GetTrueTypeEngineType() { if (disposed) throw new ObjectDisposedException("Library", "Cannot access a disposed object."); return FT.FT_Get_TrueType_Engine_Type(Reference); } #endregion /// /// Disposes the Library. /// public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } internal void AddChildFace(Face child) { childFaces.Add(child); } internal void RemoveChildFace(Face child) { childFaces.Remove(child); } internal void AddChildGlyph(Glyph child) { childGlyphs.Add(child); } internal void RemoveChildGlyph(Glyph child) { childGlyphs.Remove(child); } internal void AddChildOutline(Outline child) { childOutlines.Add(child); } internal void RemoveChildOutline(Outline child) { childOutlines.Remove(child); } internal void AddChildStroker(Stroker child) { childStrokers.Add(child); } internal void RemoveChildStroker(Stroker child) { childStrokers.Remove(child); } internal void AddChildManager(Manager child) { childManagers.Add(child); } internal void RemoveChildManager(Manager child) { childManagers.Remove(child); } private void Dispose(bool disposing) { if (!disposed) { disposed = true; //dipose all the children before disposing the library. foreach (Face f in childFaces) f.Dispose(); foreach (Glyph g in childGlyphs) g.Dispose(); foreach (Outline o in childOutlines) o.Dispose(); foreach (Stroker s in childStrokers) s.Dispose(); foreach (Manager m in childManagers) m.Dispose(); childFaces.Clear(); childGlyphs.Clear(); childOutlines.Clear(); childStrokers.Clear(); childManagers.Clear(); Error err = customMemory ? FT.FT_Done_Library(reference) : FT.FT_Done_FreeType(reference); if (err != Error.Ok) throw new FreeTypeException(err); reference = IntPtr.Zero; } } #endregion } }