/** * Copyright (c) 2006-2017 LOVE Development Team * * This software is provided 'as-is', without any express or implied * warranty. In no event will the authors be held liable for any damages * arising from the use of this software. * * Permission is granted to anyone to use this software for any purpose, * including commercial applications, and to alter it and redistribute it * freely, subject to the following restrictions: * * 1. The origin of this software must not be misrepresented; you must not * claim that you wrote the original software. If you use this software * in a product, an acknowledgment in the product documentation would be * appreciated but is not required. * 2. Altered source versions must be plainly marked as such, and must not be * misrepresented as being the original software. * 3. This notice may not be removed or altered from any source distribution. **/ // LOVE #include "ImageRasterizer.h" #include "common/Exception.h" #include namespace love { namespace font { static_assert(sizeof(Color) == 4, "sizeof(Color) must equal 4 bytes!"); ImageRasterizer::ImageRasterizer(love::image::ImageData *data, uint32 *glyphs, int numglyphs, int extraspacing, float pixeldensity) : imageData(data) , glyphs(glyphs) , numglyphs(numglyphs) , extraSpacing(extraspacing) { this->pixelDensity = pixeldensity; if (data->getFormat() != PIXELFORMAT_RGBA8) throw love::Exception("Only 32-bit RGBA images are supported in Image Fonts!"); load(); } ImageRasterizer::~ImageRasterizer() { } int ImageRasterizer::getLineHeight() const { return getHeight(); } GlyphData *ImageRasterizer::getGlyphData(uint32 glyph) const { GlyphMetrics gm = {}; // Set relevant glyph metrics if the glyph is in this ImageFont std::map::const_iterator it = imageGlyphs.find(glyph); if (it != imageGlyphs.end()) { gm.width = it->second.width; gm.advance = it->second.width + extraSpacing; } gm.height = metrics.height; GlyphData *g = new GlyphData(glyph, gm, PIXELFORMAT_RGBA8); if (gm.width == 0) return g; // We don't want another thread modifying our ImageData mid-copy. love::thread::Lock lock(imageData->getMutex()); Color *gdpixels = (Color *) g->getData(); const Color *imagepixels = (const Color *) imageData->getData(); // copy glyph pixels from imagedata to glyphdata for (int i = 0; i < g->getWidth() * g->getHeight(); i++) { Color p = imagepixels[it->second.x + (i % gm.width) + (imageData->getWidth() * (i / gm.width))]; // Use transparency instead of the spacer color if (p == spacer) gdpixels[i] = Color(0, 0, 0, 0); else gdpixels[i] = p; } return g; } void ImageRasterizer::load() { const Color *pixels = (const Color *) imageData->getData(); int imgw = imageData->getWidth(); int imgh = imageData->getHeight(); // We don't want another thread modifying our ImageData mid-parse. love::thread::Lock lock(imageData->getMutex()); // Set the only metric that matters metrics.height = imgh; // Reading texture data begins spacer = pixels[0]; int start = 0; int end = 0; for (int i = 0; i < numglyphs; ++i) { start = end; // Finds out where the first character starts while (start < imgw && pixels[start] == spacer) ++start; end = start; // Find where glyph ends. while (end < imgw && pixels[end] != spacer) ++end; if (start >= end) break; ImageGlyphData imageGlyph; imageGlyph.x = start; imageGlyph.width = end - start; imageGlyphs[glyphs[i]] = imageGlyph; } } int ImageRasterizer::getGlyphCount() const { return numglyphs; } bool ImageRasterizer::hasGlyph(uint32 glyph) const { return imageGlyphs.find(glyph) != imageGlyphs.end(); } Rasterizer::DataType ImageRasterizer::getDataType() const { return DATA_IMAGE; } } // font } // love