10 #include "../../stdafx.h"
12 #include "../../string_func.h"
13 #include "../../strings_func.h"
14 #include "../../table/control_codes.h"
15 #include "../../fontcache.h"
18 #include <CoreFoundation/CoreFoundation.h>
22 #ifndef HAVE_OSX_109_SDK
24 typedef const struct __CTRunDelegate * CTRunDelegateRef;
26 typedef void (*CTRunDelegateDeallocateCallback) (
void* refCon);
27 typedef CGFloat (*CTRunDelegateGetAscentCallback) (
void* refCon);
28 typedef CGFloat (*CTRunDelegateGetDescentCallback) (
void* refCon);
29 typedef CGFloat (*CTRunDelegateGetWidthCallback) (
void* refCon);
32 CTRunDelegateDeallocateCallback dealloc;
33 CTRunDelegateGetAscentCallback getAscent;
34 CTRunDelegateGetDescentCallback getDescent;
35 CTRunDelegateGetWidthCallback getWidth;
39 kCTRunDelegateVersion1 = 1,
40 kCTRunDelegateCurrentVersion = kCTRunDelegateVersion1
43 extern const CFStringRef kCTRunDelegateAttributeName AVAILABLE_MAC_OS_X_VERSION_10_5_AND_LATER;
45 CTRunDelegateRef CTRunDelegateCreate(
const CTRunDelegateCallbacks* callbacks,
void* refCon) AVAILABLE_MAC_OS_X_VERSION_10_5_AND_LATER;
72 std::vector<GlyphID> glyphs;
73 std::vector<float> positions;
74 std::vector<int> glyph_to_char;
76 int total_advance = 0;
83 const GlyphID *GetGlyphs()
const override {
return &this->glyphs[0]; }
84 const float *GetPositions()
const override {
return &this->positions[0]; }
85 const int *GetGlyphToCharMap()
const override {
return &this->glyph_to_char[0]; }
87 const Font *GetFont()
const override {
return this->font; }
88 int GetLeading()
const override {
return this->font->fc->GetHeight(); }
89 int GetGlyphCount()
const override {
return (
int)this->glyphs.size(); }
90 int GetAdvance()
const {
return this->total_advance; }
98 CFArrayRef runs = CTLineGetGlyphRuns(line.get());
99 for (CFIndex i = 0; i < CFArrayGetCount(runs); i++) {
100 CTRunRef run = (CTRunRef)CFArrayGetValueAtIndex(runs, i);
103 CFRange chars = CTRunGetStringRange(run);
104 auto map = fontMapping.begin();
105 while (map < fontMapping.end() - 1 && map->first <= chars.location) map++;
107 this->emplace_back(run, map->second, buff);
113 int CountRuns()
const override {
return this->size(); }
114 const VisualRun &GetVisualRun(
int run)
const override {
return this->at(run); }
116 int GetInternalCharLength(
WChar c)
const override
119 return c >= 0x010000U ? 2 : 1;
128 void Reflow()
override
130 this->cur_offset = 0;
133 std::unique_ptr<const Line> NextLine(
int max_width)
override;
141 WChar c = (
WChar)((
size_t)ref_con & 0xFFFFFF);
147 kCTRunDelegateCurrentVersion,
nullptr,
nullptr,
nullptr,
156 ptrdiff_t length = buff_end - buff;
157 if (length == 0)
return nullptr;
160 for (
const auto &i : fontMapping) {
161 if (i.second->fc->IsBuiltInFont())
return nullptr;
166 CFAttributedStringBeginEditing(str.get());
169 CFAttributedStringReplaceString(str.get(), CFRangeMake(0, 0), base.get());
174 for (
const auto &i : fontMapping) {
175 if (i.first - last == 0)
continue;
179 CFAutoRelease<CFStringRef> font_name(CFStringCreateWithCString(kCFAllocatorDefault, i.second->fc->GetFontName(), kCFStringEncodingUTF8));
180 _font_cache[i.second->fc->GetSize()].reset(CTFontCreateWithName(font_name.get(), i.second->fc->GetFontSize(),
nullptr));
182 CFAttributedStringSetAttribute(str.get(), CFRangeMake(last, i.first - last), kCTFontAttributeName,
_font_cache[i.second->fc->GetSize()].get());
184 CGColorRef color = CGColorCreateGenericGray((uint8)i.second->colour / 255.0f, 1.0f);
185 CFAttributedStringSetAttribute(str.get(), CFRangeMake(last, i.first - last), kCTForegroundColorAttributeName, color);
186 CGColorRelease(color);
189 for (ssize_t c = last; c < i.first; c++) {
190 if (buff[c] >= SCC_SPRITE_START && buff[c] <= SCC_SPRITE_END) {
192 CFAttributedStringSetAttribute(str.get(), CFRangeMake(c, 1), kCTRunDelegateAttributeName, del.get());
198 CFAttributedStringEndEditing(str.get());
206 std::unique_ptr<const ParagraphLayouter::Line> CoreTextParagraphLayout::NextLine(
int max_width)
208 if (this->
cur_offset >= this->length)
return nullptr;
211 CFIndex len = CTTypesetterSuggestLineBreak(this->typesetter.get(), this->cur_offset, max_width);
212 if (len <= 0) len = CTTypesetterSuggestClusterBreak(this->typesetter.get(), this->cur_offset, max_width);
218 return std::unique_ptr<const Line>(line ?
new CoreTextLine(std::move(line), this->font_map, this->text_buffer) :
nullptr);
223 this->glyphs.resize(CTRunGetGlyphCount(run));
226 CFIndex map[this->glyphs.size()];
227 CTRunGetStringIndices(run, CFRangeMake(0, 0), map);
229 this->glyph_to_char.resize(this->glyphs.size());
230 for (
size_t i = 0; i < this->glyph_to_char.size(); i++) this->glyph_to_char[i] = (
int)map[i];
232 CGPoint pts[this->glyphs.size()];
233 CTRunGetPositions(run, CFRangeMake(0, 0), pts);
234 this->positions.resize(this->glyphs.size() * 2 + 2);
238 CGGlyph gl[this->glyphs.size()];
239 CTRunGetGlyphs(run, CFRangeMake(0, 0), gl);
240 for (
size_t i = 0; i < this->glyphs.size(); i++) {
241 if (buff[this->glyph_to_char[i]] >= SCC_SPRITE_START && buff[this->glyph_to_char[i]] <= SCC_SPRITE_END) {
242 this->glyphs[i] = font->fc->MapCharToGlyph(buff[this->glyph_to_char[i]]);
243 this->positions[i * 2 + 0] = pts[i].x;
244 this->positions[i * 2 + 1] = font->fc->GetAscender() - font->fc->GetGlyph(this->glyphs[i])->height - 1;
246 this->glyphs[i] = gl[i];
247 this->positions[i * 2 + 0] = pts[i].x;
248 this->positions[i * 2 + 1] = pts[i].y;
251 this->total_advance = (int)CTRunGetTypographicBounds(run, CFRangeMake(0, 0),
nullptr,
nullptr,
nullptr);
252 this->positions[this->glyphs.size() * 2] = this->positions[0] + this->total_advance;
262 for (
const auto &run : *
this) {
263 leading = std::max(leading, run.GetLeading());
275 if (this->size() == 0)
return 0;
278 for (
const auto &run : *
this) {
279 total_width += run.GetAdvance();
298 CFAutoRelease<CFURLRef> url(CFURLCreateWithFileSystemPath(kCFAllocatorDefault, path.get(), kCFURLPOSIXPathStyle,
false));
300 CTFontManagerRegisterFontsForURL(url.get(), kCTFontManagerScopeProcess,
nullptr);
309 _osx_locale.reset(CFLocaleCreate(kCFAllocatorDefault, iso.get()));
322 if (!supported)
return 0;
324 CFStringCompareFlags flags = kCFCompareCaseInsensitive | kCFCompareNumerically | kCFCompareLocalized | kCFCompareWidthInsensitive | kCFCompareForcedOrdering;
330 if (cf1 ==
nullptr || cf2 ==
nullptr)
return 0;
332 return (
int)CFStringCompareWithOptionsAndLocale(cf1.get(), cf2.get(), CFRangeMake(0, CFStringGetLength(cf1.get())), flags,
_osx_locale.get()) + 2;
338 const char *string_base = s;
340 this->utf16_to_utf8.clear();
341 this->str_info.clear();
346 std::vector<UniChar> utf16_str;
348 size_t idx = s - string_base;
350 WChar c = Utf8Consume(&s);
352 utf16_str.push_back((UniChar)c);
355 utf16_str.push_back((UniChar)(0xD800 + ((c - 0x10000) >> 10)));
356 utf16_str.push_back((UniChar)(0xDC00 + ((c - 0x10000) & 0x3FF)));
357 this->utf16_to_utf8.push_back(idx);
359 this->utf16_to_utf8.push_back(idx);
361 this->utf16_to_utf8.push_back(s - string_base);
364 this->str_info.resize(utf16_to_utf8.size());
366 if (utf16_str.size() > 0) {
367 CFAutoRelease<CFStringRef> str(CFStringCreateWithCharactersNoCopy(kCFAllocatorDefault, &utf16_str[0], utf16_str.size(), kCFAllocatorNull));
370 for (CFIndex i = 0; i < CFStringGetLength(str.get()); ) {
371 CFRange r = CFStringGetRangeOfComposedCharactersAtIndex(str.get(), i);
372 this->str_info[r.location].char_stop =
true;
380 CFStringTokenizerTokenType tokenType = kCFStringTokenizerTokenNone;
381 while ((tokenType = CFStringTokenizerAdvanceToNextToken(tokenizer.get())) != kCFStringTokenizerTokenNone) {
383 if ((tokenType & kCFStringTokenizerTokenHasNonLettersMask) != kCFStringTokenizerTokenHasNonLettersMask) {
384 CFRange r = CFStringTokenizerGetCurrentTokenRange(tokenizer.get());
385 this->str_info[r.location].word_stop =
true;
391 this->str_info.back().char_stop =
true;
392 this->str_info.back().word_stop =
true;
398 size_t utf16_pos = 0;
399 for (
size_t i = 0; i < this->utf16_to_utf8.size(); i++) {
400 if (this->utf16_to_utf8[i] == pos) {
407 while (utf16_pos > 0 && !this->str_info[utf16_pos].char_stop) utf16_pos--;
408 this->cur_pos = utf16_pos;
410 return this->utf16_to_utf8[this->cur_pos];
415 assert(this->cur_pos <= this->utf16_to_utf8.size());
418 if (this->cur_pos == this->utf16_to_utf8.size())
return END;
422 }
while (this->cur_pos < this->utf16_to_utf8.size() && (what == ITER_WORD ? !this->str_info[this->cur_pos].word_stop : !this->str_info[this->cur_pos].char_stop));
424 return this->cur_pos == this->utf16_to_utf8.size() ? END : this->utf16_to_utf8[this->cur_pos];
429 assert(this->cur_pos <= this->utf16_to_utf8.size());
432 if (this->cur_pos == 0)
return END;
436 }
while (this->cur_pos > 0 && (what == ITER_WORD ? !this->str_info[this->cur_pos].word_stop : !this->str_info[this->cur_pos].char_stop));
438 return this->utf16_to_utf8[this->cur_pos];