You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
902 lines
30 KiB
902 lines
30 KiB
2 years ago
|
<?php
|
||
|
/**
|
||
|
* Provides static methods for charset and locale safe string manipulation.
|
||
|
*
|
||
|
* Copyright 2003-2017 Horde LLC (http://www.horde.org/)
|
||
|
*
|
||
|
* See the enclosed file COPYING for license information (LGPL). If you
|
||
|
* did not receive this file, see http://www.horde.org/licenses/lgpl21.
|
||
|
*
|
||
|
* @todo Split up in Horde_String_Multibyte for multibyte-safe methods and
|
||
|
* Horde_String_Locale for locale-safe methods.
|
||
|
*
|
||
|
* @author Jan Schneider <jan@horde.org>
|
||
|
* @category Horde
|
||
|
* @license http://www.horde.org/licenses/lgpl21 LGPL 2.1
|
||
|
* @package Util
|
||
|
*/
|
||
|
class Horde_String
|
||
|
{
|
||
|
/**
|
||
|
* lower() cache.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
protected static $_lowers = array();
|
||
|
|
||
|
/**
|
||
|
* upper() cache.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
protected static $_uppers = array();
|
||
|
|
||
|
/**
|
||
|
* Converts a string from one charset to another.
|
||
|
*
|
||
|
* Uses the iconv or the mbstring extensions.
|
||
|
* The original string is returned if conversion failed or none
|
||
|
* of the extensions were available.
|
||
|
*
|
||
|
* @param mixed $input The data to be converted. If $input is an an
|
||
|
* array, the array's values get converted
|
||
|
* recursively.
|
||
|
* @param string $from The string's current charset.
|
||
|
* @param string $to The charset to convert the string to.
|
||
|
* @param boolean $force Force conversion?
|
||
|
*
|
||
|
* @return mixed The converted input data.
|
||
|
*/
|
||
|
public static function convertCharset($input, $from, $to, $force = false)
|
||
|
{
|
||
|
/* Don't bother converting numbers. */
|
||
|
if (is_numeric($input)) {
|
||
|
return $input;
|
||
|
}
|
||
|
|
||
|
/* If the from and to character sets are identical, return now. */
|
||
|
if (!$force && $from == $to) {
|
||
|
return $input;
|
||
|
}
|
||
|
$from = self::lower($from);
|
||
|
$to = self::lower($to);
|
||
|
if (!$force && $from == $to) {
|
||
|
return $input;
|
||
|
}
|
||
|
|
||
|
if (is_array($input)) {
|
||
|
$tmp = array();
|
||
|
foreach ($input as $key => $val) {
|
||
|
$tmp[self::_convertCharset($key, $from, $to)] = self::convertCharset($val, $from, $to, $force);
|
||
|
}
|
||
|
return $tmp;
|
||
|
}
|
||
|
|
||
|
if (is_object($input)) {
|
||
|
// PEAR_Error/Exception objects are almost guaranteed to contain
|
||
|
// recursion, which will cause a segfault in PHP. We should never
|
||
|
// reach this line, but add a check.
|
||
|
if (($input instanceof Exception) ||
|
||
|
($input instanceof PEAR_Error)) {
|
||
|
return '';
|
||
|
}
|
||
|
|
||
|
$input = clone $input;
|
||
|
$vars = get_object_vars($input);
|
||
|
foreach ($vars as $key => $val) {
|
||
|
$input->$key = self::convertCharset($val, $from, $to, $force);
|
||
|
}
|
||
|
return $input;
|
||
|
}
|
||
|
|
||
|
if (!is_string($input)) {
|
||
|
return $input;
|
||
|
}
|
||
|
|
||
|
return self::_convertCharset($input, $from, $to);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Internal function used to do charset conversion.
|
||
|
*
|
||
|
* @param string $input See self::convertCharset().
|
||
|
* @param string $from See self::convertCharset().
|
||
|
* @param string $to See self::convertCharset().
|
||
|
*
|
||
|
* @return string The converted string.
|
||
|
*/
|
||
|
protected static function _convertCharset($input, $from, $to)
|
||
|
{
|
||
|
/* Use utf8_[en|de]code() if possible and if the string isn't too
|
||
|
* large (less than 16 MB = 16 * 1024 * 1024 = 16777216 bytes) - these
|
||
|
* functions use more memory. */
|
||
|
if (Horde_Util::extensionExists('xml') &&
|
||
|
((strlen($input) < 16777216) ||
|
||
|
!Horde_Util::extensionExists('iconv') ||
|
||
|
!Horde_Util::extensionExists('mbstring'))) {
|
||
|
if (($to == 'utf-8') &&
|
||
|
in_array($from, array('iso-8859-1', 'us-ascii', 'utf-8'))) {
|
||
|
return utf8_encode($input);
|
||
|
}
|
||
|
|
||
|
if (($from == 'utf-8') &&
|
||
|
in_array($to, array('iso-8859-1', 'us-ascii', 'utf-8'))) {
|
||
|
return utf8_decode($input);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Try UTF7-IMAP conversions. */
|
||
|
if (($from == 'utf7-imap') || ($to == 'utf7-imap')) {
|
||
|
try {
|
||
|
if ($from == 'utf7-imap') {
|
||
|
return self::convertCharset(Horde_Imap_Client_Utf7imap::Utf7ImapToUtf8($input), 'UTF-8', $to);
|
||
|
} else {
|
||
|
if ($from == 'utf-8') {
|
||
|
$conv = $input;
|
||
|
} else {
|
||
|
$conv = self::convertCharset($input, $from, 'UTF-8');
|
||
|
}
|
||
|
return Horde_Imap_Client_Utf7imap::Utf8ToUtf7Imap($conv);
|
||
|
}
|
||
|
} catch (Horde_Imap_Client_Exception $e) {
|
||
|
return $input;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Try iconv with transliteration. */
|
||
|
if (Horde_Util::extensionExists('iconv')) {
|
||
|
unset($php_errormsg);
|
||
|
ini_set('track_errors', 1);
|
||
|
$out = @iconv($from, $to . '//TRANSLIT', $input);
|
||
|
$errmsg = isset($php_errormsg);
|
||
|
ini_restore('track_errors');
|
||
|
if (!$errmsg && $out !== false) {
|
||
|
return $out;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Try mbstring. */
|
||
|
if (Horde_Util::extensionExists('mbstring')) {
|
||
|
$out = @mb_convert_encoding($input, $to, self::_mbstringCharset($from));
|
||
|
if (!empty($out)) {
|
||
|
return $out;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $input;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Makes a string lowercase.
|
||
|
*
|
||
|
* @param string $string The string to be converted.
|
||
|
* @param boolean $locale If true the string will be converted based on
|
||
|
* a given charset, locale independent else.
|
||
|
* @param string $charset If $locale is true, the charset to use when
|
||
|
* converting.
|
||
|
*
|
||
|
* @return string The string with lowercase characters.
|
||
|
*/
|
||
|
public static function lower($string, $locale = false, $charset = null)
|
||
|
{
|
||
|
if ($locale) {
|
||
|
if (Horde_Util::extensionExists('mbstring')) {
|
||
|
if (is_null($charset)) {
|
||
|
throw new InvalidArgumentException('$charset argument must not be null');
|
||
|
}
|
||
|
$ret = @mb_strtolower($string, self::_mbstringCharset($charset));
|
||
|
if (!empty($ret)) {
|
||
|
return $ret;
|
||
|
}
|
||
|
}
|
||
|
return strtolower($string);
|
||
|
}
|
||
|
|
||
|
if (!isset(self::$_lowers[$string])) {
|
||
|
$language = setlocale(LC_CTYPE, 0);
|
||
|
setlocale(LC_CTYPE, 'C');
|
||
|
self::$_lowers[$string] = strtolower($string);
|
||
|
setlocale(LC_CTYPE, $language);
|
||
|
}
|
||
|
|
||
|
return self::$_lowers[$string];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Makes a string uppercase.
|
||
|
*
|
||
|
* @param string $string The string to be converted.
|
||
|
* @param boolean $locale If true the string will be converted based on a
|
||
|
* given charset, locale independent else.
|
||
|
* @param string $charset If $locale is true, the charset to use when
|
||
|
* converting. If not provided the current charset.
|
||
|
*
|
||
|
* @return string The string with uppercase characters.
|
||
|
*/
|
||
|
public static function upper($string, $locale = false, $charset = null)
|
||
|
{
|
||
|
if ($locale) {
|
||
|
if (Horde_Util::extensionExists('mbstring')) {
|
||
|
if (is_null($charset)) {
|
||
|
throw new InvalidArgumentException('$charset argument must not be null');
|
||
|
}
|
||
|
$ret = @mb_strtoupper($string, self::_mbstringCharset($charset));
|
||
|
if (!empty($ret)) {
|
||
|
return $ret;
|
||
|
}
|
||
|
}
|
||
|
return strtoupper($string);
|
||
|
}
|
||
|
|
||
|
if (!isset(self::$_uppers[$string])) {
|
||
|
$language = setlocale(LC_CTYPE, 0);
|
||
|
setlocale(LC_CTYPE, 'C');
|
||
|
self::$_uppers[$string] = strtoupper($string);
|
||
|
setlocale(LC_CTYPE, $language);
|
||
|
}
|
||
|
|
||
|
return self::$_uppers[$string];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns a string with the first letter capitalized if it is
|
||
|
* alphabetic.
|
||
|
*
|
||
|
* @param string $string The string to be capitalized.
|
||
|
* @param boolean $locale If true the string will be converted based on a
|
||
|
* given charset, locale independent else.
|
||
|
* @param string $charset The charset to use, defaults to current charset.
|
||
|
*
|
||
|
* @return string The capitalized string.
|
||
|
*/
|
||
|
public static function ucfirst($string, $locale = false, $charset = null)
|
||
|
{
|
||
|
if ($locale) {
|
||
|
if (is_null($charset)) {
|
||
|
throw new InvalidArgumentException('$charset argument must not be null');
|
||
|
}
|
||
|
$first = self::substr($string, 0, 1, $charset);
|
||
|
if (self::isAlpha($first, $charset)) {
|
||
|
$string = self::upper($first, true, $charset) . self::substr($string, 1, null, $charset);
|
||
|
}
|
||
|
} else {
|
||
|
$string = self::upper(substr($string, 0, 1), false) . substr($string, 1);
|
||
|
}
|
||
|
|
||
|
return $string;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns a string with the first letter of each word capitalized if it is
|
||
|
* alphabetic.
|
||
|
*
|
||
|
* Sentences are splitted into words at whitestrings.
|
||
|
*
|
||
|
* @param string $string The string to be capitalized.
|
||
|
* @param boolean $locale If true the string will be converted based on a
|
||
|
* given charset, locale independent else.
|
||
|
* @param string $charset The charset to use, defaults to current charset.
|
||
|
*
|
||
|
* @return string The capitalized string.
|
||
|
*/
|
||
|
public static function ucwords($string, $locale = false, $charset = null)
|
||
|
{
|
||
|
$words = preg_split('/(\s+)/', $string, -1, PREG_SPLIT_DELIM_CAPTURE);
|
||
|
for ($i = 0, $c = count($words); $i < $c; $i += 2) {
|
||
|
$words[$i] = self::ucfirst($words[$i], $locale, $charset);
|
||
|
}
|
||
|
return implode('', $words);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns part of a string.
|
||
|
*
|
||
|
* @param string $string The string to be converted.
|
||
|
* @param integer $start The part's start position, zero based.
|
||
|
* @param integer $length The part's length.
|
||
|
* @param string $charset The charset to use when calculating the part's
|
||
|
* position and length, defaults to current
|
||
|
* charset.
|
||
|
*
|
||
|
* @return string The string's part.
|
||
|
*/
|
||
|
public static function substr($string, $start, $length = null,
|
||
|
$charset = 'UTF-8')
|
||
|
{
|
||
|
if (is_null($length)) {
|
||
|
$length = self::length($string, $charset) - $start;
|
||
|
}
|
||
|
|
||
|
if ($length === 0) {
|
||
|
return '';
|
||
|
}
|
||
|
|
||
|
$error = false;
|
||
|
|
||
|
/* Try mbstring. */
|
||
|
if (Horde_Util::extensionExists('mbstring')) {
|
||
|
$ret = @mb_substr($string, $start, $length, self::_mbstringCharset($charset));
|
||
|
|
||
|
/* mb_substr() returns empty string on failure. */
|
||
|
if (strlen($ret)) {
|
||
|
return $ret;
|
||
|
}
|
||
|
$error = true;
|
||
|
}
|
||
|
|
||
|
/* Try iconv. */
|
||
|
if (Horde_Util::extensionExists('iconv')) {
|
||
|
$ret = @iconv_substr($string, $start, $length, $charset);
|
||
|
|
||
|
/* iconv_substr() returns false on failure. */
|
||
|
if ($ret !== false) {
|
||
|
return $ret;
|
||
|
}
|
||
|
$error = true;
|
||
|
}
|
||
|
|
||
|
/* Try intl. */
|
||
|
if (Horde_Util::extensionExists('intl')) {
|
||
|
$ret = self::convertCharset(
|
||
|
@grapheme_substr(
|
||
|
self::convertCharset($string, $charset, 'UTF-8'),
|
||
|
$start,
|
||
|
$length
|
||
|
),
|
||
|
'UTF-8',
|
||
|
$charset
|
||
|
);
|
||
|
|
||
|
/* grapheme_substr() returns false on failure. */
|
||
|
if ($ret !== false) {
|
||
|
return $ret;
|
||
|
}
|
||
|
$error = true;
|
||
|
}
|
||
|
|
||
|
return $error
|
||
|
? ''
|
||
|
: substr($string, $start, $length);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the character (not byte) length of a string.
|
||
|
*
|
||
|
* @param string $string The string to return the length of.
|
||
|
* @param string $charset The charset to use when calculating the string's
|
||
|
* length.
|
||
|
*
|
||
|
* @return integer The string's length.
|
||
|
*/
|
||
|
public static function length($string, $charset = 'UTF-8')
|
||
|
{
|
||
|
$charset = self::lower($charset);
|
||
|
|
||
|
if ($charset == 'utf-8' || $charset == 'utf8') {
|
||
|
return strlen(utf8_decode($string));
|
||
|
}
|
||
|
|
||
|
if (Horde_Util::extensionExists('mbstring')) {
|
||
|
$ret = @mb_strlen($string, self::_mbstringCharset($charset));
|
||
|
if (!empty($ret)) {
|
||
|
return $ret;
|
||
|
}
|
||
|
}
|
||
|
if (Horde_Util::extensionExists('intl')) {
|
||
|
return grapheme_strlen(
|
||
|
self::convertCharset($string, $charset, 'UTF-8')
|
||
|
);
|
||
|
}
|
||
|
|
||
|
return strlen($string);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the numeric position of the first occurrence of $needle
|
||
|
* in the $haystack string.
|
||
|
*
|
||
|
* @param string $haystack The string to search through.
|
||
|
* @param string $needle The string to search for.
|
||
|
* @param integer $offset Character in $haystack to start searching at.
|
||
|
* @param string $charset Charset of $needle.
|
||
|
*
|
||
|
* @return integer The position of first occurrence.
|
||
|
*/
|
||
|
public static function pos(
|
||
|
$haystack, $needle, $offset = 0, $charset = 'UTF-8'
|
||
|
)
|
||
|
{
|
||
|
return self::_pos($haystack, $needle, $offset, $charset, 'strpos');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the numeric position of the first case-insensitive occurrence
|
||
|
* of $needle in the $haystack string.
|
||
|
*
|
||
|
* @since 2.5.0
|
||
|
*
|
||
|
* @param string $haystack The string to search through.
|
||
|
* @param string $needle The string to search for.
|
||
|
* @param integer $offset Character in $haystack to start searching at.
|
||
|
* @param string $charset Charset of $needle.
|
||
|
*
|
||
|
* @return integer The position of first case-insensitive occurrence.
|
||
|
*/
|
||
|
public static function ipos(
|
||
|
$haystack, $needle, $offset = 0, $charset = 'UTF-8'
|
||
|
)
|
||
|
{
|
||
|
return self::_pos($haystack, $needle, $offset, $charset, 'stripos');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the numeric position of the last occurrence of $needle
|
||
|
* in the $haystack string.
|
||
|
*
|
||
|
* @param string $haystack The string to search through.
|
||
|
* @param string $needle The string to search for.
|
||
|
* @param integer $offset Character in $haystack to start searching at.
|
||
|
* @param string $charset Charset of $needle.
|
||
|
*
|
||
|
* @return integer The position of last occurrence.
|
||
|
*/
|
||
|
public static function rpos(
|
||
|
$haystack, $needle, $offset = 0, $charset = 'UTF-8'
|
||
|
)
|
||
|
{
|
||
|
return self::_pos($haystack, $needle, $offset, $charset, 'strrpos');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the numeric position of the last case-insensitive occurrence of
|
||
|
* $needle in the $haystack string.
|
||
|
*
|
||
|
* @since 2.5.0
|
||
|
*
|
||
|
* @param string $haystack The string to search through.
|
||
|
* @param string $needle The string to search for.
|
||
|
* @param integer $offset Character in $haystack to start searching at.
|
||
|
* @param string $charset Charset of $needle.
|
||
|
*
|
||
|
* @return integer The position of last case-insensitive occurrence.
|
||
|
*/
|
||
|
public static function ripos(
|
||
|
$haystack, $needle, $offset = 0, $charset = 'UTF-8'
|
||
|
)
|
||
|
{
|
||
|
return self::_pos($haystack, $needle, $offset, $charset, 'strripos');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Perform string position searches.
|
||
|
*
|
||
|
* @param string $haystack The string to search through.
|
||
|
* @param string $needle The string to search for.
|
||
|
* @param integer $offset Character in $haystack to start searching at.
|
||
|
* @param string $charset Charset of $needle.
|
||
|
* @param string $func Function to use.
|
||
|
*
|
||
|
* @return integer The position of occurrence.
|
||
|
*
|
||
|
*/
|
||
|
protected static function _pos(
|
||
|
$haystack, $needle, $offset, $charset, $func
|
||
|
)
|
||
|
{
|
||
|
if (Horde_Util::extensionExists('mbstring')) {
|
||
|
unset($php_errormsg);
|
||
|
$track_errors = ini_set('track_errors', 1);
|
||
|
$ret = @call_user_func('mb_' . $func, $haystack, $needle, $offset, self::_mbstringCharset($charset));
|
||
|
ini_set('track_errors', $track_errors);
|
||
|
if (!isset($php_errormsg)) {
|
||
|
return $ret;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (Horde_Util::extensionExists('intl')) {
|
||
|
unset($php_errormsg);
|
||
|
$track_errors = ini_set('track_errors', 1);
|
||
|
$ret = self::convertCharset(
|
||
|
@call_user_func(
|
||
|
'grapheme_' . $func,
|
||
|
self::convertCharset($haystack, $charset, 'UTF-8'),
|
||
|
self::convertCharset($needle, $charset, 'UTF-8'),
|
||
|
$offset
|
||
|
),
|
||
|
'UTF-8',
|
||
|
$charset
|
||
|
);
|
||
|
ini_set('track_errors', $track_errors);
|
||
|
if (!isset($php_errormsg)) {
|
||
|
return $ret;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $func($haystack, $needle, $offset);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns a string padded to a certain length with another string.
|
||
|
* This method behaves exactly like str_pad() but is multibyte safe.
|
||
|
*
|
||
|
* @param string $input The string to be padded.
|
||
|
* @param integer $length The length of the resulting string.
|
||
|
* @param string $pad The string to pad the input string with. Must
|
||
|
* be in the same charset like the input string.
|
||
|
* @param const $type The padding type. One of STR_PAD_LEFT,
|
||
|
* STR_PAD_RIGHT, or STR_PAD_BOTH.
|
||
|
* @param string $charset The charset of the input and the padding
|
||
|
* strings.
|
||
|
*
|
||
|
* @return string The padded string.
|
||
|
*/
|
||
|
public static function pad($input, $length, $pad = ' ',
|
||
|
$type = STR_PAD_RIGHT, $charset = 'UTF-8')
|
||
|
{
|
||
|
$mb_length = self::length($input, $charset);
|
||
|
$sb_length = strlen($input);
|
||
|
$pad_length = self::length($pad, $charset);
|
||
|
|
||
|
/* Return if we already have the length. */
|
||
|
if ($mb_length >= $length) {
|
||
|
return $input;
|
||
|
}
|
||
|
|
||
|
/* Shortcut for single byte strings. */
|
||
|
if ($mb_length == $sb_length && $pad_length == strlen($pad)) {
|
||
|
return str_pad($input, $length, $pad, $type);
|
||
|
}
|
||
|
|
||
|
switch ($type) {
|
||
|
case STR_PAD_LEFT:
|
||
|
$left = $length - $mb_length;
|
||
|
$output = self::substr(str_repeat($pad, ceil($left / $pad_length)), 0, $left, $charset) . $input;
|
||
|
break;
|
||
|
|
||
|
case STR_PAD_BOTH:
|
||
|
$left = floor(($length - $mb_length) / 2);
|
||
|
$right = ceil(($length - $mb_length) / 2);
|
||
|
$output = self::substr(str_repeat($pad, ceil($left / $pad_length)), 0, $left, $charset) .
|
||
|
$input .
|
||
|
self::substr(str_repeat($pad, ceil($right / $pad_length)), 0, $right, $charset);
|
||
|
break;
|
||
|
|
||
|
case STR_PAD_RIGHT:
|
||
|
$right = $length - $mb_length;
|
||
|
$output = $input . self::substr(str_repeat($pad, ceil($right / $pad_length)), 0, $right, $charset);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return $output;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Wraps the text of a message.
|
||
|
*
|
||
|
* @param string $string String containing the text to wrap.
|
||
|
* @param integer $width Wrap the string at this number of
|
||
|
* characters.
|
||
|
* @param string $break Character(s) to use when breaking lines.
|
||
|
* @param boolean $cut Whether to cut inside words if a line
|
||
|
* can't be wrapped.
|
||
|
* @param boolean $line_folding Whether to apply line folding rules per
|
||
|
* RFC 822 or similar. The correct break
|
||
|
* characters including leading whitespace
|
||
|
* have to be specified too.
|
||
|
*
|
||
|
* @return string String containing the wrapped text.
|
||
|
*/
|
||
|
public static function wordwrap($string, $width = 75, $break = "\n",
|
||
|
$cut = false, $line_folding = false)
|
||
|
{
|
||
|
$wrapped = '';
|
||
|
|
||
|
while (self::length($string, 'UTF-8') > $width) {
|
||
|
$line = self::substr($string, 0, $width, 'UTF-8');
|
||
|
$string = self::substr($string, self::length($line, 'UTF-8'), null, 'UTF-8');
|
||
|
|
||
|
// Make sure we didn't cut a word, unless we want hard breaks
|
||
|
// anyway.
|
||
|
if (!$cut && preg_match('/^(.+?)((\s|\r?\n).*)/us', $string, $match)) {
|
||
|
$line .= $match[1];
|
||
|
$string = $match[2];
|
||
|
}
|
||
|
|
||
|
// Wrap at existing line breaks.
|
||
|
if (preg_match('/^(.*?)(\r?\n)(.*)$/su', $line, $match)) {
|
||
|
$wrapped .= $match[1] . $match[2];
|
||
|
$string = $match[3] . $string;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Wrap at the last colon or semicolon followed by a whitespace if
|
||
|
// doing line folding.
|
||
|
if ($line_folding &&
|
||
|
preg_match('/^(.*?)(;|:)(\s+.*)$/u', $line, $match)) {
|
||
|
$wrapped .= $match[1] . $match[2] . $break;
|
||
|
$string = $match[3] . $string;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Wrap at the last whitespace of $line.
|
||
|
$sub = $line_folding
|
||
|
? '(.+[^\s])'
|
||
|
: '(.*)';
|
||
|
|
||
|
if (preg_match('/^' . $sub . '(\s+)(.*)$/u', $line, $match)) {
|
||
|
$wrapped .= $match[1] . $break;
|
||
|
$string = ($line_folding ? $match[2] : '') . $match[3] . $string;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Hard wrap if necessary.
|
||
|
if ($cut) {
|
||
|
$wrapped .= $line . $break;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$wrapped .= $line;
|
||
|
}
|
||
|
|
||
|
return $wrapped . $string;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Wraps the text of a message.
|
||
|
*
|
||
|
* @param string $text String containing the text to wrap.
|
||
|
* @param integer $length Wrap $text at this number of characters.
|
||
|
* @param string $break_char Character(s) to use when breaking lines.
|
||
|
* @param boolean $quote Ignore lines that are wrapped with the '>'
|
||
|
* character (RFC 2646)? If true, we don't
|
||
|
* remove any padding whitespace at the end of
|
||
|
* the string.
|
||
|
*
|
||
|
* @return string String containing the wrapped text.
|
||
|
*/
|
||
|
public static function wrap($text, $length = 80, $break_char = "\n",
|
||
|
$quote = false)
|
||
|
{
|
||
|
$paragraphs = array();
|
||
|
|
||
|
foreach (preg_split('/\r?\n/', $text) as $input) {
|
||
|
if ($quote && (strpos($input, '>') === 0)) {
|
||
|
$line = $input;
|
||
|
} else {
|
||
|
/* We need to handle the Usenet-style signature line
|
||
|
* separately; since the space after the two dashes is
|
||
|
* REQUIRED, we don't want to trim the line. */
|
||
|
if ($input != '-- ') {
|
||
|
$input = rtrim($input);
|
||
|
}
|
||
|
$line = self::wordwrap($input, $length, $break_char);
|
||
|
}
|
||
|
|
||
|
$paragraphs[] = $line;
|
||
|
}
|
||
|
|
||
|
return implode($break_char, $paragraphs);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return a truncated string, suitable for notifications.
|
||
|
*
|
||
|
* @param string $text The original string.
|
||
|
* @param integer $length The maximum length.
|
||
|
*
|
||
|
* @return string The truncated string, if longer than $length.
|
||
|
*/
|
||
|
public static function truncate($text, $length = 100)
|
||
|
{
|
||
|
return (self::length($text) > $length)
|
||
|
? rtrim(self::substr($text, 0, $length - 3)) . '...'
|
||
|
: $text;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return an abbreviated string, with characters in the middle of the
|
||
|
* excessively long string replaced by '...'.
|
||
|
*
|
||
|
* @param string $text The original string.
|
||
|
* @param integer $length The length at which to abbreviate.
|
||
|
*
|
||
|
* @return string The abbreviated string, if longer than $length.
|
||
|
*/
|
||
|
public static function abbreviate($text, $length = 20)
|
||
|
{
|
||
|
return (self::length($text) > $length)
|
||
|
? rtrim(self::substr($text, 0, round(($length - 3) / 2))) . '...' . ltrim(self::substr($text, (($length - 3) / 2) * -1))
|
||
|
: $text;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the common leading part of two strings.
|
||
|
*
|
||
|
* @param string $str1 A string.
|
||
|
* @param string $str2 Another string.
|
||
|
*
|
||
|
* @return string The start of $str1 and $str2 that is identical in both.
|
||
|
*/
|
||
|
public static function common($str1, $str2)
|
||
|
{
|
||
|
for ($result = '', $i = 0;
|
||
|
isset($str1[$i]) && isset($str2[$i]) && $str1[$i] == $str2[$i];
|
||
|
$i++) {
|
||
|
$result .= $str1[$i];
|
||
|
}
|
||
|
return $result;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns true if the every character in the parameter is an alphabetic
|
||
|
* character.
|
||
|
*
|
||
|
* @param string $string The string to test.
|
||
|
* @param string $charset The charset to use when testing the string.
|
||
|
*
|
||
|
* @return boolean True if the parameter was alphabetic only.
|
||
|
*/
|
||
|
public static function isAlpha($string, $charset)
|
||
|
{
|
||
|
if (!Horde_Util::extensionExists('mbstring')) {
|
||
|
return ctype_alpha($string);
|
||
|
}
|
||
|
|
||
|
$charset = self::_mbstringCharset($charset);
|
||
|
$old_charset = mb_regex_encoding();
|
||
|
|
||
|
if ($charset != $old_charset) {
|
||
|
@mb_regex_encoding($charset);
|
||
|
}
|
||
|
$alpha = !@mb_ereg_match('[^[:alpha:]]', $string);
|
||
|
if ($charset != $old_charset) {
|
||
|
@mb_regex_encoding($old_charset);
|
||
|
}
|
||
|
|
||
|
return $alpha;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns true if ever character in the parameter is a lowercase letter in
|
||
|
* the current locale.
|
||
|
*
|
||
|
* @param string $string The string to test.
|
||
|
* @param string $charset The charset to use when testing the string.
|
||
|
*
|
||
|
* @return boolean True if the parameter was lowercase.
|
||
|
*/
|
||
|
public static function isLower($string, $charset)
|
||
|
{
|
||
|
return ((self::lower($string, true, $charset) === $string) &&
|
||
|
self::isAlpha($string, $charset));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns true if every character in the parameter is an uppercase letter
|
||
|
* in the current locale.
|
||
|
*
|
||
|
* @param string $string The string to test.
|
||
|
* @param string $charset The charset to use when testing the string.
|
||
|
*
|
||
|
* @return boolean True if the parameter was uppercase.
|
||
|
*/
|
||
|
public static function isUpper($string, $charset)
|
||
|
{
|
||
|
return ((self::upper($string, true, $charset) === $string) &&
|
||
|
self::isAlpha($string, $charset));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Performs a multibyte safe regex match search on the text provided.
|
||
|
*
|
||
|
* @param string $text The text to search.
|
||
|
* @param array $regex The regular expressions to use, without perl
|
||
|
* regex delimiters (e.g. '/' or '|').
|
||
|
* @param string $charset The character set of the text.
|
||
|
*
|
||
|
* @return array The matches array from the first regex that matches.
|
||
|
*/
|
||
|
public static function regexMatch($text, $regex, $charset = null)
|
||
|
{
|
||
|
if (!empty($charset)) {
|
||
|
$regex = self::convertCharset($regex, $charset, 'utf-8');
|
||
|
$text = self::convertCharset($text, $charset, 'utf-8');
|
||
|
}
|
||
|
|
||
|
$matches = array();
|
||
|
foreach ($regex as $val) {
|
||
|
if (preg_match('/' . $val . '/u', $text, $matches)) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!empty($charset)) {
|
||
|
$matches = self::convertCharset($matches, 'utf-8', $charset);
|
||
|
}
|
||
|
|
||
|
return $matches;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Check to see if a string is valid UTF-8.
|
||
|
*
|
||
|
* @param string $text The text to check.
|
||
|
*
|
||
|
* @return boolean True if valid UTF-8.
|
||
|
*/
|
||
|
public static function validUtf8($text)
|
||
|
{
|
||
|
$text = strval($text);
|
||
|
|
||
|
// First check for illegal surrogate pair sequences. See RFC 3629.
|
||
|
if (preg_match('/\xE0[\x80-\x9F][\x80-\xBF]|\xED[\xA0-\xBF][\x80-\xBF]/S', $text)) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
for ($i = 0, $len = strlen($text); $i < $len; ++$i) {
|
||
|
$c = ord($text[$i]);
|
||
|
if ($c > 128) {
|
||
|
if ($c > 247) {
|
||
|
// STD 63 (RFC 3629) eliminates 5 & 6-byte characters.
|
||
|
return false;
|
||
|
} elseif ($c > 239) {
|
||
|
$j = 3;
|
||
|
} elseif ($c > 223) {
|
||
|
$j = 2;
|
||
|
} elseif ($c > 191) {
|
||
|
$j = 1;
|
||
|
} else {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if (($i + $j) > $len) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
do {
|
||
|
$c = ord($text[++$i]);
|
||
|
if (($c < 128) || ($c > 191)) {
|
||
|
return false;
|
||
|
}
|
||
|
} while (--$j);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Workaround charsets that don't work with mbstring functions.
|
||
|
*
|
||
|
* @param string $charset The original charset.
|
||
|
*
|
||
|
* @return string The charset to use with mbstring functions.
|
||
|
*/
|
||
|
protected static function _mbstringCharset($charset)
|
||
|
{
|
||
|
/* mbstring functions do not handle the 'ks_c_5601-1987' &
|
||
|
* 'ks_c_5601-1989' charsets. However, these charsets are used, for
|
||
|
* example, by various versions of Outlook to send Korean characters.
|
||
|
* Use UHC (CP949) encoding instead. See, e.g.,
|
||
|
* http://lists.w3.org/Archives/Public/ietf-charsets/2001AprJun/0030.html */
|
||
|
return in_array(self::lower($charset), array('ks_c_5601-1987', 'ks_c_5601-1989'))
|
||
|
? 'UHC'
|
||
|
: $charset;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Strip UTF-8 byte order mark (BOM) from string data.
|
||
|
*
|
||
|
* @param string $str Input string (UTF-8).
|
||
|
*
|
||
|
* @return string Stripped string (UTF-8).
|
||
|
*/
|
||
|
public static function trimUtf8Bom($str)
|
||
|
{
|
||
|
return (substr($str, 0, 3) == pack('CCC', 239, 187, 191))
|
||
|
? substr($str, 3)
|
||
|
: $str;
|
||
|
}
|
||
|
|
||
|
}
|