xref: /dokuwiki/inc/utf8.php (revision 8225761095e9517e20d997e58efd5e3dda2c477f)
1*82257610Sandi<?
2*82257610Sandi/**
3*82257610Sandi * UTF8 helper functions
4*82257610Sandi *
5*82257610Sandi * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
6*82257610Sandi * @author     Andreas Gohr <andi@splitbrain.org>
7*82257610Sandi */
8*82257610Sandi
9*82257610Sandi/**
10*82257610Sandi * This is a unicode aware replacement for strtolower()
11*82257610Sandi *
12*82257610Sandi * Uses mb_string extension if available
13*82257610Sandi *
14*82257610Sandi * @author Andreas Gohr <andi@splitbrain.org>
15*82257610Sandi * @see    strtolower()
16*82257610Sandi * @see    utf8_strtoupper()
17*82257610Sandi */
18*82257610Sandifunction utf8_strtolower($string){
19*82257610Sandi  if(!defined('UTF8_NOMBSTRING') && function_exists('mb_strtolower'))
20*82257610Sandi    return mb_strtolower($string,'utf-8');
21*82257610Sandi
22*82257610Sandi  global $UTF8_UPPER_TO_LOWER;
23*82257610Sandi  $uni = utf8_to_unicode($string);
24*82257610Sandi  for ($i=0; $i < count($uni); $i++){
25*82257610Sandi    if($UTF8_UPPER_TO_LOWER[$uni[$i]]){
26*82257610Sandi      $uni[$i] = $UTF8_UPPER_TO_LOWER[$uni[$i]];
27*82257610Sandi    }
28*82257610Sandi  }
29*82257610Sandi  return unicode_to_utf8($uni);
30*82257610Sandi}
31*82257610Sandi
32*82257610Sandi/**
33*82257610Sandi * This is a unicode aware replacement for strtoupper()
34*82257610Sandi *
35*82257610Sandi * Uses mb_string extension if available
36*82257610Sandi *
37*82257610Sandi * @author Andreas Gohr <andi@splitbrain.org>
38*82257610Sandi * @see    strtoupper()
39*82257610Sandi * @see    utf8_strtoupper()
40*82257610Sandi */
41*82257610Sandifunction utf8_strtoupper($string){
42*82257610Sandi  if(!defined('UTF8_NOMBSTRING') && function_exists('mb_strtolower'))
43*82257610Sandi    return mb_strtolower($string,'utf-8');
44*82257610Sandi
45*82257610Sandi  global $UTF8_LOWER_TO_UPPER;
46*82257610Sandi  $uni = utf8_to_unicode($string);
47*82257610Sandi  for ($i=0; $i < count($uni); $i++){
48*82257610Sandi    if($UTF8_LOWER_TO_UPPER[$uni[$i]]){
49*82257610Sandi      $uni[$i] = $UTF8_LOWER_TO_UPPER[$uni[$i]];
50*82257610Sandi    }
51*82257610Sandi  }
52*82257610Sandi  return unicode_to_utf8($uni);
53*82257610Sandi}
54*82257610Sandi
55*82257610Sandi/**
56*82257610Sandi * Replace accented UTF-8 characters by unaccented ASCII-7 equivalents
57*82257610Sandi *
58*82257610Sandi * Use the optional parameter to just deaccent lower ($case = -1) or upper ($case = 1)
59*82257610Sandi * letters. Default is to deaccent both cases ($case = 0)
60*82257610Sandi *
61*82257610Sandi * @author Andreas Gohr <andi@splitbrain.org>
62*82257610Sandi */
63*82257610Sandifunction utf8_deaccent($string,$case=0){
64*82257610Sandi  if($case <= 0){
65*82257610Sandi    global $UTF8_LOWER_ACCENTS;
66*82257610Sandi    $string = str_replace(array_keys($UTF8_LOWER_ACCENTS),array_values($UTF8_LOWER_ACCENTS),$string);
67*82257610Sandi  }
68*82257610Sandi  if($case >= 0){
69*82257610Sandi    global $UTF8_UPPER_ACCENTS;
70*82257610Sandi    $string = str_replace(array_keys($UTF8_UPPER_ACCENTS),array_values($UTF8_UPPER_ACCENTS),$string);
71*82257610Sandi  }
72*82257610Sandi  return $string;
73*82257610Sandi}
74*82257610Sandi
75*82257610Sandi/**
76*82257610Sandi * This function will any UTF-8 encoded text and return it as
77*82257610Sandi * a list of Unicode values:
78*82257610Sandi *
79*82257610Sandi * @author Scott Michael Reynen <scott@randomchaos.com>
80*82257610Sandi * @link   http://www.randomchaos.com/document.php?source=php_and_unicode
81*82257610Sandi * @see    unicode_to_utf8()
82*82257610Sandi */
83*82257610Sandifunction utf8_to_unicode( $str ) {
84*82257610Sandi  $unicode = array();
85*82257610Sandi  $values = array();
86*82257610Sandi  $lookingFor = 1;
87*82257610Sandi
88*82257610Sandi  for ($i = 0; $i < strlen( $str ); $i++ ) {
89*82257610Sandi    $thisValue = ord( $str[ $i ] );
90*82257610Sandi    if ( $thisValue < 128 ) $unicode[] = $thisValue;
91*82257610Sandi    else {
92*82257610Sandi      if ( count( $values ) == 0 ) $lookingFor = ( $thisValue < 224 ) ? 2 : 3;
93*82257610Sandi      $values[] = $thisValue;
94*82257610Sandi      if ( count( $values ) == $lookingFor ) {
95*82257610Sandi  $number = ( $lookingFor == 3 ) ?
96*82257610Sandi    ( ( $values[0] % 16 ) * 4096 ) + ( ( $values[1] % 64 ) * 64 ) + ( $values[2] % 64 ):
97*82257610Sandi  	( ( $values[0] % 32 ) * 64 ) + ( $values[1] % 64 );
98*82257610Sandi  $unicode[] = $number;
99*82257610Sandi  $values = array();
100*82257610Sandi  $lookingFor = 1;
101*82257610Sandi      }
102*82257610Sandi    }
103*82257610Sandi  }
104*82257610Sandi  return $unicode;
105*82257610Sandi}
106*82257610Sandi
107*82257610Sandi/**
108*82257610Sandi * This function will convert a Unicode array back to its UTF-8 representation
109*82257610Sandi *
110*82257610Sandi * @author Scott Michael Reynen <scott@randomchaos.com>
111*82257610Sandi * @link   http://www.randomchaos.com/document.php?source=php_and_unicode
112*82257610Sandi * @see    utf8_to_unicode()
113*82257610Sandi */
114*82257610Sandifunction unicode_to_utf8( $str ) {
115*82257610Sandi  $utf8 = '';
116*82257610Sandi  foreach( $str as $unicode ) {
117*82257610Sandi    if ( $unicode < 128 ) {
118*82257610Sandi      $utf8.= chr( $unicode );
119*82257610Sandi    } elseif ( $unicode < 2048 ) {
120*82257610Sandi      $utf8.= chr( 192 +  ( ( $unicode - ( $unicode % 64 ) ) / 64 ) );
121*82257610Sandi      $utf8.= chr( 128 + ( $unicode % 64 ) );
122*82257610Sandi    } else {
123*82257610Sandi      $utf8.= chr( 224 + ( ( $unicode - ( $unicode % 4096 ) ) / 4096 ) );
124*82257610Sandi      $utf8.= chr( 128 + ( ( ( $unicode % 4096 ) - ( $unicode % 64 ) ) / 64 ) );
125*82257610Sandi      $utf8.= chr( 128 + ( $unicode % 64 ) );
126*82257610Sandi    }
127*82257610Sandi  }
128*82257610Sandi  return $utf8;
129*82257610Sandi}
130*82257610Sandi
131*82257610Sandi/**
132*82257610Sandi * UTF-8 Case lookup table
133*82257610Sandi *
134*82257610Sandi * This lookuptable defines the upper case letters to their correspponding
135*82257610Sandi * lower case letter in UTF-8
136*82257610Sandi *
137*82257610Sandi * @author Andreas Gohr <andi@splitbrain.org>
138*82257610Sandi */
139*82257610Sandi$UTF8_LOWER_TO_UPPER = array(
140*82257610Sandi  0x0061=>0x0041, 0x03C6=>0x03A6, 0x0163=>0x0162, 0x00E5=>0x00C5, 0x0062=>0x0042,
141*82257610Sandi  0x013A=>0x0139, 0x00E1=>0x00C1, 0x0142=>0x0141, 0x03CD=>0x038E, 0x0101=>0x0100,
142*82257610Sandi  0x0491=>0x0490, 0x03B4=>0x0394, 0x015B=>0x015A, 0x0064=>0x0044, 0x03B3=>0x0393,
143*82257610Sandi  0x00F4=>0x00D4, 0x044A=>0x042A, 0x0439=>0x0419, 0x0113=>0x0112, 0x043C=>0x041C,
144*82257610Sandi  0x015F=>0x015E, 0x0144=>0x0143, 0x00EE=>0x00CE, 0x045E=>0x040E, 0x044F=>0x042F,
145*82257610Sandi  0x03BA=>0x039A, 0x0155=>0x0154, 0x0069=>0x0049, 0x0073=>0x0053, 0x1E1F=>0x1E1E,
146*82257610Sandi  0x0135=>0x0134, 0x0447=>0x0427, 0x03C0=>0x03A0, 0x0438=>0x0418, 0x00F3=>0x00D3,
147*82257610Sandi  0x0440=>0x0420, 0x0454=>0x0404, 0x0435=>0x0415, 0x0449=>0x0429, 0x014B=>0x014A,
148*82257610Sandi  0x0431=>0x0411, 0x0459=>0x0409, 0x1E03=>0x1E02, 0x00F6=>0x00D6, 0x00F9=>0x00D9,
149*82257610Sandi  0x006E=>0x004E, 0x0451=>0x0401, 0x03C4=>0x03A4, 0x0443=>0x0423, 0x015D=>0x015C,
150*82257610Sandi  0x0453=>0x0403, 0x03C8=>0x03A8, 0x0159=>0x0158, 0x0067=>0x0047, 0x00E4=>0x00C4,
151*82257610Sandi  0x03AC=>0x0386, 0x03AE=>0x0389, 0x0167=>0x0166, 0x03BE=>0x039E, 0x0165=>0x0164,
152*82257610Sandi  0x0117=>0x0116, 0x0109=>0x0108, 0x0076=>0x0056, 0x00FE=>0x00DE, 0x0157=>0x0156,
153*82257610Sandi  0x00FA=>0x00DA, 0x1E61=>0x1E60, 0x1E83=>0x1E82, 0x00E2=>0x00C2, 0x0119=>0x0118,
154*82257610Sandi  0x0146=>0x0145, 0x0070=>0x0050, 0x0151=>0x0150, 0x044E=>0x042E, 0x0129=>0x0128,
155*82257610Sandi  0x03C7=>0x03A7, 0x013E=>0x013D, 0x0442=>0x0422, 0x007A=>0x005A, 0x0448=>0x0428,
156*82257610Sandi  0x03C1=>0x03A1, 0x1E81=>0x1E80, 0x016D=>0x016C, 0x00F5=>0x00D5, 0x0075=>0x0055,
157*82257610Sandi  0x0177=>0x0176, 0x00FC=>0x00DC, 0x1E57=>0x1E56, 0x03C3=>0x03A3, 0x043A=>0x041A,
158*82257610Sandi  0x006D=>0x004D, 0x016B=>0x016A, 0x0171=>0x0170, 0x0444=>0x0424, 0x00EC=>0x00CC,
159*82257610Sandi  0x0169=>0x0168, 0x03BF=>0x039F, 0x006B=>0x004B, 0x00F2=>0x00D2, 0x00E0=>0x00C0,
160*82257610Sandi  0x0434=>0x0414, 0x03C9=>0x03A9, 0x1E6B=>0x1E6A, 0x00E3=>0x00C3, 0x044D=>0x042D,
161*82257610Sandi  0x0436=>0x0416, 0x01A1=>0x01A0, 0x010D=>0x010C, 0x011D=>0x011C, 0x00F0=>0x00D0,
162*82257610Sandi  0x013C=>0x013B, 0x045F=>0x040F, 0x045A=>0x040A, 0x00E8=>0x00C8, 0x03C5=>0x03A5,
163*82257610Sandi  0x0066=>0x0046, 0x00FD=>0x00DD, 0x0063=>0x0043, 0x021B=>0x021A, 0x00EA=>0x00CA,
164*82257610Sandi  0x03B9=>0x0399, 0x017A=>0x0179, 0x00EF=>0x00CF, 0x01B0=>0x01AF, 0x0065=>0x0045,
165*82257610Sandi  0x03BB=>0x039B, 0x03B8=>0x0398, 0x03BC=>0x039C, 0x045C=>0x040C, 0x043F=>0x041F,
166*82257610Sandi  0x044C=>0x042C, 0x00FE=>0x00DE, 0x00F0=>0x00D0, 0x1EF3=>0x1EF2, 0x0068=>0x0048,
167*82257610Sandi  0x00EB=>0x00CB, 0x0111=>0x0110, 0x0433=>0x0413, 0x012F=>0x012E, 0x00E6=>0x00C6,
168*82257610Sandi  0x0078=>0x0058, 0x0161=>0x0160, 0x016F=>0x016E, 0x03B1=>0x0391, 0x0457=>0x0407,
169*82257610Sandi  0x0173=>0x0172, 0x00FF=>0x0178, 0x006F=>0x004F, 0x043B=>0x041B, 0x03B5=>0x0395,
170*82257610Sandi  0x0445=>0x0425, 0x0121=>0x0120, 0x017E=>0x017D, 0x017C=>0x017B, 0x03B6=>0x0396,
171*82257610Sandi  0x03B2=>0x0392, 0x03AD=>0x0388, 0x1E85=>0x1E84, 0x0175=>0x0174, 0x0071=>0x0051,
172*82257610Sandi  0x0437=>0x0417, 0x1E0B=>0x1E0A, 0x0148=>0x0147, 0x0105=>0x0104, 0x0458=>0x0408,
173*82257610Sandi  0x014D=>0x014C, 0x00ED=>0x00CD, 0x0079=>0x0059, 0x010B=>0x010A, 0x03CE=>0x038F,
174*82257610Sandi  0x0072=>0x0052, 0x0430=>0x0410, 0x0455=>0x0405, 0x0452=>0x0402, 0x0127=>0x0126,
175*82257610Sandi  0x0137=>0x0136, 0x012B=>0x012A, 0x03AF=>0x038A, 0x044B=>0x042B, 0x006C=>0x004C,
176*82257610Sandi  0x03B7=>0x0397, 0x0125=>0x0124, 0x0219=>0x0218, 0x00FB=>0x00DB, 0x011F=>0x011E,
177*82257610Sandi  0x043E=>0x041E, 0x1E41=>0x1E40, 0x03BD=>0x039D, 0x0107=>0x0106, 0x03CB=>0x03AB,
178*82257610Sandi  0x0446=>0x0426, 0x00FE=>0x00DE, 0x00E7=>0x00C7, 0x03CA=>0x03AA, 0x0441=>0x0421,
179*82257610Sandi  0x0432=>0x0412, 0x010F=>0x010E, 0x00F8=>0x00D8, 0x0077=>0x0057, 0x011B=>0x011A,
180*82257610Sandi  0x0074=>0x0054, 0x006A=>0x004A, 0x045B=>0x040B, 0x0456=>0x0406, 0x0103=>0x0102,
181*82257610Sandi  0x03BB=>0x039B, 0x00F1=>0x00D1, 0x043D=>0x041D, 0x03CC=>0x038C, 0x00E9=>0x00C9,
182*82257610Sandi  0x00F0=>0x00D0, 0x0457=>0x0407, 0x0123=>0x0122,
183*82257610Sandi);
184*82257610Sandi
185*82257610Sandi/**
186*82257610Sandi * UTF-8 Case lookup table
187*82257610Sandi *
188*82257610Sandi * This lookuptable defines the lower case letters to their correspponding
189*82257610Sandi * upper case letter in UTF-8 (it does so by flipping $UTF8_LOWER_TO_UPPER)
190*82257610Sandi *
191*82257610Sandi * @author Andreas Gohr <andi@splitbrain.org>
192*82257610Sandi */
193*82257610Sandi$UTF8_UPPER_TO_LOWER = @array_flip($UTF8_LOWER_TO_UPPER);
194*82257610Sandi
195*82257610Sandi/**
196*82257610Sandi * UTF-8 lookup table for lower case accented letters
197*82257610Sandi *
198*82257610Sandi * This lookuptable defines replacements for accented characters from the ASCII-7
199*82257610Sandi * range. This are lower case letters only.
200*82257610Sandi *
201*82257610Sandi * FIXME missing chars eg: æ
202*82257610Sandi *
203*82257610Sandi * @author Andreas Gohr <andi@splitbrain.org>
204*82257610Sandi * @see    utf8_deaccent()
205*82257610Sandi */
206*82257610Sandi$UTF8_LOWER_ACCENTS = array(
207*82257610Sandi  'à' => 'a', 'ô' => 'o', 'ď' => 'd', 'ḟ' => 'f', 'ë' => 'e', 'š' => 's', 'ơ' => 'o',
208*82257610Sandi  'ß' => 'ss', 'ă' => 'a', 'ř' => 'r', 'ț' => 't', 'ň' => 'n', 'ā' => 'a', 'ķ' => 'k',
209*82257610Sandi  'ŝ' => 's', 'ỳ' => 'y', 'ņ' => 'n', 'ĺ' => 'l', 'ħ' => 'h', 'ṗ' => 'p', 'ó' => 'o',
210*82257610Sandi  'ú' => 'u', 'ě' => 'e', 'é' => 'e', 'ç' => 'c', 'ẁ' => 'w', 'ċ' => 'c', 'õ' => 'o',
211*82257610Sandi  'ṡ' => 's', 'ø' => 'o', 'ģ' => 'g', 'ŧ' => 't', 'ș' => 's', 'ė' => 'e', 'ĉ' => 'c',
212*82257610Sandi  'ś' => 's', 'î' => 'i', 'ű' => 'u', 'ć' => 'c', 'ę' => 'e', 'ŵ' => 'w', 'ṫ' => 't',
213*82257610Sandi  'ū' => 'u', 'č' => 'c', 'ö' => 'oe', 'è' => 'e', 'ŷ' => 'y', 'ą' => 'a', 'ł' => 'l',
214*82257610Sandi  'ų' => 'u', 'ů' => 'u', 'ş' => 's', 'ğ' => 'g', 'ļ' => 'l', 'ƒ' => 'f', 'ž' => 'z',
215*82257610Sandi  'ẃ' => 'w', 'ḃ' => 'b', 'å' => 'a', 'ì' => 'i', 'ï' => 'i', 'ḋ' => 'd', 'ť' => 't',
216*82257610Sandi  'ŗ' => 'r', 'ä' => 'ae', 'í' => 'i', 'ŕ' => 'r', 'ê' => 'e', 'ü' => 'ue', 'ò' => 'o',
217*82257610Sandi  'ē' => 'e', 'ñ' => 'n', 'ń' => 'n', 'ĥ' => 'h', 'ĝ' => 'g', 'đ' => 'd', 'ĵ' => 'j',
218*82257610Sandi  'ÿ' => 'y', 'ũ' => 'u', 'ŭ' => 'u', 'ư' => 'u', 'ţ' => 't', 'ý' => 'y', 'ő' => 'o',
219*82257610Sandi  'â' => 'a', 'ľ' => 'l', 'ẅ' => 'w', 'ż' => 'z', 'ī' => 'i', 'ã' => 'a', 'ġ' => 'g',
220*82257610Sandi  'ṁ' => 'm', 'ō' => 'o', 'ĩ' => 'i', 'ù' => 'u', 'į' => 'i', 'ź' => 'z', 'á' => 'a',
221*82257610Sandi  'û' => 'u',
222*82257610Sandi);
223*82257610Sandi
224*82257610Sandi/**
225*82257610Sandi * UTF-8 lookup table for upper case accented letters
226*82257610Sandi *
227*82257610Sandi * This lookuptable defines replacements for accented characters from the ASCII-7
228*82257610Sandi * range. This are upper case letters only.
229*82257610Sandi *
230*82257610Sandi * FIXME missing chars eg: æ
231*82257610Sandi *
232*82257610Sandi * @author Andreas Gohr <andi@splitbrain.org>
233*82257610Sandi * @see    utf8_deaccent()
234*82257610Sandi */
235*82257610Sandi$UTF8_UPPER_ACCENTS = array(
236*82257610Sandi  'à' => 'A', 'ô' => 'O', 'ď' => 'D', 'ḟ' => 'F', 'ë' => 'E', 'š' => 'S', 'ơ' => 'O',
237*82257610Sandi  'ß' => 'Ss', 'ă' => 'A', 'ř' => 'R', 'ț' => 'T', 'ň' => 'N', 'ā' => 'A', 'ķ' => 'K',
238*82257610Sandi  'ŝ' => 'S', 'ỳ' => 'Y', 'ņ' => 'N', 'ĺ' => 'L', 'ħ' => 'H', 'ṗ' => 'P', 'ó' => 'O',
239*82257610Sandi  'ú' => 'U', 'ě' => 'E', 'é' => 'E', 'ç' => 'C', 'ẁ' => 'W', 'ċ' => 'C', 'õ' => 'O',
240*82257610Sandi  'ṡ' => 'S', 'ø' => 'O', 'ģ' => 'G', 'ŧ' => 'T', 'ș' => 'S', 'ė' => 'E', 'ĉ' => 'C',
241*82257610Sandi  'ś' => 'S', 'î' => 'I', 'ű' => 'U', 'ć' => 'C', 'ę' => 'E', 'ŵ' => 'W', 'ṫ' => 'T',
242*82257610Sandi  'ū' => 'U', 'č' => 'C', 'ö' => 'Oe', 'è' => 'E', 'ŷ' => 'Y', 'ą' => 'A', 'ł' => 'L',
243*82257610Sandi  'ų' => 'U', 'ů' => 'U', 'ş' => 'S', 'ğ' => 'G', 'ļ' => 'L', 'ƒ' => 'F', 'ž' => 'Z',
244*82257610Sandi  'ẃ' => 'W', 'ḃ' => 'B', 'å' => 'A', 'ì' => 'I', 'ï' => 'I', 'ḋ' => 'D', 'ť' => 'T',
245*82257610Sandi  'ŗ' => 'R', 'ä' => 'Ae', 'í' => 'I', 'ŕ' => 'R', 'ê' => 'E', 'ü' => 'Ue', 'ò' => 'O',
246*82257610Sandi  'ē' => 'E', 'ñ' => 'N', 'ń' => 'N', 'ĥ' => 'H', 'ĝ' => 'G', 'đ' => 'D', 'ĵ' => 'J',
247*82257610Sandi  'ÿ' => 'Y', 'ũ' => 'U', 'ŭ' => 'U', 'ư' => 'U', 'ţ' => 'T', 'ý' => 'Y', 'ő' => 'O',
248*82257610Sandi  'â' => 'A', 'ľ' => 'L', 'ẅ' => 'W', 'ż' => 'Z', 'ī' => 'I', 'ã' => 'A', 'ġ' => 'G',
249*82257610Sandi  'ṁ' => 'M', 'ō' => 'O', 'ĩ' => 'I', 'ù' => 'U', 'į' => 'I', 'ź' => 'Z', 'á' => 'A',
250*82257610Sandi  'û' => 'U',
251*82257610Sandi);
252*82257610Sandi
253*82257610Sandi?>
254