From 5257752b44f942085afed70764a18b21bb72baba Mon Sep 17 00:00:00 2001 From: Chris Loer Date: Fri, 1 Dec 2017 11:53:43 -0800 Subject: Add font family configuration option for LocalGlyphRasterizer Add support for "bold" font stacks Somewhat rationalize class names, add comments. --- .../mapboxsdk/text/LocalGlyphRasterizer.java | 4 +- platform/android/src/jni.cpp | 2 +- .../android/src/text/local_glyph_rasterizer.cpp | 82 ++++++++++++++++------ .../src/text/local_glyph_rasterizer_impl.hpp | 14 +++- 4 files changed, 75 insertions(+), 27 deletions(-) diff --git a/platform/android/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxsdk/text/LocalGlyphRasterizer.java b/platform/android/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxsdk/text/LocalGlyphRasterizer.java index a7a2c5216e..6e0e7daaa8 100644 --- a/platform/android/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxsdk/text/LocalGlyphRasterizer.java +++ b/platform/android/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxsdk/text/LocalGlyphRasterizer.java @@ -7,13 +7,13 @@ import android.graphics.Typeface; public class LocalGlyphRasterizer { - public static Bitmap drawGlyphBitmap(char glyphID) { + public static Bitmap drawGlyphBitmap(String fontFamily, boolean bold, char glyphID) { Bitmap bitmap = Bitmap.createBitmap(35, 35, Bitmap.Config.ARGB_8888); Paint paint = new Paint(); paint.setAntiAlias(true); paint.setTextSize(24); - paint.setTypeface(Typeface.create("Noto Sans", Typeface.NORMAL)); + paint.setTypeface(Typeface.create(fontFamily, bold ? Typeface.BOLD : Typeface.NORMAL)); Canvas canvas = new Canvas(); canvas.setBitmap(bitmap); diff --git a/platform/android/src/jni.cpp b/platform/android/src/jni.cpp index 354c7594c2..b18f471a37 100755 --- a/platform/android/src/jni.cpp +++ b/platform/android/src/jni.cpp @@ -191,7 +191,7 @@ void registerNatives(JavaVM *vm) { MapSnapshot::registerNative(env); // text - AndroidLocalGlyphRasterizer::registerNative(env); + LocalGlyphRasterizer::registerNative(env); } } // namespace android diff --git a/platform/android/src/text/local_glyph_rasterizer.cpp b/platform/android/src/text/local_glyph_rasterizer.cpp index dd58ee2228..4845234944 100644 --- a/platform/android/src/text/local_glyph_rasterizer.cpp +++ b/platform/android/src/text/local_glyph_rasterizer.cpp @@ -1,62 +1,102 @@ #include #include +#include #include #include "../attach_env.hpp" #include "../bitmap.hpp" -#include "local_glyph_rasterizer_impl.hpp" // Actually AndroidLocalGlyphRasterizer +#include "local_glyph_rasterizer_impl.hpp" + +/* + Android implementation of LocalGlyphRasterizer: + Draws CJK glyphs using locally available fonts. + + Follows pattern of GL JS implementation in that: + - Only CJK glyphs are drawn locally (because we can guess their metrics effectively) + * Render size/metrics determined experimentally using Noto Sans + - Configuration is done at map creation time by setting a "font family" + * JS uses a CSS font-family, this uses android.graphics.Typeface + https://developer.android.com/reference/android/graphics/Typeface.html + - We use heuristics to extract a font-weight based on the incoming font stack + * JS tries to extract multiple weights, this implementation only looks for + "bold" + + mbgl::LocalGlyphRasterizer is the portable interface + mbgl::LocalGlyphRasterizer::Impl stores platform-specific configuration data + mbgl::android::LocalGlyphRasterizer is the JNI wrapper + com.mapbox.mapboxsdk.text.LocalGlyphRasterizer is the Java implementation that + actually does the drawing + */ namespace mbgl { - namespace android { -PremultipliedImage AndroidLocalGlyphRasterizer::drawGlyphBitmap(const FontStack&, unsigned short glyphID, Size) { - UniqueEnv env { AttachEnv() }; // TODO: How should this be hooked up? +PremultipliedImage LocalGlyphRasterizer::drawGlyphBitmap(const std::string& fontFamily, const bool bold, const GlyphID glyphID) { + UniqueEnv env { AttachEnv() }; - // TODO: Pass in font stack and size (and configuration) - // For now, just try to hard-wire any rendering at all - using Signature = jni::Object(jni::jchar); + using Signature = jni::Object(jni::String, jni::jboolean, jni::jchar); auto method = javaClass.GetStaticMethod(*env, "drawGlyphBitmap"); - auto result = javaClass.Call(*env, method, glyphID); + + jni::String jniFontFamily = jni::Make(*env, fontFamily); + + auto result = javaClass.Call(*env, + method, + jniFontFamily, + static_cast(bold), + static_cast(glyphID)); return Bitmap::GetImage(*env, result); } -void AndroidLocalGlyphRasterizer::registerNative(jni::JNIEnv& env) { - javaClass = *jni::Class::Find(env).NewGlobalRef(env).release(); +void LocalGlyphRasterizer::registerNative(jni::JNIEnv& env) { + javaClass = *jni::Class::Find(env).NewGlobalRef(env).release(); } -jni::Class AndroidLocalGlyphRasterizer::javaClass; +jni::Class LocalGlyphRasterizer::javaClass; } // namespace android class LocalGlyphRasterizer::Impl { public: - bool hasFont(const FontStack&) const { - return true; + Impl(const optional fontFamily_) + : fontFamily(fontFamily_) + {} + + bool isConfigured() const { + return bool(fontFamily); } - PremultipliedImage drawGlyphBitmap(const FontStack& fontStack, GlyphID glyphID, Size size) { - return android::AndroidLocalGlyphRasterizer::drawGlyphBitmap(fontStack, (unsigned short)glyphID, size); + PremultipliedImage drawGlyphBitmap(const FontStack& fontStack, GlyphID glyphID) { + bool bold = false; + for (auto font : fontStack) { + std::string lowercaseFont = platform::lowercase(font); + if (lowercaseFont.find("bold") != std::string::npos) { + bold = true; + } + } + return android::LocalGlyphRasterizer::drawGlyphBitmap(*fontFamily, bold, glyphID); } + +private: + optional fontFamily; }; -LocalGlyphRasterizer::LocalGlyphRasterizer(const optional) - : impl(std::make_unique()) +LocalGlyphRasterizer::LocalGlyphRasterizer(const optional fontFamily) + : impl(std::make_unique(fontFamily)) {} LocalGlyphRasterizer::~LocalGlyphRasterizer() {} -bool LocalGlyphRasterizer::canRasterizeGlyph(const FontStack& fontStack, GlyphID glyphID) { - return util::i18n::allowsFixedWidthGlyphGeneration(glyphID) && impl->hasFont(fontStack); +bool LocalGlyphRasterizer::canRasterizeGlyph(const FontStack&, GlyphID glyphID) { + return util::i18n::allowsFixedWidthGlyphGeneration(glyphID) && impl->isConfigured(); } Glyph LocalGlyphRasterizer::rasterizeGlyph(const FontStack& fontStack, GlyphID glyphID) { Glyph fixedMetrics; - if (!impl->hasFont(fontStack)) { + if (!impl->isConfigured()) { return fixedMetrics; } @@ -70,7 +110,7 @@ Glyph LocalGlyphRasterizer::rasterizeGlyph(const FontStack& fontStack, GlyphID g fixedMetrics.metrics.top = -10; fixedMetrics.metrics.advance = 24; - PremultipliedImage rgbaBitmap = impl->drawGlyphBitmap(fontStack, glyphID, size); + PremultipliedImage rgbaBitmap = impl->drawGlyphBitmap(fontStack, glyphID); // Copy alpha values from RGBA bitmap into the AlphaImage output fixedMetrics.bitmap = AlphaImage(size); diff --git a/platform/android/src/text/local_glyph_rasterizer_impl.hpp b/platform/android/src/text/local_glyph_rasterizer_impl.hpp index 2e14f87e34..38d98d5368 100644 --- a/platform/android/src/text/local_glyph_rasterizer_impl.hpp +++ b/platform/android/src/text/local_glyph_rasterizer_impl.hpp @@ -4,16 +4,24 @@ #include +/* + android::LocalGlyphRasterizer is the JNI wrapper of + com/mapbox/mapboxsdk/text/LocalGlyphRasterizer + + mbgl::LocalGlyphRasterizer is the portable interface + Both implementations are in local_glyph_rasterizer.cpp + */ + namespace mbgl { namespace android { -class AndroidLocalGlyphRasterizer { +class LocalGlyphRasterizer { public: - static PremultipliedImage drawGlyphBitmap(const FontStack&, unsigned short glyphID, Size size); + static PremultipliedImage drawGlyphBitmap(const std::string& fontFamily, const bool bold, const char16_t glyphID); static constexpr auto Name() { return "com/mapbox/mapboxsdk/text/LocalGlyphRasterizer"; }; - static jni::Class javaClass; + static jni::Class javaClass; static void registerNative(jni::JNIEnv&); -- cgit v1.2.1