bidi.js 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. /* Copyright 2017 Mozilla Foundation
  2. *
  3. * Licensed under the Apache License, Version 2.0 (the "License");
  4. * you may not use this file except in compliance with the License.
  5. * You may obtain a copy of the License at
  6. *
  7. * http://www.apache.org/licenses/LICENSE-2.0
  8. *
  9. * Unless required by applicable law or agreed to in writing, software
  10. * distributed under the License is distributed on an "AS IS" BASIS,
  11. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. * See the License for the specific language governing permissions and
  13. * limitations under the License.
  14. */
  15. 'use strict';
  16. Object.defineProperty(exports, "__esModule", {
  17. value: true
  18. });
  19. exports.bidi = undefined;
  20. var _util = require('../shared/util');
  21. var baseTypes = ['BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'S', 'B', 'S', 'WS', 'B', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'B', 'B', 'B', 'S', 'WS', 'ON', 'ON', 'ET', 'ET', 'ET', 'ON', 'ON', 'ON', 'ON', 'ON', 'ES', 'CS', 'ES', 'CS', 'CS', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'CS', 'ON', 'ON', 'ON', 'ON', 'ON', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'ON', 'ON', 'ON', 'ON', 'ON', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'ON', 'ON', 'ON', 'ON', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'B', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'BN', 'CS', 'ON', 'ET', 'ET', 'ET', 'ET', 'ON', 'ON', 'ON', 'ON', 'L', 'ON', 'ON', 'BN', 'ON', 'ON', 'ET', 'ET', 'EN', 'EN', 'ON', 'L', 'ON', 'ON', 'ON', 'EN', 'L', 'ON', 'ON', 'ON', 'ON', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'ON', 'L', 'L', 'L', 'L', 'L', 'L', 'L', 'L'];
  22. var arabicTypes = ['AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'ON', 'ON', 'AL', 'ET', 'ET', 'AL', 'CS', 'AL', 'ON', 'ON', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'AL', 'AL', '', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'AN', 'ET', 'AN', 'AN', 'AL', 'AL', 'AL', 'NSM', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'AN', 'ON', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'NSM', 'AL', 'AL', 'NSM', 'NSM', 'ON', 'NSM', 'NSM', 'NSM', 'NSM', 'AL', 'AL', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'EN', 'AL', 'AL', 'AL', 'AL', 'AL', 'AL'];
  23. function isOdd(i) {
  24. return (i & 1) !== 0;
  25. }
  26. function isEven(i) {
  27. return (i & 1) === 0;
  28. }
  29. function findUnequal(arr, start, value) {
  30. for (var j = start, jj = arr.length; j < jj; ++j) {
  31. if (arr[j] !== value) {
  32. return j;
  33. }
  34. }
  35. return j;
  36. }
  37. function setValues(arr, start, end, value) {
  38. for (var j = start; j < end; ++j) {
  39. arr[j] = value;
  40. }
  41. }
  42. function reverseValues(arr, start, end) {
  43. for (var i = start, j = end - 1; i < j; ++i, --j) {
  44. var temp = arr[i];
  45. arr[i] = arr[j];
  46. arr[j] = temp;
  47. }
  48. }
  49. function createBidiText(str, isLTR, vertical) {
  50. return {
  51. str: str,
  52. dir: vertical ? 'ttb' : isLTR ? 'ltr' : 'rtl'
  53. };
  54. }
  55. var chars = [];
  56. var types = [];
  57. function bidi(str, startLevel, vertical) {
  58. var isLTR = true;
  59. var strLength = str.length;
  60. if (strLength === 0 || vertical) {
  61. return createBidiText(str, isLTR, vertical);
  62. }
  63. chars.length = strLength;
  64. types.length = strLength;
  65. var numBidi = 0;
  66. var i, ii;
  67. for (i = 0; i < strLength; ++i) {
  68. chars[i] = str.charAt(i);
  69. var charCode = str.charCodeAt(i);
  70. var charType = 'L';
  71. if (charCode <= 0x00ff) {
  72. charType = baseTypes[charCode];
  73. } else if (0x0590 <= charCode && charCode <= 0x05f4) {
  74. charType = 'R';
  75. } else if (0x0600 <= charCode && charCode <= 0x06ff) {
  76. charType = arabicTypes[charCode & 0xff];
  77. if (!charType) {
  78. (0, _util.warn)('Bidi: invalid Unicode character ' + charCode.toString(16));
  79. }
  80. } else if (0x0700 <= charCode && charCode <= 0x08AC) {
  81. charType = 'AL';
  82. }
  83. if (charType === 'R' || charType === 'AL' || charType === 'AN') {
  84. numBidi++;
  85. }
  86. types[i] = charType;
  87. }
  88. if (numBidi === 0) {
  89. isLTR = true;
  90. return createBidiText(str, isLTR);
  91. }
  92. if (startLevel === -1) {
  93. if (numBidi / strLength < 0.3) {
  94. isLTR = true;
  95. startLevel = 0;
  96. } else {
  97. isLTR = false;
  98. startLevel = 1;
  99. }
  100. }
  101. var levels = [];
  102. for (i = 0; i < strLength; ++i) {
  103. levels[i] = startLevel;
  104. }
  105. var e = isOdd(startLevel) ? 'R' : 'L';
  106. var sor = e;
  107. var eor = sor;
  108. var lastType = sor;
  109. for (i = 0; i < strLength; ++i) {
  110. if (types[i] === 'NSM') {
  111. types[i] = lastType;
  112. } else {
  113. lastType = types[i];
  114. }
  115. }
  116. lastType = sor;
  117. var t;
  118. for (i = 0; i < strLength; ++i) {
  119. t = types[i];
  120. if (t === 'EN') {
  121. types[i] = lastType === 'AL' ? 'AN' : 'EN';
  122. } else if (t === 'R' || t === 'L' || t === 'AL') {
  123. lastType = t;
  124. }
  125. }
  126. for (i = 0; i < strLength; ++i) {
  127. t = types[i];
  128. if (t === 'AL') {
  129. types[i] = 'R';
  130. }
  131. }
  132. for (i = 1; i < strLength - 1; ++i) {
  133. if (types[i] === 'ES' && types[i - 1] === 'EN' && types[i + 1] === 'EN') {
  134. types[i] = 'EN';
  135. }
  136. if (types[i] === 'CS' && (types[i - 1] === 'EN' || types[i - 1] === 'AN') && types[i + 1] === types[i - 1]) {
  137. types[i] = types[i - 1];
  138. }
  139. }
  140. for (i = 0; i < strLength; ++i) {
  141. if (types[i] === 'EN') {
  142. var j;
  143. for (j = i - 1; j >= 0; --j) {
  144. if (types[j] !== 'ET') {
  145. break;
  146. }
  147. types[j] = 'EN';
  148. }
  149. for (j = i + 1; j < strLength; ++j) {
  150. if (types[j] !== 'ET') {
  151. break;
  152. }
  153. types[j] = 'EN';
  154. }
  155. }
  156. }
  157. for (i = 0; i < strLength; ++i) {
  158. t = types[i];
  159. if (t === 'WS' || t === 'ES' || t === 'ET' || t === 'CS') {
  160. types[i] = 'ON';
  161. }
  162. }
  163. lastType = sor;
  164. for (i = 0; i < strLength; ++i) {
  165. t = types[i];
  166. if (t === 'EN') {
  167. types[i] = lastType === 'L' ? 'L' : 'EN';
  168. } else if (t === 'R' || t === 'L') {
  169. lastType = t;
  170. }
  171. }
  172. for (i = 0; i < strLength; ++i) {
  173. if (types[i] === 'ON') {
  174. var end = findUnequal(types, i + 1, 'ON');
  175. var before = sor;
  176. if (i > 0) {
  177. before = types[i - 1];
  178. }
  179. var after = eor;
  180. if (end + 1 < strLength) {
  181. after = types[end + 1];
  182. }
  183. if (before !== 'L') {
  184. before = 'R';
  185. }
  186. if (after !== 'L') {
  187. after = 'R';
  188. }
  189. if (before === after) {
  190. setValues(types, i, end, before);
  191. }
  192. i = end - 1;
  193. }
  194. }
  195. for (i = 0; i < strLength; ++i) {
  196. if (types[i] === 'ON') {
  197. types[i] = e;
  198. }
  199. }
  200. for (i = 0; i < strLength; ++i) {
  201. t = types[i];
  202. if (isEven(levels[i])) {
  203. if (t === 'R') {
  204. levels[i] += 1;
  205. } else if (t === 'AN' || t === 'EN') {
  206. levels[i] += 2;
  207. }
  208. } else {
  209. if (t === 'L' || t === 'AN' || t === 'EN') {
  210. levels[i] += 1;
  211. }
  212. }
  213. }
  214. var highestLevel = -1;
  215. var lowestOddLevel = 99;
  216. var level;
  217. for (i = 0, ii = levels.length; i < ii; ++i) {
  218. level = levels[i];
  219. if (highestLevel < level) {
  220. highestLevel = level;
  221. }
  222. if (lowestOddLevel > level && isOdd(level)) {
  223. lowestOddLevel = level;
  224. }
  225. }
  226. for (level = highestLevel; level >= lowestOddLevel; --level) {
  227. var start = -1;
  228. for (i = 0, ii = levels.length; i < ii; ++i) {
  229. if (levels[i] < level) {
  230. if (start >= 0) {
  231. reverseValues(chars, start, i);
  232. start = -1;
  233. }
  234. } else if (start < 0) {
  235. start = i;
  236. }
  237. }
  238. if (start >= 0) {
  239. reverseValues(chars, start, levels.length);
  240. }
  241. }
  242. for (i = 0, ii = chars.length; i < ii; ++i) {
  243. var ch = chars[i];
  244. if (ch === '<' || ch === '>') {
  245. chars[i] = '';
  246. }
  247. }
  248. return createBidiText(chars.join(''), isLTR);
  249. }
  250. exports.bidi = bidi;