Home | History | Annotate | Download | only in mac
      1 /*
      2  * Copyright (C) 2005, 2006, 2012 Apple Inc. All rights reserved.
      3  * Copyright (C) 2006 Alexey Proskuryakov
      4  *
      5  * Redistribution and use in source and binary forms, with or without
      6  * modification, are permitted provided that the following conditions
      7  * are met:
      8  * 1. Redistributions of source code must retain the above copyright
      9  *    notice, this list of conditions and the following disclaimer.
     10  * 2. Redistributions in binary form must reproduce the above copyright
     11  *    notice, this list of conditions and the following disclaimer in the
     12  *    documentation and/or other materials provided with the distribution.
     13  *
     14  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
     15  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
     16  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     17  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
     18  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     19  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     20  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     21  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     22  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     23  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
     24  * THE POSSIBILITY OF SUCH DAMAGE.
     25  */
     26 
     27 #include "config.h"
     28 #include "platform/fonts/SimpleFontData.h"
     29 
     30 #include "platform/fonts/Character.h"
     31 #include "platform/fonts/Font.h"
     32 #include "platform/fonts/GlyphPage.h"
     33 #include <ApplicationServices/ApplicationServices.h>
     34 
     35 // Forward declare Mac SPIs.
     36 // Request for public API: rdar://13787589
     37 extern "C" {
     38 void CGFontGetGlyphsForUnichars(CGFontRef, const UniChar chars[], CGGlyph glyphs[], size_t length);
     39 }
     40 
     41 namespace blink {
     42 
     43 CFDictionaryRef SimpleFontData::getCFStringAttributes(TypesettingFeatures typesettingFeatures, FontOrientation orientation) const
     44 {
     45     unsigned key = typesettingFeatures + 1;
     46     HashMap<unsigned, RetainPtr<CFDictionaryRef> >::AddResult addResult = m_CFStringAttributes.add(key, RetainPtr<CFDictionaryRef>());
     47     RetainPtr<CFDictionaryRef>& attributesDictionary = addResult.storedValue->value;
     48     if (!addResult.isNewEntry)
     49         return attributesDictionary.get();
     50 
     51     attributesDictionary.adoptCF(CFDictionaryCreateMutable(kCFAllocatorDefault, 4, &kCFCopyStringDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks));
     52     CFMutableDictionaryRef mutableAttributes = (CFMutableDictionaryRef)attributesDictionary.get();
     53 
     54     CFDictionarySetValue(mutableAttributes, kCTFontAttributeName, platformData().ctFont());
     55 
     56     if (!(typesettingFeatures & Kerning)) {
     57         const float zero = 0;
     58         static CFNumberRef zeroKerningValue = CFNumberCreate(kCFAllocatorDefault, kCFNumberFloatType, &zero);
     59         CFDictionarySetValue(mutableAttributes, kCTKernAttributeName, zeroKerningValue);
     60     }
     61 
     62     bool allowLigatures = (orientation == Horizontal && platformData().allowsLigatures()) || (typesettingFeatures & Ligatures);
     63     if (!allowLigatures) {
     64         const int zero = 0;
     65         static CFNumberRef essentialLigaturesValue = CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &zero);
     66         CFDictionarySetValue(mutableAttributes, kCTLigatureAttributeName, essentialLigaturesValue);
     67     }
     68 
     69     if (orientation == Vertical)
     70         CFDictionarySetValue(mutableAttributes, kCTVerticalFormsAttributeName, kCFBooleanTrue);
     71 
     72     return attributesDictionary.get();
     73 }
     74 
     75 static bool shouldUseCoreText(UChar* buffer, unsigned bufferLength, const SimpleFontData* fontData)
     76 {
     77     if (fontData->platformData().isCompositeFontReference())
     78         return true;
     79 
     80     // CoreText doesn't have vertical glyphs of surrogate pair characters.
     81     // Therefore, we should not use CoreText, but this always returns horizontal glyphs.
     82     // FIXME: We should use vertical glyphs. https://code.google.com/p/chromium/issues/detail?id=340173
     83     if (bufferLength >= 2 && U_IS_SURROGATE(buffer[0]) && fontData->hasVerticalGlyphs()) {
     84         ASSERT(U_IS_SURROGATE_LEAD(buffer[0]));
     85         ASSERT(U_IS_TRAIL(buffer[1]));
     86         return false;
     87     }
     88 
     89     if (fontData->platformData().widthVariant() != RegularWidth || fontData->hasVerticalGlyphs()) {
     90         // Ideographs don't have a vertical variant or width variants.
     91         for (unsigned i = 0; i < bufferLength; ++i) {
     92             if (!Character::isCJKIdeograph(buffer[i]))
     93                 return true;
     94         }
     95     }
     96 
     97     return false;
     98 }
     99 
    100 bool SimpleFontData::fillGlyphPage(GlyphPage* pageToFill, unsigned offset, unsigned length, UChar* buffer, unsigned bufferLength) const
    101 {
    102     bool haveGlyphs = false;
    103 
    104     Vector<CGGlyph, 512> glyphs(bufferLength);
    105     if (!shouldUseCoreText(buffer, bufferLength, this)) {
    106         CGFontGetGlyphsForUnichars(platformData().cgFont(), buffer, glyphs.data(), bufferLength);
    107         for (unsigned i = 0; i < length; ++i) {
    108             if (glyphs[i]) {
    109                 pageToFill->setGlyphDataForIndex(offset + i, glyphs[i], this);
    110                 haveGlyphs = true;
    111             }
    112         }
    113     } else if (!platformData().isCompositeFontReference() && platformData().widthVariant() != RegularWidth
    114         && CTFontGetGlyphsForCharacters(platformData().ctFont(), buffer, glyphs.data(), bufferLength)) {
    115         // When buffer consists of surrogate pairs, CTFontGetGlyphsForCharacters
    116         // places the glyphs at indices corresponding to the first character of each pair.
    117         unsigned glyphStep = bufferLength / length;
    118         for (unsigned i = 0; i < length; ++i) {
    119             if (glyphs[i * glyphStep]) {
    120                 pageToFill->setGlyphDataForIndex(offset + i, glyphs[i * glyphStep], this);
    121                 haveGlyphs = true;
    122             }
    123         }
    124     } else {
    125         // We ask CoreText for possible vertical variant glyphs
    126         RetainPtr<CFStringRef> string(AdoptCF, CFStringCreateWithCharactersNoCopy(kCFAllocatorDefault, buffer, bufferLength, kCFAllocatorNull));
    127         RetainPtr<CFAttributedStringRef> attributedString(AdoptCF, CFAttributedStringCreate(kCFAllocatorDefault, string.get(), getCFStringAttributes(0, hasVerticalGlyphs() ? Vertical : Horizontal)));
    128         RetainPtr<CTLineRef> line(AdoptCF, CTLineCreateWithAttributedString(attributedString.get()));
    129 
    130         CFArrayRef runArray = CTLineGetGlyphRuns(line.get());
    131         CFIndex runCount = CFArrayGetCount(runArray);
    132 
    133         Vector<CGGlyph, 512> glyphVector;
    134         Vector<CFIndex, 512> indexVector;
    135         bool done = false;
    136 
    137         // For the CGFont comparison in the loop, use the CGFont that Core Text assigns to the CTFont. This may
    138         // be non-CFEqual to platformData().cgFont().
    139         RetainPtr<CGFontRef> cgFont(AdoptCF, CTFontCopyGraphicsFont(platformData().ctFont(), 0));
    140 
    141         for (CFIndex r = 0; r < runCount && !done ; ++r) {
    142             // CTLine could map characters over multiple fonts using its own font fallback list.
    143             // We need to pick runs that use the exact font we need, i.e., platformData().ctFont().
    144             CTRunRef ctRun = static_cast<CTRunRef>(CFArrayGetValueAtIndex(runArray, r));
    145             ASSERT(CFGetTypeID(ctRun) == CTRunGetTypeID());
    146 
    147             CFDictionaryRef attributes = CTRunGetAttributes(ctRun);
    148             CTFontRef runFont = static_cast<CTFontRef>(CFDictionaryGetValue(attributes, kCTFontAttributeName));
    149             RetainPtr<CGFontRef> runCGFont(AdoptCF, CTFontCopyGraphicsFont(runFont, 0));
    150             // Use CGFont here as CFEqual for CTFont counts all attributes for font.
    151             bool gotBaseFont = CFEqual(cgFont.get(), runCGFont.get());
    152             if (gotBaseFont || platformData().isCompositeFontReference()) {
    153                 // This run uses the font we want. Extract glyphs.
    154                 CFIndex glyphCount = CTRunGetGlyphCount(ctRun);
    155                 const CGGlyph* glyphs = CTRunGetGlyphsPtr(ctRun);
    156                 if (!glyphs) {
    157                     glyphVector.resize(glyphCount);
    158                     CTRunGetGlyphs(ctRun, CFRangeMake(0, 0), glyphVector.data());
    159                     glyphs = glyphVector.data();
    160                 }
    161                 const CFIndex* stringIndices = CTRunGetStringIndicesPtr(ctRun);
    162                 if (!stringIndices) {
    163                     indexVector.resize(glyphCount);
    164                     CTRunGetStringIndices(ctRun, CFRangeMake(0, 0), indexVector.data());
    165                     stringIndices = indexVector.data();
    166                 }
    167 
    168                 if (gotBaseFont) {
    169                     for (CFIndex i = 0; i < glyphCount; ++i) {
    170                         if (stringIndices[i] >= static_cast<CFIndex>(length)) {
    171                             done = true;
    172                             break;
    173                         }
    174                         if (glyphs[i]) {
    175                             pageToFill->setGlyphDataForIndex(offset + stringIndices[i], glyphs[i], this);
    176                             haveGlyphs = true;
    177                         }
    178                     }
    179                 } else {
    180                     const SimpleFontData* runSimple = getCompositeFontReferenceFontData((NSFont *)runFont);
    181                     if (runSimple) {
    182                         for (CFIndex i = 0; i < glyphCount; ++i) {
    183                             if (stringIndices[i] >= static_cast<CFIndex>(length)) {
    184                                 done = true;
    185                                 break;
    186                             }
    187                             if (glyphs[i]) {
    188                                 pageToFill->setGlyphDataForIndex(offset + stringIndices[i], glyphs[i], runSimple);
    189                                 haveGlyphs = true;
    190                             }
    191                         }
    192                     }
    193                 }
    194             }
    195         }
    196     }
    197 
    198     return haveGlyphs;
    199 }
    200 
    201 } // namespace blink
    202