2012-11-09 23:18:50 +11:00
|
|
|
<?php defined('SYSPATH') OR die('No direct script access.');
|
2010-08-21 14:43:03 +10:00
|
|
|
/**
|
|
|
|
* Inflector helper class. Inflection is changing the form of a word based on
|
|
|
|
* the context it is used in. For example, changing a word into a plural form.
|
|
|
|
*
|
|
|
|
* [!!] Inflection is only tested with English, and is will not work with other languages.
|
|
|
|
*
|
|
|
|
* @package Kohana
|
|
|
|
* @category Helpers
|
|
|
|
* @author Kohana Team
|
2012-11-09 23:18:50 +11:00
|
|
|
* @copyright (c) 2007-2012 Kohana Team
|
2011-05-13 16:00:25 +10:00
|
|
|
* @license http://kohanaframework.org/license
|
2010-08-21 14:43:03 +10:00
|
|
|
*/
|
|
|
|
class Kohana_Inflector {
|
|
|
|
|
2011-05-13 16:00:25 +10:00
|
|
|
/**
|
|
|
|
* @var array cached inflections
|
|
|
|
*/
|
2010-08-21 14:43:03 +10:00
|
|
|
protected static $cache = array();
|
|
|
|
|
2011-05-13 16:00:25 +10:00
|
|
|
/**
|
|
|
|
* @var array uncountable words
|
|
|
|
*/
|
2010-08-21 14:43:03 +10:00
|
|
|
protected static $uncountable;
|
2011-05-13 16:00:25 +10:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array irregular words
|
|
|
|
*/
|
2010-08-21 14:43:03 +10:00
|
|
|
protected static $irregular;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if a word is defined as uncountable. An uncountable word has a
|
|
|
|
* single form. For instance, one "fish" and many "fish", not "fishes".
|
|
|
|
*
|
|
|
|
* Inflector::uncountable('fish'); // TRUE
|
|
|
|
* Inflector::uncountable('cat'); // FALSE
|
|
|
|
*
|
|
|
|
* If you find a word is being pluralized improperly, it has probably not
|
|
|
|
* been defined as uncountable in `config/inflector.php`. If this is the
|
|
|
|
* case, please report [an issue](http://dev.kohanaphp.com/projects/kohana3/issues).
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $str word to check
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public static function uncountable($str)
|
|
|
|
{
|
|
|
|
if (Inflector::$uncountable === NULL)
|
|
|
|
{
|
|
|
|
// Cache uncountables
|
2012-11-09 23:18:50 +11:00
|
|
|
Inflector::$uncountable = Kohana::$config->load('inflector')->uncountable;
|
2010-08-21 14:43:03 +10:00
|
|
|
|
|
|
|
// Make uncountables mirrored
|
|
|
|
Inflector::$uncountable = array_combine(Inflector::$uncountable, Inflector::$uncountable);
|
|
|
|
}
|
|
|
|
|
|
|
|
return isset(Inflector::$uncountable[strtolower($str)]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Makes a plural word singular.
|
|
|
|
*
|
|
|
|
* echo Inflector::singular('cats'); // "cat"
|
|
|
|
* echo Inflector::singular('fish'); // "fish", uncountable
|
|
|
|
*
|
|
|
|
* You can also provide the count to make inflection more intelligent.
|
|
|
|
* In this case, it will only return the singular value if the count is
|
|
|
|
* greater than one and not zero.
|
|
|
|
*
|
|
|
|
* echo Inflector::singular('cats', 2); // "cats"
|
|
|
|
*
|
|
|
|
* [!!] Special inflections are defined in `config/inflector.php`.
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $str word to singularize
|
|
|
|
* @param integer $count count of thing
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return string
|
|
|
|
* @uses Inflector::uncountable
|
|
|
|
*/
|
|
|
|
public static function singular($str, $count = NULL)
|
|
|
|
{
|
|
|
|
// $count should always be a float
|
|
|
|
$count = ($count === NULL) ? 1.0 : (float) $count;
|
|
|
|
|
|
|
|
// Do nothing when $count is not 1
|
|
|
|
if ($count != 1)
|
|
|
|
return $str;
|
|
|
|
|
|
|
|
// Remove garbage
|
|
|
|
$str = strtolower(trim($str));
|
|
|
|
|
|
|
|
// Cache key name
|
|
|
|
$key = 'singular_'.$str.$count;
|
|
|
|
|
|
|
|
if (isset(Inflector::$cache[$key]))
|
|
|
|
return Inflector::$cache[$key];
|
|
|
|
|
|
|
|
if (Inflector::uncountable($str))
|
|
|
|
return Inflector::$cache[$key] = $str;
|
|
|
|
|
|
|
|
if (empty(Inflector::$irregular))
|
|
|
|
{
|
|
|
|
// Cache irregular words
|
2012-11-09 23:18:50 +11:00
|
|
|
Inflector::$irregular = Kohana::$config->load('inflector')->irregular;
|
2010-08-21 14:43:03 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($irregular = array_search($str, Inflector::$irregular))
|
|
|
|
{
|
|
|
|
$str = $irregular;
|
|
|
|
}
|
|
|
|
elseif (preg_match('/us$/', $str))
|
|
|
|
{
|
|
|
|
// http://en.wikipedia.org/wiki/Plural_form_of_words_ending_in_-us
|
|
|
|
// Already singular, do nothing
|
|
|
|
}
|
|
|
|
elseif (preg_match('/[sxz]es$/', $str) OR preg_match('/[^aeioudgkprt]hes$/', $str))
|
|
|
|
{
|
|
|
|
// Remove "es"
|
|
|
|
$str = substr($str, 0, -2);
|
|
|
|
}
|
|
|
|
elseif (preg_match('/[^aeiou]ies$/', $str))
|
|
|
|
{
|
|
|
|
// Replace "ies" with "y"
|
|
|
|
$str = substr($str, 0, -3).'y';
|
|
|
|
}
|
|
|
|
elseif (substr($str, -1) === 's' AND substr($str, -2) !== 'ss')
|
|
|
|
{
|
|
|
|
// Remove singular "s"
|
|
|
|
$str = substr($str, 0, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Inflector::$cache[$key] = $str;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Makes a singular word plural.
|
|
|
|
*
|
|
|
|
* echo Inflector::plural('fish'); // "fish", uncountable
|
|
|
|
* echo Inflector::plural('cat'); // "cats"
|
|
|
|
*
|
|
|
|
* You can also provide the count to make inflection more intelligent.
|
|
|
|
* In this case, it will only return the plural value if the count is
|
|
|
|
* not one.
|
|
|
|
*
|
|
|
|
* echo Inflector::singular('cats', 3); // "cats"
|
|
|
|
*
|
|
|
|
* [!!] Special inflections are defined in `config/inflector.php`.
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $str word to pluralize
|
|
|
|
* @param integer $count count of thing
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return string
|
|
|
|
* @uses Inflector::uncountable
|
|
|
|
*/
|
|
|
|
public static function plural($str, $count = NULL)
|
|
|
|
{
|
|
|
|
// $count should always be a float
|
|
|
|
$count = ($count === NULL) ? 0.0 : (float) $count;
|
|
|
|
|
|
|
|
// Do nothing with singular
|
|
|
|
if ($count == 1)
|
|
|
|
return $str;
|
|
|
|
|
|
|
|
// Remove garbage
|
2011-05-13 16:00:25 +10:00
|
|
|
$str = trim($str);
|
2010-08-21 14:43:03 +10:00
|
|
|
|
|
|
|
// Cache key name
|
|
|
|
$key = 'plural_'.$str.$count;
|
|
|
|
|
2011-05-13 16:00:25 +10:00
|
|
|
// Check uppercase
|
|
|
|
$is_uppercase = ctype_upper($str);
|
|
|
|
|
2010-08-21 14:43:03 +10:00
|
|
|
if (isset(Inflector::$cache[$key]))
|
|
|
|
return Inflector::$cache[$key];
|
|
|
|
|
|
|
|
if (Inflector::uncountable($str))
|
|
|
|
return Inflector::$cache[$key] = $str;
|
|
|
|
|
|
|
|
if (empty(Inflector::$irregular))
|
|
|
|
{
|
|
|
|
// Cache irregular words
|
2012-11-09 23:18:50 +11:00
|
|
|
Inflector::$irregular = Kohana::$config->load('inflector')->irregular;
|
2010-08-21 14:43:03 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
if (isset(Inflector::$irregular[$str]))
|
|
|
|
{
|
|
|
|
$str = Inflector::$irregular[$str];
|
|
|
|
}
|
|
|
|
elseif (preg_match('/[sxz]$/', $str) OR preg_match('/[^aeioudgkprt]h$/', $str))
|
|
|
|
{
|
|
|
|
$str .= 'es';
|
|
|
|
}
|
|
|
|
elseif (preg_match('/[^aeiou]y$/', $str))
|
|
|
|
{
|
|
|
|
// Change "y" to "ies"
|
|
|
|
$str = substr_replace($str, 'ies', -1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$str .= 's';
|
|
|
|
}
|
|
|
|
|
2011-05-13 16:00:25 +10:00
|
|
|
// Convert to uppsecase if nessasary
|
|
|
|
if ($is_uppercase)
|
|
|
|
{
|
|
|
|
$str = strtoupper($str);
|
|
|
|
}
|
|
|
|
|
2010-08-21 14:43:03 +10:00
|
|
|
// Set the cache and return
|
|
|
|
return Inflector::$cache[$key] = $str;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Makes a phrase camel case. Spaces and underscores will be removed.
|
|
|
|
*
|
|
|
|
* $str = Inflector::camelize('mother cat'); // "motherCat"
|
|
|
|
* $str = Inflector::camelize('kittens in bed'); // "kittensInBed"
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $str phrase to camelize
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function camelize($str)
|
|
|
|
{
|
|
|
|
$str = 'x'.strtolower(trim($str));
|
|
|
|
$str = ucwords(preg_replace('/[\s_]+/', ' ', $str));
|
|
|
|
|
|
|
|
return substr(str_replace(' ', '', $str), 1);
|
|
|
|
}
|
|
|
|
|
2011-05-13 16:00:25 +10:00
|
|
|
/**
|
|
|
|
* Converts a camel case phrase into a spaced phrase.
|
|
|
|
*
|
|
|
|
* $str = Inflector::decamelize('houseCat'); // "house cat"
|
|
|
|
* $str = Inflector::decamelize('kingAllyCat'); // "king ally cat"
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $str phrase to camelize
|
|
|
|
* @param string $sep word separator
|
2011-05-13 16:00:25 +10:00
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function decamelize($str, $sep = ' ')
|
|
|
|
{
|
|
|
|
return strtolower(preg_replace('/([a-z])([A-Z])/', '$1'.$sep.'$2', trim($str)));
|
|
|
|
}
|
|
|
|
|
2010-08-21 14:43:03 +10:00
|
|
|
/**
|
|
|
|
* Makes a phrase underscored instead of spaced.
|
|
|
|
*
|
|
|
|
* $str = Inflector::underscore('five cats'); // "five_cats";
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $str phrase to underscore
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function underscore($str)
|
|
|
|
{
|
|
|
|
return preg_replace('/\s+/', '_', trim($str));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Makes an underscored or dashed phrase human-readable.
|
|
|
|
*
|
|
|
|
* $str = Inflector::humanize('kittens-are-cats'); // "kittens are cats"
|
|
|
|
* $str = Inflector::humanize('dogs_as_well'); // "dogs as well"
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $str phrase to make human-readable
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function humanize($str)
|
|
|
|
{
|
|
|
|
return preg_replace('/[_-]+/', ' ', trim($str));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // End Inflector
|