Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "ui/gfx/render_text_win.h" | 5 #include "ui/gfx/render_text_win.h" |
| 6 | 6 |
| 7 #include <algorithm> | 7 #include <algorithm> |
| 8 | 8 |
| 9 #include "base/i18n/break_iterator.h" | 9 #include "base/i18n/break_iterator.h" |
| 10 #include "base/i18n/char_iterator.h" | 10 #include "base/i18n/char_iterator.h" |
| (...skipping 19 matching lines...) Expand all Loading... | |
| 30 const size_t kMaxUniscribeTextLength = 10000; | 30 const size_t kMaxUniscribeTextLength = 10000; |
| 31 | 31 |
| 32 // The initial guess and maximum supported number of runs; arbitrary values. | 32 // The initial guess and maximum supported number of runs; arbitrary values. |
| 33 // TODO(msw): Support more runs, determine a better initial guess, etc. | 33 // TODO(msw): Support more runs, determine a better initial guess, etc. |
| 34 const int kGuessRuns = 100; | 34 const int kGuessRuns = 100; |
| 35 const size_t kMaxRuns = 10000; | 35 const size_t kMaxRuns = 10000; |
| 36 | 36 |
| 37 // The maximum number of glyphs per run; ScriptShape fails on larger values. | 37 // The maximum number of glyphs per run; ScriptShape fails on larger values. |
| 38 const size_t kMaxGlyphs = 65535; | 38 const size_t kMaxGlyphs = 65535; |
| 39 | 39 |
| 40 // Callback to |EnumEnhMetaFile()| to intercept font creation. | |
| 41 int CALLBACK MetaFileEnumProc(HDC hdc, | |
| 42 HANDLETABLE* table, | |
| 43 CONST ENHMETARECORD* record, | |
| 44 int table_entries, | |
| 45 LPARAM log_font) { | |
| 46 if (record->iType == EMR_EXTCREATEFONTINDIRECTW) { | |
| 47 const EMREXTCREATEFONTINDIRECTW* create_font_record = | |
| 48 reinterpret_cast<const EMREXTCREATEFONTINDIRECTW*>(record); | |
| 49 *reinterpret_cast<LOGFONT*>(log_font) = create_font_record->elfw.elfLogFont; | |
| 50 } | |
| 51 return 1; | |
| 52 } | |
| 53 | |
| 54 // Finds a fallback font to use to render the specified |text| with respect to | |
| 55 // an initial |font|. Returns the resulting font via out param |result|. Returns | |
| 56 // |true| if a fallback font was found. | |
| 57 // Adapted from WebKit's |FontCache::GetFontDataForCharacters()|. | |
| 58 // TODO(asvitkine): This should be moved to font_fallback_win.cc. | |
| 59 bool ChooseFallbackFont(HDC hdc, | |
| 60 const Font& font, | |
| 61 const wchar_t* text, | |
| 62 int text_length, | |
| 63 Font* result) { | |
| 64 // Use a meta file to intercept the fallback font chosen by Uniscribe. | |
| 65 HDC meta_file_dc = CreateEnhMetaFile(hdc, NULL, NULL, NULL); | |
| 66 if (!meta_file_dc) | |
| 67 return false; | |
| 68 | |
| 69 SelectObject(meta_file_dc, font.GetNativeFont()); | |
| 70 | |
| 71 SCRIPT_STRING_ANALYSIS script_analysis; | |
| 72 HRESULT hresult = | |
| 73 ScriptStringAnalyse(meta_file_dc, text, text_length, 0, -1, | |
| 74 SSA_METAFILE | SSA_FALLBACK | SSA_GLYPHS | SSA_LINK, | |
| 75 0, NULL, NULL, NULL, NULL, NULL, &script_analysis); | |
| 76 | |
| 77 if (SUCCEEDED(hresult)) { | |
| 78 hresult = ScriptStringOut(script_analysis, 0, 0, 0, NULL, 0, 0, FALSE); | |
| 79 ScriptStringFree(&script_analysis); | |
| 80 } | |
| 81 | |
| 82 bool found_fallback = false; | |
| 83 HENHMETAFILE meta_file = CloseEnhMetaFile(meta_file_dc); | |
| 84 if (SUCCEEDED(hresult)) { | |
| 85 LOGFONT log_font; | |
| 86 log_font.lfFaceName[0] = 0; | |
| 87 EnumEnhMetaFile(0, meta_file, MetaFileEnumProc, &log_font, NULL); | |
| 88 if (log_font.lfFaceName[0]) { | |
| 89 *result = Font(base::UTF16ToUTF8(log_font.lfFaceName), | |
| 90 font.GetFontSize()); | |
| 91 found_fallback = true; | |
| 92 } | |
| 93 } | |
| 94 DeleteEnhMetaFile(meta_file); | |
| 95 | |
| 96 return found_fallback; | |
| 97 } | |
| 98 | |
| 99 // Changes |font| to have the specified |font_size| (or |font_height| on Windows | 40 // Changes |font| to have the specified |font_size| (or |font_height| on Windows |
| 100 // XP) and |font_style| if it is not the case already. Only considers bold and | 41 // XP) and |font_style| if it is not the case already. Only considers bold and |
| 101 // italic styles, since the underlined style has no effect on glyph shaping. | 42 // italic styles, since the underlined style has no effect on glyph shaping. |
| 102 void DeriveFontIfNecessary(int font_size, | 43 void DeriveFontIfNecessary(int font_size, |
| 103 int font_height, | 44 int font_height, |
| 104 int font_style, | 45 int font_style, |
| 105 Font* font) { | 46 Font* font) { |
| 106 const int kStyleMask = (Font::BOLD | Font::ITALIC); | 47 const int kStyleMask = (Font::BOLD | Font::ITALIC); |
| 107 const int target_style = (font_style & kStyleMask); | 48 const int target_style = (font_style & kStyleMask); |
| 108 | 49 |
| (...skipping 939 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1048 run->width = abc.abcA + abc.abcB + abc.abcC; | 989 run->width = abc.abcA + abc.abcB + abc.abcC; |
| 1049 preceding_run_widths += run->width; | 990 preceding_run_widths += run->width; |
| 1050 } | 991 } |
| 1051 string_width_ = preceding_run_widths; | 992 string_width_ = preceding_run_widths; |
| 1052 } | 993 } |
| 1053 | 994 |
| 1054 void RenderTextWin::LayoutTextRun(internal::TextRun* run) { | 995 void RenderTextWin::LayoutTextRun(internal::TextRun* run) { |
| 1055 const size_t run_length = run->range.length(); | 996 const size_t run_length = run->range.length(); |
| 1056 const wchar_t* run_text = &(GetLayoutText()[run->range.start()]); | 997 const wchar_t* run_text = &(GetLayoutText()[run->range.start()]); |
| 1057 Font original_font = run->font; | 998 Font original_font = run->font; |
| 1058 LinkedFontsIterator fonts(original_font); | |
| 1059 | 999 |
| 1060 run->logical_clusters.reset(new WORD[run_length]); | 1000 run->logical_clusters.reset(new WORD[run_length]); |
| 1061 | 1001 |
| 1062 // Try to shape with the first font in the fallback list, which is | 1002 // Try shaping with |original_font|. |
| 1063 // |original_font|. | 1003 Font current_font(original_font.GetFontName(), original_font.GetFontSize()); |
|
msw
2014/07/18 17:09:18
nit: just do "Font current_font = original_font;"
ckocagil
2014/07/18 17:25:30
Done.
| |
| 1064 Font current_font; | |
| 1065 fonts.NextFont(¤t_font); | |
| 1066 int missing_count = CountCharsWithMissingGlyphs(run, | 1004 int missing_count = CountCharsWithMissingGlyphs(run, |
| 1067 ShapeTextRunWithFont(run, current_font)); | 1005 ShapeTextRunWithFont(run, current_font)); |
| 1068 if (missing_count == 0) | 1006 if (missing_count == 0) |
| 1069 return; | 1007 return; |
| 1070 | 1008 |
| 1071 // Keep track of the font that is able to display the greatest number of | 1009 // Keep track of the font that is able to display the greatest number of |
| 1072 // characters for which ScriptShape() returned S_OK. This font will be used | 1010 // characters for which ScriptShape() returned S_OK. This font will be used |
| 1073 // in the case where no font is able to display the entire run. | 1011 // in the case where no font is able to display the entire run. |
| 1074 int best_partial_font_missing_char_count = missing_count; | 1012 int best_partial_font_missing_char_count = missing_count; |
| 1075 Font best_partial_font = current_font; | 1013 Font best_partial_font = current_font; |
| 1076 | 1014 |
| 1077 // Try to shape with the cached font from previous runs, if any. | 1015 // Try to shape with the cached font from previous runs, if any. |
| 1078 std::map<std::string, Font>::const_iterator it = | 1016 std::map<std::string, Font>::const_iterator it = |
| 1079 successful_substitute_fonts_.find(original_font.GetFontName()); | 1017 successful_substitute_fonts_.find(original_font.GetFontName()); |
| 1080 if (it != successful_substitute_fonts_.end()) { | 1018 if (it != successful_substitute_fonts_.end()) { |
| 1081 current_font = it->second; | 1019 current_font = it->second; |
| 1082 missing_count = CountCharsWithMissingGlyphs(run, | 1020 missing_count = CountCharsWithMissingGlyphs(run, |
| 1083 ShapeTextRunWithFont(run, current_font)); | 1021 ShapeTextRunWithFont(run, current_font)); |
| 1084 if (missing_count == 0) | 1022 if (missing_count == 0) |
| 1085 return; | 1023 return; |
| 1086 if (missing_count < best_partial_font_missing_char_count) { | 1024 if (missing_count < best_partial_font_missing_char_count) { |
| 1087 best_partial_font_missing_char_count = missing_count; | 1025 best_partial_font_missing_char_count = missing_count; |
| 1088 best_partial_font = current_font; | 1026 best_partial_font = current_font; |
| 1089 } | 1027 } |
| 1090 } | 1028 } |
| 1091 | 1029 |
| 1092 // Try finding a fallback font using a meta file. | 1030 // Try finding a fallback font using a meta file. |
| 1093 // TODO(msw|asvitkine): Support RenderText's font_list()? | 1031 // TODO(msw|asvitkine): Support RenderText's font_list()? |
| 1094 if (ChooseFallbackFont(cached_hdc_, run->font, run_text, run_length, | 1032 if (GetUniscribeFallbackFont(run->font, run_text, run_length, |
|
msw
2014/07/18 17:09:18
Use original_font here? ShapeTextRunWithFont modif
ckocagil
2014/07/18 17:25:30
AFAICT this is identical to the logic before my un
msw
2014/07/18 17:37:10
Acknowledged.
| |
| 1095 ¤t_font)) { | 1033 ¤t_font)) { |
| 1096 missing_count = CountCharsWithMissingGlyphs(run, | 1034 missing_count = CountCharsWithMissingGlyphs(run, |
| 1097 ShapeTextRunWithFont(run, current_font)); | 1035 ShapeTextRunWithFont(run, current_font)); |
| 1098 if (missing_count == 0) { | 1036 if (missing_count == 0) { |
| 1099 successful_substitute_fonts_[original_font.GetFontName()] = current_font; | 1037 successful_substitute_fonts_[original_font.GetFontName()] = current_font; |
| 1100 return; | 1038 return; |
| 1101 } | 1039 } |
| 1102 if (missing_count < best_partial_font_missing_char_count) { | 1040 if (missing_count < best_partial_font_missing_char_count) { |
| 1103 best_partial_font_missing_char_count = missing_count; | 1041 best_partial_font_missing_char_count = missing_count; |
| 1104 best_partial_font = current_font; | 1042 best_partial_font = current_font; |
| 1105 } | 1043 } |
| 1106 } | 1044 } |
| 1107 | 1045 |
| 1108 // Try the rest of fonts in the fallback list. | 1046 // Try fonts in the fallback list except the first, which is |original_font|. |
| 1109 while (fonts.NextFont(¤t_font)) { | 1047 std::vector<std::string> fonts = |
| 1048 GetFallbackFontFamilies(original_font.GetFontName()); | |
| 1049 for (size_t i = 1; i < fonts.size(); ++i) { | |
| 1110 missing_count = CountCharsWithMissingGlyphs(run, | 1050 missing_count = CountCharsWithMissingGlyphs(run, |
| 1111 ShapeTextRunWithFont(run, current_font)); | 1051 ShapeTextRunWithFont(run, Font(fonts[i], original_font.GetFontSize()))); |
| 1112 if (missing_count == 0) { | 1052 if (missing_count == 0) { |
| 1113 successful_substitute_fonts_[original_font.GetFontName()] = current_font; | 1053 successful_substitute_fonts_[original_font.GetFontName()] = current_font; |
| 1114 return; | 1054 return; |
| 1115 } | 1055 } |
| 1116 if (missing_count < best_partial_font_missing_char_count) { | 1056 if (missing_count < best_partial_font_missing_char_count) { |
| 1117 best_partial_font_missing_char_count = missing_count; | 1057 best_partial_font_missing_char_count = missing_count; |
| 1118 best_partial_font = current_font; | 1058 best_partial_font = current_font; |
| 1119 } | 1059 } |
| 1120 } | 1060 } |
| 1121 | 1061 |
| (...skipping 159 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1281 size_t position = LayoutIndexToTextIndex(run->range.end()); | 1221 size_t position = LayoutIndexToTextIndex(run->range.end()); |
| 1282 position = IndexOfAdjacentGrapheme(position, CURSOR_BACKWARD); | 1222 position = IndexOfAdjacentGrapheme(position, CURSOR_BACKWARD); |
| 1283 return SelectionModel(position, CURSOR_FORWARD); | 1223 return SelectionModel(position, CURSOR_FORWARD); |
| 1284 } | 1224 } |
| 1285 | 1225 |
| 1286 RenderText* RenderText::CreateNativeInstance() { | 1226 RenderText* RenderText::CreateNativeInstance() { |
| 1287 return new RenderTextWin; | 1227 return new RenderTextWin; |
| 1288 } | 1228 } |
| 1289 | 1229 |
| 1290 } // namespace gfx | 1230 } // namespace gfx |
| OLD | NEW |