Skip to content

Latest commit

 

History

History
233 lines (201 loc) · 15.8 KB

develop.md

File metadata and controls

233 lines (201 loc) · 15.8 KB

How to develop?

  • Toolkit can probably run on W7, but Qt Creator, main development environment, absolutely needs W10
  • Download/find all tools for building.
  • Ensure that your Qt has MinGW 11+. Or make a custom kit of some MinGW Qt and MSYS compiler; tested Qt 6.1.3, 6.7.0. I actually use 6.1.3 as the last supporting W7 and fast enough on W10.
  • Compile and run AutoBuilder.
  • Move UcAuto.cpp to Unicodia/Uc.
  • Compile Unicodia.
  • Then you need to recreate working directory, symlinking all files/folders existing in original Unicodia.
  • Congratulations! You can now run Unicodia directly from Qt Creator!

Hidden commands

  • Ctrl+Shift+T — tofu stats
  • F12 — reload translation from disk. Locale does NOT reload
  • Ctrl+F12 — dump Library tile info to opt.xml, for access optimization. After placing it into NotoEmoji and running tape.bat the first chunk of emoji.zip will contains all emoji needed for tiles
  • Ctrl+Shift+Q — test emoji repainting engine

Update font: what to pay attention to?

  • First of all tofu stats (About → Tofu statistics / Ctrl+T): there should be no new tofu
  • Then display of characters and samples
  • If some character hangs → check whether it was fixed professionally. Most hanging characters are done in UnicodiaFunky font, some in *Fixup.
  • Check license what was done by author of Unicodia, and whether we need it now.

How to update emoji?

  • Simple emoji (no skintones)
    • Ensure that it has no <clipPath>, edit if needed.
      • Originals is directory for original SVGs, Remakes if for remade but unoptimized
    • It’s better to run SvgCleaner on it
    • If emoji-related library data changed → copy opt.xml from AutoBuilder to NotoEmoji
    • Make sure Unicodia is closed, run tape.bat
  • Two or more skintones (e.g. interracial kiss/handshake): same. They are neither repainted nor checked.
  • One skintone (one person, same-race handshake…), repainted versions are stored
    • After that press Ctrl+Shift+Q in Unicodia.
    • Copy resulting BMP files to MiscFiles\SkinToneQa. You can do it manually, or using “Good” button in QaCompareBmp tool.
  • One skintone, repainted versions are created on-the-fly (e.g. because of <clipPath>)
    • So-called “preparation”: we need to remember what was originally
      • First copy ORIGINAL files with all skintones to NotoEmoji
      • It’s better to run SvgCleaner
      • Make sure Unicodia is closed, run tape.bat
      • Press Ctrl+Shift+Q in Unicodia.
      • Copy resulting BMP files to MiscFiles\SkinToneQa. You can do it manually, or using “Good” button in QaCompareBmp tool.
    • Then normal work commences.
      • It’s absolutely needed to change nose colour E59600 → E49600, to prevent confusion with ears
      • Remove original files, copy your remake.
      • It’s absolutely needed to run SvgCleaner, repainting engine relies on it (e.g. lowercase, XML attributes rather than CSS)
      • Make sure Unicodia is closed, run tape.bat
      • Press Ctrl+Shift+Q in Unicodia.
      • Check what’s different using QaCompareBmp. If some colour is repainted erroneously (teacher/blackboard, fireman/insignia) → change colour a bit and repeat
      • If you are satisfied with difference → press “Good” button in QaCompareBmp.

How to update to next Unicode

Adapting to new Unicode data

  • Go to MiscFiles/RawData, modify load.bat and load bases of the next Unicode
  • Make AutoBuilder run (who knows what will be added to the next bases)
    • Warning, while Unicode is alpha/beta, AutoBuilder may contain manual patches: see bug #349
  • Run it
  • Copy all UcAuto* to Unicodia/Uc
  • Make Unicodia compile, first of all manually update version/script/block data:
    • VERSIONS: add new Unicode and emoji versions (EcVersion). Write when it’s going to be released. If it’s beta, add Vfg::BETA flag
    • BLOCKS: they are in UcBlocks.cpp. While the block is full of tofu, add Ifg::MISSING flag to synthIcon field
    • SCRIPTS: they are in UcScripts.cpp. Do not mess up alphabetic order!
  • Now Unicodia runs somehow!!

Urgent first-party data: don’t release until you fill all somehow

  • Block extension history:
    • Add the new version to BlockExtensionHistory
    • Run it, check if the block was extended/shrunk
    • Write that to UcBlocks.cpp
      • Sample: during U2…16 three blocks were extended (Tibt, Ahom, Egyp format) and one shrunk (Tang supp)
    • As an alternative, you may check what changed in the newest Blocks.txt, as older data is already written
  • Texts
    • Run UTranslator, open en.uorig
    • For example, you decided to add Klingon script:
      • Go to Script folder, press Original → Add group. Ensure that folder is in Script
      • Name it Klin
      • Better move the group into its alphabetic place using commands Original → Move up/Move down
      • Press Original → Add text. Ensure that text is in Klin
      • Name it Name, write Klingon
      • Also add Lang and Text texts
    • If you know languages, translate to Russian/Ukrainian using the same UTranslator
      • File → Update data, Go → Find warnings → All, and close those changes
    • If some blocks should be logically near, but specific language makes a mess (Greek → Ancient Greek numbers), add an ellipsis to the beginning using locale.xml.
      • This ellipsis will automatically go if you sort blocks by code
  • Decapitalization rules: “LATIN CAPITAL LETTER A” → “Latin capital letter A”
    • They are in AutoBuilder/data.cpp
    • See Decapitalization rules below
    • Of course, run AutoBuilder
    • Always compare before and after with some diff tool! Heal one thing and hurt another — that’s frequent!
    • Copy CPP files again
  • Block icons
    • There are two ways to make a block icon, SVG and programmatic. I’ll talk about the former, default
    • 16×16 px, scale 1, no background, foreground is arbitrary. Put them to Resources/Scripts, optimize somehow, write a new filename to scripts.qrc
      • Do not deviate w/o reason from foreground of synthesized icons. MISSING → white FFF, international → black 000, Asia → Tango mid orange F57900, the rest continents → see SVGs.
      • What to draw if the block is still full of tofu? In U14 and U15 I drew “U ↵ 14”, in U16 I drew very crude images of characters.
    • Go to UcBlocks.cpp. Write in synthIcon field what characters (up to two) you drew and how to align that picture to pixels on HiDPI
      • Example 1: Devanagari 0900 has upper line at 2px and vertical stem at 9px — thus coordinates of cardinal stem are (9, 2)
      • Example 2: Basic Latin 0000 has vertical stem at 5px and horizontal line at 15px, but to make it aligned to pixels, I greatly shifted to the right — I think that imbalance is 0.3px to the right.
      • Example 3: Kawi 11F00 has almost square letter with 2px frame — decided to make a custom engine that deflates rectangle a bit, aligning all three sides to pixels
      • Example 4: Tangut 17000 has historical icon and modern font — thus use Ifg::HISTORICAL

When Unicode transitions from beta to release:

  • Go to MiscFiles/RawData and change addresses to final ones
  • Load data, run AutoBuilder, replace files
  • Delete LAST cached Blocks.txt from BlockExtensionHistory build directory
  • Check block extension history once again
  • Do not forget to remove Vfg::BETA flag

Adding really optional data Unicodia can live without

  • Placeholders for emoji: see above
  • Version texts. If the last version tells a story (an emergency release, a key person died, someone tried to hijack Unicode etc)…
    • Add Vfg::TEXT flag to this version
    • Write about this version in UTranslator. Again, original is English, the rest are translations
  • Fonts
    • Find somehow, write in UcData.h/UcFonts.cpp
    • There are two ways to write a font, tie to script or tie to block. Block is stronger, and I tie to script if possible
    • Modify block icon to permanent, see above
    • Do not forget to remove Ifg::MISSING
    • When you remove this flag, synthesized icon will change, too. Do the following.
      • Find one of your new characters OR open Terms and scroll to Versions
      • Open your new Unicode version, click a tiny “Find” 🔍 icon next to # of new characters (now it’s the only, but things may change)
      • Find your block in search results, you’ll see a SYNTHESIZED icon. Make sure that icon is in harmony with hand-drawn 16×16, change synthIcon field if needed
  • GlyphWiki’s placeholder of CJK fonts
    • (See GlyphWiki below)
  • Latin in Library
    • AutoBuilder has some sort of forget-checker, and when the tofu is filled, you may write it to MiscFiles/library.xml and reduce number of forgotten characters

Decapitalization rules

I repeat, decapitalization rules convert “LATIN CAPITAL LETTER A” → “Latin capital letter A”.

There are several types of decapitalization rules, but the most frequent three are dictionary, idioms, script-specific and exceptions.

Exceptions

  • It means: human manually wrote how the text should decapitalize.
  • If exception triggers, nothing else works.
  • If string’s case is mixed, the rest rules are just skipped, but exceptions still work (see extractSynonym function).
  • Otherwise first dictionary works, then idioms.
  • Script-specific rules are just inserted somewhere.

Dictionary

Some types of dictionary entries:

  • Scripts (TRIG_SCRIPT, e.g. CYRILLIC): capitalize, trigger script-based rules
  • Keywords (TRIG_CAP_NEXT, e.g. TONE): capitalize next to it
  • Adjectives (PART_ADJECTIVE, e.g. BREATHY): if capitalized, push capitalization to the right
    • If chain of capitalization occurred, e.g. “letter Breathy long Foobar”, only the first adjective is capitalized
  • Nouns (PART_NOUN, e.g. COMMA): if capitalization pushed, stop and do not capitalize: “letter Breathy long Foobar”, but “letter Breathy long comma”

Script-specific

Right now only Cuneiform uses script-specific rules: e.g. “cuneiform sign LAM times KUR plus RU”. See line if (words[0].original == "CUNEIFORM"sv) {

Examples

  1. Garay capital letter Ca
    • You should turn script mode on (it’ll automatically capitalize CA using keyword LETTER), and need one rule “GARAY”, flag TRIG_SCRIPT
  2. Myanmar Eastern Pwo Karen digit Zero
    • Myanmar/Karen/Zero already work, but need to do something with “Eastern Pwo”, and need two-word script EASTERN+PWO
  3. Ol Onal letter O
    • Already got two-word script “Ol Chiki”, but it’s possible to add “Ol Onal” to the same dictionary entry (AFAIR, up to 3 or 4 options)
  4. Tulu-Tigalari letter A
    • If you just write as in example 1, it’ll automatically capitalize to “Tulu-tigalari”, and need a custom capitalization “Tulu-Tigalari”
  5. Garay vowel length mark
    • Add idiom VOWEL LENGTH MARK
  6. Tulu-Tigalari Vedic tone Svarita
    • Gagged with idiom VEDIC TONE
  7. logic gate Or
    • You may make idiom LOGIC GATE. Or may add two exceptions
  8. musical symbol With fingernails
    • Our rules misfire here (musical symbol “With fingernails” if have quotes), a clear exception!
  9. Vedic tone Svarita
    • Exception, but why? It’s found in NamesList.txt in mixed case
  10. Toto letter Breathy Iu
    • Two rules here, script TOTO and adjective BREATHY

If you want to rebuild GlyphWiki font

Need FontForge, TtfAutoHint, Qt Creator.

  1. Data is in Font-Source/GlyphWiki
  2. Font consists of manual and automatic glyphs. Manual ones are in UnicodiaHan.sfd and are taken from various sources. But we need automatic ones now!
  3. Rename GlyphWiki font temporarily. Open Unicodia, find Tofu Statistics, extract CJK tofu. Put to hani-tofu.txt. (See comments in hani-tofu.txt, we also need Kangxi radicals just because there’s no good font!
  4. Run GwLoader. Put contents of hani-tofu.txt to editing box. Load data. Empty Font-Source/GlyphWiki/Raw (except files starting with ! and $, they are just tests), move resulting SVGs here. This directory is versioned!
  5. Run GwRemake. Get AutoRemade directory. This directory is not versioned, just exists, and may contain lots of harmless garbage!
  6. Open load-glyphs.py, set path to your TtfAutoHint
  7. Open UnicodiaHan.sfd. Run script load-glyphs.py. Do not save!
  8. VCS (Git etc) should notice changes in Font-Source/GlyphWiki/Raw and UnicodiaHan.ttf.

What do utilities do?

  • AutoBuilder — build UcAuto.cpp from Unicode base.
    • Warning: transition to older/newer Unicode requires a bit of handwork.
  • BlockExtensionHistory:
  • GwLoader — loads glyphs from GlyphWiki
  • GwRemake — does an auto-remake of GlyphWiki
  • PanoseTool — early tool that used to remove fonts’ declared script support. Left for history. Current Unicodia uses custom font loading code based on PanoseTool + font matching flags.
  • QaCompareBmp — to check what changed in single-skintone emoji (e.g. changed repainting engine, stored→repainted)
  • SmartCopy — copies the file if it actually differs.
  • TapeMaker — creates a tape file for graphic emoji.
  • UTranslator — translation tool.

Unicodia emoji style guide

If you decide to make a new character, there are two decisions available: 1) Mimic existing style; 2) Use new style. If you decide the latter (right now just three emoji have new style, placard and pushing hands)…

I discovered that most of professional emoji… do not work! Do not work because they have too much detail and too little contrast. The last professional emoji that really worked is… 2018 Microsoft!

High priority: emoji that are really badly grasped.

Canvas size: 64×64. For oblong/round object: leave 2px from edge. For square object: leave 6px from edge. For faces: leave 7px from edge.

Colour scheme: GNOME Human Interface Guidelines unless REALLY needed (e.g. military helmet). Use plain colours. Use black #000 and white #FFF sparingly, mainly for outlines and small details. Hands/people use Noto’s colours for skin.

Projection: frontal/sideways unless the object is better identified in other projection. Bilateral animals, vehicles and people face to the left.

Style: avoid toyish look. Machinery should be based on real-life models.

Outline style: outer 2px, the darkest colour of palette (unless outlining grey/white/luminous thing). Inner may be thinner if present. Butts are round for natural objects and square/slanted for boxy ones. Same for corners.

Outline presence: outer may be partly omitted to gain more dynamic look; the object should continue from the OUTER edge of outline (like in OpenMoji). Inner lines are mostly absent. White things are outlined more aggressively. Truncation lines are never outlined. You may omit outline of small outer details (e.g. hooves); but if they really define object’s shape, it’s better to outline them at least partly, at least with 1px.

Grid: 1px; extremum nodes are snapped.

Flags

These rules are intended to work around renderer deficiencies. Emoji are intended to be drawn in small size, and rendering SVG leaves half-pixel lines. Also white and black are main colours everywhere.

  • Joining fields:
    • Fields always overlap to at least 1/16 width. Example (England): you just draw a white rectangle, and a cross over it.
    • When fields are large, that overlap should shun the edge: Example (Ukraine, Poland and other bicolours): one half is a rectangle, the other is a hexagon, with two vertices making that overlap.
  • Colour scheme:
    • White is pure white FFF, black is Twitter black 141414.
  • White fields adjacent to background:
    • Small (Norway, US): just white
    • Medium, both ends are of the same colour (Argentina, Oman): outline of that colour
    • Medium, ends have different colour (France, Luxembourg): grey outline
    • Large (England, Slovakia): also grey outline
    • Difference: small = white field makes a negative space rather than a hole; large = coloured outline makes too much visual noise