| OLD | NEW |
| (Empty) |
| 1 // Copyright 2014 PDFium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com | |
| 6 | |
| 7 #include "core/include/fxcrt/fx_ucd.h" | |
| 8 #include "core/src/fxcrt/fx_arabic.h" | |
| 9 | |
| 10 namespace { | |
| 11 | |
| 12 const FX_ARBFORMTABLE g_FX_ArabicFormTables[] = { | |
| 13 {0xFE81, 0xFE82, 0xFE81, 0xFE82}, | |
| 14 {0xFE83, 0xFE84, 0xFE83, 0xFE84}, | |
| 15 {0xFE85, 0xFE86, 0xFE85, 0xFE86}, | |
| 16 {0xFE87, 0xFE88, 0xFE87, 0xFE88}, | |
| 17 {0xFE89, 0xFE8A, 0xFE8B, 0xFE8C}, | |
| 18 {0xFE8D, 0xFE8E, 0xFE8D, 0xFE8E}, | |
| 19 {0xFE8F, 0xFE90, 0xFE91, 0xFE92}, | |
| 20 {0xFE93, 0xFE94, 0xFE93, 0xFE94}, | |
| 21 {0xFE95, 0xFE96, 0xFE97, 0xFE98}, | |
| 22 {0xFE99, 0xFE9A, 0xFE9B, 0xFE9C}, | |
| 23 {0xFE9D, 0xFE9E, 0xFE9F, 0xFEA0}, | |
| 24 {0xFEA1, 0xFEA2, 0xFEA3, 0xFEA4}, | |
| 25 {0xFEA5, 0xFEA6, 0xFEA7, 0xFEA8}, | |
| 26 {0xFEA9, 0xFEAA, 0xFEA9, 0xFEAA}, | |
| 27 {0xFEAB, 0xFEAC, 0xFEAB, 0xFEAC}, | |
| 28 {0xFEAD, 0xFEAE, 0xFEAD, 0xFEAE}, | |
| 29 {0xFEAF, 0xFEB0, 0xFEAF, 0xFEB0}, | |
| 30 {0xFEB1, 0xFEB2, 0xFEB3, 0xFEB4}, | |
| 31 {0xFEB5, 0xFEB6, 0xFEB7, 0xFEB8}, | |
| 32 {0xFEB9, 0xFEBA, 0xFEBB, 0xFEBC}, | |
| 33 {0xFEBD, 0xFEBE, 0xFEBF, 0xFEC0}, | |
| 34 {0xFEC1, 0xFEC2, 0xFEC3, 0xFEC4}, | |
| 35 {0xFEC5, 0xFEC6, 0xFEC7, 0xFEC8}, | |
| 36 {0xFEC9, 0xFECA, 0xFECB, 0xFECC}, | |
| 37 {0xFECD, 0xFECE, 0xFECF, 0xFED0}, | |
| 38 {0x063B, 0x063B, 0x063B, 0x063B}, | |
| 39 {0x063C, 0x063C, 0x063C, 0x063C}, | |
| 40 {0x063D, 0x063D, 0x063D, 0x063D}, | |
| 41 {0x063E, 0x063E, 0x063E, 0x063E}, | |
| 42 {0x063F, 0x063F, 0x063F, 0x063F}, | |
| 43 {0x0640, 0x0640, 0x0640, 0x0640}, | |
| 44 {0xFED1, 0xFED2, 0xFED3, 0xFED4}, | |
| 45 {0xFED5, 0xFED6, 0xFED7, 0xFED8}, | |
| 46 {0xFED9, 0xFEDA, 0xFEDB, 0xFEDC}, | |
| 47 {0xFEDD, 0xFEDE, 0xFEDF, 0xFEE0}, | |
| 48 {0xFEE1, 0xFEE2, 0xFEE3, 0xFEE4}, | |
| 49 {0xFEE5, 0xFEE6, 0xFEE7, 0xFEE8}, | |
| 50 {0xFEE9, 0xFEEA, 0xFEEB, 0xFEEC}, | |
| 51 {0xFEED, 0xFEEE, 0xFEED, 0xFEEE}, | |
| 52 {0xFEEF, 0xFEF0, 0xFBFE, 0xFBFF}, | |
| 53 {0xFEF1, 0xFEF2, 0xFEF3, 0xFEF4}, | |
| 54 {0x064B, 0x064B, 0x064B, 0x064B}, | |
| 55 {0x064C, 0x064C, 0x064C, 0x064C}, | |
| 56 {0x064D, 0x064D, 0x064D, 0x064D}, | |
| 57 {0x064E, 0x064E, 0x064E, 0x064E}, | |
| 58 {0x064F, 0x064F, 0x064F, 0x064F}, | |
| 59 {0x0650, 0x0650, 0x0650, 0x0650}, | |
| 60 {0x0651, 0x0651, 0x0651, 0x0651}, | |
| 61 {0x0652, 0x0652, 0x0652, 0x0652}, | |
| 62 {0x0653, 0x0653, 0x0653, 0x0653}, | |
| 63 {0x0654, 0x0654, 0x0654, 0x0654}, | |
| 64 {0x0655, 0x0655, 0x0655, 0x0655}, | |
| 65 {0x0656, 0x0656, 0x0656, 0x0656}, | |
| 66 {0x0657, 0x0657, 0x0657, 0x0657}, | |
| 67 {0x0658, 0x0658, 0x0658, 0x0658}, | |
| 68 {0x0659, 0x0659, 0x0659, 0x0659}, | |
| 69 {0x065A, 0x065A, 0x065A, 0x065A}, | |
| 70 {0x065B, 0x065B, 0x065B, 0x065B}, | |
| 71 {0x065C, 0x065C, 0x065C, 0x065C}, | |
| 72 {0x065D, 0x065D, 0x065D, 0x065D}, | |
| 73 {0x065E, 0x065E, 0x065E, 0x065E}, | |
| 74 {0x065F, 0x065F, 0x065F, 0x065F}, | |
| 75 {0x0660, 0x0660, 0x0660, 0x0660}, | |
| 76 {0x0661, 0x0661, 0x0661, 0x0661}, | |
| 77 {0x0662, 0x0662, 0x0662, 0x0662}, | |
| 78 {0x0663, 0x0663, 0x0663, 0x0663}, | |
| 79 {0x0664, 0x0664, 0x0664, 0x0664}, | |
| 80 {0x0665, 0x0665, 0x0665, 0x0665}, | |
| 81 {0x0666, 0x0666, 0x0666, 0x0666}, | |
| 82 {0x0667, 0x0667, 0x0667, 0x0667}, | |
| 83 {0x0668, 0x0668, 0x0668, 0x0668}, | |
| 84 {0x0669, 0x0669, 0x0669, 0x0669}, | |
| 85 {0x066A, 0x066A, 0x066A, 0x066A}, | |
| 86 {0x066B, 0x066B, 0x066B, 0x066B}, | |
| 87 {0x066C, 0x066C, 0x066C, 0x066C}, | |
| 88 {0x066D, 0x066D, 0x066D, 0x066D}, | |
| 89 {0x066E, 0x066E, 0x066E, 0x066E}, | |
| 90 {0x066F, 0x066F, 0x066F, 0x066F}, | |
| 91 {0x0670, 0x0670, 0x0670, 0x0670}, | |
| 92 {0xFB50, 0xFB51, 0xFB50, 0xFB51}, | |
| 93 {0x0672, 0x0672, 0x0672, 0x0672}, | |
| 94 {0x0673, 0x0673, 0x0673, 0x0673}, | |
| 95 {0x0674, 0x0674, 0x0674, 0x0674}, | |
| 96 {0x0675, 0x0675, 0x0675, 0x0675}, | |
| 97 {0x0676, 0x0676, 0x0676, 0x0676}, | |
| 98 {0x0677, 0x0677, 0x0677, 0x0677}, | |
| 99 {0x0678, 0x0678, 0x0678, 0x0678}, | |
| 100 {0xFB66, 0xFB67, 0xFB68, 0xFB69}, | |
| 101 {0xFB5E, 0xFB5F, 0xFB60, 0xFB61}, | |
| 102 {0xFB52, 0xFB53, 0xFB54, 0xFB55}, | |
| 103 {0x067C, 0x067C, 0x067C, 0x067C}, | |
| 104 {0x067D, 0x067D, 0x067D, 0x067D}, | |
| 105 {0xFB56, 0xFB57, 0xFB58, 0xFB59}, | |
| 106 {0xFB62, 0xFB63, 0xFB64, 0xFB65}, | |
| 107 {0xFB5A, 0xFB5B, 0xFB5C, 0xFB5D}, | |
| 108 {0x0681, 0x0681, 0x0681, 0x0681}, | |
| 109 {0x0682, 0x0682, 0x0682, 0x0682}, | |
| 110 {0xFB76, 0xFB77, 0xFB78, 0xFB79}, | |
| 111 {0xFB72, 0xFB73, 0xFB74, 0xFB75}, | |
| 112 {0x0685, 0x0685, 0x0685, 0x0685}, | |
| 113 {0xFB7A, 0xFB7B, 0xFB7C, 0xFB7D}, | |
| 114 {0xFB7E, 0xFB7F, 0xFB80, 0xFB81}, | |
| 115 {0xFB88, 0xFB89, 0xFB88, 0xFB89}, | |
| 116 {0x0689, 0x0689, 0x0689, 0x0689}, | |
| 117 {0x068A, 0x068A, 0x068A, 0x068A}, | |
| 118 {0x068B, 0x068B, 0x068B, 0x068B}, | |
| 119 {0xFB84, 0xFB85, 0xFB84, 0xFB85}, | |
| 120 {0xFB82, 0xFB83, 0xFB82, 0xFB83}, | |
| 121 {0xFB86, 0xFB87, 0xFB86, 0xFB87}, | |
| 122 {0x068F, 0x068F, 0x068F, 0x068F}, | |
| 123 {0x0690, 0x0690, 0x0690, 0x0690}, | |
| 124 {0xFB8C, 0xFB8D, 0xFB8C, 0xFB8D}, | |
| 125 {0x0692, 0x0692, 0x0692, 0x0692}, | |
| 126 {0x0693, 0x0693, 0x0693, 0x0693}, | |
| 127 {0x0694, 0x0694, 0x0694, 0x0694}, | |
| 128 {0x0695, 0x0695, 0x0695, 0x0695}, | |
| 129 {0x0696, 0x0696, 0x0696, 0x0696}, | |
| 130 {0x0697, 0x0697, 0x0697, 0x0697}, | |
| 131 {0xFB8A, 0xFB8B, 0xFB8A, 0xFB8B}, | |
| 132 {0x0699, 0x0699, 0x0699, 0x0699}, | |
| 133 {0x069A, 0x069A, 0x069A, 0x069A}, | |
| 134 {0x069B, 0x069B, 0x069B, 0x069B}, | |
| 135 {0x069C, 0x069C, 0x069C, 0x069C}, | |
| 136 {0x069D, 0x069D, 0x069D, 0x069D}, | |
| 137 {0x069E, 0x069E, 0x069E, 0x069E}, | |
| 138 {0x069F, 0x069F, 0x069F, 0x069F}, | |
| 139 {0x06A0, 0x06A0, 0x06A0, 0x06A0}, | |
| 140 {0x06A1, 0x06A1, 0x06A1, 0x06A1}, | |
| 141 {0x06A2, 0x06A2, 0x06A2, 0x06A2}, | |
| 142 {0x06A3, 0x06A3, 0x06A3, 0x06A3}, | |
| 143 {0xFB6A, 0xFB6B, 0xFB6C, 0xFB6D}, | |
| 144 {0x06A5, 0x06A5, 0x06A5, 0x06A5}, | |
| 145 {0xFB6E, 0xFB6F, 0xFB70, 0xFB71}, | |
| 146 {0x06A7, 0x06A7, 0x06A7, 0x06A7}, | |
| 147 {0x06A8, 0x06A8, 0x06A8, 0x06A8}, | |
| 148 {0xFB8E, 0xFB8F, 0xFB90, 0xFB91}, | |
| 149 {0x06AA, 0x06AA, 0x06AA, 0x06AA}, | |
| 150 {0x06AB, 0x06AB, 0x06AB, 0x06AB}, | |
| 151 {0x06AC, 0x06AC, 0x06AC, 0x06AC}, | |
| 152 {0xFBD3, 0xFBD4, 0xFBD5, 0xFBD6}, | |
| 153 {0x06AE, 0x06AE, 0x06AE, 0x06AE}, | |
| 154 {0xFB92, 0xFB93, 0xFB94, 0xFB95}, | |
| 155 {0x06B0, 0x06B0, 0x06B0, 0x06B0}, | |
| 156 {0xFB9A, 0xFB9B, 0xFB9C, 0xFB9D}, | |
| 157 {0x06B2, 0x06B2, 0x06B2, 0x06B2}, | |
| 158 {0xFB96, 0xFB97, 0xFB98, 0xFB99}, | |
| 159 {0x06B4, 0x06B4, 0x06B4, 0x06B4}, | |
| 160 {0x06B5, 0x06B5, 0x06B5, 0x06B5}, | |
| 161 {0x06B6, 0x06B6, 0x06B6, 0x06B6}, | |
| 162 {0x06B7, 0x06B7, 0x06B7, 0x06B7}, | |
| 163 {0x06B8, 0x06B8, 0x06B8, 0x06B8}, | |
| 164 {0x06B9, 0x06B9, 0x06B9, 0x06B9}, | |
| 165 {0xFB9E, 0xFB9F, 0xFBE8, 0xFBE9}, | |
| 166 {0xFBA0, 0xFBA1, 0xFBA2, 0xFBA3}, | |
| 167 {0x06BC, 0x06BC, 0x06BC, 0x06BC}, | |
| 168 {0x06BD, 0x06BD, 0x06BD, 0x06BD}, | |
| 169 {0xFBAA, 0xFBAB, 0xFBAC, 0xFBAD}, | |
| 170 {0x06BF, 0x06BF, 0x06BF, 0x06BF}, | |
| 171 {0xFBA4, 0xFBA5, 0xFBA4, 0xFBA5}, | |
| 172 {0xFBA6, 0xFBA7, 0xFBA8, 0xFBA9}, | |
| 173 {0x06C2, 0x06C2, 0x06C2, 0x06C2}, | |
| 174 {0x06C3, 0x06C3, 0x06C3, 0x06C3}, | |
| 175 {0x06C4, 0x06C4, 0x06C4, 0x06C4}, | |
| 176 {0xFBE0, 0xFBE1, 0xFBE0, 0xFBE1}, | |
| 177 {0xFBD9, 0xFBDA, 0xFBD9, 0xFBDA}, | |
| 178 {0xFBD7, 0xFBD8, 0xFBD7, 0xFBD8}, | |
| 179 {0xFBDB, 0xFBDC, 0xFBDB, 0xFBDC}, | |
| 180 {0xFBE2, 0xFBE3, 0xFBE2, 0xFBE3}, | |
| 181 {0x06CA, 0x06CA, 0x06CA, 0x06CA}, | |
| 182 {0xFBDE, 0xFBDF, 0xFBDE, 0xFBDF}, | |
| 183 {0xFBFC, 0xFBFD, 0xFBFE, 0xFBFF}, | |
| 184 {0x06CD, 0x06CD, 0x06CD, 0x06CD}, | |
| 185 {0x06CE, 0x06CE, 0x06CE, 0x06CE}, | |
| 186 {0x06CF, 0x06CF, 0x06CF, 0x06CF}, | |
| 187 {0xFBE4, 0xFBE5, 0xFBE6, 0xFBE7}, | |
| 188 {0x06D1, 0x06D1, 0x06D1, 0x06D1}, | |
| 189 {0xFBAE, 0xFBAF, 0xFBAE, 0xFBAF}, | |
| 190 {0xFBB0, 0xFBB1, 0xFBB0, 0xFBB1}, | |
| 191 {0x06D4, 0x06D4, 0x06D4, 0x06D4}, | |
| 192 {0x06D5, 0x06D5, 0x06D5, 0x06D5}, | |
| 193 }; | |
| 194 | |
| 195 const FX_ARAALEF gs_FX_AlefTable[] = { | |
| 196 {0x0622, 0xFEF5}, | |
| 197 {0x0623, 0xFEF7}, | |
| 198 {0x0625, 0xFEF9}, | |
| 199 {0x0627, 0xFEFB}, | |
| 200 }; | |
| 201 | |
| 202 const FX_ARASHADDA gs_FX_ShaddaTable[] = { | |
| 203 {0x064C, 0xFC5E}, | |
| 204 {0x064D, 0xFC5F}, | |
| 205 {0x064E, 0xFC60}, | |
| 206 {0x064F, 0xFC61}, | |
| 207 {0x0650, 0xFC62}, | |
| 208 }; | |
| 209 | |
| 210 } // namespace | |
| 211 | |
| 212 const FX_ARBFORMTABLE* FX_GetArabicFormTable(FX_WCHAR unicode) { | |
| 213 if (unicode < 0x622 || unicode > 0x6d5) { | |
| 214 return NULL; | |
| 215 } | |
| 216 return g_FX_ArabicFormTables + unicode - 0x622; | |
| 217 } | |
| 218 FX_WCHAR FX_GetArabicFromAlefTable(FX_WCHAR alef) { | |
| 219 static const int32_t s_iAlefCount = | |
| 220 sizeof(gs_FX_AlefTable) / sizeof(FX_ARAALEF); | |
| 221 for (int32_t iStart = 0; iStart < s_iAlefCount; iStart++) { | |
| 222 const FX_ARAALEF& v = gs_FX_AlefTable[iStart]; | |
| 223 if (v.wAlef == alef) { | |
| 224 return v.wIsolated; | |
| 225 } | |
| 226 } | |
| 227 return alef; | |
| 228 } | |
| 229 FX_WCHAR FX_GetArabicFromShaddaTable(FX_WCHAR shadda) { | |
| 230 static const int32_t s_iShaddaCount = | |
| 231 sizeof(gs_FX_ShaddaTable) / sizeof(FX_ARASHADDA); | |
| 232 for (int32_t iStart = 0; iStart < s_iShaddaCount; iStart++) { | |
| 233 const FX_ARASHADDA& v = gs_FX_ShaddaTable[iStart]; | |
| 234 if (v.wShadda == shadda) { | |
| 235 return v.wIsolated; | |
| 236 } | |
| 237 } | |
| 238 return shadda; | |
| 239 } | |
| 240 | |
| 241 IFX_ArabicChar* IFX_ArabicChar::Create() { | |
| 242 return new CFX_ArabicChar; | |
| 243 } | |
| 244 FX_BOOL CFX_ArabicChar::IsArabicChar(FX_WCHAR wch) const { | |
| 245 FX_DWORD dwRet = | |
| 246 kTextLayoutCodeProperties[(FX_WORD)wch] & FX_CHARTYPEBITSMASK; | |
| 247 return dwRet >= FX_CHARTYPE_ArabicAlef; | |
| 248 } | |
| 249 FX_BOOL CFX_ArabicChar::IsArabicFormChar(FX_WCHAR wch) const { | |
| 250 return (kTextLayoutCodeProperties[(FX_WORD)wch] & FX_CHARTYPEBITSMASK) == | |
| 251 FX_CHARTYPE_ArabicForm; | |
| 252 } | |
| 253 FX_WCHAR CFX_ArabicChar::GetFormChar(FX_WCHAR wch, | |
| 254 FX_WCHAR prev, | |
| 255 FX_WCHAR next) const { | |
| 256 CFX_Char c(wch, kTextLayoutCodeProperties[(FX_WORD)wch]); | |
| 257 CFX_Char p(prev, kTextLayoutCodeProperties[(FX_WORD)prev]); | |
| 258 CFX_Char n(next, kTextLayoutCodeProperties[(FX_WORD)next]); | |
| 259 return GetFormChar(&c, &p, &n); | |
| 260 } | |
| 261 FX_WCHAR CFX_ArabicChar::GetFormChar(const CFX_Char* cur, | |
| 262 const CFX_Char* prev, | |
| 263 const CFX_Char* next) const { | |
| 264 FX_CHARTYPE eCur; | |
| 265 FX_WCHAR wCur; | |
| 266 const FX_ARBFORMTABLE* ft = ParseChar(cur, wCur, eCur); | |
| 267 if (eCur < FX_CHARTYPE_ArabicAlef || eCur >= FX_CHARTYPE_ArabicNormal) { | |
| 268 return wCur; | |
| 269 } | |
| 270 FX_CHARTYPE ePrev; | |
| 271 FX_WCHAR wPrev; | |
| 272 ParseChar(prev, wPrev, ePrev); | |
| 273 if (wPrev == 0x0644 && eCur == FX_CHARTYPE_ArabicAlef) { | |
| 274 return 0xFEFF; | |
| 275 } | |
| 276 FX_CHARTYPE eNext; | |
| 277 FX_WCHAR wNext; | |
| 278 ParseChar(next, wNext, eNext); | |
| 279 bool bAlef = (eNext == FX_CHARTYPE_ArabicAlef && wCur == 0x644); | |
| 280 if (ePrev < FX_CHARTYPE_ArabicAlef) { | |
| 281 if (bAlef) { | |
| 282 return FX_GetArabicFromAlefTable(wNext); | |
| 283 } else { | |
| 284 return (eNext < FX_CHARTYPE_ArabicAlef) ? ft->wIsolated : ft->wInitial; | |
| 285 } | |
| 286 } else { | |
| 287 if (bAlef) { | |
| 288 wCur = FX_GetArabicFromAlefTable(wNext); | |
| 289 return (ePrev != FX_CHARTYPE_ArabicDistortion) ? wCur : ++wCur; | |
| 290 } else if (ePrev == FX_CHARTYPE_ArabicAlef || | |
| 291 ePrev == FX_CHARTYPE_ArabicSpecial) { | |
| 292 return (eNext < FX_CHARTYPE_ArabicAlef) ? ft->wIsolated : ft->wInitial; | |
| 293 } else { | |
| 294 return (eNext < FX_CHARTYPE_ArabicAlef) ? ft->wFinal : ft->wMedial; | |
| 295 } | |
| 296 } | |
| 297 } | |
| 298 const FX_ARBFORMTABLE* CFX_ArabicChar::ParseChar(const CFX_Char* pTC, | |
| 299 FX_WCHAR& wChar, | |
| 300 FX_CHARTYPE& eType) const { | |
| 301 if (pTC == NULL) { | |
| 302 eType = FX_CHARTYPE_Unknown; | |
| 303 wChar = 0xFEFF; | |
| 304 return NULL; | |
| 305 } | |
| 306 eType = (FX_CHARTYPE)pTC->GetCharType(); | |
| 307 wChar = (FX_WCHAR)pTC->m_wCharCode; | |
| 308 const FX_ARBFORMTABLE* pFT = FX_GetArabicFormTable(wChar); | |
| 309 if (pFT == NULL || eType >= FX_CHARTYPE_ArabicNormal) { | |
| 310 eType = FX_CHARTYPE_Unknown; | |
| 311 } | |
| 312 return pFT; | |
| 313 } | |
| 314 void FX_BidiReverseString(CFX_WideString& wsText, | |
| 315 int32_t iStart, | |
| 316 int32_t iCount) { | |
| 317 FXSYS_assert(iStart > -1 && iStart < wsText.GetLength()); | |
| 318 FXSYS_assert(iCount >= 0 && iStart + iCount <= wsText.GetLength()); | |
| 319 FX_WCHAR wch; | |
| 320 FX_WCHAR* pStart = (FX_WCHAR*)(const FX_WCHAR*)wsText; | |
| 321 pStart += iStart; | |
| 322 FX_WCHAR* pEnd = pStart + iCount - 1; | |
| 323 while (pStart < pEnd) { | |
| 324 wch = *pStart; | |
| 325 *pStart++ = *pEnd; | |
| 326 *pEnd-- = wch; | |
| 327 } | |
| 328 } | |
| 329 void FX_BidiSetDeferredRun(CFX_Int32Array& values, | |
| 330 int32_t iStart, | |
| 331 int32_t iCount, | |
| 332 int32_t iValue) { | |
| 333 FXSYS_assert(iStart > -1 && iStart <= values.GetSize()); | |
| 334 FXSYS_assert(iStart - iCount > -1); | |
| 335 for (int32_t i = iStart - 1; i >= iStart - iCount; i--) { | |
| 336 values.SetAt(i, iValue); | |
| 337 } | |
| 338 } | |
| 339 const int32_t gc_FX_BidiNTypes[] = { | |
| 340 FX_BIDICLASS_N, FX_BIDICLASS_L, FX_BIDICLASS_R, FX_BIDICLASS_AN, | |
| 341 FX_BIDICLASS_EN, FX_BIDICLASS_AL, FX_BIDICLASS_NSM, FX_BIDICLASS_CS, | |
| 342 FX_BIDICLASS_ES, FX_BIDICLASS_ET, FX_BIDICLASS_BN, FX_BIDICLASS_BN, | |
| 343 FX_BIDICLASS_N, FX_BIDICLASS_B, FX_BIDICLASS_RLO, FX_BIDICLASS_RLE, | |
| 344 FX_BIDICLASS_LRO, FX_BIDICLASS_LRE, FX_BIDICLASS_PDF, FX_BIDICLASS_ON, | |
| 345 }; | |
| 346 void FX_BidiClassify(const CFX_WideString& wsText, | |
| 347 CFX_Int32Array& classes, | |
| 348 FX_BOOL bWS) { | |
| 349 FXSYS_assert(wsText.GetLength() == classes.GetSize()); | |
| 350 int32_t iCount = wsText.GetLength(); | |
| 351 const FX_WCHAR* pwsStart = (const FX_WCHAR*)wsText; | |
| 352 FX_WCHAR wch; | |
| 353 int32_t iCls; | |
| 354 if (bWS) { | |
| 355 for (int32_t i = 0; i < iCount; i++) { | |
| 356 wch = *pwsStart++; | |
| 357 iCls = | |
| 358 ((kTextLayoutCodeProperties[(FX_WORD)wch] & FX_BIDICLASSBITSMASK) >> | |
| 359 FX_BIDICLASSBITS); | |
| 360 classes.SetAt(i, iCls); | |
| 361 } | |
| 362 } else { | |
| 363 for (int32_t i = 0; i < iCount; i++) { | |
| 364 wch = *pwsStart++; | |
| 365 iCls = | |
| 366 ((kTextLayoutCodeProperties[(FX_WORD)wch] & FX_BIDICLASSBITSMASK) >> | |
| 367 FX_BIDICLASSBITS); | |
| 368 classes.SetAt(i, gc_FX_BidiNTypes[iCls]); | |
| 369 } | |
| 370 } | |
| 371 } | |
| 372 int32_t FX_BidiResolveExplicit(int32_t iBaseLevel, | |
| 373 int32_t iDirection, | |
| 374 CFX_Int32Array& classes, | |
| 375 CFX_Int32Array& levels, | |
| 376 int32_t iStart, | |
| 377 int32_t iCount, | |
| 378 int32_t iNest) { | |
| 379 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL && iNest >= 0); | |
| 380 FXSYS_assert(classes.GetSize() == levels.GetSize()); | |
| 381 FXSYS_assert(iStart >= 0 && iStart < classes.GetSize()); | |
| 382 FXSYS_assert(iCount >= 0 && iStart + iCount <= classes.GetSize()); | |
| 383 if (iCount < 1) { | |
| 384 return 0; | |
| 385 } | |
| 386 | |
| 387 int32_t iSize = classes.GetSize(); | |
| 388 int32_t i = iStart, iCur, iCls; | |
| 389 for (; i < iSize && iCount > 0; i++, iCount--) { | |
| 390 iCur = iCls = classes.GetAt(i); | |
| 391 | |
| 392 if (iDirection != FX_BIDICLASS_N) { | |
| 393 iCls = iDirection; | |
| 394 } | |
| 395 if (iCur != FX_BIDICLASS_BN) { | |
| 396 classes.SetAt(i, iCls); | |
| 397 } | |
| 398 levels.SetAt(i, iBaseLevel); | |
| 399 } | |
| 400 return i - iStart; | |
| 401 } | |
| 402 const int32_t gc_FX_BidiWeakStates[][10] = { | |
| 403 {FX_BWSao, FX_BWSxl, FX_BWSxr, FX_BWScn, FX_BWScn, FX_BWSxa, FX_BWSxa, | |
| 404 FX_BWSao, FX_BWSao, FX_BWSao}, | |
| 405 {FX_BWSro, FX_BWSxl, FX_BWSxr, FX_BWSra, FX_BWSre, FX_BWSxa, FX_BWSxr, | |
| 406 FX_BWSro, FX_BWSro, FX_BWSrt}, | |
| 407 {FX_BWSlo, FX_BWSxl, FX_BWSxr, FX_BWSla, FX_BWSle, FX_BWSxa, FX_BWSxl, | |
| 408 FX_BWSlo, FX_BWSlo, FX_BWSlt}, | |
| 409 {FX_BWSao, FX_BWSxl, FX_BWSxr, FX_BWScn, FX_BWScn, FX_BWSxa, FX_BWSao, | |
| 410 FX_BWSao, FX_BWSao, FX_BWSao}, | |
| 411 {FX_BWSro, FX_BWSxl, FX_BWSxr, FX_BWSra, FX_BWSre, FX_BWSxa, FX_BWSro, | |
| 412 FX_BWSro, FX_BWSro, FX_BWSrt}, | |
| 413 {FX_BWSlo, FX_BWSxl, FX_BWSxr, FX_BWSla, FX_BWSle, FX_BWSxa, FX_BWSlo, | |
| 414 FX_BWSlo, FX_BWSlo, FX_BWSlt}, | |
| 415 {FX_BWSro, FX_BWSxl, FX_BWSxr, FX_BWSra, FX_BWSre, FX_BWSxa, FX_BWSrt, | |
| 416 FX_BWSro, FX_BWSro, FX_BWSrt}, | |
| 417 {FX_BWSlo, FX_BWSxl, FX_BWSxr, FX_BWSla, FX_BWSle, FX_BWSxa, FX_BWSlt, | |
| 418 FX_BWSlo, FX_BWSlo, FX_BWSlt}, | |
| 419 {FX_BWSao, FX_BWSxl, FX_BWSxr, FX_BWScn, FX_BWScn, FX_BWSxa, FX_BWScn, | |
| 420 FX_BWSac, FX_BWSao, FX_BWSao}, | |
| 421 {FX_BWSro, FX_BWSxl, FX_BWSxr, FX_BWSra, FX_BWSre, FX_BWSxa, FX_BWSra, | |
| 422 FX_BWSrc, FX_BWSro, FX_BWSrt}, | |
| 423 {FX_BWSro, FX_BWSxl, FX_BWSxr, FX_BWSra, FX_BWSre, FX_BWSxa, FX_BWSre, | |
| 424 FX_BWSrs, FX_BWSrs, FX_BWSret}, | |
| 425 {FX_BWSlo, FX_BWSxl, FX_BWSxr, FX_BWSla, FX_BWSle, FX_BWSxa, FX_BWSla, | |
| 426 FX_BWSlc, FX_BWSlo, FX_BWSlt}, | |
| 427 {FX_BWSlo, FX_BWSxl, FX_BWSxr, FX_BWSla, FX_BWSle, FX_BWSxa, FX_BWSle, | |
| 428 FX_BWSls, FX_BWSls, FX_BWSlet}, | |
| 429 {FX_BWSao, FX_BWSxl, FX_BWSxr, FX_BWScn, FX_BWScn, FX_BWSxa, FX_BWSao, | |
| 430 FX_BWSao, FX_BWSao, FX_BWSao}, | |
| 431 {FX_BWSro, FX_BWSxl, FX_BWSxr, FX_BWSra, FX_BWSre, FX_BWSxa, FX_BWSro, | |
| 432 FX_BWSro, FX_BWSro, FX_BWSrt}, | |
| 433 {FX_BWSro, FX_BWSxl, FX_BWSxr, FX_BWSra, FX_BWSre, FX_BWSxa, FX_BWSro, | |
| 434 FX_BWSro, FX_BWSro, FX_BWSrt}, | |
| 435 {FX_BWSlo, FX_BWSxl, FX_BWSxr, FX_BWSla, FX_BWSle, FX_BWSxa, FX_BWSlo, | |
| 436 FX_BWSlo, FX_BWSlo, FX_BWSlt}, | |
| 437 {FX_BWSlo, FX_BWSxl, FX_BWSxr, FX_BWSla, FX_BWSle, FX_BWSxa, FX_BWSlo, | |
| 438 FX_BWSlo, FX_BWSlo, FX_BWSlt}, | |
| 439 {FX_BWSro, FX_BWSxl, FX_BWSxr, FX_BWSra, FX_BWSre, FX_BWSxa, FX_BWSret, | |
| 440 FX_BWSro, FX_BWSro, FX_BWSret}, | |
| 441 {FX_BWSlo, FX_BWSxl, FX_BWSxr, FX_BWSla, FX_BWSle, FX_BWSxa, FX_BWSlet, | |
| 442 FX_BWSlo, FX_BWSlo, FX_BWSlet}, | |
| 443 }; | |
| 444 const int32_t gc_FX_BidiWeakActions[][10] = { | |
| 445 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxA, FX_BWAxxR, | |
| 446 FX_BWAxxR, FX_BWAxxN, FX_BWAxxN, FX_BWAxxN}, | |
| 447 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxE, FX_BWAxxR, | |
| 448 FX_BWAxxR, FX_BWAxxN, FX_BWAxxN, FX_BWAxIx}, | |
| 449 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxL, FX_BWAxxR, | |
| 450 FX_BWAxxL, FX_BWAxxN, FX_BWAxxN, FX_BWAxIx}, | |
| 451 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxA, FX_BWAxxR, | |
| 452 FX_BWAxxN, FX_BWAxxN, FX_BWAxxN, FX_BWAxxN}, | |
| 453 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxE, FX_BWAxxR, | |
| 454 FX_BWAxxN, FX_BWAxxN, FX_BWAxxN, FX_BWAxIx}, | |
| 455 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxL, FX_BWAxxR, | |
| 456 FX_BWAxxN, FX_BWAxxN, FX_BWAxxN, FX_BWAxIx}, | |
| 457 {FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWAExE, FX_BWANxR, | |
| 458 FX_BWAxIx, FX_BWANxN, FX_BWANxN, FX_BWAxIx}, | |
| 459 {FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWALxL, FX_BWANxR, | |
| 460 FX_BWAxIx, FX_BWANxN, FX_BWANxN, FX_BWAxIx}, | |
| 461 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxA, FX_BWAxxR, | |
| 462 FX_BWAxxA, FX_BWAxIx, FX_BWAxxN, FX_BWAxxN}, | |
| 463 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxE, FX_BWAxxR, | |
| 464 FX_BWAxxA, FX_BWAxIx, FX_BWAxxN, FX_BWAxIx}, | |
| 465 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxE, FX_BWAxxR, | |
| 466 FX_BWAxxE, FX_BWAxIx, FX_BWAxIx, FX_BWAxxE}, | |
| 467 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxL, FX_BWAxxR, | |
| 468 FX_BWAxxA, FX_BWAxIx, FX_BWAxxN, FX_BWAxIx}, | |
| 469 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxL, FX_BWAxxR, | |
| 470 FX_BWAxxL, FX_BWAxIx, FX_BWAxIx, FX_BWAxxL}, | |
| 471 {FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWAAxx, FX_BWAAxA, FX_BWANxR, | |
| 472 FX_BWANxN, FX_BWANxN, FX_BWANxN, FX_BWANxN}, | |
| 473 {FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWAAxx, FX_BWANxE, FX_BWANxR, | |
| 474 FX_BWANxN, FX_BWANxN, FX_BWANxN, FX_BWANIx}, | |
| 475 {FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWAExE, FX_BWANxR, | |
| 476 FX_BWANxN, FX_BWANxN, FX_BWANxN, FX_BWANIx}, | |
| 477 {FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWAAxx, FX_BWANxL, FX_BWANxR, | |
| 478 FX_BWANxN, FX_BWANxN, FX_BWANxN, FX_BWANIx}, | |
| 479 {FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWANxx, FX_BWALxL, FX_BWANxR, | |
| 480 FX_BWANxN, FX_BWANxN, FX_BWANxN, FX_BWANIx}, | |
| 481 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxE, FX_BWAxxR, | |
| 482 FX_BWAxxE, FX_BWAxxN, FX_BWAxxN, FX_BWAxxE}, | |
| 483 {FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxx, FX_BWAxxL, FX_BWAxxR, | |
| 484 FX_BWAxxL, FX_BWAxxN, FX_BWAxxN, FX_BWAxxL}, | |
| 485 }; | |
| 486 void FX_BidiResolveWeak(int32_t iBaseLevel, | |
| 487 CFX_Int32Array& classes, | |
| 488 CFX_Int32Array& levels) { | |
| 489 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 490 FXSYS_assert(classes.GetSize() == levels.GetSize()); | |
| 491 int32_t iSize = classes.GetSize(); | |
| 492 if (iSize < 1) { | |
| 493 return; | |
| 494 } | |
| 495 iSize--; | |
| 496 int32_t iLevelCur = iBaseLevel; | |
| 497 int32_t iState = FX_IsOdd(iBaseLevel) ? FX_BWSxr : FX_BWSxl; | |
| 498 int32_t i = 0, iCount = 0, iClsCur, iClsRun, iClsNew, iAction; | |
| 499 for (; i <= iSize; i++) { | |
| 500 iClsCur = classes.GetAt(i); | |
| 501 | |
| 502 FXSYS_assert(iClsCur <= FX_BIDICLASS_BN); | |
| 503 iAction = gc_FX_BidiWeakActions[iState][iClsCur]; | |
| 504 iClsRun = FX_BidiGetDeferredType(iAction); | |
| 505 if (iClsRun != FX_BIDIWEAKACTION_XX && iCount > 0) { | |
| 506 FX_BidiSetDeferredRun(classes, i, iCount, iClsRun); | |
| 507 iCount = 0; | |
| 508 } | |
| 509 iClsNew = FX_BidiGetResolvedType(iAction); | |
| 510 if (iClsNew != FX_BIDIWEAKACTION_XX) { | |
| 511 classes.SetAt(i, iClsNew); | |
| 512 } | |
| 513 if (FX_BIDIWEAKACTION_IX & iAction) { | |
| 514 iCount++; | |
| 515 } | |
| 516 iState = gc_FX_BidiWeakStates[iState][iClsCur]; | |
| 517 } | |
| 518 iClsCur = FX_BidiDirection(iLevelCur); | |
| 519 iClsRun = FX_BidiGetDeferredType(gc_FX_BidiWeakActions[iState][iClsCur]); | |
| 520 if (iClsRun != FX_BIDIWEAKACTION_XX && iCount > 0) { | |
| 521 FX_BidiSetDeferredRun(classes, i, iCount, iClsRun); | |
| 522 } | |
| 523 } | |
| 524 const int32_t gc_FX_BidiNeutralStates[][5] = { | |
| 525 {FX_BNSrn, FX_BNSl, FX_BNSr, FX_BNSr, FX_BNSr}, | |
| 526 {FX_BNSln, FX_BNSl, FX_BNSr, FX_BNSa, FX_BNSl}, | |
| 527 {FX_BNSrn, FX_BNSl, FX_BNSr, FX_BNSr, FX_BNSr}, | |
| 528 {FX_BNSln, FX_BNSl, FX_BNSr, FX_BNSa, FX_BNSl}, | |
| 529 {FX_BNSna, FX_BNSl, FX_BNSr, FX_BNSa, FX_BNSl}, | |
| 530 {FX_BNSna, FX_BNSl, FX_BNSr, FX_BNSa, FX_BNSl}, | |
| 531 }; | |
| 532 const int32_t gc_FX_BidiNeutralActions[][5] = { | |
| 533 {FX_BNAIn, 0, 0, 0, 0}, | |
| 534 {FX_BNAIn, 0, 0, 0, FX_BCL}, | |
| 535 {FX_BNAIn, FX_BNAEn, FX_BNARn, FX_BNARn, FX_BNARn}, | |
| 536 {FX_BNAIn, FX_BNALn, FX_BNAEn, FX_BNAEn, FX_BNALnL}, | |
| 537 {FX_BNAIn, 0, 0, 0, FX_BCL}, | |
| 538 {FX_BNAIn, FX_BNAEn, FX_BNARn, FX_BNARn, FX_BNAEn}, | |
| 539 }; | |
| 540 int32_t FX_BidiGetDeferredNeutrals(int32_t iAction, int32_t iLevel) { | |
| 541 iAction = (iAction >> 4) & 0xF; | |
| 542 if (iAction == (FX_BIDINEUTRALACTION_En >> 4)) { | |
| 543 return FX_BidiDirection(iLevel); | |
| 544 } else { | |
| 545 return iAction; | |
| 546 } | |
| 547 } | |
| 548 int32_t FX_BidiGetResolvedNeutrals(int32_t iAction) { | |
| 549 iAction = (iAction & 0xF); | |
| 550 if (iAction == FX_BIDINEUTRALACTION_In) { | |
| 551 return 0; | |
| 552 } else { | |
| 553 return iAction; | |
| 554 } | |
| 555 } | |
| 556 void FX_BidiResolveNeutrals(int32_t iBaseLevel, | |
| 557 CFX_Int32Array& classes, | |
| 558 const CFX_Int32Array& levels) { | |
| 559 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 560 FXSYS_assert(classes.GetSize() == levels.GetSize()); | |
| 561 int32_t iSize = classes.GetSize(); | |
| 562 if (iSize < 1) { | |
| 563 return; | |
| 564 } | |
| 565 iSize--; | |
| 566 int32_t iLevel = iBaseLevel; | |
| 567 int32_t iState = FX_IsOdd(iBaseLevel) ? FX_BNSr : FX_BNSl; | |
| 568 int32_t i = 0, iCount = 0, iClsCur, iClsRun, iClsNew, iAction; | |
| 569 for (; i <= iSize; i++) { | |
| 570 iClsCur = classes.GetAt(i); | |
| 571 if (iClsCur == FX_BIDICLASS_BN) { | |
| 572 if (iCount) { | |
| 573 iCount++; | |
| 574 } | |
| 575 continue; | |
| 576 } | |
| 577 FXSYS_assert(iClsCur < FX_BIDICLASS_AL); | |
| 578 iAction = gc_FX_BidiNeutralActions[iState][iClsCur]; | |
| 579 iClsRun = FX_BidiGetDeferredNeutrals(iAction, iLevel); | |
| 580 if (iClsRun != FX_BIDICLASS_N && iCount > 0) { | |
| 581 FX_BidiSetDeferredRun(classes, i, iCount, iClsRun); | |
| 582 iCount = 0; | |
| 583 } | |
| 584 iClsNew = FX_BidiGetResolvedNeutrals(iAction); | |
| 585 if (iClsNew != FX_BIDICLASS_N) { | |
| 586 classes.SetAt(i, iClsNew); | |
| 587 } | |
| 588 if (FX_BIDINEUTRALACTION_In & iAction) { | |
| 589 iCount++; | |
| 590 } | |
| 591 iState = gc_FX_BidiNeutralStates[iState][iClsCur]; | |
| 592 iLevel = levels.GetAt(i); | |
| 593 } | |
| 594 iClsCur = FX_BidiDirection(iLevel); | |
| 595 iClsRun = FX_BidiGetDeferredNeutrals( | |
| 596 gc_FX_BidiNeutralActions[iState][iClsCur], iLevel); | |
| 597 if (iClsRun != FX_BIDICLASS_N && iCount > 0) { | |
| 598 FX_BidiSetDeferredRun(classes, i, iCount, iClsRun); | |
| 599 } | |
| 600 } | |
| 601 const int32_t gc_FX_BidiAddLevel[][4] = { | |
| 602 {0, 1, 2, 2}, | |
| 603 {1, 0, 1, 1}, | |
| 604 }; | |
| 605 void FX_BidiResolveImplicit(const CFX_Int32Array& classes, | |
| 606 CFX_Int32Array& levels) { | |
| 607 FXSYS_assert(classes.GetSize() == levels.GetSize()); | |
| 608 int32_t iSize = classes.GetSize(); | |
| 609 if (iSize < 1) { | |
| 610 return; | |
| 611 } | |
| 612 iSize--; | |
| 613 int32_t iCls, iLevel; | |
| 614 for (int32_t i = 0; i <= iSize; i++) { | |
| 615 iCls = classes.GetAt(i); | |
| 616 if (iCls == FX_BIDICLASS_BN) { | |
| 617 continue; | |
| 618 } | |
| 619 FXSYS_assert(iCls > FX_BIDICLASS_ON && iCls < FX_BIDICLASS_AL); | |
| 620 iLevel = levels.GetAt(i); | |
| 621 iLevel += gc_FX_BidiAddLevel[FX_IsOdd(iLevel)][iCls - 1]; | |
| 622 levels.SetAt(i, iLevel); | |
| 623 } | |
| 624 } | |
| 625 void FX_BidiResolveWhitespace(int32_t iBaseLevel, | |
| 626 const CFX_Int32Array& classes, | |
| 627 CFX_Int32Array& levels) { | |
| 628 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 629 FXSYS_assert(classes.GetSize() == levels.GetSize()); | |
| 630 int32_t iSize = classes.GetSize(); | |
| 631 if (iSize < 1) { | |
| 632 return; | |
| 633 } | |
| 634 iSize--; | |
| 635 int32_t iLevel = iBaseLevel; | |
| 636 int32_t i = 0, iCount = 0; | |
| 637 for (; i <= iSize; i++) { | |
| 638 switch (classes.GetAt(i)) { | |
| 639 case FX_BIDICLASS_WS: | |
| 640 iCount++; | |
| 641 break; | |
| 642 case FX_BIDICLASS_RLE: | |
| 643 case FX_BIDICLASS_LRE: | |
| 644 case FX_BIDICLASS_LRO: | |
| 645 case FX_BIDICLASS_RLO: | |
| 646 case FX_BIDICLASS_PDF: | |
| 647 case FX_BIDICLASS_BN: | |
| 648 levels.SetAt(i, iLevel); | |
| 649 iCount++; | |
| 650 break; | |
| 651 case FX_BIDICLASS_S: | |
| 652 case FX_BIDICLASS_B: | |
| 653 if (iCount > 0) { | |
| 654 FX_BidiSetDeferredRun(levels, i, iCount, iBaseLevel); | |
| 655 } | |
| 656 levels.SetAt(i, iBaseLevel); | |
| 657 iCount = 0; | |
| 658 break; | |
| 659 default: | |
| 660 iCount = 0; | |
| 661 break; | |
| 662 } | |
| 663 iLevel = levels.GetAt(i); | |
| 664 } | |
| 665 if (iCount > 0) { | |
| 666 FX_BidiSetDeferredRun(levels, i, iCount, iBaseLevel); | |
| 667 } | |
| 668 } | |
| 669 int32_t FX_BidiReorderLevel(int32_t iBaseLevel, | |
| 670 CFX_WideString& wsText, | |
| 671 const CFX_Int32Array& levels, | |
| 672 int32_t iStart, | |
| 673 FX_BOOL bReverse) { | |
| 674 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 675 FXSYS_assert(wsText.GetLength() == levels.GetSize()); | |
| 676 FXSYS_assert(iStart >= 0 && iStart < wsText.GetLength()); | |
| 677 int32_t iSize = wsText.GetLength(); | |
| 678 if (iSize < 1) { | |
| 679 return 0; | |
| 680 } | |
| 681 bReverse = bReverse || FX_IsOdd(iBaseLevel); | |
| 682 int32_t i = iStart, iLevel; | |
| 683 for (; i < iSize; i++) { | |
| 684 if ((iLevel = levels.GetAt(i)) == iBaseLevel) { | |
| 685 continue; | |
| 686 } | |
| 687 if (iLevel < iBaseLevel) { | |
| 688 break; | |
| 689 } | |
| 690 i += FX_BidiReorderLevel(iBaseLevel + 1, wsText, levels, i, bReverse) - 1; | |
| 691 } | |
| 692 int32_t iCount = i - iStart; | |
| 693 if (bReverse && iCount > 1) { | |
| 694 FX_BidiReverseString(wsText, iStart, iCount); | |
| 695 } | |
| 696 return iCount; | |
| 697 } | |
| 698 void FX_BidiReorder(int32_t iBaseLevel, | |
| 699 CFX_WideString& wsText, | |
| 700 const CFX_Int32Array& levels) { | |
| 701 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 702 FXSYS_assert(wsText.GetLength() == levels.GetSize()); | |
| 703 int32_t iSize = wsText.GetLength(); | |
| 704 if (iSize < 1) { | |
| 705 return; | |
| 706 } | |
| 707 int32_t i = 0; | |
| 708 while (i < iSize) { | |
| 709 i += FX_BidiReorderLevel(iBaseLevel, wsText, levels, i, FALSE); | |
| 710 } | |
| 711 } | |
| 712 void FX_BidiLine(CFX_WideString& wsText, int32_t iBaseLevel) { | |
| 713 int32_t iLength = wsText.GetLength(); | |
| 714 if (iLength < 2) { | |
| 715 return; | |
| 716 } | |
| 717 CFX_Int32Array classes, levels; | |
| 718 classes.SetAtGrow(iLength - 1, 0); | |
| 719 levels.SetAtGrow(iLength - 1, 0); | |
| 720 FX_BidiClassify(wsText, classes, FALSE); | |
| 721 FX_BidiResolveExplicit(iBaseLevel, FX_BIDICLASS_N, classes, levels, 0, | |
| 722 iLength, 0); | |
| 723 FX_BidiResolveWeak(iBaseLevel, classes, levels); | |
| 724 FX_BidiResolveNeutrals(iBaseLevel, classes, levels); | |
| 725 FX_BidiResolveImplicit(classes, levels); | |
| 726 FX_BidiClassify(wsText, classes, TRUE); | |
| 727 FX_BidiResolveWhitespace(iBaseLevel, classes, levels); | |
| 728 FX_BidiReorder(iBaseLevel, wsText, levels); | |
| 729 classes.RemoveAll(); | |
| 730 levels.RemoveAll(); | |
| 731 } | |
| 732 template <class baseType> | |
| 733 class CFX_BidiLineTemplate { | |
| 734 public: | |
| 735 void FX_BidiReverseString(CFX_ArrayTemplate<baseType>& chars, | |
| 736 int32_t iStart, | |
| 737 int32_t iCount) { | |
| 738 FXSYS_assert(iStart > -1 && iStart < chars.GetSize()); | |
| 739 FXSYS_assert(iCount >= 0 && iStart + iCount <= chars.GetSize()); | |
| 740 baseType *pStart, *pEnd; | |
| 741 int32_t iEnd = iStart + iCount - 1, iTemp; | |
| 742 while (iStart < iEnd) { | |
| 743 pStart = chars.GetDataPtr(iStart++); | |
| 744 pEnd = chars.GetDataPtr(iEnd--); | |
| 745 iTemp = pStart->m_iBidiPos; | |
| 746 pStart->m_iBidiPos = pEnd->m_iBidiPos; | |
| 747 pEnd->m_iBidiPos = iTemp; | |
| 748 } | |
| 749 } | |
| 750 void FX_BidiSetDeferredRun(CFX_ArrayTemplate<baseType>& chars, | |
| 751 FX_BOOL bClass, | |
| 752 int32_t iStart, | |
| 753 int32_t iCount, | |
| 754 int32_t iValue) { | |
| 755 FXSYS_assert(iStart > -1 && iStart <= chars.GetSize()); | |
| 756 FXSYS_assert(iStart - iCount > -1); | |
| 757 baseType* pTC; | |
| 758 int32_t iLast = iStart - iCount; | |
| 759 if (bClass) { | |
| 760 for (int32_t i = iStart - 1; i >= iLast; i--) { | |
| 761 pTC = chars.GetDataPtr(i); | |
| 762 pTC->m_iBidiClass = (int16_t)iValue; | |
| 763 } | |
| 764 } else { | |
| 765 for (int32_t i = iStart - 1; i >= iLast; i--) { | |
| 766 pTC = chars.GetDataPtr(i); | |
| 767 pTC->m_iBidiLevel = (int16_t)iValue; | |
| 768 } | |
| 769 } | |
| 770 } | |
| 771 void FX_BidiClassify(CFX_ArrayTemplate<baseType>& chars, | |
| 772 int32_t iCount, | |
| 773 FX_BOOL bWS) { | |
| 774 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 775 baseType* pTC; | |
| 776 if (bWS) { | |
| 777 for (int32_t i = 0; i < iCount; i++) { | |
| 778 pTC = chars.GetDataPtr(i); | |
| 779 pTC->m_iBidiClass = | |
| 780 (int16_t)(pTC->m_dwCharProps & FX_BIDICLASSBITSMASK) >> | |
| 781 FX_BIDICLASSBITS; | |
| 782 } | |
| 783 } else { | |
| 784 for (int32_t i = 0; i < iCount; i++) { | |
| 785 pTC = chars.GetDataPtr(i); | |
| 786 pTC->m_iBidiClass = (int16_t) | |
| 787 gc_FX_BidiNTypes[(pTC->m_dwCharProps & FX_BIDICLASSBITSMASK) >> | |
| 788 FX_BIDICLASSBITS]; | |
| 789 } | |
| 790 } | |
| 791 } | |
| 792 void FX_BidiResolveExplicit(CFX_ArrayTemplate<baseType>& chars, | |
| 793 int32_t iCount, | |
| 794 int32_t iBaseLevel) { | |
| 795 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 796 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 797 if (iCount < 1) { | |
| 798 return; | |
| 799 } | |
| 800 baseType* pTC; | |
| 801 for (int32_t i = 0; i < iCount; i++) { | |
| 802 pTC = chars.GetDataPtr(i); | |
| 803 pTC->m_iBidiLevel = (int16_t)iBaseLevel; | |
| 804 } | |
| 805 } | |
| 806 void FX_BidiResolveWeak(CFX_ArrayTemplate<baseType>& chars, | |
| 807 int32_t iCount, | |
| 808 int32_t iBaseLevel) { | |
| 809 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 810 iCount--; | |
| 811 if (iCount < 1) { | |
| 812 return; | |
| 813 } | |
| 814 baseType *pTC, *pTCNext; | |
| 815 int32_t iLevelCur = iBaseLevel; | |
| 816 int32_t iState = FX_IsOdd(iBaseLevel) ? FX_BWSxr : FX_BWSxl; | |
| 817 int32_t i = 0, iNum = 0, iClsCur, iClsRun, iClsNew, iAction; | |
| 818 for (; i <= iCount; i++) { | |
| 819 pTC = chars.GetDataPtr(i); | |
| 820 iClsCur = pTC->m_iBidiClass; | |
| 821 if (iClsCur == FX_BIDICLASS_BN) { | |
| 822 pTC->m_iBidiLevel = (int16_t)iLevelCur; | |
| 823 if (i == iCount && iLevelCur != iBaseLevel) { | |
| 824 iClsCur = FX_BidiDirection(iLevelCur); | |
| 825 pTC->m_iBidiClass = (int16_t)iClsCur; | |
| 826 } else if (i < iCount) { | |
| 827 pTCNext = chars.GetDataPtr(i + 1); | |
| 828 int32_t iLevelNext, iLevelNew; | |
| 829 iClsNew = pTCNext->m_iBidiClass; | |
| 830 iLevelNext = pTCNext->m_iBidiLevel; | |
| 831 if (iClsNew != FX_BIDICLASS_BN && iLevelCur != iLevelNext) { | |
| 832 iLevelNew = iLevelNext; | |
| 833 if (iLevelCur > iLevelNew) { | |
| 834 iLevelNew = iLevelCur; | |
| 835 } | |
| 836 pTC->m_iBidiLevel = (int16_t)iLevelNew; | |
| 837 iClsCur = FX_BidiDirection(iLevelNew); | |
| 838 pTC->m_iBidiClass = (int16_t)iClsCur; | |
| 839 iLevelCur = iLevelNext; | |
| 840 } else { | |
| 841 if (iNum > 0) { | |
| 842 iNum++; | |
| 843 } | |
| 844 continue; | |
| 845 } | |
| 846 } else { | |
| 847 if (iNum > 0) { | |
| 848 iNum++; | |
| 849 } | |
| 850 continue; | |
| 851 } | |
| 852 } | |
| 853 FXSYS_assert(iClsCur <= FX_BIDICLASS_BN); | |
| 854 iAction = gc_FX_BidiWeakActions[iState][iClsCur]; | |
| 855 iClsRun = FX_BidiGetDeferredType(iAction); | |
| 856 if (iClsRun != FX_BIDIWEAKACTION_XX && iNum > 0) { | |
| 857 FX_BidiSetDeferredRun(chars, TRUE, i, iNum, iClsRun); | |
| 858 iNum = 0; | |
| 859 } | |
| 860 iClsNew = FX_BidiGetResolvedType(iAction); | |
| 861 if (iClsNew != FX_BIDIWEAKACTION_XX) { | |
| 862 pTC->m_iBidiClass = (int16_t)iClsNew; | |
| 863 } | |
| 864 if (FX_BIDIWEAKACTION_IX & iAction) { | |
| 865 iNum++; | |
| 866 } | |
| 867 iState = gc_FX_BidiWeakStates[iState][iClsCur]; | |
| 868 } | |
| 869 if (iNum > 0) { | |
| 870 iClsCur = FX_BidiDirection(iBaseLevel); | |
| 871 iClsRun = FX_BidiGetDeferredType(gc_FX_BidiWeakActions[iState][iClsCur]); | |
| 872 if (iClsRun != FX_BIDIWEAKACTION_XX) { | |
| 873 FX_BidiSetDeferredRun(chars, TRUE, i, iNum, iClsRun); | |
| 874 } | |
| 875 } | |
| 876 } | |
| 877 void FX_BidiResolveNeutrals(CFX_ArrayTemplate<baseType>& chars, | |
| 878 int32_t iCount, | |
| 879 int32_t iBaseLevel) { | |
| 880 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 881 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 882 iCount--; | |
| 883 if (iCount < 1) { | |
| 884 return; | |
| 885 } | |
| 886 baseType* pTC; | |
| 887 int32_t iLevel = iBaseLevel; | |
| 888 int32_t iState = FX_IsOdd(iBaseLevel) ? FX_BNSr : FX_BNSl; | |
| 889 int32_t i = 0, iNum = 0, iClsCur, iClsRun, iClsNew, iAction; | |
| 890 for (; i <= iCount; i++) { | |
| 891 pTC = chars.GetDataPtr(i); | |
| 892 iClsCur = pTC->m_iBidiClass; | |
| 893 if (iClsCur == FX_BIDICLASS_BN) { | |
| 894 if (iNum) { | |
| 895 iNum++; | |
| 896 } | |
| 897 continue; | |
| 898 } | |
| 899 FXSYS_assert(iClsCur < FX_BIDICLASS_AL); | |
| 900 iAction = gc_FX_BidiNeutralActions[iState][iClsCur]; | |
| 901 iClsRun = FX_BidiGetDeferredNeutrals(iAction, iLevel); | |
| 902 if (iClsRun != FX_BIDICLASS_N && iNum > 0) { | |
| 903 FX_BidiSetDeferredRun(chars, TRUE, i, iNum, iClsRun); | |
| 904 iNum = 0; | |
| 905 } | |
| 906 iClsNew = FX_BidiGetResolvedNeutrals(iAction); | |
| 907 if (iClsNew != FX_BIDICLASS_N) { | |
| 908 pTC->m_iBidiClass = (int16_t)iClsNew; | |
| 909 } | |
| 910 if (FX_BIDINEUTRALACTION_In & iAction) { | |
| 911 iNum++; | |
| 912 } | |
| 913 iState = gc_FX_BidiNeutralStates[iState][iClsCur]; | |
| 914 iLevel = pTC->m_iBidiLevel; | |
| 915 } | |
| 916 if (iNum > 0) { | |
| 917 iClsCur = FX_BidiDirection(iLevel); | |
| 918 iClsRun = FX_BidiGetDeferredNeutrals( | |
| 919 gc_FX_BidiNeutralActions[iState][iClsCur], iLevel); | |
| 920 if (iClsRun != FX_BIDICLASS_N) { | |
| 921 FX_BidiSetDeferredRun(chars, TRUE, i, iNum, iClsRun); | |
| 922 } | |
| 923 } | |
| 924 } | |
| 925 void FX_BidiResolveImplicit(CFX_ArrayTemplate<baseType>& chars, | |
| 926 int32_t iCount) { | |
| 927 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 928 baseType* pTC; | |
| 929 int32_t iCls, iLevel; | |
| 930 for (int32_t i = 0; i < iCount; i++) { | |
| 931 pTC = chars.GetDataPtr(i); | |
| 932 iCls = pTC->m_iBidiClass; | |
| 933 if (iCls == FX_BIDICLASS_BN) { | |
| 934 continue; | |
| 935 } | |
| 936 FXSYS_assert(iCls > FX_BIDICLASS_ON && iCls < FX_BIDICLASS_AL); | |
| 937 iLevel = pTC->m_iBidiLevel; | |
| 938 iLevel += gc_FX_BidiAddLevel[FX_IsOdd(iLevel)][iCls - 1]; | |
| 939 pTC->m_iBidiLevel = (int16_t)iLevel; | |
| 940 } | |
| 941 } | |
| 942 void FX_BidiResolveWhitespace(CFX_ArrayTemplate<baseType>& chars, | |
| 943 int32_t iCount, | |
| 944 int32_t iBaseLevel) { | |
| 945 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 946 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 947 if (iCount < 1) { | |
| 948 return; | |
| 949 } | |
| 950 iCount--; | |
| 951 int32_t iLevel = iBaseLevel; | |
| 952 int32_t i = 0, iNum = 0; | |
| 953 baseType* pTC; | |
| 954 for (; i <= iCount; i++) { | |
| 955 pTC = chars.GetDataPtr(i); | |
| 956 switch (pTC->m_iBidiClass) { | |
| 957 case FX_BIDICLASS_WS: | |
| 958 iNum++; | |
| 959 break; | |
| 960 case FX_BIDICLASS_RLE: | |
| 961 case FX_BIDICLASS_LRE: | |
| 962 case FX_BIDICLASS_LRO: | |
| 963 case FX_BIDICLASS_RLO: | |
| 964 case FX_BIDICLASS_PDF: | |
| 965 case FX_BIDICLASS_BN: | |
| 966 pTC->m_iBidiLevel = (int16_t)iLevel; | |
| 967 iNum++; | |
| 968 break; | |
| 969 case FX_BIDICLASS_S: | |
| 970 case FX_BIDICLASS_B: | |
| 971 if (iNum > 0) { | |
| 972 FX_BidiSetDeferredRun(chars, FALSE, i, iNum, iBaseLevel); | |
| 973 } | |
| 974 pTC->m_iBidiLevel = (int16_t)iBaseLevel; | |
| 975 iNum = 0; | |
| 976 break; | |
| 977 default: | |
| 978 iNum = 0; | |
| 979 break; | |
| 980 } | |
| 981 iLevel = pTC->m_iBidiLevel; | |
| 982 } | |
| 983 if (iNum > 0) { | |
| 984 FX_BidiSetDeferredRun(chars, FALSE, i, iNum, iBaseLevel); | |
| 985 } | |
| 986 } | |
| 987 int32_t FX_BidiReorderLevel(CFX_ArrayTemplate<baseType>& chars, | |
| 988 int32_t iCount, | |
| 989 int32_t iBaseLevel, | |
| 990 int32_t iStart, | |
| 991 FX_BOOL bReverse) { | |
| 992 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 993 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 994 FXSYS_assert(iStart >= 0 && iStart < iCount); | |
| 995 if (iCount < 1) { | |
| 996 return 0; | |
| 997 } | |
| 998 baseType* pTC; | |
| 999 bReverse = bReverse || FX_IsOdd(iBaseLevel); | |
| 1000 int32_t i = iStart, iLevel; | |
| 1001 for (; i < iCount; i++) { | |
| 1002 pTC = chars.GetDataPtr(i); | |
| 1003 if ((iLevel = pTC->m_iBidiLevel) == iBaseLevel) { | |
| 1004 continue; | |
| 1005 } | |
| 1006 if (iLevel < iBaseLevel) { | |
| 1007 break; | |
| 1008 } | |
| 1009 i += FX_BidiReorderLevel(chars, iCount, iBaseLevel + 1, i, bReverse) - 1; | |
| 1010 } | |
| 1011 int32_t iNum = i - iStart; | |
| 1012 if (bReverse && iNum > 1) { | |
| 1013 FX_BidiReverseString(chars, iStart, iNum); | |
| 1014 } | |
| 1015 return iNum; | |
| 1016 } | |
| 1017 void FX_BidiReorder(CFX_ArrayTemplate<baseType>& chars, | |
| 1018 int32_t iCount, | |
| 1019 int32_t iBaseLevel) { | |
| 1020 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 1021 FXSYS_assert(iBaseLevel >= 0 && iBaseLevel <= FX_BIDIMAXLEVEL); | |
| 1022 int32_t i = 0; | |
| 1023 while (i < iCount) { | |
| 1024 i += FX_BidiReorderLevel(chars, iCount, iBaseLevel, i, FALSE); | |
| 1025 } | |
| 1026 } | |
| 1027 void FX_BidiPosition(CFX_ArrayTemplate<baseType>& chars, int32_t iCount) { | |
| 1028 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 1029 baseType* pTC; | |
| 1030 int32_t i = 0; | |
| 1031 while (i < iCount) { | |
| 1032 pTC = chars.GetDataPtr(i); | |
| 1033 pTC = chars.GetDataPtr(pTC->m_iBidiPos); | |
| 1034 pTC->m_iBidiOrder = i++; | |
| 1035 } | |
| 1036 } | |
| 1037 | |
| 1038 void FX_BidiLine(CFX_ArrayTemplate<baseType>& chars, | |
| 1039 int32_t iCount, | |
| 1040 int32_t iBaseLevel) { | |
| 1041 FXSYS_assert(iCount > -1 && iCount <= chars.GetSize()); | |
| 1042 if (iCount < 2) { | |
| 1043 return; | |
| 1044 } | |
| 1045 FX_BidiClassify(chars, iCount, FALSE); | |
| 1046 FX_BidiResolveExplicit(chars, iCount, iBaseLevel); | |
| 1047 FX_BidiResolveWeak(chars, iCount, iBaseLevel); | |
| 1048 FX_BidiResolveNeutrals(chars, iCount, iBaseLevel); | |
| 1049 FX_BidiResolveImplicit(chars, iCount); | |
| 1050 FX_BidiClassify(chars, iCount, TRUE); | |
| 1051 FX_BidiResolveWhitespace(chars, iCount, iBaseLevel); | |
| 1052 FX_BidiReorder(chars, iCount, iBaseLevel); | |
| 1053 FX_BidiPosition(chars, iCount); | |
| 1054 } | |
| 1055 }; | |
| 1056 void FX_BidiLine(CFX_TxtCharArray& chars, int32_t iCount, int32_t iBaseLevel) { | |
| 1057 CFX_BidiLineTemplate<CFX_TxtChar> blt; | |
| 1058 blt.FX_BidiLine(chars, iCount, iBaseLevel); | |
| 1059 } | |
| 1060 void FX_BidiLine(CFX_RTFCharArray& chars, int32_t iCount, int32_t iBaseLevel) { | |
| 1061 CFX_BidiLineTemplate<CFX_RTFChar> blt; | |
| 1062 blt.FX_BidiLine(chars, iCount, iBaseLevel); | |
| 1063 } | |
| OLD | NEW |