<?php /** * @copyright Copyright © Kartik Visweswaran, Krajee.com, 2014 * @package yii2-widgets * @subpackage yii2-widget-growl * @version 1.1.1 */ namespace kartik\growl; use Yii; use yii\helpers\Html; use yii\helpers\Json; use yii\helpers\ArrayHelper; use kartik\base\AnimateAsset; /** * Widget that wraps the Bootstrap Growl plugin by remabledesigns. * * @http://bootstrap-growl.remabledesigns.com/ * * @author Kartik Visweswaran <kartikv2@gmail.com> * @since 1.0 */ class Growl extends \kartik\base\Widget { const TYPE_INFO = 'info'; const TYPE_DANGER = 'danger'; const TYPE_SUCCESS = 'success'; const TYPE_WARNING = 'warning'; const TYPE_GROWL = 'growl'; const TYPE_MINIMALIST = 'minimalist'; const TYPE_PASTEL = 'pastel'; const TYPE_CUSTOM = 'custom'; /** * @var string the type of the alert to be displayed. One of the `TYPE_` constants. * Defaults to `TYPE_INFO` */ public $type = self::TYPE_INFO; /** * @var string the class name for the icon */ public $icon = ''; /** * @var string the title for the alert */ public $title = ''; /** * @var string the url to redirect to on clicking the alert. If this is <code>null</code> or not set, * the alert will not be clickable. */ public $linkUrl = ''; /** * @var string the target to open the linked notification */ public $linkTarget = '_blank'; /** * @var bool show title separator. Only applicable if `title` is set. */ public $showSeparator = false; /** * @var string the alert message body */ public $body = ''; /** * @var array the HTML options and settings for the bootstrap progress bar. Defaults to: * ``` * [ * 'role' => 'progressbar', * 'aria-valuenow' => '0', * 'aria-valuemin' => '0', * 'aria-valuemax' => '100', * 'style' => '100', * ] * ``` * The following special options are recognized: * - `title`: the progress bar title text/markup. */ public $progressBarOptions = []; /** * @var integer the delay in microseconds after which the alert will be displayed. * Will be useful when multiple alerts are to be shown. */ public $delay; /** * @var array the options for rendering the close button tag. */ public $closeButton = []; /** * @var use animations */ public $useAnimation = true; /** * @var array the HTML attributes for the growl icon container. */ public $iconOptions = []; /** * @var array the HTML attributes for the growl title container. */ public $titleOptions = []; /** * @var array the HTML attributes for the growl message body. */ public $bodyOptions = []; /** * @var array the HTML attributes for the growl progress bar container. */ public $progressContainerOptions = []; /** * @var array the HTML attributes for the growl url link */ public $linkOptions = []; /** * @var array the bootstrap growl plugin configuration options * @see http://bootstrap-growl.remabledesigns.com/ */ public $pluginOptions = []; /** * @var array the list of themes. */ protected static $_themes = [ self::TYPE_GROWL, self::TYPE_MINIMALIST, self::TYPE_PASTEL ]; /** * @var array the first part of growl plugin settings/options */ private $_settings; /** * Initializes the widget */ public function init() { parent::init(); $this->initOptions(); } /** * Initializes the widget options. * This method sets the default values for various options. */ protected function initOptions() { if (empty($this->options['id'])) { $this->options['id'] = $this->getId(); } $this->_settings = [ 'message' => $this->body, 'icon' => $this->icon, 'title' => $this->title, 'url' => $this->linkUrl, 'target' => $this->linkTarget ]; $this->progressBarOptions += [ 'role' => 'progressbar', 'aria-valuenow' => '0', 'aria-valuemin' => '0', 'aria-valuemax' => '100', 'style' => 'width:100%', ]; $this->pluginOptions['type'] = $this->type; $class = 'progress'; $progressTitle = ArrayHelper::remove($this->progressBarOptions, 'title', ''); if (empty($this->progressContainerOptions['class'])) { $class .= ' kv-progress-bar'; } Html::addCssClass($this->progressContainerOptions, $class); Html::addCssClass($this->progressBarOptions, 'progress-bar progress-bar-{0}'); $class = "alert alert-{0}"; if (empty($this->options['class'])) { $this->options['class'] = "col-xs-11 col-sm-3 {$class}"; } else { Html::addCssClass($this->options, $class); } $divider = !empty($this->showSeparator) && !empty($this->title) ? '<hr class="kv-alert-separator">' . "\n" : ''; $this->options['role'] = 'alert'; $this->options['data-notify'] = 'container'; $this->iconOptions['data-notify'] = 'icon'; $this->titleOptions['data-notify'] = 'title'; $this->bodyOptions['data-notify'] = 'message'; $this->progressContainerOptions['data-notify'] = 'progressbar'; $this->linkOptions['data-notify'] = 'url'; $this->linkOptions['target'] = '{4}'; $iconTag = ArrayHelper::getValue($this->pluginOptions, 'icon_type', 'class') === 'class' ? 'span' : 'img'; $content = $this->renderCloseButton() . "\n" . Html::tag($iconTag, '', $this->iconOptions) . "\n" . Html::tag('span', '{1}', $this->titleOptions) . "\n" . $divider . Html::tag('span', '{2}', $this->bodyOptions) . "\n" . Html::tag('div', Html::tag('div', $progressTitle, $this->progressBarOptions), $this->progressContainerOptions) . "\n" . Html::a('', '{3}', $this->linkOptions); $this->pluginOptions['template'] = Html::tag('div', $content, $this->options); $this->registerAssets(); } /** * Renders the close button. * * @return string the rendering result */ protected function renderCloseButton() { if ($this->closeButton !== null) { $tag = ArrayHelper::remove($this->closeButton, 'tag', 'button'); $label = ArrayHelper::remove($this->closeButton, 'label', '×'); $label = '<span aria-hidden="true">' . $label . '</span>'; Html::addCssClass($this->closeButton, 'close'); if ($tag === 'button' && !isset($this->closeButton['type'])) { $this->closeButton['type'] = 'button'; } $this->closeButton['data-notify'] = 'dismiss'; return Html::tag($tag, $label, $this->closeButton); } else { return ''; } } /** * Register client assets */ protected function registerAssets() { $view = $this->getView(); if (in_array($this->type, self::$_themes)) { GrowlAsset::register($view)->addTheme($this->type); } else { GrowlAsset::register($view); } if ($this->useAnimation) { AnimateAsset::register($view); } $this->registerPluginOptions('notify'); $js = '$.notify(' . Json::encode($this->_settings) . ', ' . $this->_hashVar . ');'; if (!empty($this->delay) && $this->delay > 0) { $js = 'setTimeout(function () {' . $js . '}, ' . $this->delay . ');'; } $view->registerJs($js); } }