FontFamily.cpp revision 9cc9bbe1461f359f0b27c5e7645c17dda001ab1d
19cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien/*
29cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * Copyright (C) 2013 The Android Open Source Project
39cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien *
49cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * Licensed under the Apache License, Version 2.0 (the "License");
59cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * you may not use this file except in compliance with the License.
69cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * You may obtain a copy of the License at
79cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien *
89cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien *      http://www.apache.org/licenses/LICENSE-2.0
99cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien *
109cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * Unless required by applicable law or agreed to in writing, software
119cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * distributed under the License is distributed on an "AS IS" BASIS,
129cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * See the License for the specific language governing permissions and
149cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * limitations under the License.
159cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien */
169cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
179cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#define LOG_TAG "Minikin"
189cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
199cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include <cutils/log.h>
209cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include <stdlib.h>
219cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include <stdint.h>
229cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include <utils/UniquePtr.h>
239cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include <ft2build.h>
249cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include FT_FREETYPE_H
259cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include FT_TRUETYPE_TABLES_H
269cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include <minikin/AnalyzeStyle.h>
279cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include <minikin/FontFamily.h>
289cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
299cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levienusing std::vector;
309cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
319cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Leviennamespace android {
329cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
339cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levienbool FontFamily::addFont(FT_Face typeface) {
349cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    const uint32_t os2Tag = FT_MAKE_TAG('O', 'S', '/', '2');
359cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    FT_ULong os2Size = 0;
369cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    FT_Error error = FT_Load_Sfnt_Table(typeface, os2Tag, 0, NULL, &os2Size);
379cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    if (error != 0) return false;
389cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    UniquePtr<uint8_t[]> os2Data(new uint8_t[os2Size]);
399cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    error = FT_Load_Sfnt_Table(typeface, os2Tag, 0, os2Data.get(), &os2Size);
409cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    if (error != 0) return false;
419cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    int weight;
429cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    bool italic;
439cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    if (analyzeStyle(os2Data.get(), os2Size, &weight, &italic)) {
449cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        //ALOGD("analyzed weight = %d, italic = %s", weight, italic ? "true" : "false");
459cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        FontStyle style(weight, italic);
469cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        addFont(typeface, style);
479cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        return true;
489cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    } else {
499cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        ALOGD("failed to analyze style");
509cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    }
519cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    return false;
529cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}
539cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
549cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levienvoid FontFamily::addFont(FT_Face typeface, FontStyle style) {
559cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    mFonts.push_back(Font(typeface, style));
569cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    ALOGD("added font, mFonts.size() = %d", mFonts.size());
579cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}
589cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
599cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien// Compute a matching metric between two styles - 0 is an exact match
609cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levienint computeMatch(FontStyle style1, FontStyle style2) {
619cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    if (style1 == style2) return 0;
629cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    int score = abs(style1.getWeight() - style2.getWeight());
639cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    if (style1.getItalic() != style2.getItalic()) {
649cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        score += 2;
659cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    }
669cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    return score;
679cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}
689cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
699cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph LevienFT_Face FontFamily::getClosestMatch(FontStyle style) const {
709cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    const Font* bestFont = NULL;
719cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    int bestMatch = 0;
729cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    for (size_t i = 0; i < mFonts.size(); i++) {
739cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        const Font& font = mFonts[i];
749cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        int match = computeMatch(font.style, style);
759cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        if (i == 0 || match < bestMatch) {
769cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien            bestFont = &font;
779cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien            bestMatch = match;
789cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien        }
799cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    }
809cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    return bestFont == NULL ? NULL : bestFont->typeface;
819cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}
829cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
839cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Leviensize_t FontFamily::getNumFonts() const {
849cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    return mFonts.size();
859cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}
869cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
879cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph LevienFT_Face FontFamily::getFont(size_t index) const {
889cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    return mFonts[index].typeface;
899cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}
909cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
919cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph LevienFontStyle FontFamily::getStyle(size_t index) const {
929cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien    return mFonts[index].style;
939cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}
949cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien
959cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}  // namespace android
96