/*************************************************************************/ /* dynamicfont_import_settings.cpp */ /*************************************************************************/ /* This file is part of: */ /* GODOT ENGINE */ /* https://godotengine.org */ /*************************************************************************/ /* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */ /* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */ /* */ /* Permission is hereby granted, free of charge, to any person obtaining */ /* a copy of this software and associated documentation files (the */ /* "Software"), to deal in the Software without restriction, including */ /* without limitation the rights to use, copy, modify, merge, publish, */ /* distribute, sublicense, and/or sell copies of the Software, and to */ /* permit persons to whom the Software is furnished to do so, subject to */ /* the following conditions: */ /* */ /* The above copyright notice and this permission notice shall be */ /* included in all copies or substantial portions of the Software. */ /* */ /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /*************************************************************************/ #include "dynamicfont_import_settings.h" #include "editor/editor_node.h" #include "editor/editor_scale.h" /*************************************************************************/ /* Settings data */ /*************************************************************************/ class DynamicFontImportSettingsData : public RefCounted { GDCLASS(DynamicFontImportSettingsData, RefCounted) friend class DynamicFontImportSettings; Map settings; Map defaults; List options; DynamicFontImportSettings *owner = nullptr; bool _set(const StringName &p_name, const Variant &p_value) { if (defaults.has(p_name) && defaults[p_name] == p_value) { settings.erase(p_name); } else { settings[p_name] = p_value; } return true; } bool _get(const StringName &p_name, Variant &r_ret) const { if (settings.has(p_name)) { r_ret = settings[p_name]; return true; } if (defaults.has(p_name)) { r_ret = defaults[p_name]; return true; } return false; } void _get_property_list(List *p_list) const { for (const List::Element *E = options.front(); E; E = E->next()) { if (owner && owner->import_settings_data.is_valid()) { if (owner->import_settings_data->get("multichannel_signed_distance_field") && (E->get().option.name == "size" || E->get().option.name == "outline_size" || E->get().option.name == "oversampling")) { continue; } if (!owner->import_settings_data->get("multichannel_signed_distance_field") && (E->get().option.name == "msdf_pixel_range" || E->get().option.name == "msdf_size")) { continue; } } p_list->push_back(E->get().option); } } }; /*************************************************************************/ /* Glyph ranges */ /*************************************************************************/ struct UniRange { int32_t start; int32_t end; String name; }; // Unicode Character Blocks // Source: https://www.unicode.org/Public/14.0.0/ucd/Blocks.txt static UniRange unicode_ranges[] = { { 0x0000, 0x007F, U"Basic Latin" }, { 0x0080, 0x00FF, U"Latin-1 Supplement" }, { 0x0100, 0x017F, U"Latin Extended-A" }, { 0x0180, 0x024F, U"Latin Extended-B" }, { 0x0250, 0x02AF, U"IPA Extensions" }, { 0x02B0, 0x02FF, U"Spacing Modifier Letters" }, { 0x0300, 0x036F, U"Combining Diacritical Marks" }, { 0x0370, 0x03FF, U"Greek and Coptic" }, { 0x0400, 0x04FF, U"Cyrillic" }, { 0x0500, 0x052F, U"Cyrillic Supplement" }, { 0x0530, 0x058F, U"Armenian" }, { 0x0590, 0x05FF, U"Hebrew" }, { 0x0600, 0x06FF, U"Arabic" }, { 0x0700, 0x074F, U"Syriac" }, { 0x0750, 0x077F, U"Arabic Supplement" }, { 0x0780, 0x07BF, U"Thaana" }, { 0x07C0, 0x07FF, U"NKo" }, { 0x0800, 0x083F, U"Samaritan" }, { 0x0840, 0x085F, U"Mandaic" }, { 0x0860, 0x086F, U"Syriac Supplement" }, { 0x0870, 0x089F, U"Arabic Extended-B" }, { 0x08A0, 0x08FF, U"Arabic Extended-A" }, { 0x0900, 0x097F, U"Devanagari" }, { 0x0980, 0x09FF, U"Bengali" }, { 0x0A00, 0x0A7F, U"Gurmukhi" }, { 0x0A80, 0x0AFF, U"Gujarati" }, { 0x0B00, 0x0B7F, U"Oriya" }, { 0x0B80, 0x0BFF, U"Tamil" }, { 0x0C00, 0x0C7F, U"Telugu" }, { 0x0C80, 0x0CFF, U"Kannada" }, { 0x0D00, 0x0D7F, U"Malayalam" }, { 0x0D80, 0x0DFF, U"Sinhala" }, { 0x0E00, 0x0E7F, U"Thai" }, { 0x0E80, 0x0EFF, U"Lao" }, { 0x0F00, 0x0FFF, U"Tibetan" }, { 0x1000, 0x109F, U"Myanmar" }, { 0x10A0, 0x10FF, U"Georgian" }, { 0x1100, 0x11FF, U"Hangul Jamo" }, { 0x1200, 0x137F, U"Ethiopic" }, { 0x1380, 0x139F, U"Ethiopic Supplement" }, { 0x13A0, 0x13FF, U"Cherokee" }, { 0x1400, 0x167F, U"Unified Canadian Aboriginal Syllabics" }, { 0x1680, 0x169F, U"Ogham" }, { 0x16A0, 0x16FF, U"Runic" }, { 0x1700, 0x171F, U"Tagalog" }, { 0x1720, 0x173F, U"Hanunoo" }, { 0x1740, 0x175F, U"Buhid" }, { 0x1760, 0x177F, U"Tagbanwa" }, { 0x1780, 0x17FF, U"Khmer" }, { 0x1800, 0x18AF, U"Mongolian" }, { 0x18B0, 0x18FF, U"Unified Canadian Aboriginal Syllabics Extended" }, { 0x1900, 0x194F, U"Limbu" }, { 0x1950, 0x197F, U"Tai Le" }, { 0x1980, 0x19DF, U"New Tai Lue" }, { 0x19E0, 0x19FF, U"Khmer Symbols" }, { 0x1A00, 0x1A1F, U"Buginese" }, { 0x1A20, 0x1AAF, U"Tai Tham" }, { 0x1AB0, 0x1AFF, U"Combining Diacritical Marks Extended" }, { 0x1B00, 0x1B7F, U"Balinese" }, { 0x1B80, 0x1BBF, U"Sundanese" }, { 0x1BC0, 0x1BFF, U"Batak" }, { 0x1C00, 0x1C4F, U"Lepcha" }, { 0x1C50, 0x1C7F, U"Ol Chiki" }, { 0x1C80, 0x1C8F, U"Cyrillic Extended-C" }, { 0x1C90, 0x1CBF, U"Georgian Extended" }, { 0x1CC0, 0x1CCF, U"Sundanese Supplement" }, { 0x1CD0, 0x1CFF, U"Vedic Extensions" }, { 0x1D00, 0x1D7F, U"Phonetic Extensions" }, { 0x1D80, 0x1DBF, U"Phonetic Extensions Supplement" }, { 0x1DC0, 0x1DFF, U"Combining Diacritical Marks Supplement" }, { 0x1E00, 0x1EFF, U"Latin Extended Additional" }, { 0x1F00, 0x1FFF, U"Greek Extended" }, { 0x2000, 0x206F, U"General Punctuation" }, { 0x2070, 0x209F, U"Superscripts and Subscripts" }, { 0x20A0, 0x20CF, U"Currency Symbols" }, { 0x20D0, 0x20FF, U"Combining Diacritical Marks for Symbols" }, { 0x2100, 0x214F, U"Letterlike Symbols" }, { 0x2150, 0x218F, U"Number Forms" }, { 0x2190, 0x21FF, U"Arrows" }, { 0x2200, 0x22FF, U"Mathematical Operators" }, { 0x2300, 0x23FF, U"Miscellaneous Technical" }, { 0x2400, 0x243F, U"Control Pictures" }, { 0x2440, 0x245F, U"Optical Character Recognition" }, { 0x2460, 0x24FF, U"Enclosed Alphanumerics" }, { 0x2500, 0x257F, U"Box Drawing" }, { 0x2580, 0x259F, U"Block Elements" }, { 0x25A0, 0x25FF, U"Geometric Shapes" }, { 0x2600, 0x26FF, U"Miscellaneous Symbols" }, { 0x2700, 0x27BF, U"Dingbats" }, { 0x27C0, 0x27EF, U"Miscellaneous Mathematical Symbols-A" }, { 0x27F0, 0x27FF, U"Supplemental Arrows-A" }, { 0x2800, 0x28FF, U"Braille Patterns" }, { 0x2900, 0x297F, U"Supplemental Arrows-B" }, { 0x2980, 0x29FF, U"Miscellaneous Mathematical Symbols-B" }, { 0x2A00, 0x2AFF, U"Supplemental Mathematical Operators" }, { 0x2B00, 0x2BFF, U"Miscellaneous Symbols and Arrows" }, { 0x2C00, 0x2C5F, U"Glagolitic" }, { 0x2C60, 0x2C7F, U"Latin Extended-C" }, { 0x2C80, 0x2CFF, U"Coptic" }, { 0x2D00, 0x2D2F, U"Georgian Supplement" }, { 0x2D30, 0x2D7F, U"Tifinagh" }, { 0x2D80, 0x2DDF, U"Ethiopic Extended" }, { 0x2DE0, 0x2DFF, U"Cyrillic Extended-A" }, { 0x2E00, 0x2E7F, U"Supplemental Punctuation" }, { 0x2E80, 0x2EFF, U"CJK Radicals Supplement" }, { 0x2F00, 0x2FDF, U"Kangxi Radicals" }, { 0x2FF0, 0x2FFF, U"Ideographic Description Characters" }, { 0x3000, 0x303F, U"CJK Symbols and Punctuation" }, { 0x3040, 0x309F, U"Hiragana" }, { 0x30A0, 0x30FF, U"Katakana" }, { 0x3100, 0x312F, U"Bopomofo" }, { 0x3130, 0x318F, U"Hangul Compatibility Jamo" }, { 0x3190, 0x319F, U"Kanbun" }, { 0x31A0, 0x31BF, U"Bopomofo Extended" }, { 0x31C0, 0x31EF, U"CJK Strokes" }, { 0x31F0, 0x31FF, U"Katakana Phonetic Extensions" }, { 0x3200, 0x32FF, U"Enclosed CJK Letters and Months" }, { 0x3300, 0x33FF, U"CJK Compatibility" }, { 0x3400, 0x4DBF, U"CJK Unified Ideographs Extension A" }, { 0x4DC0, 0x4DFF, U"Yijing Hexagram Symbols" }, { 0x4E00, 0x9FFF, U"CJK Unified Ideographs" }, { 0xA000, 0xA48F, U"Yi Syllables" }, { 0xA490, 0xA4CF, U"Yi Radicals" }, { 0xA4D0, 0xA4FF, U"Lisu" }, { 0xA500, 0xA63F, U"Vai" }, { 0xA640, 0xA69F, U"Cyrillic Extended-B" }, { 0xA6A0, 0xA6FF, U"Bamum" }, { 0xA700, 0xA71F, U"Modifier Tone Letters" }, { 0xA720, 0xA7FF, U"Latin Extended-D" }, { 0xA800, 0xA82F, U"Syloti Nagri" }, { 0xA830, 0xA83F, U"Common Indic Number Forms" }, { 0xA840, 0xA87F, U"Phags-pa" }, { 0xA880, 0xA8DF, U"Saurashtra" }, { 0xA8E0, 0xA8FF, U"Devanagari Extended" }, { 0xA900, 0xA92F, U"Kayah Li" }, { 0xA930, 0xA95F, U"Rejang" }, { 0xA960, 0xA97F, U"Hangul Jamo Extended-A" }, { 0xA980, 0xA9DF, U"Javanese" }, { 0xA9E0, 0xA9FF, U"Myanmar Extended-B" }, { 0xAA00, 0xAA5F, U"Cham" }, { 0xAA60, 0xAA7F, U"Myanmar Extended-A" }, { 0xAA80, 0xAADF, U"Tai Viet" }, { 0xAAE0, 0xAAFF, U"Meetei Mayek Extensions" }, { 0xAB00, 0xAB2F, U"Ethiopic Extended-A" }, { 0xAB30, 0xAB6F, U"Latin Extended-E" }, { 0xAB70, 0xABBF, U"Cherokee Supplement" }, { 0xABC0, 0xABFF, U"Meetei Mayek" }, { 0xAC00, 0xD7AF, U"Hangul Syllables" }, { 0xD7B0, 0xD7FF, U"Hangul Jamo Extended-B" }, //{ 0xD800, 0xDB7F, U"High Surrogates" }, //{ 0xDB80, 0xDBFF, U"High Private Use Surrogates" }, //{ 0xDC00, 0xDFFF, U"Low Surrogates" }, { 0xE000, 0xF8FF, U"Private Use Area" }, { 0xF900, 0xFAFF, U"CJK Compatibility Ideographs" }, { 0xFB00, 0xFB4F, U"Alphabetic Presentation Forms" }, { 0xFB50, 0xFDFF, U"Arabic Presentation Forms-A" }, //{ 0xFE00, 0xFE0F, U"Variation Selectors" }, { 0xFE10, 0xFE1F, U"Vertical Forms" }, { 0xFE20, 0xFE2F, U"Combining Half Marks" }, { 0xFE30, 0xFE4F, U"CJK Compatibility Forms" }, { 0xFE50, 0xFE6F, U"Small Form Variants" }, { 0xFE70, 0xFEFF, U"Arabic Presentation Forms-B" }, { 0xFF00, 0xFFEF, U"Halfwidth and Fullwidth Forms" }, //{ 0xFFF0, 0xFFFF, U"Specials" }, { 0x10000, 0x1007F, U"Linear B Syllabary" }, { 0x10080, 0x100FF, U"Linear B Ideograms" }, { 0x10100, 0x1013F, U"Aegean Numbers" }, { 0x10140, 0x1018F, U"Ancient Greek Numbers" }, { 0x10190, 0x101CF, U"Ancient Symbols" }, { 0x101D0, 0x101FF, U"Phaistos Disc" }, { 0x10280, 0x1029F, U"Lycian" }, { 0x102A0, 0x102DF, U"Carian" }, { 0x102E0, 0x102FF, U"Coptic Epact Numbers" }, { 0x10300, 0x1032F, U"Old Italic" }, { 0x10330, 0x1034F, U"Gothic" }, { 0x10350, 0x1037F, U"Old Permic" }, { 0x10380, 0x1039F, U"Ugaritic" }, { 0x103A0, 0x103DF, U"Old Persian" }, { 0x10400, 0x1044F, U"Deseret" }, { 0x10450, 0x1047F, U"Shavian" }, { 0x10480, 0x104AF, U"Osmanya" }, { 0x104B0, 0x104FF, U"Osage" }, { 0x10500, 0x1052F, U"Elbasan" }, { 0x10530, 0x1056F, U"Caucasian Albanian" }, { 0x10570, 0x105BF, U"Vithkuqi" }, { 0x10600, 0x1077F, U"Linear A" }, { 0x10780, 0x107BF, U"Latin Extended-F" }, { 0x10800, 0x1083F, U"Cypriot Syllabary" }, { 0x10840, 0x1085F, U"Imperial Aramaic" }, { 0x10860, 0x1087F, U"Palmyrene" }, { 0x10880, 0x108AF, U"Nabataean" }, { 0x108E0, 0x108FF, U"Hatran" }, { 0x10900, 0x1091F, U"Phoenician" }, { 0x10920, 0x1093F, U"Lydian" }, { 0x10980, 0x1099F, U"Meroitic Hieroglyphs" }, { 0x109A0, 0x109FF, U"Meroitic Cursive" }, { 0x10A00, 0x10A5F, U"Kharoshthi" }, { 0x10A60, 0x10A7F, U"Old South Arabian" }, { 0x10A80, 0x10A9F, U"Old North Arabian" }, { 0x10AC0, 0x10AFF, U"Manichaean" }, { 0x10B00, 0x10B3F, U"Avestan" }, { 0x10B40, 0x10B5F, U"Inscriptional Parthian" }, { 0x10B60, 0x10B7F, U"Inscriptional Pahlavi" }, { 0x10B80, 0x10BAF, U"Psalter Pahlavi" }, { 0x10C00, 0x10C4F, U"Old Turkic" }, { 0x10C80, 0x10CFF, U"Old Hungarian" }, { 0x10D00, 0x10D3F, U"Hanifi Rohingya" }, { 0x10E60, 0x10E7F, U"Rumi Numeral Symbols" }, { 0x10E80, 0x10EBF, U"Yezidi" }, { 0x10F00, 0x10F2F, U"Old Sogdian" }, { 0x10F30, 0x10F6F, U"Sogdian" }, { 0x10F70, 0x10FAF, U"Old Uyghur" }, { 0x10FB0, 0x10FDF, U"Chorasmian" }, { 0x10FE0, 0x10FFF, U"Elymaic" }, { 0x11000, 0x1107F, U"Brahmi" }, { 0x11080, 0x110CF, U"Kaithi" }, { 0x110D0, 0x110FF, U"Sora Sompeng" }, { 0x11100, 0x1114F, U"Chakma" }, { 0x11150, 0x1117F, U"Mahajani" }, { 0x11180, 0x111DF, U"Sharada" }, { 0x111E0, 0x111FF, U"Sinhala Archaic Numbers" }, { 0x11200, 0x1124F, U"Khojki" }, { 0x11280, 0x112AF, U"Multani" }, { 0x112B0, 0x112FF, U"Khudawadi" }, { 0x11300, 0x1137F, U"Grantha" }, { 0x11400, 0x1147F, U"Newa" }, { 0x11480, 0x114DF, U"Tirhuta" }, { 0x11580, 0x115FF, U"Siddham" }, { 0x11600, 0x1165F, U"Modi" }, { 0x11660, 0x1167F, U"Mongolian Supplement" }, { 0x11680, 0x116CF, U"Takri" }, { 0x11700, 0x1174F, U"Ahom" }, { 0x11800, 0x1184F, U"Dogra" }, { 0x118A0, 0x118FF, U"Warang Citi" }, { 0x11900, 0x1195F, U"Dives Akuru" }, { 0x119A0, 0x119FF, U"Nandinagari" }, { 0x11A00, 0x11A4F, U"Zanabazar Square" }, { 0x11A50, 0x11AAF, U"Soyombo" }, { 0x11AB0, 0x11ABF, U"Unified Canadian Aboriginal Syllabics Extended-A" }, { 0x11AC0, 0x11AFF, U"Pau Cin Hau" }, { 0x11C00, 0x11C6F, U"Bhaiksuki" }, { 0x11C70, 0x11CBF, U"Marchen" }, { 0x11D00, 0x11D5F, U"Masaram Gondi" }, { 0x11D60, 0x11DAF, U"Gunjala Gondi" }, { 0x11EE0, 0x11EFF, U"Makasar" }, { 0x11FB0, 0x11FBF, U"Lisu Supplement" }, { 0x11FC0, 0x11FFF, U"Tamil Supplement" }, { 0x12000, 0x123FF, U"Cuneiform" }, { 0x12400, 0x1247F, U"Cuneiform Numbers and Punctuation" }, { 0x12480, 0x1254F, U"Early Dynastic Cuneiform" }, { 0x12F90, 0x12FFF, U"Cypro-Minoan" }, { 0x13000, 0x1342F, U"Egyptian Hieroglyphs" }, { 0x13430, 0x1343F, U"Egyptian Hieroglyph Format Controls" }, { 0x14400, 0x1467F, U"Anatolian Hieroglyphs" }, { 0x16800, 0x16A3F, U"Bamum Supplement" }, { 0x16A40, 0x16A6F, U"Mro" }, { 0x16A70, 0x16ACF, U"Tangsa" }, { 0x16AD0, 0x16AFF, U"Bassa Vah" }, { 0x16B00, 0x16B8F, U"Pahawh Hmong" }, { 0x16E40, 0x16E9F, U"Medefaidrin" }, { 0x16F00, 0x16F9F, U"Miao" }, { 0x16FE0, 0x16FFF, U"Ideographic Symbols and Punctuation" }, { 0x17000, 0x187FF, U"Tangut" }, { 0x18800, 0x18AFF, U"Tangut Components" }, { 0x18B00, 0x18CFF, U"Khitan Small Script" }, { 0x18D00, 0x18D7F, U"Tangut Supplement" }, { 0x1AFF0, 0x1AFFF, U"Kana Extended-B" }, { 0x1B000, 0x1B0FF, U"Kana Supplement" }, { 0x1B100, 0x1B12F, U"Kana Extended-A" }, { 0x1B130, 0x1B16F, U"Small Kana Extension" }, { 0x1B170, 0x1B2FF, U"Nushu" }, { 0x1BC00, 0x1BC9F, U"Duployan" }, { 0x1BCA0, 0x1BCAF, U"Shorthand Format Controls" }, { 0x1CF00, 0x1CFCF, U"Znamenny Musical Notation" }, { 0x1D000, 0x1D0FF, U"Byzantine Musical Symbols" }, { 0x1D100, 0x1D1FF, U"Musical Symbols" }, { 0x1D200, 0x1D24F, U"Ancient Greek Musical Notation" }, { 0x1D2E0, 0x1D2FF, U"Mayan Numerals" }, { 0x1D300, 0x1D35F, U"Tai Xuan Jing Symbols" }, { 0x1D360, 0x1D37F, U"Counting Rod Numerals" }, { 0x1D400, 0x1D7FF, U"Mathematical Alphanumeric Symbols" }, { 0x1D800, 0x1DAAF, U"Sutton SignWriting" }, { 0x1DF00, 0x1DFFF, U"Latin Extended-G" }, { 0x1E000, 0x1E02F, U"Glagolitic Supplement" }, { 0x1E100, 0x1E14F, U"Nyiakeng Puachue Hmong" }, { 0x1E290, 0x1E2BF, U"Toto" }, { 0x1E2C0, 0x1E2FF, U"Wancho" }, { 0x1E7E0, 0x1E7FF, U"Ethiopic Extended-B" }, { 0x1E800, 0x1E8DF, U"Mende Kikakui" }, { 0x1E900, 0x1E95F, U"Adlam" }, { 0x1EC70, 0x1ECBF, U"Indic Siyaq Numbers" }, { 0x1ED00, 0x1ED4F, U"Ottoman Siyaq Numbers" }, { 0x1EE00, 0x1EEFF, U"Arabic Mathematical Alphabetic Symbols" }, { 0x1F000, 0x1F02F, U"Mahjong Tiles" }, { 0x1F030, 0x1F09F, U"Domino Tiles" }, { 0x1F0A0, 0x1F0FF, U"Playing Cards" }, { 0x1F100, 0x1F1FF, U"Enclosed Alphanumeric Supplement" }, { 0x1F200, 0x1F2FF, U"Enclosed Ideographic Supplement" }, { 0x1F300, 0x1F5FF, U"Miscellaneous Symbols and Pictographs" }, { 0x1F600, 0x1F64F, U"Emoticons" }, { 0x1F650, 0x1F67F, U"Ornamental Dingbats" }, { 0x1F680, 0x1F6FF, U"Transport and Map Symbols" }, { 0x1F700, 0x1F77F, U"Alchemical Symbols" }, { 0x1F780, 0x1F7FF, U"Geometric Shapes Extended" }, { 0x1F800, 0x1F8FF, U"Supplemental Arrows-C" }, { 0x1F900, 0x1F9FF, U"Supplemental Symbols and Pictographs" }, { 0x1FA00, 0x1FA6F, U"Chess Symbols" }, { 0x1FA70, 0x1FAFF, U"Symbols and Pictographs Extended-A" }, { 0x1FB00, 0x1FBFF, U"Symbols for Legacy Computing" }, { 0x20000, 0x2A6DF, U"CJK Unified Ideographs Extension B" }, { 0x2A700, 0x2B73F, U"CJK Unified Ideographs Extension C" }, { 0x2B740, 0x2B81F, U"CJK Unified Ideographs Extension D" }, { 0x2B820, 0x2CEAF, U"CJK Unified Ideographs Extension E" }, { 0x2CEB0, 0x2EBEF, U"CJK Unified Ideographs Extension F" }, { 0x2F800, 0x2FA1F, U"CJK Compatibility Ideographs Supplement" }, { 0x30000, 0x3134F, U"CJK Unified Ideographs Extension G" }, //{ 0xE0000, 0xE007F, U"Tags" }, //{ 0xE0100, 0xE01EF, U"Variation Selectors Supplement" }, { 0xF0000, 0xFFFFF, U"Supplementary Private Use Area-A" }, { 0x100000, 0x10FFFF, U"Supplementary Private Use Area-B" }, { 0x10FFFF, 0x10FFFF, String() } }; void DynamicFontImportSettings::_add_glyph_range_item(int32_t p_start, int32_t p_end, const String &p_name) { const int page_size = 512; int pages = (p_end - p_start) / page_size; int remain = (p_end - p_start) % page_size; int32_t start = p_start; for (int i = 0; i < pages; i++) { TreeItem *item = glyph_tree->create_item(glyph_root); ERR_FAIL_NULL(item); item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK); item->set_text(0, _pad_zeros(String::num_int64(start, 16)) + " - " + _pad_zeros(String::num_int64(start + page_size, 16))); item->set_text(1, p_name); item->set_metadata(0, Vector2i(start, start + page_size)); start += page_size; } if (remain > 0) { TreeItem *item = glyph_tree->create_item(glyph_root); ERR_FAIL_NULL(item); item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK); item->set_text(0, _pad_zeros(String::num_int64(start, 16)) + " - " + _pad_zeros(String::num_int64(p_end, 16))); item->set_text(1, p_name); item->set_metadata(0, Vector2i(start, p_end)); } } /*************************************************************************/ /* Page 1 callbacks: Rendering Options */ /*************************************************************************/ void DynamicFontImportSettings::_main_prop_changed(const String &p_edited_property) { // Update font preview. if (p_edited_property == "antialiased") { if (font_preview->get_data_count() > 0) { font_preview->get_data(0)->set_antialiased(import_settings_data->get("antialiased")); } } else if (p_edited_property == "multichannel_signed_distance_field") { if (font_preview->get_data_count() > 0) { font_preview->get_data(0)->set_multichannel_signed_distance_field(import_settings_data->get("multichannel_signed_distance_field")); } _variation_selected(); _variations_validate(); } else if (p_edited_property == "msdf_pixel_range") { if (font_preview->get_data_count() > 0) { font_preview->get_data(0)->set_msdf_pixel_range(import_settings_data->get("msdf_pixel_range")); } } else if (p_edited_property == "msdf_size") { if (font_preview->get_data_count() > 0) { font_preview->get_data(0)->set_msdf_size(import_settings_data->get("msdf_size")); } } else if (p_edited_property == "force_autohinter") { if (font_preview->get_data_count() > 0) { font_preview->get_data(0)->set_force_autohinter(import_settings_data->get("force_autohinter")); } } else if (p_edited_property == "hinting") { if (font_preview->get_data_count() > 0) { font_preview->get_data(0)->set_hinting((TextServer::Hinting)import_settings_data->get("hinting").operator int()); } } else if (p_edited_property == "oversampling") { if (font_preview->get_data_count() > 0) { font_preview->get_data(0)->set_oversampling(import_settings_data->get("oversampling")); } } font_preview_label->add_theme_font_override("font", font_preview); font_preview_label->update(); } /*************************************************************************/ /* Page 2 callbacks: Configurations */ /*************************************************************************/ void DynamicFontImportSettings::_variation_add() { TreeItem *vars_item = vars_list->create_item(vars_list_root); ERR_FAIL_NULL(vars_item); vars_item->set_text(0, TTR("New configuration")); vars_item->set_editable(0, true); vars_item->add_button(1, vars_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove Variation")); vars_item->set_button_color(1, 0, Color(1, 1, 1, 0.75)); Ref import_variation_data; import_variation_data.instantiate(); import_variation_data->owner = this; ERR_FAIL_NULL(import_variation_data); for (List::Element *E = options_variations.front(); E; E = E->next()) { import_variation_data->defaults[E->get().option.name] = E->get().default_value; } import_variation_data->options = options_variations; inspector_vars->edit(import_variation_data.ptr()); import_variation_data->notify_property_list_changed(); vars_item->set_metadata(0, import_variation_data); _variations_validate(); } void DynamicFontImportSettings::_variation_selected() { TreeItem *vars_item = vars_list->get_selected(); if (vars_item) { Ref import_variation_data = vars_item->get_metadata(0); ERR_FAIL_NULL(import_variation_data); inspector_vars->edit(import_variation_data.ptr()); import_variation_data->notify_property_list_changed(); } } void DynamicFontImportSettings::_variation_remove(Object *p_item, int p_column, int p_id) { TreeItem *vars_item = (TreeItem *)p_item; ERR_FAIL_NULL(vars_item); inspector_vars->edit(nullptr); vars_list_root->remove_child(vars_item); memdelete(vars_item); if (vars_list_root->get_first_child()) { Ref import_variation_data = vars_list_root->get_first_child()->get_metadata(0); inspector_vars->edit(import_variation_data.ptr()); import_variation_data->notify_property_list_changed(); } _variations_validate(); } void DynamicFontImportSettings::_variation_changed(const String &p_edited_property) { _variations_validate(); } void DynamicFontImportSettings::_variations_validate() { String warn; if (!vars_list_root->get_first_child()) { warn = TTR("Warinig: There are no configurations specified, no glyphs will be pre-rendered."); } for (TreeItem *vars_item_a = vars_list_root->get_first_child(); vars_item_a; vars_item_a = vars_item_a->get_next()) { Ref import_variation_data_a = vars_item_a->get_metadata(0); ERR_FAIL_NULL(import_variation_data_a); for (TreeItem *vars_item_b = vars_list_root->get_first_child(); vars_item_b; vars_item_b = vars_item_b->get_next()) { if (vars_item_b != vars_item_a) { bool match = true; for (Map::Element *E = import_variation_data_a->settings.front(); E; E = E->next()) { Ref import_variation_data_b = vars_item_b->get_metadata(0); ERR_FAIL_NULL(import_variation_data_b); match = match && (import_variation_data_b->settings[E->key()] == E->get()); } if (match) { warn = TTR("Warinig: Multiple configurations have identical settings. Duplicates will be ignored."); break; } } } } if (warn.is_empty()) { label_warn->set_text(""); label_warn->hide(); } else { label_warn->set_text(warn); label_warn->show(); } } /*************************************************************************/ /* Page 3 callbacks: Text to select glyphs */ /*************************************************************************/ void DynamicFontImportSettings::_change_text_opts() { Vector ftr = ftr_edit->get_text().split(","); for (int i = 0; i < ftr.size(); i++) { Vector tokens = ftr[i].split("="); if (tokens.size() == 2) { text_edit->set_opentype_feature(tokens[0], tokens[1].to_int()); } else if (tokens.size() == 1) { text_edit->set_opentype_feature(tokens[0], 1); } } text_edit->set_language(lang_edit->get_text()); } void DynamicFontImportSettings::_glyph_clear() { selected_glyphs.clear(); label_glyphs->set_text(TTR("Preloaded glyphs: ") + itos(selected_glyphs.size())); _range_selected(); } void DynamicFontImportSettings::_glyph_text_selected() { Dictionary ftrs; Vector ftr = ftr_edit->get_text().split(","); for (int i = 0; i < ftr.size(); i++) { Vector tokens = ftr[i].split("="); if (tokens.size() == 2) { ftrs[tokens[0]] = tokens[1].to_int(); } else if (tokens.size() == 1) { ftrs[tokens[0]] = 1; } } RID text_rid = TS->create_shaped_text(); if (text_rid.is_valid()) { TS->shaped_text_add_string(text_rid, text_edit->get_text(), font_main->get_rids(), 16, ftrs, text_edit->get_language()); TS->shaped_text_shape(text_rid); const Glyph *gl = TS->shaped_text_get_glyphs(text_rid); const int gl_size = TS->shaped_text_get_glyph_count(text_rid); for (int i = 0; i < gl_size; i++) { if (gl[i].font_rid.is_valid() && gl[i].index != 0) { selected_glyphs.insert(gl[i].index); } } TS->free(text_rid); label_glyphs->set_text(TTR("Preloaded glyphs: ") + itos(selected_glyphs.size())); } _range_selected(); } /*************************************************************************/ /* Page 4 callbacks: Character map */ /*************************************************************************/ void DynamicFontImportSettings::_glyph_selected() { TreeItem *item = glyph_table->get_selected(); ERR_FAIL_NULL(item); Color scol = glyph_table->get_theme_color(SNAME("box_selection_fill_color"), SNAME("Editor")); Color fcol = glyph_table->get_theme_color(SNAME("font_selected_color"), SNAME("Editor")); scol.a = 1.f; int32_t c = item->get_metadata(glyph_table->get_selected_column()); if (font_main->has_char(c)) { if (_char_update(c)) { item->set_custom_color(glyph_table->get_selected_column(), fcol); item->set_custom_bg_color(glyph_table->get_selected_column(), scol); } else { item->clear_custom_color(glyph_table->get_selected_column()); item->clear_custom_bg_color(glyph_table->get_selected_column()); } } label_glyphs->set_text(TTR("Preloaded glyphs: ") + itos(selected_glyphs.size())); item = glyph_tree->get_selected(); ERR_FAIL_NULL(item); Vector2i range = item->get_metadata(0); int total_chars = range.y - range.x; int selected_count = 0; for (int i = range.x; i < range.y; i++) { if (!font_main->has_char(i)) { total_chars--; } if (selected_chars.has(i)) { selected_count++; } } if (selected_count == total_chars) { item->set_checked(0, true); } else if (selected_count > 0) { item->set_indeterminate(0, true); } else { item->set_checked(0, false); } } void DynamicFontImportSettings::_range_edited() { TreeItem *item = glyph_tree->get_selected(); ERR_FAIL_NULL(item); Vector2i range = item->get_metadata(0); _range_update(range.x, range.y); } void DynamicFontImportSettings::_range_selected() { TreeItem *item = glyph_tree->get_selected(); if (item) { Vector2i range = item->get_metadata(0); _edit_range(range.x, range.y); } } void DynamicFontImportSettings::_edit_range(int32_t p_start, int32_t p_end) { glyph_table->clear(); TreeItem *root = glyph_table->create_item(); ERR_FAIL_NULL(root); Color scol = glyph_table->get_theme_color(SNAME("box_selection_fill_color"), SNAME("Editor")); Color fcol = glyph_table->get_theme_color(SNAME("font_selected_color"), SNAME("Editor")); scol.a = 1.f; TreeItem *item = nullptr; int col = 0; for (int32_t c = p_start; c <= p_end; c++) { if (col == 0) { item = glyph_table->create_item(root); ERR_FAIL_NULL(item); item->set_text(0, _pad_zeros(String::num_int64(c, 16))); item->set_text_alignment(0, HORIZONTAL_ALIGNMENT_LEFT); item->set_selectable(0, false); item->set_custom_bg_color(0, glyph_table->get_theme_color(SNAME("dark_color_3"), SNAME("Editor"))); } if (font_main->has_char(c)) { item->set_text(col + 1, String::chr(c)); item->set_custom_color(col + 1, Color(1, 1, 1)); if (selected_chars.has(c) || (font_main->get_data(0).is_valid() && selected_glyphs.has(font_main->get_data(0)->get_glyph_index(get_theme_font_size(SNAME("font_size")) * 2, c)))) { item->set_custom_color(col + 1, fcol); item->set_custom_bg_color(col + 1, scol); } else { item->clear_custom_color(col + 1); item->clear_custom_bg_color(col + 1); } } else { item->set_custom_bg_color(col + 1, glyph_table->get_theme_color(SNAME("dark_color_2"), SNAME("Editor"))); } item->set_metadata(col + 1, c); item->set_text_alignment(col + 1, HORIZONTAL_ALIGNMENT_CENTER); item->set_selectable(col + 1, true); item->set_custom_font(col + 1, font_main); item->set_custom_font_size(col + 1, get_theme_font_size(SNAME("font_size")) * 2); col++; if (col == 16) { col = 0; } } label_glyphs->set_text(TTR("Preloaded glyphs: ") + itos(selected_glyphs.size())); } bool DynamicFontImportSettings::_char_update(int32_t p_char) { if (selected_chars.has(p_char)) { selected_chars.erase(p_char); return false; } else if (font_main->get_data(0).is_valid() && selected_glyphs.has(font_main->get_data(0)->get_glyph_index(get_theme_font_size(SNAME("font_size")) * 2, p_char))) { selected_glyphs.erase(font_main->get_data(0)->get_glyph_index(get_theme_font_size(SNAME("font_size")) * 2, p_char)); return false; } else { selected_chars.insert(p_char); return true; } label_glyphs->set_text(TTR("Preloaded glyphs: ") + itos(selected_glyphs.size())); } void DynamicFontImportSettings::_range_update(int32_t p_start, int32_t p_end) { bool all_selected = true; for (int32_t i = p_start; i <= p_end; i++) { if (font_main->has_char(i)) { if (font_main->get_data(0).is_valid()) { all_selected = all_selected && (selected_chars.has(i) || (font_main->get_data(0).is_valid() && selected_glyphs.has(font_main->get_data(0)->get_glyph_index(get_theme_font_size(SNAME("font_size")) * 2, i)))); } else { all_selected = all_selected && selected_chars.has(i); } } } for (int32_t i = p_start; i <= p_end; i++) { if (font_main->has_char(i)) { if (!all_selected) { selected_chars.insert(i); } else { selected_chars.erase(i); if (font_main->get_data(0).is_valid()) { selected_glyphs.erase(font_main->get_data(0)->get_glyph_index(get_theme_font_size(SNAME("font_size")) * 2, i)); } } } } _edit_range(p_start, p_end); TreeItem *item = glyph_tree->get_selected(); ERR_FAIL_NULL(item); item->set_checked(0, !all_selected); } /*************************************************************************/ /* Page 5 callbacks: CMetadata override */ /*************************************************************************/ void DynamicFontImportSettings::_lang_add() { locale_select->popup_locale_dialog(); } void DynamicFontImportSettings::_lang_add_item(const String &p_locale) { TreeItem *lang_item = lang_list->create_item(lang_list_root); ERR_FAIL_NULL(lang_item); lang_item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK); lang_item->set_editable(0, true); lang_item->set_checked(0, false); lang_item->set_text(1, p_locale); lang_item->set_editable(1, true); lang_item->add_button(2, lang_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove")); lang_item->set_button_color(2, 0, Color(1, 1, 1, 0.75)); } void DynamicFontImportSettings::_lang_remove(Object *p_item, int p_column, int p_id) { TreeItem *lang_item = (TreeItem *)p_item; ERR_FAIL_NULL(lang_item); lang_list_root->remove_child(lang_item); memdelete(lang_item); } void DynamicFontImportSettings::_ot_add() { menu_ot->set_position(ot_list->get_screen_transform().xform(ot_list->get_local_mouse_position())); menu_ot->set_size(Vector2(1, 1)); menu_ot->popup(); } void DynamicFontImportSettings::_ot_add_item(int p_option) { String name = TS->tag_to_name(p_option); for (TreeItem *ot_item = ot_list_root->get_first_child(); ot_item; ot_item = ot_item->get_next()) { if (ot_item->get_text(0) == name) { return; } } TreeItem *ot_item = ot_list->create_item(ot_list_root); ERR_FAIL_NULL(ot_item); ot_item->set_text(0, name); ot_item->set_editable(0, false); ot_item->set_text(1, "1"); ot_item->set_editable(1, true); ot_item->add_button(2, ot_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove")); ot_item->set_button_color(2, 0, Color(1, 1, 1, 0.75)); } void DynamicFontImportSettings::_ot_remove(Object *p_item, int p_column, int p_id) { TreeItem *ot_item = (TreeItem *)p_item; ERR_FAIL_NULL(ot_item); ot_list_root->remove_child(ot_item); memdelete(ot_item); } void DynamicFontImportSettings::_script_add() { menu_scripts->set_position(script_list->get_screen_position() + script_list->get_local_mouse_position()); menu_scripts->reset_size(); menu_scripts->popup(); } void DynamicFontImportSettings::_script_add_item(int p_option) { TreeItem *script_item = script_list->create_item(script_list_root); ERR_FAIL_NULL(script_item); script_item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK); script_item->set_editable(0, true); script_item->set_checked(0, false); script_item->set_text(1, script_codes[p_option]); script_item->set_editable(1, true); script_item->add_button(2, lang_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove")); script_item->set_button_color(2, 0, Color(1, 1, 1, 0.75)); } void DynamicFontImportSettings::_script_remove(Object *p_item, int p_column, int p_id) { TreeItem *script_item = (TreeItem *)p_item; ERR_FAIL_NULL(script_item); script_list_root->remove_child(script_item); memdelete(script_item); } /*************************************************************************/ /* Common */ /*************************************************************************/ DynamicFontImportSettings *DynamicFontImportSettings::singleton = nullptr; String DynamicFontImportSettings::_pad_zeros(const String &p_hex) const { int len = CLAMP(5 - p_hex.length(), 0, 5); return String("0").repeat(len) + p_hex; } void DynamicFontImportSettings::_notification(int p_what) { if (p_what == NOTIFICATION_READY) { connect("confirmed", callable_mp(this, &DynamicFontImportSettings::_re_import)); } else if (p_what == NOTIFICATION_ENTER_TREE || p_what == NOTIFICATION_THEME_CHANGED) { add_lang->set_icon(add_var->get_theme_icon(SNAME("Add"), SNAME("EditorIcons"))); add_script->set_icon(add_var->get_theme_icon(SNAME("Add"), SNAME("EditorIcons"))); add_var->set_icon(add_var->get_theme_icon(SNAME("Add"), SNAME("EditorIcons"))); add_ot->set_icon(add_var->get_theme_icon(SNAME("Add"), SNAME("EditorIcons"))); } } void DynamicFontImportSettings::_re_import() { Map main_settings; main_settings["antialiased"] = import_settings_data->get("antialiased"); main_settings["multichannel_signed_distance_field"] = import_settings_data->get("multichannel_signed_distance_field"); main_settings["msdf_pixel_range"] = import_settings_data->get("msdf_pixel_range"); main_settings["msdf_size"] = import_settings_data->get("msdf_size"); main_settings["force_autohinter"] = import_settings_data->get("force_autohinter"); main_settings["hinting"] = import_settings_data->get("hinting"); main_settings["oversampling"] = import_settings_data->get("oversampling"); main_settings["compress"] = import_settings_data->get("compress"); Vector variations; for (TreeItem *vars_item = vars_list_root->get_first_child(); vars_item; vars_item = vars_item->get_next()) { String variation; Ref import_variation_data = vars_item->get_metadata(0); ERR_FAIL_NULL(import_variation_data); String name = vars_item->get_text(0); variation += ("name=" + name); for (Map::Element *E = import_variation_data->settings.front(); E; E = E->next()) { if (!variation.is_empty()) { variation += ","; } variation += (String(E->key()) + "=" + String(E->get())); } variations.push_back(variation); } main_settings["preload/configurations"] = variations; Vector langs_enabled; Vector langs_disabled; for (TreeItem *lang_item = lang_list_root->get_first_child(); lang_item; lang_item = lang_item->get_next()) { bool selected = lang_item->is_checked(0); String name = lang_item->get_text(1); if (selected) { langs_enabled.push_back(name); } else { langs_disabled.push_back(name); } } main_settings["support_overrides/language_enabled"] = langs_enabled; main_settings["support_overrides/language_disabled"] = langs_disabled; Vector scripts_enabled; Vector scripts_disabled; for (TreeItem *script_item = script_list_root->get_first_child(); script_item; script_item = script_item->get_next()) { bool selected = script_item->is_checked(0); String name = script_item->get_text(1); if (selected) { scripts_enabled.push_back(name); } else { scripts_disabled.push_back(name); } } main_settings["support_overrides/script_enabled"] = scripts_enabled; main_settings["support_overrides/script_disabled"] = scripts_disabled; if (!selected_chars.is_empty()) { Vector ranges; char32_t start = selected_chars.front()->get(); for (Set::Element *E = selected_chars.front()->next(); E; E = E->next()) { if (E->prev() && ((E->prev()->get() + 1) != E->get())) { ranges.push_back(String("0x") + String::num_int64(start, 16) + String("-0x") + String::num_int64(E->prev()->get(), 16)); start = E->get(); } } ranges.push_back(String("0x") + String::num_int64(start, 16) + String("-0x") + String::num_int64(selected_chars.back()->get(), 16)); main_settings["preload/char_ranges"] = ranges; } if (!selected_glyphs.is_empty()) { Vector ranges; int32_t start = selected_glyphs.front()->get(); for (Set::Element *E = selected_glyphs.front()->next(); E; E = E->next()) { if (E->prev() && ((E->prev()->get() + 1) != E->get())) { ranges.push_back(String("0x") + String::num_int64(start, 16) + String("-0x") + String::num_int64(E->prev()->get(), 16)); start = E->get(); } } ranges.push_back(String("0x") + String::num_int64(start, 16) + String("-0x") + String::num_int64(selected_glyphs.back()->get(), 16)); main_settings["preload/glyph_ranges"] = ranges; } Dictionary ot_ov; for (TreeItem *ot_item = ot_list_root->get_first_child(); ot_item; ot_item = ot_item->get_next()) { String tag = ot_item->get_text(0); int32_t value = ot_item->get_text(1).to_int(); ot_ov[tag] = value; } main_settings["opentype_feature_overrides"] = ot_ov; if (OS::get_singleton()->is_stdout_verbose()) { print_line("Import settings:"); for (Map::Element *E = main_settings.front(); E; E = E->next()) { print_line(String(" ") + String(E->key()).utf8().get_data() + " == " + String(E->get()).utf8().get_data()); } } EditorFileSystem::get_singleton()->reimport_file_with_custom_parameters(base_path, "font_data_dynamic", main_settings); } void DynamicFontImportSettings::open_settings(const String &p_path) { // Load base font data. Vector data = FileAccess::get_file_as_array(p_path); // Load font for preview. Ref dfont_prev; dfont_prev.instantiate(); dfont_prev->set_data(data); font_preview.instantiate(); font_preview->add_data(dfont_prev); String sample; static const String sample_base = U"12漢字ԱբΑαАбΑαאבابܐܒހށआআਆઆଆஆఆಆആආกิກິༀကႠა한글ሀᎣᐁᚁᚠᜀᜠᝀᝠកᠠᤁᥐAb😀"; for (int i = 0; i < sample_base.length(); i++) { if (dfont_prev->has_char(sample_base[i])) { sample += sample_base[i]; } } if (sample.is_empty()) { sample = dfont_prev->get_supported_chars().substr(0, 6); } font_preview_label->set_text(sample); // Load second copy of font with MSDF disabled for the glyph table and metadata extraction. Ref dfont_main; dfont_main.instantiate(); dfont_main->set_data(data); dfont_main->set_multichannel_signed_distance_field(false); font_main.instantiate(); font_main->add_data(dfont_main); text_edit->add_theme_font_override("font", font_main); base_path = p_path; inspector_vars->edit(nullptr); inspector_general->edit(nullptr); int gww = get_theme_font(SNAME("font"))->get_string_size("00000", get_theme_font_size(SNAME("font_size"))).x + 50; glyph_table->set_column_custom_minimum_width(0, gww); glyph_table->clear(); vars_list->clear(); lang_list->clear(); script_list->clear(); ot_list->clear(); selected_chars.clear(); selected_glyphs.clear(); text_edit->set_text(String()); vars_list_root = vars_list->create_item(); lang_list_root = lang_list->create_item(); script_list_root = script_list->create_item(); ot_list_root = ot_list->create_item(); options_variations.clear(); Dictionary var_list = dfont_main->get_supported_variation_list(); for (int i = 0; i < var_list.size(); i++) { int32_t tag = var_list.get_key_at_index(i); Vector3i value = var_list.get_value_at_index(i); options_variations.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::FLOAT, TS->tag_to_name(tag), PROPERTY_HINT_RANGE, itos(value.x) + "," + itos(value.y) + ",1"), value.z)); } options_variations.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::INT, "size", PROPERTY_HINT_RANGE, "0,127,1"), 16)); options_variations.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::INT, "outline_size", PROPERTY_HINT_RANGE, "0,127,1"), 0)); options_variations.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::INT, "extra_spacing_glyph"), 0)); options_variations.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::INT, "extra_spacing_space"), 0)); import_settings_data->defaults.clear(); for (List::Element *E = options_general.front(); E; E = E->next()) { import_settings_data->defaults[E->get().option.name] = E->get().default_value; } Ref config; config.instantiate(); ERR_FAIL_NULL(config); Error err = config->load(p_path + ".import"); print_verbose("Loading import settings:"); if (err == OK) { List keys; config->get_section_keys("params", &keys); for (List::Element *E = keys.front(); E; E = E->next()) { String key = E->get(); print_verbose(String(" ") + key + " == " + String(config->get_value("params", key))); if (key == "preload/char_ranges") { Vector ranges = config->get_value("params", key); for (int i = 0; i < ranges.size(); i++) { int32_t start, end; Vector tokens = ranges[i].split("-"); if (tokens.size() == 2) { if (!ResourceImporterDynamicFont::_decode_range(tokens[0], start) || !ResourceImporterDynamicFont::_decode_range(tokens[1], end)) { WARN_PRINT("Invalid range: \"" + ranges[i] + "\""); continue; } } else if (tokens.size() == 1) { if (!ResourceImporterDynamicFont::_decode_range(tokens[0], start)) { WARN_PRINT("Invalid range: \"" + ranges[i] + "\""); continue; } end = start; } else { WARN_PRINT("Invalid range: \"" + ranges[i] + "\""); continue; } for (int32_t j = start; j <= end; j++) { selected_chars.insert(j); } } } else if (key == "preload/glyph_ranges") { Vector ranges = config->get_value("params", key); for (int i = 0; i < ranges.size(); i++) { int32_t start, end; Vector tokens = ranges[i].split("-"); if (tokens.size() == 2) { if (!ResourceImporterDynamicFont::_decode_range(tokens[0], start) || !ResourceImporterDynamicFont::_decode_range(tokens[1], end)) { WARN_PRINT("Invalid range: \"" + ranges[i] + "\""); continue; } } else if (tokens.size() == 1) { if (!ResourceImporterDynamicFont::_decode_range(tokens[0], start)) { WARN_PRINT("Invalid range: \"" + ranges[i] + "\""); continue; } end = start; } else { WARN_PRINT("Invalid range: \"" + ranges[i] + "\""); continue; } for (int32_t j = start; j <= end; j++) { selected_glyphs.insert(j); } } } else if (key == "preload/configurations") { Vector variations = config->get_value("params", key); for (int i = 0; i < variations.size(); i++) { TreeItem *vars_item = vars_list->create_item(vars_list_root); ERR_FAIL_NULL(vars_item); vars_item->set_text(0, TTR("Configuration") + " " + itos(i)); vars_item->set_editable(0, true); vars_item->add_button(1, vars_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove Variation")); vars_item->set_button_color(1, 0, Color(1, 1, 1, 0.75)); Ref import_variation_data_custom; import_variation_data_custom.instantiate(); import_variation_data_custom->owner = this; ERR_FAIL_NULL(import_variation_data_custom); for (List::Element *F = options_variations.front(); F; F = F->next()) { import_variation_data_custom->defaults[F->get().option.name] = F->get().default_value; } import_variation_data_custom->options = options_variations; vars_item->set_metadata(0, import_variation_data_custom); Vector variation_tags = variations[i].split(","); for (int j = 0; j < variation_tags.size(); j++) { Vector tokens = variation_tags[j].split("="); if (tokens[0] == "name") { vars_item->set_text(0, tokens[1]); } else if (tokens[0] == "size" || tokens[0] == "outline_size" || tokens[0] == "extra_spacing_space" || tokens[0] == "extra_spacing_glyph") { import_variation_data_custom->set(tokens[0], tokens[1].to_int()); } else { import_variation_data_custom->set(tokens[0], tokens[1].to_float()); } } } } else if (key == "support_overrides/language_enabled") { PackedStringArray _langs = config->get_value("params", key); for (int i = 0; i < _langs.size(); i++) { TreeItem *lang_item = lang_list->create_item(lang_list_root); ERR_FAIL_NULL(lang_item); lang_item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK); lang_item->set_editable(0, true); lang_item->set_checked(0, true); lang_item->set_text(1, _langs[i]); lang_item->set_editable(1, true); lang_item->add_button(2, lang_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove")); } } else if (key == "support_overrides/language_disabled") { PackedStringArray _langs = config->get_value("params", key); for (int i = 0; i < _langs.size(); i++) { TreeItem *lang_item = lang_list->create_item(lang_list_root); ERR_FAIL_NULL(lang_item); lang_item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK); lang_item->set_editable(0, true); lang_item->set_checked(0, false); lang_item->set_text(1, _langs[i]); lang_item->set_editable(1, true); lang_item->add_button(2, lang_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove")); } } else if (key == "support_overrides/script_enabled") { PackedStringArray _scripts = config->get_value("params", key); for (int i = 0; i < _scripts.size(); i++) { TreeItem *script_item = script_list->create_item(script_list_root); ERR_FAIL_NULL(script_item); script_item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK); script_item->set_editable(0, true); script_item->set_checked(0, true); script_item->set_text(1, _scripts[i]); script_item->set_editable(1, true); script_item->add_button(2, lang_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove")); } } else if (key == "support_overrides/script_disabled") { PackedStringArray _scripts = config->get_value("params", key); for (int i = 0; i < _scripts.size(); i++) { TreeItem *script_item = script_list->create_item(script_list_root); ERR_FAIL_NULL(script_item); script_item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK); script_item->set_editable(0, true); script_item->set_checked(0, false); script_item->set_text(1, _scripts[i]); script_item->set_editable(1, true); script_item->add_button(2, lang_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove")); } } else if (key == "opentype_feature_overrides") { Dictionary features = config->get_value("params", key); for (const Variant *ftr = features.next(nullptr); ftr != nullptr; ftr = features.next(ftr)) { TreeItem *ot_item = ot_list->create_item(ot_list_root); ERR_FAIL_NULL(ot_item); int32_t value = features[*ftr]; if (ftr->get_type() == Variant::STRING) { ot_item->set_text(0, *ftr); } else { ot_item->set_text(0, TS->tag_to_name(*ftr)); } ot_item->set_editable(0, false); ot_item->set_text(1, itos(value)); ot_item->set_editable(1, true); ot_item->add_button(2, ot_list->get_theme_icon(SNAME("Remove"), SNAME("EditorIcons")), BUTTON_REMOVE_VAR, false, TTR("Remove")); ot_item->set_button_color(2, 0, Color(1, 1, 1, 0.75)); } } else { Variant value = config->get_value("params", key); import_settings_data->defaults[key] = value; } } } label_glyphs->set_text(TTR("Preloaded glyphs: ") + itos(selected_glyphs.size())); import_settings_data->options = options_general; inspector_general->edit(import_settings_data.ptr()); import_settings_data->notify_property_list_changed(); if (font_preview->get_data_count() > 0) { font_preview->get_data(0)->set_antialiased(import_settings_data->get("antialiased")); font_preview->get_data(0)->set_multichannel_signed_distance_field(import_settings_data->get("multichannel_signed_distance_field")); font_preview->get_data(0)->set_msdf_pixel_range(import_settings_data->get("msdf_pixel_range")); font_preview->get_data(0)->set_msdf_size(import_settings_data->get("msdf_size")); font_preview->get_data(0)->set_force_autohinter(import_settings_data->get("force_autohinter")); font_preview->get_data(0)->set_hinting((TextServer::Hinting)import_settings_data->get("hinting").operator int()); font_preview->get_data(0)->set_oversampling(import_settings_data->get("oversampling")); } font_preview_label->add_theme_font_override("font", font_preview); font_preview_label->update(); menu_ot->clear(); menu_ot_ss->clear(); menu_ot_cv->clear(); menu_ot_cu->clear(); bool have_ss = false; bool have_cv = false; bool have_cu = false; Dictionary features = font_preview->get_feature_list(); for (const Variant *ftr = features.next(nullptr); ftr != nullptr; ftr = features.next(ftr)) { String ftr_name = TS->tag_to_name(*ftr); if (ftr_name.begins_with("stylistic_set_")) { menu_ot_ss->add_item(ftr_name.capitalize(), (int32_t)*ftr); have_ss = true; } else if (ftr_name.begins_with("character_variant_")) { menu_ot_cv->add_item(ftr_name.capitalize(), (int32_t)*ftr); have_cv = true; } else if (ftr_name.begins_with("custom_")) { menu_ot_cu->add_item(ftr_name.replace("custom_", ""), (int32_t)*ftr); have_cu = true; } else { menu_ot->add_item(ftr_name.capitalize(), (int32_t)*ftr); } } if (have_ss) { menu_ot->add_submenu_item(RTR("Stylistic Sets"), "SSMenu"); } if (have_cv) { menu_ot->add_submenu_item(RTR("Character Variants"), "CVMenu"); } if (have_cu) { menu_ot->add_submenu_item(RTR("Custom"), "CUMenu"); } _variations_validate(); popup_centered_ratio(); set_title(vformat(TTR("Advanced Import Settings for '%s'"), base_path.get_file())); } DynamicFontImportSettings *DynamicFontImportSettings::get_singleton() { return singleton; } DynamicFontImportSettings::DynamicFontImportSettings() { singleton = this; options_general.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::BOOL, "antialiased"), true)); options_general.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::BOOL, "multichannel_signed_distance_field", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT | PROPERTY_USAGE_UPDATE_ALL_IF_MODIFIED), true)); options_general.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::INT, "msdf_pixel_range", PROPERTY_HINT_RANGE, "1,100,1"), 8)); options_general.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::INT, "msdf_size", PROPERTY_HINT_RANGE, "1,250,1"), 48)); options_general.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::BOOL, "force_autohinter"), false)); options_general.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::INT, "hinting", PROPERTY_HINT_ENUM, "None,Light,Normal"), 1)); options_general.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::FLOAT, "oversampling", PROPERTY_HINT_RANGE, "0,10,0.1"), 0.0)); options_general.push_back(ResourceImporter::ImportOption(PropertyInfo(Variant::BOOL, "compress", PROPERTY_HINT_NONE, ""), false)); // Popup menus locale_select = memnew(EditorLocaleDialog); locale_select->connect("locale_selected", callable_mp(this, &DynamicFontImportSettings::_lang_add_item)); add_child(locale_select); menu_scripts = memnew(PopupMenu); menu_scripts->set_name("Script"); script_codes = TranslationServer::get_singleton()->get_all_scripts(); for (int i = 0; i < script_codes.size(); i++) { menu_scripts->add_item(TranslationServer::get_singleton()->get_script_name(script_codes[i]) + " (" + script_codes[i] + ")", i); } add_child(menu_scripts); menu_scripts->connect("id_pressed", callable_mp(this, &DynamicFontImportSettings::_script_add_item)); menu_ot = memnew(PopupMenu); add_child(menu_ot); menu_ot->connect("id_pressed", callable_mp(this, &DynamicFontImportSettings::_ot_add_item)); menu_ot_cv = memnew(PopupMenu); menu_ot_cv->set_name("CVMenu"); menu_ot->add_child(menu_ot_cv); menu_ot_cv->connect("id_pressed", callable_mp(this, &DynamicFontImportSettings::_ot_add_item)); menu_ot_ss = memnew(PopupMenu); menu_ot_ss->set_name("SSMenu"); menu_ot->add_child(menu_ot_ss); menu_ot_ss->connect("id_pressed", callable_mp(this, &DynamicFontImportSettings::_ot_add_item)); menu_ot_cu = memnew(PopupMenu); menu_ot_cu->set_name("CUMenu"); menu_ot->add_child(menu_ot_cu); menu_ot_cu->connect("id_pressed", callable_mp(this, &DynamicFontImportSettings::_ot_add_item)); Color warn_color = (EditorNode::get_singleton()) ? EditorNode::get_singleton()->get_gui_base()->get_theme_color(SNAME("warning_color"), SNAME("Editor")) : Color(1, 1, 0); // Root layout VBoxContainer *root_vb = memnew(VBoxContainer); add_child(root_vb); main_pages = memnew(TabContainer); main_pages->set_v_size_flags(Control::SIZE_EXPAND_FILL); main_pages->set_h_size_flags(Control::SIZE_EXPAND_FILL); root_vb->add_child(main_pages); label_warn = memnew(Label); label_warn->set_horizontal_alignment(HORIZONTAL_ALIGNMENT_CENTER); label_warn->set_text(""); root_vb->add_child(label_warn); label_warn->add_theme_color_override("font_color", warn_color); label_warn->hide(); // Page 1 layout: Rendering Options VBoxContainer *page1_vb = memnew(VBoxContainer); page1_vb->set_meta("_tab_name", TTR("Rendering options")); main_pages->add_child(page1_vb); page1_description = memnew(Label); page1_description->set_text(TTR("Select font rendering options:")); page1_description->set_h_size_flags(Control::SIZE_EXPAND_FILL); page1_vb->add_child(page1_description); HSplitContainer *page1_hb = memnew(HSplitContainer); page1_hb->set_v_size_flags(Control::SIZE_EXPAND_FILL); page1_hb->set_h_size_flags(Control::SIZE_EXPAND_FILL); page1_vb->add_child(page1_hb); font_preview_label = memnew(Label); font_preview_label->add_theme_font_size_override("font_size", 200 * EDSCALE); font_preview_label->set_horizontal_alignment(HORIZONTAL_ALIGNMENT_CENTER); font_preview_label->set_vertical_alignment(VERTICAL_ALIGNMENT_CENTER); font_preview_label->set_autowrap_mode(Label::AUTOWRAP_ARBITRARY); font_preview_label->set_clip_text(true); font_preview_label->set_v_size_flags(Control::SIZE_EXPAND_FILL); font_preview_label->set_h_size_flags(Control::SIZE_EXPAND_FILL); page1_hb->add_child(font_preview_label); inspector_general = memnew(EditorInspector); inspector_general->set_v_size_flags(Control::SIZE_EXPAND_FILL); inspector_general->set_custom_minimum_size(Size2(300 * EDSCALE, 250 * EDSCALE)); inspector_general->connect("property_edited", callable_mp(this, &DynamicFontImportSettings::_main_prop_changed)); page1_hb->add_child(inspector_general); // Page 2 layout: Configurations VBoxContainer *page2_vb = memnew(VBoxContainer); page2_vb->set_meta("_tab_name", TTR("Sizes and variations")); main_pages->add_child(page2_vb); page2_description = memnew(Label); page2_description->set_text(TTR("Add font size, variation coordinates, and extra spacing combinations to pre-render:")); page2_description->set_h_size_flags(Control::SIZE_EXPAND_FILL); page2_description->set_autowrap_mode(Label::AUTOWRAP_WORD_SMART); page2_vb->add_child(page2_description); HSplitContainer *page2_hb = memnew(HSplitContainer); page2_hb->set_v_size_flags(Control::SIZE_EXPAND_FILL); page2_hb->set_h_size_flags(Control::SIZE_EXPAND_FILL); page2_vb->add_child(page2_hb); VBoxContainer *page2_side_vb = memnew(VBoxContainer); page2_hb->add_child(page2_side_vb); HBoxContainer *page2_hb_vars = memnew(HBoxContainer); page2_side_vb->add_child(page2_hb_vars); label_vars = memnew(Label); page2_hb_vars->add_child(label_vars); label_vars->set_horizontal_alignment(HORIZONTAL_ALIGNMENT_CENTER); label_vars->set_h_size_flags(Control::SIZE_EXPAND_FILL); label_vars->set_text(TTR("Configuration:")); add_var = memnew(Button); page2_hb_vars->add_child(add_var); add_var->set_tooltip(TTR("Add configuration")); add_var->set_icon(add_var->get_theme_icon(SNAME("Add"), SNAME("EditorIcons"))); add_var->connect("pressed", callable_mp(this, &DynamicFontImportSettings::_variation_add)); vars_list = memnew(Tree); page2_side_vb->add_child(vars_list); vars_list->set_custom_minimum_size(Size2(300 * EDSCALE, 0)); vars_list->set_hide_root(true); vars_list->set_columns(2); vars_list->set_column_expand(0, true); vars_list->set_column_custom_minimum_width(0, 80 * EDSCALE); vars_list->set_column_expand(1, false); vars_list->set_column_custom_minimum_width(1, 50 * EDSCALE); vars_list->connect("item_selected", callable_mp(this, &DynamicFontImportSettings::_variation_selected)); vars_list->connect("button_pressed", callable_mp(this, &DynamicFontImportSettings::_variation_remove)); vars_list->set_v_size_flags(Control::SIZE_EXPAND_FILL); inspector_vars = memnew(EditorInspector); inspector_vars->set_v_size_flags(Control::SIZE_EXPAND_FILL); inspector_vars->connect("property_edited", callable_mp(this, &DynamicFontImportSettings::_variation_changed)); page2_hb->add_child(inspector_vars); // Page 3 layout: Text to select glyphs VBoxContainer *page3_vb = memnew(VBoxContainer); page3_vb->set_meta("_tab_name", TTR("Glyphs from the text")); main_pages->add_child(page3_vb); page3_description = memnew(Label); page3_description->set_text(TTR("Enter a text to shape and add all required glyphs to pre-render list:")); page3_description->set_h_size_flags(Control::SIZE_EXPAND_FILL); page3_description->set_autowrap_mode(Label::AUTOWRAP_WORD_SMART); page3_vb->add_child(page3_description); HBoxContainer *ot_hb = memnew(HBoxContainer); page3_vb->add_child(ot_hb); ot_hb->set_h_size_flags(Control::SIZE_EXPAND_FILL); Label *label_ed_ftr = memnew(Label); ot_hb->add_child(label_ed_ftr); label_ed_ftr->set_text(TTR("OpenType features:")); ftr_edit = memnew(LineEdit); ot_hb->add_child(ftr_edit); ftr_edit->connect("text_changed", callable_mp(this, &DynamicFontImportSettings::_change_text_opts)); ftr_edit->set_h_size_flags(Control::SIZE_EXPAND_FILL); Label *label_ed_lang = memnew(Label); ot_hb->add_child(label_ed_lang); label_ed_lang->set_text(TTR("Text language:")); lang_edit = memnew(LineEdit); ot_hb->add_child(lang_edit); lang_edit->connect("text_changed", callable_mp(this, &DynamicFontImportSettings::_change_text_opts)); lang_edit->set_h_size_flags(Control::SIZE_EXPAND_FILL); text_edit = memnew(TextEdit); page3_vb->add_child(text_edit); text_edit->set_v_size_flags(Control::SIZE_EXPAND_FILL); text_edit->set_h_size_flags(Control::SIZE_EXPAND_FILL); HBoxContainer *text_hb = memnew(HBoxContainer); page3_vb->add_child(text_hb); text_hb->set_h_size_flags(Control::SIZE_EXPAND_FILL); label_glyphs = memnew(Label); text_hb->add_child(label_glyphs); label_glyphs->set_text(TTR("Preloaded glyphs: ") + itos(0)); label_glyphs->set_custom_minimum_size(Size2(50 * EDSCALE, 0)); Button *btn_fill = memnew(Button); text_hb->add_child(btn_fill); btn_fill->set_text(TTR("Shape text and add glyphs")); btn_fill->connect("pressed", callable_mp(this, &DynamicFontImportSettings::_glyph_text_selected)); Button *btn_clear = memnew(Button); text_hb->add_child(btn_clear); btn_clear->set_text(TTR("Clear glyph list")); btn_clear->connect("pressed", callable_mp(this, &DynamicFontImportSettings::_glyph_clear)); // Page 4 layout: Character map VBoxContainer *page4_vb = memnew(VBoxContainer); page4_vb->set_meta("_tab_name", TTR("Glyphs from the character map")); main_pages->add_child(page4_vb); page4_description = memnew(Label); page4_description->set_text(TTR("Add or remove additional glyphs from the character map to pre-render list:\nNote: Some stylistic alternatives and glyph variants do not have one-to-one correspondence to character, and not shown in this map, use \"Glyphs from the text\" to add these.")); page4_description->set_h_size_flags(Control::SIZE_EXPAND_FILL); page4_description->set_autowrap_mode(Label::AUTOWRAP_WORD_SMART); page4_vb->add_child(page4_description); HSplitContainer *glyphs_split = memnew(HSplitContainer); glyphs_split->set_v_size_flags(Control::SIZE_EXPAND_FILL); glyphs_split->set_h_size_flags(Control::SIZE_EXPAND_FILL); page4_vb->add_child(glyphs_split); glyph_table = memnew(Tree); glyphs_split->add_child(glyph_table); glyph_table->set_custom_minimum_size(Size2((30 * 16 + 100) * EDSCALE, 0)); glyph_table->set_columns(17); glyph_table->set_column_expand(0, false); glyph_table->set_hide_root(true); glyph_table->set_allow_reselect(true); glyph_table->set_select_mode(Tree::SELECT_SINGLE); glyph_table->connect("item_activated", callable_mp(this, &DynamicFontImportSettings::_glyph_selected)); glyph_table->set_column_titles_visible(true); for (int i = 0; i < 16; i++) { glyph_table->set_column_title(i + 1, String::num_int64(i, 16)); } glyph_table->add_theme_style_override("selected", glyph_table->get_theme_stylebox(SNAME("bg"))); glyph_table->add_theme_style_override("selected_focus", glyph_table->get_theme_stylebox(SNAME("bg"))); glyph_table->add_theme_constant_override("hseparation", 0); glyph_table->set_h_size_flags(Control::SIZE_EXPAND_FILL); glyph_table->set_v_size_flags(Control::SIZE_EXPAND_FILL); glyph_tree = memnew(Tree); glyphs_split->add_child(glyph_tree); glyph_tree->set_custom_minimum_size(Size2(300 * EDSCALE, 0)); glyph_tree->set_columns(2); glyph_tree->set_hide_root(true); glyph_tree->set_column_expand(0, false); glyph_tree->set_column_expand(1, true); glyph_tree->set_column_custom_minimum_width(0, 120 * EDSCALE); glyph_tree->connect("item_activated", callable_mp(this, &DynamicFontImportSettings::_range_edited)); glyph_tree->connect("item_selected", callable_mp(this, &DynamicFontImportSettings::_range_selected)); glyph_tree->set_v_size_flags(Control::SIZE_EXPAND_FILL); glyph_root = glyph_tree->create_item(); for (int i = 0; !unicode_ranges[i].name.is_empty(); i++) { _add_glyph_range_item(unicode_ranges[i].start, unicode_ranges[i].end, unicode_ranges[i].name); } // Page 4 layout: Metadata override VBoxContainer *page5_vb = memnew(VBoxContainer); page5_vb->set_meta("_tab_name", TTR("Metadata override")); main_pages->add_child(page5_vb); page5_description = memnew(Label); page5_description->set_text(TTR("Add or remove language and script support overrides, to control fallback font selection order:")); page5_description->set_h_size_flags(Control::SIZE_EXPAND_FILL); page5_description->set_autowrap_mode(Label::AUTOWRAP_WORD_SMART); page5_vb->add_child(page5_description); HBoxContainer *hb_lang = memnew(HBoxContainer); page5_vb->add_child(hb_lang); label_langs = memnew(Label); label_langs->set_horizontal_alignment(HORIZONTAL_ALIGNMENT_CENTER); label_langs->set_h_size_flags(Control::SIZE_EXPAND_FILL); label_langs->set_text(TTR("Language support overrides")); hb_lang->add_child(label_langs); add_lang = memnew(Button); hb_lang->add_child(add_lang); add_lang->set_tooltip(TTR("Add language override")); add_lang->set_icon(add_var->get_theme_icon(SNAME("Add"), SNAME("EditorIcons"))); add_lang->connect("pressed", callable_mp(this, &DynamicFontImportSettings::_lang_add)); lang_list = memnew(Tree); page5_vb->add_child(lang_list); lang_list->set_hide_root(true); lang_list->set_columns(3); lang_list->set_column_expand(0, false); // Check lang_list->set_column_custom_minimum_width(0, 50 * EDSCALE); lang_list->set_column_expand(1, true); lang_list->set_column_custom_minimum_width(1, 80 * EDSCALE); lang_list->set_column_expand(2, false); lang_list->set_column_custom_minimum_width(2, 50 * EDSCALE); lang_list->connect("button_pressed", callable_mp(this, &DynamicFontImportSettings::_lang_remove)); lang_list->set_v_size_flags(Control::SIZE_EXPAND_FILL); HBoxContainer *hb_script = memnew(HBoxContainer); page5_vb->add_child(hb_script); label_script = memnew(Label); label_script->set_horizontal_alignment(HORIZONTAL_ALIGNMENT_CENTER); label_script->set_h_size_flags(Control::SIZE_EXPAND_FILL); label_script->set_text(TTR("Script support overrides")); hb_script->add_child(label_script); add_script = memnew(Button); hb_script->add_child(add_script); add_script->set_tooltip(TTR("Add script override")); add_script->set_icon(add_var->get_theme_icon(SNAME("Add"), SNAME("EditorIcons"))); add_script->connect("pressed", callable_mp(this, &DynamicFontImportSettings::_script_add)); script_list = memnew(Tree); page5_vb->add_child(script_list); script_list->set_hide_root(true); script_list->set_columns(3); script_list->set_column_expand(0, false); script_list->set_column_custom_minimum_width(0, 50 * EDSCALE); script_list->set_column_expand(1, true); script_list->set_column_custom_minimum_width(1, 80 * EDSCALE); script_list->set_column_expand(2, false); script_list->set_column_custom_minimum_width(2, 50 * EDSCALE); script_list->connect("button_pressed", callable_mp(this, &DynamicFontImportSettings::_script_remove)); script_list->set_v_size_flags(Control::SIZE_EXPAND_FILL); HBoxContainer *hb_ot = memnew(HBoxContainer); page5_vb->add_child(hb_ot); label_ot = memnew(Label); label_ot->set_horizontal_alignment(HORIZONTAL_ALIGNMENT_CENTER); label_ot->set_h_size_flags(Control::SIZE_EXPAND_FILL); label_ot->set_text(TTR("OpenType feature overrides")); hb_ot->add_child(label_ot); add_ot = memnew(Button); hb_ot->add_child(add_ot); add_ot->set_tooltip(TTR("Add feature override")); add_ot->set_icon(add_var->get_theme_icon(SNAME("Add"), SNAME("EditorIcons"))); add_ot->connect("pressed", callable_mp(this, &DynamicFontImportSettings::_ot_add)); ot_list = memnew(Tree); page5_vb->add_child(ot_list); ot_list->set_hide_root(true); ot_list->set_columns(3); ot_list->set_column_expand(0, true); ot_list->set_column_custom_minimum_width(0, 80 * EDSCALE); ot_list->set_column_expand(1, true); ot_list->set_column_custom_minimum_width(1, 80 * EDSCALE); ot_list->set_column_expand(2, false); ot_list->set_column_custom_minimum_width(2, 50 * EDSCALE); ot_list->connect("button_pressed", callable_mp(this, &DynamicFontImportSettings::_ot_remove)); ot_list->set_v_size_flags(Control::SIZE_EXPAND_FILL); // Common import_settings_data.instantiate(); import_settings_data->owner = this; get_ok_button()->set_text(TTR("Reimport")); get_cancel_button()->set_text(TTR("Close")); }