<?php /** * @link http://www.yiiframework.com/ * @copyright Copyright (c) 2008 Yii Software LLC * @license http://www.yiiframework.com/license/ */ namespace yii\widgets; use yii\base\InvalidConfigException; use yii\helpers\Html; use yii\helpers\Json; use yii\web\JsExpression; use yii\web\View; /** * MaskedInput generates a masked text input. * * MaskedInput is similar to [[Html::textInput()]] except that an input mask will be used to force users to enter * properly formatted data, such as phone numbers, social security numbers. * * To use MaskedInput, you must set the [[mask]] property. The following example * shows how to use MaskedInput to collect phone numbers: * * ```php * echo MaskedInput::widget([ * 'name' => 'phone', * 'mask' => '999-999-9999', * ]); * ``` * * The masked text field is implemented based on the * [jQuery input masked plugin](https://github.com/RobinHerbots/jquery.inputmask). * * @author Kartik Visweswaran <kartikv2@gmail.com> * @since 2.0 */ class MaskedInput extends InputWidget { /** * The name of the jQuery plugin to use for this widget. */ const PLUGIN_NAME = 'inputmask'; /** * @var string|array|JsExpression the input mask (e.g. '99/99/9999' for date input). The following characters * can be used in the mask and are predefined: * * - `a`: represents an alpha character (A-Z, a-z) * - `9`: represents a numeric character (0-9) * - `*`: represents an alphanumeric character (A-Z, a-z, 0-9) * - `[` and `]`: anything entered between the square brackets is considered optional user input. This is * based on the `optionalmarker` setting in [[clientOptions]]. * * Additional definitions can be set through the [[definitions]] property. */ public $mask; /** * @var array custom mask definitions to use. Should be configured as `maskSymbol => settings`, where * * - `maskSymbol` is a string, containing a character to identify your mask definition and * - `settings` is an array, consisiting of the following entries: * - `validator`: string, a JS regular expression or a JS function. * - `cardinality`: int, specifies how many characters are represented and validated for the definition. * - `prevalidator`: array, validate the characters before the definition cardinality is reached. * - `definitionSymbol`: string, allows shifting values from other definitions, with this `definitionSymbol`. */ public $definitions; /** * @var array custom aliases to use. Should be configured as `maskAlias => settings`, where * * - `maskAlias` is a string containing a text to identify your mask alias definition (e.g. 'phone') and * - `settings` is an array containing settings for the mask symbol, exactly similar to parameters as passed in [[clientOptions]]. */ public $aliases; /** * @var array the JQuery plugin options for the input mask plugin. * @see https://github.com/RobinHerbots/jquery.inputmask */ public $clientOptions = []; /** * @var array the HTML attributes for the input tag. * @see \yii\helpers\Html::renderTagAttributes() for details on how attributes are being rendered. */ public $options = ['class' => 'form-control']; /** * @var string the hashed variable to store the pluginOptions */ protected $_hashVar; /** * Initializes the widget. * * @throws InvalidConfigException if the "mask" property is not set. */ public function init() { parent::init(); if (empty($this->mask) && empty($this->clientOptions['alias'])) { throw new InvalidConfigException("Either the 'mask' property or the 'clientOptions[\"alias\"]' property must be set."); } } /** * @inheritdoc */ public function run() { if ($this->hasModel()) { echo Html::activeTextInput($this->model, $this->attribute, $this->options); } else { echo Html::textInput($this->name, $this->value, $this->options); } $this->registerClientScript(); } /** * Generates a hashed variable to store the plugin `clientOptions`. Helps in reusing the variable for similar * options passed for other widgets on the same page. The following special data attributes will also be * setup for the input widget, that can be accessed through javascript: * * - 'data-plugin-options' will store the hashed variable storing the plugin options. * - 'data-plugin-name' the name of the plugin * * @param View $view the view instance * @author [Thiago Talma](https://github.com/thiagotalma) */ protected function hashPluginOptions($view) { $encOptions = empty($this->clientOptions) ? '{}' : Json::encode($this->clientOptions); $this->_hashVar = self::PLUGIN_NAME . '_' . hash('crc32', $encOptions); $this->options['data-plugin-name'] = self::PLUGIN_NAME; $this->options['data-plugin-options'] = $this->_hashVar; $view->registerJs("var {$this->_hashVar} = {$encOptions};\n", View::POS_HEAD); } /** * Initializes client options */ protected function initClientOptions() { $options = $this->clientOptions; foreach ($options as $key => $value) { if (in_array($key, ['oncomplete', 'onincomplete', 'oncleared', 'onKeyUp', 'onKeyDown', 'onBeforeMask', 'onBeforePaste', 'onUnMask', 'isComplete', 'determineActiveMasksetIndex']) && !$value instanceof JsExpression ) { $options[$key] = new JsExpression($value); } } $this->clientOptions = $options; } /** * Registers the needed client script and options. */ public function registerClientScript() { $js = ''; $view = $this->getView(); $this->initClientOptions(); if (!empty($this->mask)) { $this->clientOptions['mask'] = $this->mask; } $this->hashPluginOptions($view); if (is_array($this->definitions) && !empty($this->definitions)) { $js .= '$.extend($.' . self::PLUGIN_NAME . '.defaults.definitions, ' . Json::encode($this->definitions) . ");\n"; } if (is_array($this->aliases) && !empty($this->aliases)) { $js .= '$.extend($.' . self::PLUGIN_NAME . '.defaults.aliases, ' . Json::encode($this->aliases) . ");\n"; } $id = $this->options['id']; $js .= '$("#' . $id . '").' . self::PLUGIN_NAME . "(" . $this->_hashVar . ");\n"; MaskedInputAsset::register($view); $view->registerJs($js); } }