一尘不染

PHP函数imagettftext()和unicode

php

我正在使用PHP函数imagettftext()将文本转换为GIF图像。我要转换的文本具有Unicode字符,包括日语。在我的本地计算机(Ubuntu
7.10)上一切正常,但是在我的Web主机服务器上,日语字符混乱。是什么引起差异?一切都应编码为UTF-8。

Web主机服务器上的图像损坏:http :
//www.ibeni.net/flashcards/imagetest.php

从我的本地机器上复制正确的图像:http :
//www.ibeni.net/flashcards/imagetest.php.gif

从我的本地计算机复制phpinfo():http :
//www.ibeni.net/flashcards/phpinfo.php.html

从我的网络主机服务器复制phpinfo():http :
//example5.nfshost.com/phpinfo

码:

mb_language('uni');
mb_internal_encoding('UTF-8');

header('Content-type: image/gif');

$text = '日本語';
$font = './Cyberbit.ttf';

// Create the image
$im = imagecreatetruecolor(160, 160);
$white = imagecolorallocate($im, 255, 255, 255);
$black = imagecolorallocate($im, 0, 0, 0);

// Create some colors
imagefilledrectangle($im, 0, 0, 159, 159, $white);

// Add the text
imagettftext($im, 12, 0, 20, 20, $black, $font, $text);
imagegif($im);
imagedestroy($im);

阅读 423

收藏
2020-05-29

共1个答案

一尘不染

这是最终对我有用的解决方案:

$text = "你好";
// Convert UTF-8 string to HTML entities
$text = mb_convert_encoding($text, 'HTML-ENTITIES',"UTF-8");
// Convert HTML entities into ISO-8859-1
$text = html_entity_decode($text,ENT_NOQUOTES, "ISO-8859-1");
// Convert characters > 127 into their hexidecimal equivalents
$out = "";
for($i = 0; $i < strlen($text); $i++) {
    $letter = $text[$i];
    $num = ord($letter);
    if($num>127) {
      $out .= "&#$num;";
    } else {
      $out .=  $letter;
    }
}

将字符串转换为HTML实体是可行的,除了函数imagettftext()不接受命名实体。例如,

&#26085;&#26412;&#35486;

可以,但是

&ccedil;

不是。转换回ISO-8859-1,将命名实体转换回字符,但是还有第二个问题。imagettftext()不支持值大于>
127的字符。最终的for循环将这些字符编码为十六进制。此解决方案对我正在使用的文本有效(包括日语,中文和葡萄牙语的加重拉丁字符),但是我不确定100%可以在所有情况下都适用。

所有这些体操都是必需的,因为imagettftext()在我的服务器上实际上并不接受UTF-8字符串。

2020-05-29