diff --git a/bootstrap.php b/bootstrap.php index fe3dd2155b8fdd0e2dbb21d249d5eeba247fadc0..c00e425c35f8278b8f27d7fc1f9edfe01848e071 100644 --- a/bootstrap.php +++ b/bootstrap.php @@ -133,6 +133,7 @@ class Bootstrap { define('SEQUENCE_TABLE', $prefix.'sequence'); define('TRANSLATION_TABLE', $prefix.'translation'); define('QUEUE_TABLE', $prefix.'queue'); + define('QUEUE_COLUMN_TABLE', $prefix.'queue_column'); define('API_KEY_TABLE',$prefix.'api_key'); define('TIMEZONE_TABLE',$prefix.'timezone'); diff --git a/include/ajax.search.php b/include/ajax.search.php index f2175a76dc4b961d7c260b823a8675f4e40fc424..40cd7311050d761e2ad71d76cadbc295db290599 100644 --- a/include/ajax.search.php +++ b/include/ajax.search.php @@ -19,6 +19,7 @@ if(!defined('INCLUDE_DIR')) die('403'); include_once(INCLUDE_DIR.'class.ticket.php'); require_once(INCLUDE_DIR.'class.ajax.php'); +require_once(INCLUDE_DIR.'class.queue.php'); class SearchAjaxAPI extends AjaxController { @@ -30,7 +31,7 @@ class SearchAjaxAPI extends AjaxController { $search = SavedSearch::create(); $form = $search->getFormFromSession('advsearch') ?: $search->getForm(); - $matches = self::_getSupportedTicketMatches(); + $matches = SavedSearch::getSupportedTicketMatches(); include STAFFINC_DIR . 'templates/advanced-search.tmpl.php'; } @@ -96,7 +97,7 @@ class SearchAjaxAPI extends AjaxController { $form = $search->getForm($_POST); if (!$form->isValid()) { - $matches = self::_getSupportedTicketMatches(); + $matches = SavedSearch::getSupportedTicketMatches(); include STAFFINC_DIR . 'templates/advanced-search.tmpl.php'; return; } @@ -148,41 +149,6 @@ class SearchAjaxAPI extends AjaxController { ))); } - function _getSupportedTicketMatches() { - // User information - $matches = array( - __('Ticket Built-In') => SavedSearch::getExtendedTicketFields(), - __('Custom Forms') => array() - ); - foreach (array('ticket'=>'TicketForm', 'user'=>'UserForm', 'organization'=>'OrganizationForm') as $k=>$F) { - $form = $F::objects()->one(); - $fields = &$matches[$form->getLocal('title')]; - foreach ($form->getFields() as $f) { - if (!$f->hasData() || $f->isPresentationOnly()) - continue; - $fields[":$k!".$f->get('id')] = __(ucfirst($k)).' / '.$f->getLocal('label'); - /* TODO: Support matches on list item properties - if (($fi = $f->getImpl()) && $fi->hasSubFields()) { - foreach ($fi->getSubFields() as $p) { - $fields[":$k.".$f->get('id').'.'.$p->get('id')] - = __(ucfirst($k)).' / '.$f->getLocal('label').' / '.$p->getLocal('label'); - } - } - */ - } - } - $fields = &$matches[__('Custom Forms')]; - foreach (DynamicForm::objects()->filter(array('type'=>'G')) as $form) { - foreach ($form->getFields() as $f) { - if (!$f->hasData() || $f->isPresentationOnly()) - continue; - $key = sprintf(':field!%d', $f->get('id'), $f->get('id')); - $fields[$key] = $form->getLocal('title').' / '.$f->getLocal('label'); - } - } - return $matches; - } - function createSearch() { global $thisstaff; @@ -208,7 +174,7 @@ class SearchAjaxAPI extends AjaxController { $form = $search->loadFromState($state); $form->loadState($state); } - $matches = self::_getSupportedTicketMatches(); + $matches = SavedSearch::getSupportedTicketMatches(); include STAFFINC_DIR . 'templates/advanced-search.tmpl.php'; } @@ -231,4 +197,120 @@ class SearchAjaxAPI extends AjaxController { 'success' => true, ))); } + + + function editColumn($queue_id, $column) { + global $thisstaff; + + if (!$thisstaff) { + Http::response(403, 'Agent login is required'); + } + elseif (!($queue = CustomQueue::lookup($queue_id))) { + Http::response(404, 'No such queue'); + } + + $data_form = new QueueDataConfigForm($_POST); + include STAFFINC_DIR . 'templates/queue-column.tmpl.php'; + } + + function previewQueue($id=false) { + global $thisstaff; + + if (!$thisstaff) { + Http::response(403, 'Agent login is required'); + } + if ($id && (!($queue = CustomQueue::lookup($id)))) { + Http::response(404, 'No such queue'); + } + + if (!$queue) { + $queue = CustomQueue::create(); + } + + $form = $queue->getForm($_POST); + + // TODO: Update queue columns (but without save) + foreach ($_POST['columns'] as $colid) { + $col = QueueColumn::create(array("id" => $colid)); + $col->update($_POST); + $queue->addColumn($col); + } + + $tickets = $queue->getQuery($form); + $count = 10; // count($queue->getBasicQuery($form)); + + include STAFFINC_DIR . 'templates/queue-tickets.tmpl.php'; + } + + function addCondition() { + global $thisstaff; + + if (!$thisstaff) { + Http::response(403, 'Agent login is required'); + } + elseif (!isset($_GET['field'])) { + Http::response(400, '`field` parameter is required'); + } + $fields = SavedSearch::getSearchableFields('Ticket'); + if (!isset($fields[$_GET['field']])) { + Http::response(400, sprintf('%s: No such searchable field'), + Format::htmlchars($_GET['field'])); + } + + $field = $fields[$_GET['field']]; + $condition = new QueueColumnCondition(); + include STAFFINC_DIR . 'templates/queue-column-condition.tmpl.php'; + } + + function addConditionProperty() { + global $thisstaff; + + if (!$thisstaff) { + Http::response(403, 'Agent login is required'); + } + elseif (!isset($_GET['prop'])) { + Http::response(400, '`prop` parameter is required'); + } + + $prop = $_GET['prop']; + include STAFFINC_DIR . 'templates/queue-column-condition-prop.tmpl.php'; + } + + function addColumn() { + global $thisstaff; + + if (!$thisstaff) { + Http::response(403, 'Agent login is required'); + } + elseif (!isset($_GET['field'])) { + Http::response(400, '`field` parameter is required'); + } + + $field = $_GET['field']; + // XXX: This method should receive a queue ID or queue root so that + // $field can be properly checked + $fields = SavedSearch::getSearchableFields('Ticket'); + if (!isset($fields[$field])) { + Http::response(400, 'Not a supported field for this queue'); + } + + // Get the tabbed column configuration + $F = $fields[$field]; + $column = QueueColumn::create(array( + "id" => (int) $_GET['id'], + "heading" => _S($F->getLabel()), + "primary" => $field, + "width" => 100, + )); + ob_start(); + include STAFFINC_DIR . 'templates/queue-column.tmpl.php'; + $config = ob_get_clean(); + + // Send back the goodies + Http::response(200, $this->encode(array( + 'config' => $config, + 'heading' => _S($F->getLabel()), + 'width' => $column->getWidth(), + )), 'application/json'); + } } diff --git a/include/class.dept.php b/include/class.dept.php index 8f09237c4b37fc721d4a9a25367984414e9e9afd..b54913af42ad14783b1b852fc55a9efaa7dab532 100644 --- a/include/class.dept.php +++ b/include/class.dept.php @@ -13,9 +13,10 @@ vim: expandtab sw=4 ts=4 sts=4: **********************************************************************/ +require_once INCLUDE_DIR . 'class.search.php'; class Dept extends VerySimpleModel -implements TemplateVariable { +implements TemplateVariable, Searchable { static $meta = array( 'table' => DEPT_TABLE, @@ -98,6 +99,17 @@ implements TemplateVariable { } } + static function getSearchableFields() { + return array( + 'name' => new TextboxField(array( + 'label' => __('Name'), + )), + 'manager' => new AgentSelectionField(array( + 'label' => __('Manager'), + )), + ); + } + function getId() { return $this->id; } diff --git a/include/class.forms.php b/include/class.forms.php index c097dd995ab0f8e48336215bb6acf578e0dfb440..9171b5f90978ab2e4cd4d639ad2d08dffcba564e 100644 --- a/include/class.forms.php +++ b/include/class.forms.php @@ -47,7 +47,10 @@ class Form { } function getId() { - return static::$id; + return @$this->id ?: static::$id; + } + function setId($id) { + $this->id = $id; } function data($source) { @@ -998,6 +1001,19 @@ class FormField { return sprintf($desc, $name, $value); } + function addToQuery($query, $name=false) { + return $query->values($name ?: $this->get('name')); + } + + /** + * Similary to to_php() and parse(), except a row from a queryset is + * passed. The value returned should be what would be retured from + * parse() or to_php() + */ + function from_query($row, $name=false) { + return $row[$name ?: $this->get('name')]; + } + function getLabel() { return $this->get('label'); } /** @@ -1042,12 +1058,25 @@ class FormField { $this->getWidget()->value = $value; } + /** + * Fetch a pseudo-random id for this form field. It is used when + * rendering the widget in the @name attribute emitted in the resulting + * HTML. The form element is based on the form id, field id and name, + * and the current user's session id. Therefor, the same form fields + * will yield differing names for different users. This is used to ward + * off bot attacks as it makes it very difficult to predict and + * correlate the form names to the data they represent. + */ function getFormName() { - if (is_numeric($this->get('id'))) + $default = $this->get('name') ?: $this->get('id'); + if ($this->_form && is_numeric($fid = $this->_form->getId())) + return substr(md5( + session_id() . '-form-field-id-' . $fid . $default), -14); + elseif (is_numeric($this->get('id'))) return substr(md5( session_id() . '-field-id-'.$this->get('id')), -16); - else - return $this->get('name') ?: $this->get('id'); + + return $default; } function setForm($form) { @@ -1173,17 +1202,6 @@ class FormField { return null; } - /** - * Indicates if the field provides for searching for something other - * than keywords. For instance, textbox fields can have hits by keyword - * searches alone, but selection fields should provide the option to - * match a specific value or set of values and therefore need to - * participate on any search builder. - */ - function hasSpecialSearch() { - return true; - } - function getConfigurationForm($source=null) { if (!$this->_cform) { $type = static::getFieldType($this->get('type')); @@ -1292,10 +1310,6 @@ class TextboxField extends FormField { ); } - function hasSpecialSearch() { - return false; - } - function validateEntry($value) { parent::validateEntry($value); $config = $this->getConfiguration(); @@ -1379,10 +1393,6 @@ class TextareaField extends FormField { ); } - function hasSpecialSearch() { - return false; - } - function display($value) { $config = $this->getConfiguration(); if ($config['html']) @@ -1434,10 +1444,6 @@ class PhoneField extends FormField { ); } - function hasSpecialSearch() { - return false; - } - function validateEntry($value) { parent::validateEntry($value); $config = $this->getConfiguration(); @@ -1847,6 +1853,10 @@ class DatetimeField extends FormField { $value, $datetime->format('T')); } + function from_query($row, $name=false) { + return strtotime(parent::from_query($row, $name)); + } + function format($timestamp, $timezone=false) { if (!$timestamp || $timestamp <= 0) @@ -2133,10 +2143,6 @@ class ThreadEntryField extends FormField { function isPresentationOnly() { return true; } - function hasSpecialSearch() { - return false; - } - function getMedia() { $config = $this->getConfiguration(); $media = parent::getMedia() ?: array(); @@ -2753,10 +2759,6 @@ class FileUploadField extends FormField { ); } - function hasSpecialSearch() { - return false; - } - /** * Called from the ajax handler for async uploads via web clients. */ @@ -3037,6 +3039,10 @@ class FileFieldAttachments { } } +class ColorChoiceField extends FormField { + static $widget = 'ColorPickerWidget'; +} + class InlineFormData extends ArrayObject { var $_form; @@ -4094,6 +4100,27 @@ class FreeTextWidget extends Widget { } } +class ColorPickerWidget extends Widget { + static $media = array( + 'css' => array( + 'css/spectrum.css', + ), + 'js' => array( + 'js/spectrum.js', + ), + ); + + function render($options=array()) { + ?><input type="color" + id="<?php echo $this->id; ?>" + <?php echo implode(' ', array_filter(array( + $classes + ))); ?> + name="<?php echo $this->name; ?>" + value="<?php echo Format::htmlchars($this->value); ?>"/><?php + } +} + class VisibilityConstraint { static $operators = array( 'eq' => 1, diff --git a/include/class.list.php b/include/class.list.php index 69106502be674220c0f2c36c2293932554245471..20bc6b87b2ce4269cfd26550e41744e13ab482b7 100644 --- a/include/class.list.php +++ b/include/class.list.php @@ -1094,7 +1094,7 @@ CustomListHandler::register('ticket-status', 'TicketStatusList'); class TicketStatus extends VerySimpleModel -implements CustomListItem, TemplateVariable { +implements CustomListItem, TemplateVariable, Searchable { static $meta = array( 'table' => TICKET_STATUS_TABLE, @@ -1273,6 +1273,18 @@ implements CustomListItem, TemplateVariable { return $base; } + // Searchable interface + static function getSearchableFields() { + return array( + 'state' => new TicketStateChoiceField(array( + 'label' => __('State'), + )), + 'name' => new TicketStatusChoiceField(array( + 'label' => __('Status Name'), + )), + ); + } + function getList() { if (!isset($this->_list)) $this->_list = DynamicList::lookup(array('type' => 'ticket-status')); diff --git a/include/class.organization.php b/include/class.organization.php index f41e04225c5ce1bf017b647644c7e73d99f3a968..db49618cb95d13701fa65e5a6ca95c394d27de07 100644 --- a/include/class.organization.php +++ b/include/class.organization.php @@ -16,6 +16,7 @@ require_once(INCLUDE_DIR . 'class.orm.php'); require_once(INCLUDE_DIR . 'class.forms.php'); require_once(INCLUDE_DIR . 'class.dynamic_forms.php'); require_once(INCLUDE_DIR . 'class.user.php'); +require_once INCLUDE_DIR . 'class.search.php'; class OrganizationModel extends VerySimpleModel { static $meta = array( @@ -160,7 +161,7 @@ class OrganizationCdata extends VerySimpleModel { } class Organization extends OrganizationModel -implements TemplateVariable { +implements TemplateVariable, Searchable { var $_entries; var $_forms; @@ -341,6 +342,20 @@ implements TemplateVariable { return $base + $extra; } + static function getSearchableFields() { + $uform = OrganizationForm::objects()->one(); + foreach ($uform->getFields() as $F) { + $fname = $F->get('name') ?: ('field_'.$F->get('id')); + if (!$F->hasData() || $F->isPresentationOnly()) + continue; + if (!$F->isStorable()) + $base[$fname] = $F; + else + $base["cdata__{$fname}"] = $F; + } + return $base; + } + function update($vars, &$errors) { $valid = true; @@ -446,6 +461,15 @@ implements TemplateVariable { return true; } + static function getLink($id) { + global $thisstaff; + + if (!$id || !$thisstaff) + return false; + + return ROOT_PATH . sprintf('orgs.php?id=%s', $id); + } + static function fromVars($vars) { $vars['name'] = Format::striptags($vars['name']); diff --git a/include/class.orm.php b/include/class.orm.php index 7539c144564aba7d5c5922632fe341dc4e9abd9e..3a51e8cdccffb525c949422f186c49b39b45baae 100644 --- a/include/class.orm.php +++ b/include/class.orm.php @@ -488,6 +488,27 @@ class VerySimpleModel { return ($key) ? $M->offsetGet($key) : $M; } + static function getOrmFields($recurse=false) { + $fks = $lfields = $fields = array(); + $myname = get_called_class(); + foreach (static::getMeta('joins') as $name=>$j) { + $fks[$j['local']] = true; + if (!$j['reverse'] && !$j['list'] && $recurse) { + foreach ($j['fkey'][0]::getOrmFields($recurse - 1) as $name2=>$f) { + $fields["{$name}__{$name2}"] = "{$name} / $f"; + } + } + } + foreach (static::getMeta('fields') as $f) { + if (isset($fks[$f])) + continue; + if (in_array($f, static::getMeta('pk'))) + continue; + $lfields[$f] = "{$f}"; + } + return $lfields + $fields; + } + /** * objects * @@ -794,17 +815,23 @@ class SqlCase extends SqlFunction { class SqlExpr extends SqlFunction { function __construct($args) { - $this->args = $args; + $this->args = (array) $args; } function toSql($compiler, $model=false, $alias=false) { $O = array(); foreach ($this->args as $field=>$value) { - list($field, $op) = $compiler->getField($field, $model); - if (is_callable($op)) - $O[] = call_user_func($op, $field, $value, $model); - else - $O[] = sprintf($op, $field, $compiler->input($value)); + if ($value instanceof Q) { + $ex = $compiler->compileQ($value); + $O[] = $ex->text; + } + else { + list($field, $op) = $compiler->getField($field, $model); + if (is_callable($op)) + $O[] = call_user_func($op, $field, $value, $model); + else + $O[] = sprintf($op, $field, $compiler->input($value)); + } } return implode(' ', $O) . ($alias ? ' AS ' . $alias : ''); } @@ -1951,10 +1978,10 @@ extends ModelResultSet { return true; } - // QuerySet delegates function count() { - return $this->objects()->count(); + return count($this->asArray()); } + // QuerySet delegates function exists() { return $this->queryset->exists(); } @@ -1977,7 +2004,8 @@ extends ModelResultSet { } function offsetSet($a, $b) { $this->fillTo($a); - $this->cache[$a]->delete(); + if ($obj = $this->cache[$a]) + $obj->delete(); $this->add($b, $a); } diff --git a/include/class.queue.php b/include/class.queue.php new file mode 100644 index 0000000000000000000000000000000000000000..38bf53236bd82d4159345fd449827f8aa449f53e --- /dev/null +++ b/include/class.queue.php @@ -0,0 +1,688 @@ +<?php +/********************************************************************* + class.queue.php + + Custom (ticket) queues for osTicket + + Jared Hancock <jared@osticket.com> + Peter Rotich <peter@osticket.com> + Copyright (c) 2006-2013 osTicket + http://www.osticket.com + + Released under the GNU General Public License WITHOUT ANY WARRANTY. + See LICENSE.TXT for details. + + vim: expandtab sw=4 ts=4 sts=4: +**********************************************************************/ +require_once INCLUDE_DIR . 'class.search.php'; + +class CustomQueue extends SavedSearch { + static $meta = array( + 'joins' => array( + 'columns' => array( + 'reverse' => 'QueueColumn.queue', + ), + ), + ); + + static function objects() { + return parent::objects()->filter(array( + 'flags__hasbit' => static::FLAG_QUEUE + )); + } + + static function getDecorations($root) { + // Ticket decorations + return array( + 'TicketThreadCount', + 'ThreadAttachmentCount', + 'OverdueFlagDecoration', + 'TicketSourceDecoration' + ); + } + + function getColumns() { + if (!count($this->columns)) { + foreach (array( + new QueueColumn(array( + "id" => 1, + "heading" => "Number", + "primary" => 'number', + "width" => 100, + )), + new QueueColumn(array( + "id" => 2, + "heading" => "Created", + "primary" => 'created', + "width" => 100, + )), + new QueueColumn(array( + "id" => 3, + "heading" => "Subject", + "primary" => 'cdata__subject', + "width" => 250, + )), + new QueueColumn(array( + "id" => 4, + "heading" => "From", + "primary" => 'user__name', + "width" => 150, + )), + new QueueColumn(array( + "id" => 5, + "heading" => "Priority", + "primary" => 'cdata__priority', + "width" => 120, + )), + new QueueColumn(array( + "id" => 6, + "heading" => "Assignee", + "primary" => 'assignee', + "secondary" => 'team__name', + "width" => 100, + )), + ) as $c) { + $this->addColumn($c); + } + } + return $this->columns; + } + + function addColumn(QueueColumn $col) { + $this->columns->add($col); + $col->queue = $this; + } + + function getRoot() { + return 'Ticket'; + } + + function getBasicQuery($form=false) { + $root = $this->getRoot(); + $query = $root::objects(); + return $this->mangleQuerySet($query, $form); + } + + /** + * Retrieve a QuerySet instance based on the type of object (root) of + * this Q, which is automatically configured with the data and criteria + * of the queue and its columns. + * + * Returns: + * <QuerySet> instance + */ + function getQuery($form=false) { + $query = $this->getBasicQuery($form); + foreach ($this->getColumns() as $C) { + $query = $C->mangleQuery($query); + } + return $query; + } +} + +abstract class QueueDecoration { + static $icon = false; + static $desc = ''; + + var $config; + + function __construct($config) { + $this->config = $config; + } + + static function fromJson($config) { + $class = $config['c']; + if (class_exists($class)) + return new $class($config); + } + + static function getDescription() { + return __(static::$desc); + } + static function getIcon() { + return static::$icon; + } + static function getPositions() { + return array( + "<" => __('Start'), + "b" => __('Before'), + "a" => __('After'), + ">" => __('End'), + ); + } + + function decorate($text, $dec) { + static $positions = array( + '<' => '<span class="pull-left">%2$s</span>%1$s', + '>' => '<span class="pull-right">%2$s</span>%1$s', + 'a' => '%1$s %2$s', + 'b' => '%2$s %1$s', + ); + + $pos = strtolower($this->config['p']); + if (!isset($positions[$pos])) + return $text; + + return sprintf($positions[$pos], $text, $dec); + } + + // Render the annotation with the database record $row. $text is the + // text of the cell before decorations were applied. + function render($row, $cell) { + if ($decoration = $this->getDecoration($row, $cell)) + return $this->decorate($cell, $decoration); + + return $cell; + } + + // Add the annotation to a QuerySet + abstract function annotate($query); + + // Fetch some HTML to render the decoration on the page. This function + // can return boolean FALSE to indicate no decoration should be applied + abstract function getDecoration($row, $text); + + function getPosition() { + return strtolower($this->config['p']) ?: 'a'; + } +} + +class TicketThreadCount +extends QueueDecoration { + static $icon = 'comments-alt'; + static $qname = '_thread_count'; + static $desc = /* @trans */ 'Thread Count'; + + function annotate($query) { + return $query->annotate(array( + static::$qname => TicketThread::objects() + ->filter(array('ticket__ticket_id' => new SqlField('ticket_id', 1))) + ->exclude(array('entries__flags__hasbit' => ThreadEntry::FLAG_HIDDEN)) + ->aggregate(array('count' => SqlAggregate::COUNT('entries__id'))) + )); + } + + function getDecoration($row, $text) { + $threadcount = $row[static::$qname]; + if ($threadcount > 1) { + return sprintf( + '<i class="icon-comments-alt"></i><small>%s</small>', + $threadcount + ); + } + } +} + +class ThreadAttachmentCount +extends QueueDecoration { + static $icon = 'paperclip'; + static $qname = '_att_count'; + static $desc = /* @trans */ 'Attachment Count'; + + function annotate($query) { + // TODO: Convert to Thread attachments + return $query->annotate(array( + static::$qname => TicketThread::objects() + ->filter(array('ticket__ticket_id' => new SqlField('ticket_id', 1))) + ->filter(array('entries__attachments__inline' => 0)) + ->aggregate(array('count' => SqlAggregate::COUNT('entries__attachments__id'))) + )); + } + + function getDecoration($row, $text) { + $count = $row[static::$qname]; + if ($count) { + return sprintf( + '<i class="small icon-paperclip icon-flip-horizontal" data-toggle="tooltip" title="%s"></i>', + $count); + } + } +} + +class OverdueFlagDecoration +extends QueueDecoration { + static $icon = 'exclamation'; + static $desc = /* @trans */ 'Overdue Icon'; + + function annotate($query) { + return $query->values('isoverdue'); + } + + function getDecoration($row, $text) { + return sprintf( + '<span class="Icon overdueTicket">%s</span>', + $text); + } +} + +class TicketSourceDecoration +extends QueueDecoration { + static $icon = 'phone'; + static $desc = /* @trans */ 'Ticket Source'; + + function annotate($query) { + return $query->values('source'); + } + + function getDecoration($row, $text) { + return sprintf( + '<span class="Icon %sTicket">%s</span>', + $row['source'], $text); + } +} + +class DataSourceField +extends ChoiceField { + function getChoices() { + $config = $this->getConfiguration(); + $root = $config['root']; + $fields = array(); + foreach (SavedSearch::getSearchableFields($root) as $path=>$f) { + $fields[$path] = $f->get('label'); + } + return $fields; + } +} + +class QueueColumnCondition { + var $config; + var $properties = array(); + + function __construct($config) { + $this->config = $config; + if (is_array($config['prop'])) + $this->properties = $config['prop']; + } + + function getProperties() { + return $this->properties; + } + + function getField() { + } + + // Add the annotation to a QuerySet + function annotate($query) { + $criteria = $this->config['crit']; + $searchable = SavedSearch::getSearchableFields('Ticket'); + + // Setup a dummy form with a source for field setup + $form = new Form($criteria); + $fields = array(); + foreach ($criteria as $k=>$v) { + if (substr($k, -7) === '+method') { + list($name,) = explode('+', $k, 2); + if (!isset($searchable[$name])) + continue; + + // Lookup the field to search this condition + $field = $searchable[$name]; + + // Get the search method and value + $breakout = SavedSearch::getSearchField($field, $name); + $method = $breakout["{$name}+method"]; + $method->setForm($form); + if (!($method = $method->getClean())) + continue; + + if (!($value = $breakout["{$name}+{$method}"])) + continue; + + // Fetch a criteria Q for the query + $value = $value->getClean(); + $Q = $field->getSearchQ($method, $value, $name); + + // Add an annotation to the query + $query = $query->annotate(array( + $this->getAnnotationName() => new SqlExpr($Q) + )); + + // Only one field can be considered in the condition + break; + } + } + return $query; + } + + function render($row, $text) { + $field = $this->getAnnotationName(); + if ($V = $row[$field]) { + $style = array(); + foreach ($this->getProperties() as $css=>$value) { + $style[] = "{$css}:{$value}"; + } + $text = sprintf('<span style="%s">%s</span>', + implode(' ', $style), $text); + } + return $text; + } + + function getAnnotationName() { + return 'howdy'; + } + + static function fromJson($config) { + if (is_string($config)) + $config = JsonDataParser::decode($cnofig); + if (!is_array($config)) + throw new BadMethodCallException('$config must be string or array'); + + return new static($config); + } +} + +class QueueColumnConditionProperty +extends ChoiceField { + static $properties = array( + 'background-color' => 'ColorChoiceField', + 'color' => 'ColorChoiceField', + 'font-family' => array( + 'monospace', 'serif', 'sans-serif', 'cursive', 'fantasy', + ), + 'font-size' => array( + 'small', 'medium', 'large', 'smaller', 'larger', + ), + 'font-style' => array( + 'normal', 'italic', 'oblique', + ), + 'font-weight' => array( + 'lighter', 'normal', 'bold', 'bolder', + ), + 'text-decoration' => array( + 'none', 'underline', + ), + 'text-transform' => array( + 'uppercase', 'lowercase', 'captalize', + ), + ); + + function __construct($property) { + $this->property = $property; + } + + static function getProperties() { + return array_keys(static::$properties[$this->property]); + } + + static function getField($prop) { + $choices = static::$properties[$prop]; + if (is_array($choices)) + return new ChoiceField(array( + 'choices' => array_combine($choices, $choices), + )); + elseif (class_exists($choices)) + return new $choices(); + } + + function getChoices() { + if (isset($this->property)) + return static::$properties[$this->property]; + + $keys = array_keys(static::$properties); + return array_combine($keys, $keys); + } +} + + +/** + * Object version of JSON-serialized column array which has several + * properties: + * + * { + * "heading": "Header Text", + * "primary": "user__name", + * "secondary": null, + * "width": 100, + * "link": 'ticket', + * "truncate": "wrap", + * "filter": "UsersName" + * "annotations": [ + * { + * "c": "ThreadCollabCount", + * "p": ">" + * } + * ], + * "conditions": [ + * { + * "crit": { + * "created+method": {"ndaysago": "in the last n days"}, "created+ndaysago": {"until":"7"} + * }, + * "prop": { + * "font-weight": "bold" + * } + * } + * ] + * } + */ +class QueueColumn +extends VerySimpleModel { + static $meta = array( + 'table' => QUEUE_COLUMN_TABLE, + 'pk' => array('id'), + 'joins' => array( + 'queue' => array( + 'constraint' => array('queue_id' => 'CustomQueue.id'), + ), + ), + ); + + var $_decorations = array(); + var $_conditions = array(); + + function __onload() { + if ($this->annotations) { + foreach ($this->annotations as $D) + $this->_decorations[] = QueueDecoration::fromJson($D) ?: array(); + } + if ($this->conditions) { + foreach ($this->conditions as $C) + $this->_conditions[] = QueueColumnCondition::fromJson($C) ?: array(); + } + } + + function getId() { + return $this->id; + } + + function getQueue() { + return $this->queue; + } + + function getHeading() { + return $this->heading; + } + + function getWidth() { + return $this->width ?: 100; + } + + function getLink($row) { + $link = $this->link; + switch (strtolower($link)) { + case 'root': + case 'ticket': + return Ticket::getLink($row['ticket_id']); + case 'user': + return User::getLink($row['user_id']); + case 'org': + return Organization::getLink($row['user__org_id']); + } + } + + function render($row) { + // Basic data + $text = $this->renderBasicValue($row); + + // Truncate + if ($class = $this->getTruncateClass()) { + $text = sprintf('<span class="%s">%s</span>', $class, $text); + } + + // Link + if ($link = $this->getLink($row)) { + $text = sprintf('<a href="%s">%s</a>', $link, $text); + } + + // Decorations and conditions + foreach ($this->_decorations as $D) { + $text = $D->render($row, $text); + } + foreach ($this->_conditions as $C) { + $text = $C->render($row, $text); + } + return $text; + } + + function renderBasicValue($row) { + $root = $this->getQueue()->getRoot(); + $fields = SavedSearch::getSearchableFields($root); + $primary = $this->getOrmPath($this->primary); + $secondary = $this->getOrmPath($this->secondary); + + // TODO: Consider data filter if configured + + if (($F = $fields[$primary]) && ($T = $F->from_query($row, $primary))) + return $F->display($F->to_php($T)); + + if (($F = $fields[$secondary]) && ($T = $F->from_query($row, $secondary))) + return $F->display($F->to_php($T)); + } + + function getTruncateClass() { + switch ($this->truncate) { + case 'ellipsis': + return 'trucate'; + case 'clip': + return 'truncate clip'; + default: + case 'wrap': + return false; + } + } + + function mangleQuery($query) { + // Basic data + $fields = SavedSearch::getSearchableFields($this->getQueue()->getRoot()); + if ($primary = $fields[$this->primary]) + $query = $primary->addToQuery($query, + $this->getOrmPath($this->primary)); + + if ($secondary = $fields[$this->secondary]) + $query = $secondary->addToQuery($query, + $this->getOrmPath($this->secondary)); + + switch ($this->link) { + // XXX: Consider the ROOT of the related queue + case 'ticket': + $query = $query->values('ticket_id'); + break; + case 'user': + $query = $query->values('user_id'); + break; + case 'org': + $query = $query->values('user__org_id'); + break; + } + + // Decorations + foreach ($this->_decorations as $D) { + $query = $D->annotate($query); + } + + // Conditions + foreach ($this->_conditions as $C) { + $query = $C->annotate($query); + } + + return $query; + } + + function getDataConfigForm($source=false) { + return new QueueColDataConfigForm($source ?: $this->ht, + array('id' => $this->id)); + } + + function getOrmPath($name) { + return $name; + } + + function getDecorations() { + return $this->_decorations; + } + + function getConditions() { + return $this->_conditions; + } + + /** + * Create a CustomQueueColumn from vars (_POST) received from an + * update request. + */ + static function create($vars=array()) { + $inst = parent::create($vars); + // TODO: Convert decorations and conditions + return $inst; + } + + function update($vars) { + $form = $this->getDataConfigForm($vars); + foreach ($form->getClean() as $k=>$v) + $this->set($k, $v); + } +} + +class QueueColDataConfigForm +extends AbstractForm { + function buildFields() { + return array( + 'primary' => new DataSourceField(array( + 'label' => __('Primary Data Source'), + 'required' => true, + 'configuration' => array( + 'root' => 'Ticket', + ), + 'layout' => new GridFluidCell(6), + )), + 'secondary' => new DataSourceField(array( + 'label' => __('Secondary Data Source'), + 'configuration' => array( + 'root' => 'Ticket', + ), + 'layout' => new GridFluidCell(6), + )), + 'heading' => new TextboxField(array( + 'label' => __('Heading'), + 'required' => true, + 'layout' => new GridFluidCell(3), + )), + 'link' => new ChoiceField(array( + 'label' => __('Link'), + 'required' => false, + 'choices' => array( + 'ticket' => __('Ticket'), + 'user' => __('User'), + 'org' => __('Organization'), + ), + 'layout' => new GridFluidCell(3), + )), + 'width' => new TextboxField(array( + 'label' => __('Width'), + 'default' => 75, + 'configuration' => array( + 'validator' => 'number', + ), + 'layout' => new GridFluidCell(3), + )), + 'truncate' => new ChoiceField(array( + 'label' => __('Text Overflow'), + 'choices' => array( + 'wrap' => __("Wrap Lines"), + 'ellipsis' => __("Add Ellipsis"), + 'clip' => __("Clip Text"), + ), + 'default' => 'wrap', + 'layout' => new GridFluidCell(3), + )), + ); + } +} diff --git a/include/class.role.php b/include/class.role.php index 0e0f89af2b493003fe2ef3437a0e554152325c36..0b34853220e1ea29bb2c607a646b1063c50efac9 100644 --- a/include/class.role.php +++ b/include/class.role.php @@ -13,6 +13,7 @@ vim: expandtab sw=4 ts=4 sts=4: **********************************************************************/ +require_once INCLUDE_DIR . 'class.forms.php'; class RoleModel extends VerySimpleModel { static $meta = array( diff --git a/include/class.search.php b/include/class.search.php index 61b56a41e61f6c14dd4fa44ccaa6fd914d36453b..674533e2a531748f9bfb8a562d34777ed45b4bee 100644 --- a/include/class.search.php +++ b/include/class.search.php @@ -21,6 +21,8 @@ vim: expandtab sw=4 ts=4 sts=4: **********************************************************************/ +require_once INCLUDE_DIR . 'class.role.php'; +require_once INCLUDE_DIR . 'class.list.php'; abstract class SearchBackend { static $id = false; @@ -672,6 +674,19 @@ class SavedSearch extends VerySimpleModel { ))); } + function getName() { + return $this->name; + } + + function getSearchForm() { + if ($state = JsonDataParser::parse($search->config)) { + $form = $search->loadFromState($state); + $form->loadState($state); + return $form; + } + return $this->getForm(); + } + function loadFromState($source=false) { // Pull out 'other' fields from the state so the fields will be // added to the form. The state will be loaded below @@ -787,6 +802,39 @@ class SavedSearch extends VerySimpleModel { } return $core; } + static function getSupportedTicketMatches() { + // User information + $matches = array( + __('Ticket Built-In') => SavedSearch::getExtendedTicketFields(), + ); + foreach (array('ticket'=>'TicketForm', 'user'=>'UserForm', 'organization'=>'OrganizationForm') as $k=>$F) { + $form = $F::objects()->one(); + $fields = &$matches[$form->getLocal('title')]; + foreach ($form->getFields() as $f) { + if (!$f->hasData() || $f->isPresentationOnly()) + continue; + $fields[":$k!".$f->get('id')] = __(ucfirst($k)).' / '.$f->getLocal('label'); + /* TODO: Support matches on list item properties + if (($fi = $f->getImpl()) && $fi->hasSubFields()) { + foreach ($fi->getSubFields() as $p) { + $fields[":$k.".$f->get('id').'.'.$p->get('id')] + = __(ucfirst($k)).' / '.$f->getLocal('label').' / '.$p->getLocal('label'); + } + } + */ + } + } + $fields = &$matches[__('Custom Forms')]; + foreach (DynamicForm::objects()->filter(array('type'=>'G')) as $form) { + foreach ($form->getFields() as $f) { + if (!$f->hasData() || $f->isPresentationOnly()) + continue; + $key = sprintf(':field!%d', $f->get('id'), $f->get('id')); + $fields[$key] = $form->getLocal('title').' / '.$f->getLocal('label'); + } + } + return $matches; + } static function getExtendedTicketFields() { return array( @@ -823,6 +871,36 @@ class SavedSearch extends VerySimpleModel { ); } + static function getSearchableFields($base, $recurse=2, $cache=true) { + static $cache; + + if (!in_array('Searchable', class_implements($base))) + return array(); + + // FIXME: The fields from dynamicFormFields seem to be cached, and + // setting the label is preserved across multiple calls to this + // function. The caching helps with this phenomenon, but a better + // mechanism should be employed + if ($cache && isset($cache[$base])) + return $cache[$base]; + + $fields = $base::getSearchableFields(); + if ($recurse) { + foreach ($base::getMeta('joins') as $path=>$j) { + $fc = $j['fkey'][0]; + if ($fc == $base || $j['list'] || $j['reverse']) + continue; + foreach (static::getSearchableFields($fc, $recurse-1, false) as $path2=>$F) { + $fields["{$path}__{$path2}"] = $F; + $F->set('label', sprintf("%s / %s", $fc, $F->get('label'))); + } + } + } + if ($cache) + $cache[$base] = $fields; + return $fields; + } + static function getSearchField($field, $name) { $baseId = $field->getId() * 20; $pieces = array(); @@ -1125,6 +1203,27 @@ class AssigneeChoiceField extends ChoiceField { return parent::describeSearchMethod($method); } } + + function addToQuery($query, $name=false) { + return $query->values('staff__firstname', 'staff__lastname', 'team__name', 'team_id'); + } + + function from_query($row, $name=false) { + if ($row['staff__firstname']) + return new AgentsName(array('first' => $row['staff__firstname'], 'last' => $row['staff__lastname'])); + if ($row['team_id']) + return Team::getLocalById($row['team_id'], 'name', $row['team__name']); + } + + function display($value) { + return (string) $value; + } +} + +class AgentSelectionField extends ChoiceField { + function getChoices() { + return Staff::getStaffMembers(); + } } class TicketStateChoiceField extends ChoiceField { @@ -1240,3 +1339,10 @@ class TicketStatusChoiceField extends SelectionField { } } } + +interface Searchable { + // Fetch an array of [ orm__path => Field() ] pairs. The field label is + // used when this list is rendered in a dropdown, and the field search + // mechanisms are use to apply query filtering based on the field. + static function getSearchableFields(); +} diff --git a/include/class.staff.php b/include/class.staff.php index e5ed7e77555e5bb72f1376d10de92bc9b579569d..ce922b1af413944042bdaeef33916a83b25f83e3 100644 --- a/include/class.staff.php +++ b/include/class.staff.php @@ -23,7 +23,7 @@ include_once(INCLUDE_DIR.'class.user.php'); include_once(INCLUDE_DIR.'class.auth.php'); class Staff extends VerySimpleModel -implements AuthenticatedUser, EmailContact, TemplateVariable { +implements AuthenticatedUser, EmailContact, TemplateVariable, Searchable { static $meta = array( 'table' => STAFF_TABLE, @@ -127,6 +127,14 @@ implements AuthenticatedUser, EmailContact, TemplateVariable { } } + static function getSearchableFields() { + return array( + 'email' => new TextboxField(array( + 'label' => __('Email Address'), + )), + ); + } + function getHashtable() { $base = $this->ht; unset($base['teams']); diff --git a/include/class.ticket.php b/include/class.ticket.php index 7cc661f9595c6a68e4dc8ee8d47b04b7fd13ddda..0fc4f021df621418c6b236d38e92c1698f528256 100644 --- a/include/class.ticket.php +++ b/include/class.ticket.php @@ -245,7 +245,7 @@ class TicketCData extends VerySimpleModel { } class Ticket extends TicketModel -implements RestrictedAccess, Threadable { +implements RestrictedAccess, Threadable, Searchable { static $meta = array( 'select_related' => array('topic', 'staff', 'user', 'team', 'dept', 'sla', 'thread', @@ -1947,6 +1947,57 @@ implements RestrictedAccess, Threadable { return $base + $extra; } + // Searchable interface + static function getSearchableFields() { + $base = array( + 'number' => new TextboxField(array( + 'label' => __('Ticket Number') + )), + 'ip_address' => new TextboxField(array( + 'label' => __('IP Address'), + 'configuration' => array('validator' => 'ip'), + )), + 'source' => new TicketSourceChoiceField(array( + 'label' => __('Ticket Source'), + )), + 'isoverdue' => new BooleanField(array( + 'label' => __('Overdue'), + )), + 'isanswered' => new BooleanField(array( + 'label' => __('Answered'), + )), + 'duedate' => new DatetimeField(array( + 'label' => __('Due Date'), + )), + 'reopened' => new DatetimeField(array( + 'label' => __('Reopen Date'), + )), + 'closed' => new DatetimeField(array( + 'label' => __('Close Date'), + )), + 'lastupdate' => new DatetimeField(array( + 'label' => __('Last Update'), + )), + 'created' => new DatetimeField(array( + 'label' => __('Create Date'), + )), + 'assignee' => new AssigneeChoiceField(array( + 'label' => __('Assignee'), + )), + ); + $tform = TicketForm::getInstance(); + foreach ($tform->getFields() as $F) { + $fname = $F->get('name') ?: ('field_'.$F->get('id')); + if (!$F->hasData() || $F->isPresentationOnly()) + continue; + if (!$F->isStorable()) + $base[$fname] = $F; + else + $base["cdata__{$fname}"] = $F; + } + return $base; + } + //Replace base variables. function replaceVars($input, $vars = array()) { global $ost; @@ -3674,5 +3725,14 @@ implements RestrictedAccess, Threadable { require STAFFINC_DIR.'templates/tickets-actions.tmpl.php'; } + + static function getLink($id) { + global $thisstaff; + + switch (true) { + case ($thisstaff instanceof Staff): + return ROOT_PATH . sprintf('scp/tickets.php?id=%s', $id); + } + } } ?> diff --git a/include/class.topic.php b/include/class.topic.php index c25ab788a4288beaae3bd1163ecb72c6b2ea183f..ead2e57b6956a36b0db0483c157f86a405ae3f32 100644 --- a/include/class.topic.php +++ b/include/class.topic.php @@ -13,12 +13,12 @@ vim: expandtab sw=4 ts=4 sts=4: **********************************************************************/ - require_once INCLUDE_DIR . 'class.sequence.php'; require_once INCLUDE_DIR . 'class.filter.php'; +require_once INCLUDE_DIR . 'class.search.php'; class Topic extends VerySimpleModel -implements TemplateVariable { +implements TemplateVariable, Searchable { static $meta = array( 'table' => TOPIC_TABLE, @@ -91,6 +91,14 @@ implements TemplateVariable { ); } + static function getSearchableFields() { + return array( + 'name' => new TextboxField(array( + 'label' => __('Name'), + )), + ); + } + function getId() { return $this->topic_id; } diff --git a/include/class.user.php b/include/class.user.php index 24cae58b9122dc8a5ee3af8ceb9183720e7524d6..a3b98f381b61d1e341ae647ce131fa60fcf28528 100644 --- a/include/class.user.php +++ b/include/class.user.php @@ -16,8 +16,9 @@ **********************************************************************/ require_once INCLUDE_DIR . 'class.orm.php'; require_once INCLUDE_DIR . 'class.util.php'; -require_once INCLUDE_DIR . 'class.organization.php'; require_once INCLUDE_DIR . 'class.variable.php'; +require_once INCLUDE_DIR . 'class.search.php'; +require_once INCLUDE_DIR . 'class.organization.php'; class UserEmailModel extends VerySimpleModel { static $meta = array( @@ -191,7 +192,7 @@ class UserCdata extends VerySimpleModel { } class User extends UserModel -implements TemplateVariable { +implements TemplateVariable, Searchable { var $_entries; var $_forms; @@ -362,6 +363,20 @@ implements TemplateVariable { return $base + $extra; } + static function getSearchableFields() { + $uform = UserForm::getUserForm(); + foreach ($uform->getFields() as $F) { + $fname = $F->get('name') ?: ('field_'.$F->get('id')); + if (!$F->hasData() || $F->isPresentationOnly()) + continue; + if (!$F->isStorable()) + $base[$fname] = $F; + else + $base["cdata__{$fname}"] = $F; + } + return $base; + } + function addDynamicData($data) { return $this->addForm(UserForm::objects()->one(), 1, $data); } @@ -599,6 +614,15 @@ implements TemplateVariable { if ($user = static::lookup($id)) return $user->getName(); } + + static function getLink($id) { + global $thisstaff; + + if (!$id || !$thisstaff) + return false; + + return ROOT_PATH . sprintf('users.php?id=%s', $id); + } } class EmailAddress diff --git a/include/staff/queue.inc.php b/include/staff/queue.inc.php new file mode 100644 index 0000000000000000000000000000000000000000..d322a912554b6bc65e91462834a1609e9f3abd27 --- /dev/null +++ b/include/staff/queue.inc.php @@ -0,0 +1,243 @@ +<?php +// vim: expandtab sw=2 ts=2 sts=2: + +if(!defined('OSTADMININC') || !$thisstaff || !$thisstaff->isAdmin()) die('Access Denied'); + +$info = $qs = array(); + +if ($_REQUEST['a']=='add'){ + if (!$queue) { + $queue = CustomQueue::create(array( + 'flags' => CustomQueue::FLAG_QUEUE, + )); + } + $title=__('Add New Queue'); + $action='create'; + $submit_text=__('Create'); +} +else { + //Editing Department. + $title=__('Manage Custom Queue'); + $action='update'; + $submit_text=__('Save Changes'); + $info['id'] = $queue->getId(); + $qs += array('id' => $queue->getId()); +} +?> + +<form action="queues.php?<?php echo Http::build_query($qs); ?>" method="post" id="save" autocomplete="off"> + <?php csrf_token(); ?> + <input type="hidden" name="do" value="<?php echo $action; ?>"> + <input type="hidden" name="a" value="<?php echo Format::htmlchars($_REQUEST['a']); ?>"> + <input type="hidden" name="id" value="<?php echo $info['id']; ?>"> + + <h2><?php echo __('Ticket Queues'); ?> // <?php echo $title; ?> + <?php if (isset($queue->id)) { ?><small> + — <?php echo $queue->getName(); ?></small> + <?php } ?> + </h2> + + + <ul class="clean tabs"> + <li class="active"><a href="#criteria"><i class="icon-filter"></i> + <?php echo __('Criteria'); ?></a></li> + <li><a href="#columns"><i class="icon-columns"></i> + <?php echo __('Columns'); ?></a></li> + <li><a href="#preview-tab"><i class="icon-eye-open"></i> + <?php echo __('Preview'); ?></a></li> + </ul> + + <div class="tab_content" id="criteria"> + <table class="table"> + <td style="width:60%; vertical-align:top"> + <div><strong><?php echo __('Queue Name'); ?>:</strong></div> + <input type="text" name="name" value="<?php + echo Format::htmlchars($queue->getName()); ?>" + style="width:100%" /> + + <br/> + <br/> + <div><strong><?php echo __("Queue Search Criteria"); ?></strong></div> + <hr/> + <div class="advanced-search"> +<?php + $form = $queue->getSearchForm(); + $search = $queue; + $matches = SavedSearch::getSupportedTicketMatches(); + include STAFFINC_DIR . 'templates/advanced-search-criteria.tmpl.php'; +?> + </div> + </td> + <td style="width:35%; padding-left:40px; vertical-align:top"> + <div><strong><?php echo __("Parent Queue"); ?>:</strong></div> + <select name="parent_id"> + <option value="0">— <?php echo __('Top-Level Queue'); ?> —</option> +<?php foreach (CustomQueue::objects() as $cq) { ?> + <option value="<?php echo $cq->id; ?>"><?php echo $cq->getName(); ?></option> +<?php } ?> + </select> + + <br/> + <br/> + <div><strong><?php echo __("Quick Filter"); ?></strong></div> + <hr/> + <select name="quick-filter"> + <option value=":p:">— <?php echo __('Inherit from parent'); ?> —</option> +<?php foreach (SavedSearch::getSearchableFields('Ticket') as $path=>$f) { ?> + <option value="<?php echo $path; ?>"><?php echo $f->get('label'); ?></option> +<?php } ?> + </select> + <br/> + <br/> + <div><strong><?php echo __("Sort Options"); ?></strong></div> + <hr/> + </td> + </table> + </div> + + <div class="hidden tab_content" id="columns"> + <h2><?php echo __("Manage columns in this queue"); ?></h2> + <p><?php echo __("Add, remove, and customize the content of the columns in this queue using the options below. Click a column header to manage or resize it"); ?></p> + + <div> + <i class="icon-plus-sign"></i> + <select id="add-column" data-next-id="0" onchange="javascript: + var $this = $(this), + selected = $this.find(':selected'), + nextId = $this.data('nextId'), + columns = $('#resizable-columns'); + $.ajax({ + url: 'ajax.php/queue/addColumn', + data: { field: selected.val(), id: nextId }, + dataType: 'json', + success: function(json) { + var div = $('<div></div>') + .addClass('column-header ui-resizable') + .text(json.heading) + .data({id: nextId, colId: 'colconfig-'+nextId, width: json.width}) + .append($('<i>') + .addClass('icon-ellipsis-vertical ui-resizable-handle ui-resizable-handle-e') + ) + .append($('<input />') + .attr({type:'hidden', name:'columns[]'}) + .val(nextId) + ); + config = $('<div></div>') + .addClass('hidden column-configuration') + .attr('id', 'colconfig-' + nextId); + config.append($(json.config)).insertAfter(columns.append(div)); + $this.data('nextId', nextId+1); + } + }); + "> + <option value="">— <?php echo __('Add a column'); ?> —</option> +<?php foreach (SavedSearch::getSearchableFields('Ticket') as $path=>$f) { ?> + <option value="<?php echo $path; ?>"><?php echo $f->get('label'); ?></option> +<?php } ?> + </select> + + <div id="resizable-columns"> +<?php foreach ($queue->getColumns() as $column) { + $colid = $column->getId(); + $maxcolid = max(@$maxcolid ?: 0, $colid); + echo sprintf('<div data-id="%s" data-col-id="colconfig-%s" class="column-header" ' + .'data-width="%s">%s' + .'<i class="icon-ellipsis-vertical ui-resizable-handle ui-resizable-handle-e"></i>' + .'<input type="hidden" name="columns[]" value="%s"/>' + .'</div>', + $colid, $colid, $column->getWidth(), $column->getHeading(), $colid); +} ?> + </div> + <script> + $(function() { + $('#add-column').data('nextId', <?php echo $maxcolid+1; ?>); + var qq = setInterval(function() { + var total = 0, + container = $('#resizable-columns'), + width = container.width(), + w2px = 1.25, + columns = $('.column-header', container); + // Await computation of the <div>'s width + if (width) + clearInterval(qq); + columns.each(function() { + total += $(this).data('width') || 100; + }); + container.data('w2px', w2px); + columns.each(function() { + // FIXME: jQuery will compensate for padding (40px) + $(this).width(w2px * ($(this).data('width') || 100) - 42); + }); + }, 20); + }); + </script> + +<?php foreach ($queue->getColumns() as $column) { + $colid = $column->getId(); + echo sprintf('<div class="hidden column-configuration" id="colconfig-%s">', + $colid); + include STAFFINC_DIR . 'templates/queue-column.tmpl.php'; + echo '</div>'; +} ?> + </div> + + <script> + var aa = setInterval(function() { + var cols = $('#resizable-columns'); + if (cols.length && cols.sortable) + clearInterval(aa); + cols.sortable({ + containment: 'parent' + }); + $('.column-header', cols).resizable({ + handles: {'e' : '.ui-resizable-handle'}, + grid: [ 20, 0 ], + maxHeight: 16, + minHeight: 16, + stop: function(event, ui) { + var w2px = ui.element.parent().data('w2px'), + width = ui.element.width() - 42; + ui.element.data('width', width / w2px); + // TODO: Update WIDTH text box in the data form + } + }); + cols.click('.column-header', function(e) { + var $this = $(event.target); + $this.parent().children().removeClass('active'); + $this.addClass('active'); + $('.column-configuration', $this.closest('.tab_content')).hide(); + $('#'+$this.data('colId')).fadeIn('fast'); + }); + }, 20); + </script> + </div> + + <div class="hidden tab_content" id="preview-tab"> + + <div id="preview"> + </div> + + <script> + $(function() { + $('#preview-tab').on('afterShow', function() { + $.ajax({ + url: 'ajax.php/queue/preview', + type: 'POST', + data: $('#save').serializeArray(), + success: function(html) { + $('#preview').html(html); + } + }); + }); + }); + </script> + + </div> + + <p style="text-align:center;"> + <input type="submit" name="submit" value="<?php echo $submit_text; ?>"> + <input type="reset" name="reset" value="<?php echo __('Reset');?>"> + <input type="button" name="cancel" value="<?php echo __('Cancel');?>" onclick="window.history.go(-1);"> + </p> + +</form> diff --git a/include/staff/queues-ticket.inc.php b/include/staff/queues-ticket.inc.php new file mode 100644 index 0000000000000000000000000000000000000000..170edafed6e22445c3a6b2ce74edea5103468272 --- /dev/null +++ b/include/staff/queues-ticket.inc.php @@ -0,0 +1,54 @@ + +<form action="queues.php?t=tickets" method="POST" name="keys"> + <div class="sticky bar opaque"> + <div class="content"> + <div class="pull-right"> + <a href="queues.php?t=tickets&a=add" class="green button action-button"><i class="icon-plus-sign"></i> <?php echo __('Add New Queue');?></a> + <span class="action-button" data-dropdown="#action-dropdown-more"> + <i class="icon-caret-down pull-right"></i> + <span ><i class="icon-cog"></i> <?php echo __('More');?></span> + </span> + <div id="action-dropdown-more" class="action-dropdown anchor-right"> + <ul id="actions"> + <li> + <a class="confirm" data-name="enable" href="queues.php?t=tickets&a=enable"> + <i class="icon-ok-sign icon-fixed-width"></i> + <?php echo __( 'Enable'); ?> + </a> + </li> + <li> + <a class="confirm" data-name="disable" href="queues.php?t=tickets&a=disable"> + <i class="icon-ban-circle icon-fixed-width"></i> + <?php echo __( 'Disable'); ?> + </a> + </li> + <li class="danger"> + <a class="confirm" data-name="delete" href="queues.php?t=tickets&a=delete#queues"> + <i class="icon-trash icon-fixed-width"></i> + <?php echo __( 'Delete'); ?> + </a> + </li> + </ul> + </div> + </div> + <h3><?php echo __('Ticket Queues');?></h3> + </div> + </div> + <div class="clear"></div> + <?php csrf_token(); ?> + <input type="hidden" name="do" value="mass_process" > +<input type="hidden" id="action" name="a" value="" > + <table class="list" border="0" cellspacing="1" cellpadding="0" width="940"> + <thead> + <tr> + <th width="4%"> </th> + <th width="46%"><a <?php echo $key_sort; ?> href="queues.php?t=tickets&<?php echo $qstr; ?>&sort=name#queues"><?php echo __('Name');?></a></th> + <th width="12%"><a <?php echo $ip_sort; ?> href="queues.php?t=tickets&<?php echo $qstr; ?>&sort=creator#queues"><?php echo __('Creator');?></a></th> + <th width="8%"><a <?php echo $status_sort; ?> href="queues.php?t=tickets&<?php echo $qstr; ?>&sort=status#queues"><?php echo __('Status');?></a></th> + <th width="10%" nowrap><a <?php echo $date_sort; ?>href="queues.php?t=tickets&<?php echo $qstr; ?>&sort=date#queues"><?php echo __('Created');?></a></th> + </tr> + </thead> + <tbody> + </tbody> +</table> +</form> diff --git a/include/staff/settings-tickets.inc.php b/include/staff/settings-tickets.inc.php index 8ec82503174c174509d51d3a923240dee7b4b1a2..454b53f64f331f4597120a61824d62dfe65b978b 100644 --- a/include/staff/settings-tickets.inc.php +++ b/include/staff/settings-tickets.inc.php @@ -15,6 +15,8 @@ if(!($maxfileuploads=ini_get('max_file_uploads'))) <?php echo __('Autoresponder'); ?></a></li> <li><a href="#alerts"><i class="icon-bell-alt"></i> <?php echo __('Alerts and Notices'); ?></a></li> + <li><a href="#queues"><i class="icon-table"></i> + <?php echo __('Queues'); ?></a></li> </ul> <div class="tab_content" id="settings"> <table class="form_table settings_table" width="940" border="0" cellspacing="0" cellpadding="2"> @@ -239,6 +241,10 @@ if(!($maxfileuploads=ini_get('max_file_uploads'))) <?php include STAFFINC_DIR . 'settings-alerts.inc.php'; ?> </div> +<div class="hidden tab_content" id="queues"> + <?php include STAFFINC_DIR . 'queues-ticket.inc.php'; ?> +</div> + <p style="text-align:center;"> <input class="button" type="submit" name="submit" value="<?php echo __('Save Changes');?>"> <input class="button" type="reset" name="reset" value="<?php echo __('Reset Changes');?>"> diff --git a/include/staff/templates/advanced-search-criteria.tmpl.php b/include/staff/templates/advanced-search-criteria.tmpl.php new file mode 100644 index 0000000000000000000000000000000000000000..ea343245a463f9e0180db2e72dcbd3d458753575 --- /dev/null +++ b/include/staff/templates/advanced-search-criteria.tmpl.php @@ -0,0 +1,108 @@ +<?php +foreach ($form->errors(true) ?: array() as $message) { + ?><div class="error-banner"><?php echo $message;?></div><?php +} + +$info = $search->getSearchFields($form); +foreach (array_keys($info) as $F) { + ?><input type="hidden" name="fields[]" value="<?php echo $F; ?>"/><?php +} +$errors = !!$form->errors(); +$inbody = false; +$first_field = true; +foreach ($form->getFields() as $name=>$field) { + @list($name, $sub) = explode('+', $field->get('name'), 2); + if ($sub === 'search') { + if (!$first_field) { + echo '</div></div>'; + } + echo '<div class="adv-search-field-container">'; + $inbody = false; + $first_field = false; + } + elseif (!$first_field && !$inbody) { + echo sprintf('<div class="adv-search-field-body %s">', + !$errors && isset($info[$name]) && $info[$name]['active'] ? 'hidden' : ''); + $inbody = true; + } +?> + <fieldset id="field<?php echo $field->getWidget()->id; ?>" <?php + $class = array(); + if (!$field->isVisible()) + $class[] = "hidden"; + if ($sub === 'method') + $class[] = "adv-search-method"; + elseif ($sub === 'search') + $class[] = "adv-search-field"; + elseif ($field->get('__searchval__')) + $class[] = "adv-search-val"; + if ($class) + echo 'class="'.implode(' ', $class).'"'; + ?>> + <?php echo $field->render(); ?> + <?php if (!$errors && $sub === 'search' && isset($info[$name]) && $info[$name]['active']) { ?> + <span style="padding-left: 5px"> + <a href="#" data-name="<?php echo Format::htmlchars($name); ?>" onclick="javascript: + var $this = $(this), + name = $this.data('name'), + expanded = $this.data('expanded') || false; + $this.closest('.adv-search-field-container').find('.adv-search-field-body').slideDown('fast'); + $this.find('span.faded').hide(); + $this.find('i').removeClass('icon-caret-right').addClass('icon-caret-down'); + return false; +"><i class="icon-caret-right"></i> + <span class="faded"><?php echo $search->describeField($info[$name]); ?></span> + </a> + </span> + <?php } ?> + <?php foreach ($field->errors() as $E) { + ?><div class="error"><?php echo $E; ?></div><?php + } ?> + </fieldset> + <?php if ($name[0] == ':' && substr($name, -7) == '+search') { + list($N,) = explode('+', $name, 2); +?> + <input type="hidden" name="fields[]" value="<?php echo $N; ?>"/> + <?php } +} +if (!$first_field) + echo '</div></div>'; +?> +<div id="extra-fields"></div> +<hr/> +<i class="icon-plus-sign"></i> +<select id="search-add-new-field" name="new-field" style="max-width: 300px;"> + <option value="">— <?php echo __('Add Other Field'); ?> —</option> +<?php +if (is_array($matches)) { +foreach ($matches as $name => $fields) { ?> + <optgroup label="<?php echo $name; ?>"> +<?php + foreach ($fields as $id => $desc) { ?> + <option value="<?php echo $id; ?>" <?php + if (isset($state[$id])) echo 'disabled="disabled"'; + ?>><?php echo ($desc instanceof FormField ? $desc->getLocal('label') : $desc); ?></option> +<?php } ?> + </optgroup> +<?php } +} ?> +</select> +<script> +$(function() { + $('#search-add-new-field').on('change', function() { + var that=this; + $.ajax({ + url: 'ajax.php/tickets/search/field/'+$(this).val(), + type: 'get', + dataType: 'json', + success: function(json) { + if (!json.success) + return false; + ff_uid = json.ff_uid; + $(that).find(':selected').prop('disabled', true); + $('#extra-fields').append($(json.html)); + } + }); + }); +}); +</script> diff --git a/include/staff/templates/advanced-search.tmpl.php b/include/staff/templates/advanced-search.tmpl.php index cc14fa18df075e28b0369cb40474791cbd5a5cde..201cb66b0af583203453613a784ad76059bf8ae6 100644 --- a/include/staff/templates/advanced-search.tmpl.php +++ b/include/staff/templates/advanced-search.tmpl.php @@ -1,4 +1,4 @@ -<div id="advanced-search"> +<div id="advanced-search" class="advanced-search"> <h3 class="drag-handle"><?php echo __('Advanced Ticket Search');?></h3> <a class="close" href=""><i class="icon-remove-circle"></i></a> <hr/> @@ -6,93 +6,7 @@ <div class="row"> <div class="span6"> <input type="hidden" name="a" value="search"> -<?php -foreach ($form->errors(true) ?: array() as $message) { - ?><div class="error-banner"><?php echo $message;?></div><?php -} - -$info = $search->getSearchFields($form); -foreach (array_keys($info) as $F) { - ?><input type="hidden" name="fields[]" value="<?php echo $F; ?>"/><?php -} -$errors = !!$form->errors(); -$inbody = false; -$first_field = true; -foreach ($form->getFields() as $name=>$field) { - @list($name, $sub) = explode('+', $field->get('name'), 2); - if ($sub === 'search') { - if (!$first_field) { - echo '</div></div>'; - } - echo '<div class="adv-search-field-container">'; - $inbody = false; - $first_field = false; - } - elseif (!$first_field && !$inbody) { - echo sprintf('<div class="adv-search-field-body %s">', - !$errors && isset($info[$name]) && $info[$name]['active'] ? 'hidden' : ''); - $inbody = true; - } -?> - <fieldset id="field<?php echo $field->getWidget()->id; ?>" <?php - $class = array(); - if (!$field->isVisible()) - $class[] = "hidden"; - if ($sub === 'method') - $class[] = "adv-search-method"; - elseif ($sub === 'search') - $class[] = "adv-search-field"; - elseif ($field->get('__searchval__')) - $class[] = "adv-search-val"; - if ($class) - echo 'class="'.implode(' ', $class).'"'; - ?>> - <?php echo $field->render(); ?> - <?php if (!$errors && $sub === 'search' && isset($info[$name]) && $info[$name]['active']) { ?> - <span style="padding-left: 5px"> - <a href="#" data-name="<?php echo Format::htmlchars($name); ?>" onclick="javascript: - var $this = $(this), - name = $this.data('name'), - expanded = $this.data('expanded') || false; - $this.closest('.adv-search-field-container').find('.adv-search-field-body').slideDown('fast'); - $this.find('span.faded').hide(); - $this.find('i').removeClass('icon-caret-right').addClass('icon-caret-down'); - return false; -"><i class="icon-caret-right"></i> - <span class="faded"><?php echo $search->describeField($info[$name]); ?></span> - </a> - </span> - <?php } ?> - <?php foreach ($field->errors() as $E) { - ?><div class="error"><?php echo $E; ?></div><?php - } ?> - </fieldset> - <?php if ($name[0] == ':' && substr($name, -7) == '+search') { - list($N,) = explode('+', $name, 2); -?> - <input type="hidden" name="fields[]" value="<?php echo $N; ?>"/> - <?php } -} -if (!$first_field) - echo '</div></div>'; -?> -<div id="extra-fields"></div> -<hr/> -<select id="search-add-new-field" name="new-field" style="max-width: 300px;"> - <option value="">— <?php echo __('Add Other Field'); ?> —</option> -<?php -foreach ($matches as $name => $fields) { ?> - <optgroup label="<?php echo $name; ?>"> -<?php - foreach ($fields as $id => $desc) { ?> - <option value="<?php echo $id; ?>" <?php - if (isset($state[$id])) echo 'disabled="disabled"'; - ?>><?php echo ($desc instanceof FormField ? $desc->getLocal('label') : $desc); ?></option> -<?php } ?> - </optgroup> -<?php } ?> -</select> - + <?php include STAFFINC_DIR . 'templates/advanced-search-criteria.tmpl.php'; ?> </div> <div class="span6" style="border-left:1px solid #888;position:relative;padding-bottom:26px;"> <div style="margin-bottom: 0.5em;"><b style="font-size: 110%;"><?php echo __('Saved Searches'); ?></b></div> @@ -208,21 +122,5 @@ $(function() { return false; }); }, 200); - - $('#search-add-new-field').on('change', function() { - var that=this; - $.ajax({ - url: 'ajax.php/tickets/search/field/'+$(this).val(), - type: 'get', - dataType: 'json', - success: function(json) { - if (!json.success) - return false; - ff_uid = json.ff_uid; - $(that).find(':selected').prop('disabled', true); - $('#extra-fields').append($(json.html)); - } - }); - }); }); </script> diff --git a/include/staff/templates/queue-column-condition-prop.tmpl.php b/include/staff/templates/queue-column-condition-prop.tmpl.php new file mode 100644 index 0000000000000000000000000000000000000000..f29f5947d99b6a57f73df9aa96648b261cb0b0af --- /dev/null +++ b/include/staff/templates/queue-column-condition-prop.tmpl.php @@ -0,0 +1,24 @@ +<?php +/** + * Calling conventions + * + * $name - condition field name, like 'thread__lastmessage' + * $prop - CSS property name from QueueColumnConditionProperty::$properties + * $v - value for the property + */ +?> +<div class="condition-property"> + <div class="pull-right"> + <a href="#" onclick="javascript:$(this).closest('.condition-property').remove()" + ><i class="icon-trash"></i></a> + </div> + <div><?php echo mb_convert_case($prop, MB_CASE_TITLE); ?></div> +<?php + $F = QueueColumnConditionProperty::getField($prop); + $F->set('name', "prop-{$name}-{$prop}"); + $F->value = $v; + $form = new SimpleForm(array($F), $_POST); + echo $F->render(); + echo $form->getMedia(); +?> +</div> diff --git a/include/staff/templates/queue-column-condition.tmpl.php b/include/staff/templates/queue-column-condition.tmpl.php new file mode 100644 index 0000000000000000000000000000000000000000..d445f4f4d1ac567018b55fc414b876673c5a605f --- /dev/null +++ b/include/staff/templates/queue-column-condition.tmpl.php @@ -0,0 +1,74 @@ +<?php +// Calling convention: +// +// $field - field for the condition (Ticket / Last Update) +// $properties - currently-configured properties for the condition +// $condition - <QueueColumnCondition> instance for this condition +?> +<div class="condition"> + <div class="pull-right"> + <a href="#" onclick="javascript: $(this).closest('.condition').remove(); + "><i class="icon-trash"></i></a> + </div> + <?php echo $field->get('label'); ?> + <div class="advanced-search"> +<?php +$name = $field->get('name'); +$parts = SavedSearch::getSearchField($field, $name); +// Drop the search checkbox field +unset($parts["{$name}+search"]); +foreach ($parts as $name=>$F) { + if (substr($name, -7) == '+method') + // XXX: Hack + unset($F->ht['visibility']); +} +$form = new SimpleForm($parts); +foreach ($form->getFields() as $F) { ?> + <fieldset id="field<?php echo $F->getWidget()->id; + ?>" <?php + $class = array(); + @list($name, $sub) = explode('+', $F->get('name'), 2); + if (!$F->isVisible()) $class[] = "hidden"; + if ($sub === 'method') + $class[] = "adv-search-method"; + elseif ($F->get('__searchval__')) + $class[] = "adv-search-val"; + if ($class) + echo 'class="'.implode(' ', $class).'"'; + ?>> + <?php echo $F->render(); ?> + <?php foreach ($F->errors() as $E) { + ?><div class="error"><?php echo $E; ?></div><?php + } ?> + </fieldset> +<?php } ?> + + <div class="properties" style="margin-left: 25px; margin-top: 10px"> +<?php foreach ($condition->getProperties() as $prop=>$v) { + include 'queue-column-condition-prop.tmpl.php'; +} ?> + <div style="margin-top: 10px"> + <i class="icon-plus-sign"></i> + <select onchange="javascript: + var $this = $(this), + selected = $this.find(':selected'), + container = $this.closest('.properties'); + $.ajax({ + url: 'ajax.php/queue/condition/addProperty', + data: { prop: selected.val() }, + dataType: 'html', + success: function(html) { + $(html).insertBefore(container); + selected.prop('disabled', true); + } + }); + "> + <option>— <?php echo __('Add a property'); ?> —</option> +<?php foreach (array_keys(QueueColumnConditionProperty::$properties) as $p) { + echo sprintf('<option value="%s">%s</option>', $p, mb_convert_case($p, MB_CASE_TITLE)); +} ?> + </select> + </div> + </div> + </div> +</div> diff --git a/include/staff/templates/queue-column.tmpl.php b/include/staff/templates/queue-column.tmpl.php new file mode 100644 index 0000000000000000000000000000000000000000..a3b4b56dea408d0faecc7cfcc997e064791e0f46 --- /dev/null +++ b/include/staff/templates/queue-column.tmpl.php @@ -0,0 +1,123 @@ +<?php +/** + * Calling conventions + * + * $column - <QueueColumn> instance for this column + */ +$colid = $column->getId(); +$data_form = $column->getDataConfigForm($_POST); +?> +<ul class="alt tabs"> + <li class="active"><a href="#<?php echo $colid; ?>-data"><?php echo __('Data'); ?></a></li> + <li><a href="#<?php echo $colid; ?>-decorations"><?php echo __('Decorations'); ?></a></li> + <li><a href="#<?php echo $colid; ?>-conditions"><?php echo __('Conditions'); ?></a></li> +</ul> + +<div class="tab_content" id="<?php echo $colid; ?>-data"> +<?php + print $data_form->asTable(); +?> +</div> + +<div class="hidden tab_content" id="<?php echo $colid; ?>-decorations" style="max-width: 400px"> + <div class="empty placeholder" style="margin-left: 20px"> + <em><?php echo __('No decorations for this field'); ?></em> + </div> + <div style="margin: 20px;"> + <div class="decoration clear template hidden"> + <input data-field="input" data-name="decorations[]" value="" type="hidden" /> + <i data-field="icon"></i> + <span data-field="name"></span> + <div class="pull-right"> + <select data-field="position"> +<?php foreach (QueueDecoration::getPositions() as $key=>$desc) { + echo sprintf('<option value="%s">%s</option>', $key, Format::htmlchars($desc)); +} ?> + </select> + <a href="#" data-field="delete" title="<?php echo __('Delete'); ?>" + onclick="javascript: + $(this).closest('.decoration').remove(); + return false;"><i class="icon-trash"></i></a> + </div> + </div> + + <div style="margin-top: 20px"> + <i class="icon-plus-sign"></i> + <select class="add-decoration"> + <option>— <?php echo __("Add a decoration"); ?> —</option> +<?php foreach (CustomQueue::getDecorations('Ticket') as $class) { + echo sprintf('<option data-icon="%s" value="%s">%s</option>', + $class::$icon, $class, $class::getDescription()); + } ?> + </select> + </div> + + <script> + $(function() { + var addDecoration = function(type, icon, pos) { + var template = $('.decoration.template', '#<?php echo $colid; ?>-decorations'), + clone = template.clone().show().removeClass('template').insertBefore(template), + input = clone.find('[data-field=input]'), + name = clone.find('[data-field=name]'), + i = clone.find('[data-field=icon]'), + position = clone.find('[data-field=position]'); + input.attr('name', input.data('name')); + i.addClass('icon-fixed-width icon-' + icon); + name.text(type); + if (pos) position.val(pos); + template.parent().find('.empty').hide(); + }; + $('select.add-decoration', '#<?php echo $colid; ?>-decorations').change(function() { + var selected = $(this).find(':selected'); + addDecoration(selected.text(), selected.data('icon')); + selected.prop('disabled', true); + }); + $('#<?php echo $colid; ?>-decorations').click('a[data-field=delete]', + function() { + var tab = $('#<?php echo $colid; ?>-decorations'); + if ($('.decoration', tab).length === 0) + tab.find('.empty').show(); + }); + <?php foreach ($column->getDecorations() as $d) { + echo sprintf('addDecoration(%s, %s, %s);', + JsonDataEncoder::encode($d::getDescription()), + JsonDataEncoder::encode($d::getIcon()), + JsonDataEncoder::encode($d->getPosition()) + ); + } ?> + }); + </script> + </div> +</div> + +<div class="hidden tab_content" id="<?php echo $colid; ?>-conditions"> + <div style="margin: 0 20px"><?php echo __("Conditions are used to change the view of the data in a row based on some conditions of the data. For instance, a column might be shown bold if some condition is met."); + ?></div> + <div class="conditions" style="margin: 20px; max-width: 400px"> +<?php foreach ($column->getConditions() as $condition) { + include STAFFINC_DIR . 'templates/queue-column-condition.tmpl.php'; +} ?> + <div style="margin-top: 20px"> + <i class="icon-plus-sign"></i> + <select id="add-condition" onchange="javascript: + var $this = $(this), + container = $this.closest('div'); + $.ajax({ + url: 'ajax.php/queue/condition/add', + data: { field: $(this).find(':selected').val() }, + dataType: 'html', + success: function(html) { + $(html).insertBefore(container); + } + }); + "> + <option>— <?php echo __("Add a condition"); ?> —</option> +<?php + foreach (SavedSearch::getSearchableFields('Ticket') as $path=>$f) { + echo sprintf('<option value="%s">%s</option>', $path, Format::htmlchars($f->get('label'))); + } +?> + </select> + </div> + </div> +</div> diff --git a/include/staff/templates/queue-tickets.tmpl.php b/include/staff/templates/queue-tickets.tmpl.php new file mode 100644 index 0000000000000000000000000000000000000000..32d664d906b4fa0d6076b0dd888275ad8de3af8e --- /dev/null +++ b/include/staff/templates/queue-tickets.tmpl.php @@ -0,0 +1,40 @@ +<?php +// Calling convention (assumed global scope): +// $tickets - <QuerySet> with all columns and annotations necessary to +// render the full page +// $count - <int> number of records matching the search / filter part of the +// query + +$page = ($_GET['p'] && is_numeric($_GET['p']))?$_GET['p']:1; +$pageNav = new Pagenate($count, $page, PAGE_LIMIT); +$pageNav->setURL('tickets.php', $args); +$tickets = $pageNav->paginate($tickets); + +// Identify columns of output +$columns = $queue->getColumns(); + +?> +<table class="list" border="0" cellspacing="1" cellpadding="2" width="940"> + <thead> + <tr> +<?php +foreach ($columns as $C) { + echo sprintf('<th width="%s">%s</th>', $C->getWidth(), + Format::htmlchars($C->getHeading())); +} ?> + </tr> + </thead> + <tbody> +<?php +foreach ($tickets as $T) { + echo '<tr>'; + foreach ($columns as $C) { + echo "<td>"; + echo $C->render($T); + echo "</td>"; + } + echo '</tr>'; +} +?> + </tbody> +</table> diff --git a/include/staff/tickets.inc.php b/include/staff/tickets.inc.php index c786e28468ab527ec60b88aaf24040d65f1601f9..15881dce53da050a89bc66d7d40cea206d7d5452 100644 --- a/include/staff/tickets.inc.php +++ b/include/staff/tickets.inc.php @@ -229,7 +229,7 @@ if (!$view_all_tickets) { if (!$thisstaff->showAssignedOnly() && ($depts=$thisstaff->getDepts())) $visibility->add(array('dept_id__in' => $depts)); - $tickets->filter(Q::any($visibility)); + $tickets->filter($visibility); } // TODO :: Apply requested quick filter diff --git a/scp/ajax.php b/scp/ajax.php index de27157dc74093cf638db8a5a3e0550384e00857..1351b9fbc58c606076286c55bbc71cd8127be9a9 100644 --- a/scp/ajax.php +++ b/scp/ajax.php @@ -253,6 +253,12 @@ $dispatcher = patterns('', url('^/reset-permissions', 'resetPermissions'), url('^/change-department', 'changeDepartment'), url('^/(?P<id>\d+)/avatar/change', 'setAvatar') + )), + url('^/queue/', patterns('ajax.search.php:SearchAjaxAPI', + url('^(?P<id>\d+/)?preview$', 'previewQueue'), + url_get('^addColumn$', 'addColumn'), + url_get('^condition/add$', 'addCondition'), + url_get('^condition/addProperty$', 'addConditionProperty') )) ); diff --git a/scp/css/scp.css b/scp/css/scp.css index 86632ba2766ff353f499c2c13b1d4b8b60d68b16..8e5e10b70b92025fa6b9fb27c9fdf92aaad77e42 100644 --- a/scp/css/scp.css +++ b/scp/css/scp.css @@ -1709,7 +1709,7 @@ time.faq { width:100%; } -.dialog fieldset { +fieldset { margin:0; padding:0 0; border:none; @@ -1771,12 +1771,12 @@ time.faq { margin-top: 5px !important; } -#advanced-search fieldset { +.advanced-search fieldset { margin-top: 3px; position: relative; } -#advanced-search .adv-search-method:before, -#advanced-search .adv-search-val:before { +.advanced-search .adv-search-method:before, +.advanced-search .adv-search-val:before { content: ""; border-left: 2px dotted #ccc; border-bottom: 2px dotted #ccc; @@ -1787,10 +1787,10 @@ time.faq { position: absolute; left: -16px; } -#advanced-search .adv-search-method { +.advanced-search .adv-search-method { margin-left: 24px; } -#advanced-search .adv-search-val { +.advanced-search .adv-search-val { margin-left: 45px; } @@ -2876,7 +2876,9 @@ table.grid.form caption { .grid.form .field textarea, .grid.form .field select { width: 100%; + max-width: 100%; display: block; + box-sizing: border-box; } .grid.form .field > label { display: block; @@ -2921,6 +2923,49 @@ a.attachment { margin-bottom: 0.3em; } +#resizable-columns { + margin: 10px 0; + position: relative; +} +#resizable-columns .column-header:hover { + cursor: pointer; +} +#resizable-columns .column-header { + display: inline-block; + padding: 5px 20px; + background-color: #ddd; + margin: 0 1px; + position: relative; + text-align: center; + box-sizing: border-box; +} +#resizable-columns .column-header.ui-resizable:not(.active) { + opacity: 0.4; +} +#resizable-columns .column-header.ui-resizable.active { + background-color: #cfe6ff; +} + +.ui-resizable-handle { + cursor: pointer; + cursor: ew-resize; + cursor: col-resize; + display: inline-block; + vertical-align: bottom; + position: absolute; + right: 5px; + color: #777; +} +.decoration + .decoration { + margin-top: 10px; +} +.advanced-search .condition-property { + margin: 7px 0 7px 25px; +} +.conditions .condition + .condition { + margin-top: 10px; +} + /* FIXME: Drop this with select2 4.0.1 * Fixes a rendering issue on Safari */ diff --git a/scp/css/spectrum.css b/scp/css/spectrum.css new file mode 100644 index 0000000000000000000000000000000000000000..ecf6fe482c01ede9de6e83edf4aaa6dbb06ce86c --- /dev/null +++ b/scp/css/spectrum.css @@ -0,0 +1,507 @@ +/*** +Spectrum Colorpicker v1.7.1 +https://github.com/bgrins/spectrum +Author: Brian Grinstead +License: MIT +***/ + +.sp-container { + position:absolute; + top:0; + left:0; + display:inline-block; + *display: inline; + *zoom: 1; + /* https://github.com/bgrins/spectrum/issues/40 */ + z-index: 9999994; + overflow: hidden; +} +.sp-container.sp-flat { + position: relative; +} + +/* Fix for * { box-sizing: border-box; } */ +.sp-container, +.sp-container * { + -webkit-box-sizing: content-box; + -moz-box-sizing: content-box; + box-sizing: content-box; +} + +/* http://ansciath.tumblr.com/post/7347495869/css-aspect-ratio */ +.sp-top { + position:relative; + width: 100%; + display:inline-block; +} +.sp-top-inner { + position:absolute; + top:0; + left:0; + bottom:0; + right:0; +} +.sp-color { + position: absolute; + top:0; + left:0; + bottom:0; + right:20%; +} +.sp-hue { + position: absolute; + top:0; + right:0; + bottom:0; + left:84%; + height: 100%; +} + +.sp-clear-enabled .sp-hue { + top:33px; + height: 77.5%; +} + +.sp-fill { + padding-top: 80%; +} +.sp-sat, .sp-val { + position: absolute; + top:0; + left:0; + right:0; + bottom:0; +} + +.sp-alpha-enabled .sp-top { + margin-bottom: 18px; +} +.sp-alpha-enabled .sp-alpha { + display: block; +} +.sp-alpha-handle { + position:absolute; + top:-4px; + bottom: -4px; + width: 6px; + left: 50%; + cursor: pointer; + border: 1px solid black; + background: white; + opacity: .8; +} +.sp-alpha { + display: none; + position: absolute; + bottom: -14px; + right: 0; + left: 0; + height: 8px; +} +.sp-alpha-inner { + border: solid 1px #333; +} + +.sp-clear { + display: none; +} + +.sp-clear.sp-clear-display { + background-position: center; +} + +.sp-clear-enabled .sp-clear { + display: block; + position:absolute; + top:0px; + right:0; + bottom:0; + left:84%; + height: 28px; +} + +/* Don't allow text selection */ +.sp-container, .sp-replacer, .sp-preview, .sp-dragger, .sp-slider, .sp-alpha, .sp-clear, .sp-alpha-handle, .sp-container.sp-dragging .sp-input, .sp-container button { + -webkit-user-select:none; + -moz-user-select: -moz-none; + -o-user-select:none; + user-select: none; +} + +.sp-container.sp-input-disabled .sp-input-container { + display: none; +} +.sp-container.sp-buttons-disabled .sp-button-container { + display: none; +} +.sp-container.sp-palette-buttons-disabled .sp-palette-button-container { + display: none; +} +.sp-palette-only .sp-picker-container { + display: none; +} +.sp-palette-disabled .sp-palette-container { + display: none; +} + +.sp-initial-disabled .sp-initial { + display: none; +} + + +/* Gradients for hue, saturation and value instead of images. Not pretty... but it works */ +.sp-sat { + background-image: -webkit-gradient(linear, 0 0, 100% 0, from(#FFF), to(rgba(204, 154, 129, 0))); + background-image: -webkit-linear-gradient(left, #FFF, rgba(204, 154, 129, 0)); + background-image: -moz-linear-gradient(left, #fff, rgba(204, 154, 129, 0)); + background-image: -o-linear-gradient(left, #fff, rgba(204, 154, 129, 0)); + background-image: -ms-linear-gradient(left, #fff, rgba(204, 154, 129, 0)); + background-image: linear-gradient(to right, #fff, rgba(204, 154, 129, 0)); + -ms-filter: "progid:DXImageTransform.Microsoft.gradient(GradientType = 1, startColorstr=#FFFFFFFF, endColorstr=#00CC9A81)"; + filter : progid:DXImageTransform.Microsoft.gradient(GradientType = 1, startColorstr='#FFFFFFFF', endColorstr='#00CC9A81'); +} +.sp-val { + background-image: -webkit-gradient(linear, 0 100%, 0 0, from(#000000), to(rgba(204, 154, 129, 0))); + background-image: -webkit-linear-gradient(bottom, #000000, rgba(204, 154, 129, 0)); + background-image: -moz-linear-gradient(bottom, #000, rgba(204, 154, 129, 0)); + background-image: -o-linear-gradient(bottom, #000, rgba(204, 154, 129, 0)); + background-image: -ms-linear-gradient(bottom, #000, rgba(204, 154, 129, 0)); + background-image: linear-gradient(to top, #000, rgba(204, 154, 129, 0)); + -ms-filter: "progid:DXImageTransform.Microsoft.gradient(startColorstr=#00CC9A81, endColorstr=#FF000000)"; + filter : progid:DXImageTransform.Microsoft.gradient(startColorstr='#00CC9A81', endColorstr='#FF000000'); +} + +.sp-hue { + background: -moz-linear-gradient(top, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%); + background: -ms-linear-gradient(top, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%); + background: -o-linear-gradient(top, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%); + background: -webkit-gradient(linear, left top, left bottom, from(#ff0000), color-stop(0.17, #ffff00), color-stop(0.33, #00ff00), color-stop(0.5, #00ffff), color-stop(0.67, #0000ff), color-stop(0.83, #ff00ff), to(#ff0000)); + background: -webkit-linear-gradient(top, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%); + background: linear-gradient(to bottom, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%); +} + +/* IE filters do not support multiple color stops. + Generate 6 divs, line them up, and do two color gradients for each. + Yes, really. + */ +.sp-1 { + height:17%; + filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff0000', endColorstr='#ffff00'); +} +.sp-2 { + height:16%; + filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffff00', endColorstr='#00ff00'); +} +.sp-3 { + height:17%; + filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#00ff00', endColorstr='#00ffff'); +} +.sp-4 { + height:17%; + filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#00ffff', endColorstr='#0000ff'); +} +.sp-5 { + height:16%; + filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#0000ff', endColorstr='#ff00ff'); +} +.sp-6 { + height:17%; + filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff00ff', endColorstr='#ff0000'); +} + +.sp-hidden { + display: none !important; +} + +/* Clearfix hack */ +.sp-cf:before, .sp-cf:after { content: ""; display: table; } +.sp-cf:after { clear: both; } +.sp-cf { *zoom: 1; } + +/* Mobile devices, make hue slider bigger so it is easier to slide */ +@media (max-device-width: 480px) { + .sp-color { right: 40%; } + .sp-hue { left: 63%; } + .sp-fill { padding-top: 60%; } +} +.sp-dragger { + border-radius: 5px; + height: 5px; + width: 5px; + border: 1px solid #fff; + background: #000; + cursor: pointer; + position:absolute; + top:0; + left: 0; +} +.sp-slider { + position: absolute; + top:0; + cursor:pointer; + height: 3px; + left: -1px; + right: -1px; + border: 1px solid #000; + background: white; + opacity: .8; +} + +/* +Theme authors: +Here are the basic themeable display options (colors, fonts, global widths). +See http://bgrins.github.io/spectrum/themes/ for instructions. +*/ + +.sp-container { + border-radius: 0; + background-color: #ECECEC; + border: solid 1px #f0c49B; + padding: 0; +} +.sp-container, .sp-container button, .sp-container input, .sp-color, .sp-hue, .sp-clear { + font: normal 12px "Lucida Grande", "Lucida Sans Unicode", "Lucida Sans", Geneva, Verdana, sans-serif; + -webkit-box-sizing: border-box; + -moz-box-sizing: border-box; + -ms-box-sizing: border-box; + box-sizing: border-box; +} +.sp-top { + margin-bottom: 3px; +} +.sp-color, .sp-hue, .sp-clear { + border: solid 1px #666; +} + +/* Input */ +.sp-input-container { + float:right; + width: 100px; + margin-bottom: 4px; +} +.sp-initial-disabled .sp-input-container { + width: 100%; +} +.sp-input { + font-size: 12px !important; + border: 1px inset; + padding: 4px 5px; + margin: 0; + width: 100%; + background:transparent; + border-radius: 3px; + color: #222; +} +.sp-input:focus { + border: 1px solid orange; +} +.sp-input.sp-validation-error { + border: 1px solid red; + background: #fdd; +} +.sp-picker-container , .sp-palette-container { + float:left; + position: relative; + padding: 10px; + padding-bottom: 300px; + margin-bottom: -290px; +} +.sp-picker-container { + width: 172px; + border-left: solid 1px #fff; +} + +/* Palettes */ +.sp-palette-container { + border-right: solid 1px #ccc; +} + +.sp-palette-only .sp-palette-container { + border: 0; +} + +.sp-palette .sp-thumb-el { + display: block; + position:relative; + float:left; + width: 24px; + height: 15px; + margin: 3px; + cursor: pointer; + border:solid 2px transparent; +} +.sp-palette .sp-thumb-el:hover, .sp-palette .sp-thumb-el.sp-thumb-active { + border-color: orange; +} +.sp-thumb-el { + position:relative; +} + +/* Initial */ +.sp-initial { + float: left; + border: solid 1px #333; +} +.sp-initial span { + width: 30px; + height: 25px; + border:none; + display:block; + float:left; + margin:0; +} + +.sp-initial .sp-clear-display { + background-position: center; +} + +/* Buttons */ +.sp-palette-button-container, +.sp-button-container { + float: right; +} + +/* Replacer (the little preview div that shows up instead of the <input>) */ +.sp-replacer { + margin:0; + overflow:hidden; + cursor:pointer; + padding: 4px; + display:inline-block; + *zoom: 1; + *display: inline; + border: solid 1px #91765d; + background: #eee; + color: #333; + vertical-align: middle; +} +.sp-replacer:hover, .sp-replacer.sp-active { + border-color: #F0C49B; + color: #111; +} +.sp-replacer.sp-disabled { + cursor:default; + border-color: silver; + color: silver; +} +.sp-dd { + padding: 2px 0; + height: 16px; + line-height: 16px; + float:left; + font-size:10px; +} +.sp-preview { + position:relative; + width:25px; + height: 20px; + border: solid 1px #222; + margin-right: 5px; + float:left; + z-index: 0; +} + +.sp-palette { + *width: 220px; + max-width: 220px; +} +.sp-palette .sp-thumb-el { + width:16px; + height: 16px; + margin:2px 1px; + border: solid 1px #d0d0d0; +} + +.sp-container { + padding-bottom:0; +} + + +/* Buttons: http://hellohappy.org/css3-buttons/ */ +.sp-container button { + background-color: #eeeeee; + background-image: -webkit-linear-gradient(top, #eeeeee, #cccccc); + background-image: -moz-linear-gradient(top, #eeeeee, #cccccc); + background-image: -ms-linear-gradient(top, #eeeeee, #cccccc); + background-image: -o-linear-gradient(top, #eeeeee, #cccccc); + background-image: linear-gradient(to bottom, #eeeeee, #cccccc); + border: 1px solid #ccc; + border-bottom: 1px solid #bbb; + border-radius: 3px; + color: #333; + font-size: 14px; + line-height: 1; + padding: 5px 4px; + text-align: center; + text-shadow: 0 1px 0 #eee; + vertical-align: middle; +} +.sp-container button:hover { + background-color: #dddddd; + background-image: -webkit-linear-gradient(top, #dddddd, #bbbbbb); + background-image: -moz-linear-gradient(top, #dddddd, #bbbbbb); + background-image: -ms-linear-gradient(top, #dddddd, #bbbbbb); + background-image: -o-linear-gradient(top, #dddddd, #bbbbbb); + background-image: linear-gradient(to bottom, #dddddd, #bbbbbb); + border: 1px solid #bbb; + border-bottom: 1px solid #999; + cursor: pointer; + text-shadow: 0 1px 0 #ddd; +} +.sp-container button:active { + border: 1px solid #aaa; + border-bottom: 1px solid #888; + -webkit-box-shadow: inset 0 0 5px 2px #aaaaaa, 0 1px 0 0 #eeeeee; + -moz-box-shadow: inset 0 0 5px 2px #aaaaaa, 0 1px 0 0 #eeeeee; + -ms-box-shadow: inset 0 0 5px 2px #aaaaaa, 0 1px 0 0 #eeeeee; + -o-box-shadow: inset 0 0 5px 2px #aaaaaa, 0 1px 0 0 #eeeeee; + box-shadow: inset 0 0 5px 2px #aaaaaa, 0 1px 0 0 #eeeeee; +} +.sp-cancel { + font-size: 11px; + color: #d93f3f !important; + margin:0; + padding:2px; + margin-right: 5px; + vertical-align: middle; + text-decoration:none; + +} +.sp-cancel:hover { + color: #d93f3f !important; + text-decoration: underline; +} + + +.sp-palette span:hover, .sp-palette span.sp-thumb-active { + border-color: #000; +} + +.sp-preview, .sp-alpha, .sp-thumb-el { + position:relative; + background-image: url(); +} +.sp-preview-inner, .sp-alpha-inner, .sp-thumb-inner { + display:block; + position:absolute; + top:0;left:0;bottom:0;right:0; +} + +.sp-palette .sp-thumb-inner { + background-position: 50% 50%; + background-repeat: no-repeat; +} + +.sp-palette .sp-thumb-light.sp-thumb-active .sp-thumb-inner { + background-image: url(); +} + +.sp-palette .sp-thumb-dark.sp-thumb-active .sp-thumb-inner { + background-image: url(); +} + +.sp-clear-display { + background-repeat:no-repeat; + background-position: center; + background-image: url(); +} diff --git a/scp/js/scp.js b/scp/js/scp.js index c825c051cc64764435c33debafd976b3c0c53324..82e2cd81cf5c3c117402c661f0d64ca782a3956c 100644 --- a/scp/js/scp.js +++ b/scp/js/scp.js @@ -945,7 +945,7 @@ $(document).on('click.tab', 'ul.tabs > li > a', function(e) { $ul.children('li.active').removeClass('active'); $(this).closest('li').addClass('active'); $container.children('.tab_content').hide(); - $tab.fadeIn('fast'); + $tab.fadeIn('fast').show(); return false; } @@ -1228,3 +1228,51 @@ window.relativeAdjust = setInterval(function() { }); }, 20000); +// Add 'afterShow' event to jQuery elements, +// thanks http://stackoverflow.com/a/1225238/1025836 +(function ($) { + var _oldShow = $.fn.show; + + $.fn.show = function (/*speed, easing, callback*/) { + var argsArray = Array.prototype.slice.call(arguments), + duration = argsArray[0], + easing, + callback, + callbackArgIndex; + + // jQuery recursively calls show sometimes; we shouldn't + // handle such situations. Pass it to original show method. + if (!this.selector) { + _oldShow.apply(this, argsArray); + return this; + } + + if (argsArray.length === 2) { + if ($.isFunction(argsArray[1])) { + callback = argsArray[1]; + callbackArgIndex = 1; + } else { + easing = argsArray[1]; + } + } else if (argsArray.length === 3) { + easing = argsArray[1]; + callback = argsArray[2]; + callbackArgIndex = 2; + } + return $(this).each(function () { + var obj = $(this), + oldCallback = callback, + newCallback = function () { + if ($.isFunction(oldCallback)) { + oldCallback.apply(obj); + } + }; + if (callback) { + argsArray[callbackArgIndex] = newCallback; + } + obj.trigger('beforeShow'); + _oldShow.apply(obj, argsArray); + obj.trigger('afterShow'); + }); + }; +})(jQuery); diff --git a/scp/js/spectrum.js b/scp/js/spectrum.js new file mode 100644 index 0000000000000000000000000000000000000000..39ffcd7e5c5c9db1942d62e7d4a915c1530cff44 --- /dev/null +++ b/scp/js/spectrum.js @@ -0,0 +1,2317 @@ +// Spectrum Colorpicker v1.7.1 +// https://github.com/bgrins/spectrum +// Author: Brian Grinstead +// License: MIT + +(function (factory) { + "use strict"; + + if (typeof define === 'function' && define.amd) { // AMD + define(['jquery'], factory); + } + else if (typeof exports == "object" && typeof module == "object") { // CommonJS + module.exports = factory; + } + else { // Browser + factory(jQuery); + } +})(function($, undefined) { + "use strict"; + + var defaultOpts = { + + // Callbacks + beforeShow: noop, + move: noop, + change: noop, + show: noop, + hide: noop, + + // Options + color: false, + flat: false, + showInput: false, + allowEmpty: false, + showButtons: true, + clickoutFiresChange: true, + showInitial: false, + showPalette: false, + showPaletteOnly: false, + hideAfterPaletteSelect: false, + togglePaletteOnly: false, + showSelectionPalette: true, + localStorageKey: false, + appendTo: "body", + maxSelectionSize: 7, + cancelText: "cancel", + chooseText: "choose", + togglePaletteMoreText: "more", + togglePaletteLessText: "less", + clearText: "Clear Color Selection", + noColorSelectedText: "No Color Selected", + preferredFormat: false, + className: "", // Deprecated - use containerClassName and replacerClassName instead. + containerClassName: "", + replacerClassName: "", + showAlpha: false, + theme: "sp-light", + palette: [["#ffffff", "#000000", "#ff0000", "#ff8000", "#ffff00", "#008000", "#0000ff", "#4b0082", "#9400d3"]], + selectionPalette: [], + disabled: false, + offset: null + }, + spectrums = [], + IE = !!/msie/i.exec( window.navigator.userAgent ), + rgbaSupport = (function() { + function contains( str, substr ) { + return !!~('' + str).indexOf(substr); + } + + var elem = document.createElement('div'); + var style = elem.style; + style.cssText = 'background-color:rgba(0,0,0,.5)'; + return contains(style.backgroundColor, 'rgba') || contains(style.backgroundColor, 'hsla'); + })(), + replaceInput = [ + "<div class='sp-replacer'>", + "<div class='sp-preview'><div class='sp-preview-inner'></div></div>", + "<div class='sp-dd'>▼</div>", + "</div>" + ].join(''), + markup = (function () { + + // IE does not support gradients with multiple stops, so we need to simulate + // that for the rainbow slider with 8 divs that each have a single gradient + var gradientFix = ""; + if (IE) { + for (var i = 1; i <= 6; i++) { + gradientFix += "<div class='sp-" + i + "'></div>"; + } + } + + return [ + "<div class='sp-container sp-hidden'>", + "<div class='sp-palette-container'>", + "<div class='sp-palette sp-thumb sp-cf'></div>", + "<div class='sp-palette-button-container sp-cf'>", + "<button type='button' class='sp-palette-toggle'></button>", + "</div>", + "</div>", + "<div class='sp-picker-container'>", + "<div class='sp-top sp-cf'>", + "<div class='sp-fill'></div>", + "<div class='sp-top-inner'>", + "<div class='sp-color'>", + "<div class='sp-sat'>", + "<div class='sp-val'>", + "<div class='sp-dragger'></div>", + "</div>", + "</div>", + "</div>", + "<div class='sp-clear sp-clear-display'>", + "</div>", + "<div class='sp-hue'>", + "<div class='sp-slider'></div>", + gradientFix, + "</div>", + "</div>", + "<div class='sp-alpha'><div class='sp-alpha-inner'><div class='sp-alpha-handle'></div></div></div>", + "</div>", + "<div class='sp-input-container sp-cf'>", + "<input class='sp-input' type='text' spellcheck='false' />", + "</div>", + "<div class='sp-initial sp-thumb sp-cf'></div>", + "<div class='sp-button-container sp-cf'>", + "<a class='sp-cancel' href='#'></a>", + "<button type='button' class='sp-choose'></button>", + "</div>", + "</div>", + "</div>" + ].join(""); + })(); + + function paletteTemplate (p, color, className, opts) { + var html = []; + for (var i = 0; i < p.length; i++) { + var current = p[i]; + if(current) { + var tiny = tinycolor(current); + var c = tiny.toHsl().l < 0.5 ? "sp-thumb-el sp-thumb-dark" : "sp-thumb-el sp-thumb-light"; + c += (tinycolor.equals(color, current)) ? " sp-thumb-active" : ""; + var formattedString = tiny.toString(opts.preferredFormat || "rgb"); + var swatchStyle = rgbaSupport ? ("background-color:" + tiny.toRgbString()) : "filter:" + tiny.toFilter(); + html.push('<span title="' + formattedString + '" data-color="' + tiny.toRgbString() + '" class="' + c + '"><span class="sp-thumb-inner" style="' + swatchStyle + ';" /></span>'); + } else { + var cls = 'sp-clear-display'; + html.push($('<div />') + .append($('<span data-color="" style="background-color:transparent;" class="' + cls + '"></span>') + .attr('title', opts.noColorSelectedText) + ) + .html() + ); + } + } + return "<div class='sp-cf " + className + "'>" + html.join('') + "</div>"; + } + + function hideAll() { + for (var i = 0; i < spectrums.length; i++) { + if (spectrums[i]) { + spectrums[i].hide(); + } + } + } + + function instanceOptions(o, callbackContext) { + var opts = $.extend({}, defaultOpts, o); + opts.callbacks = { + 'move': bind(opts.move, callbackContext), + 'change': bind(opts.change, callbackContext), + 'show': bind(opts.show, callbackContext), + 'hide': bind(opts.hide, callbackContext), + 'beforeShow': bind(opts.beforeShow, callbackContext) + }; + + return opts; + } + + function spectrum(element, o) { + + var opts = instanceOptions(o, element), + flat = opts.flat, + showSelectionPalette = opts.showSelectionPalette, + localStorageKey = opts.localStorageKey, + theme = opts.theme, + callbacks = opts.callbacks, + resize = throttle(reflow, 10), + visible = false, + isDragging = false, + dragWidth = 0, + dragHeight = 0, + dragHelperHeight = 0, + slideHeight = 0, + slideWidth = 0, + alphaWidth = 0, + alphaSlideHelperWidth = 0, + slideHelperHeight = 0, + currentHue = 0, + currentSaturation = 0, + currentValue = 0, + currentAlpha = 1, + palette = [], + paletteArray = [], + paletteLookup = {}, + selectionPalette = opts.selectionPalette.slice(0), + maxSelectionSize = opts.maxSelectionSize, + draggingClass = "sp-dragging", + shiftMovementDirection = null; + + var doc = element.ownerDocument, + body = doc.body, + boundElement = $(element), + disabled = false, + container = $(markup, doc).addClass(theme), + pickerContainer = container.find(".sp-picker-container"), + dragger = container.find(".sp-color"), + dragHelper = container.find(".sp-dragger"), + slider = container.find(".sp-hue"), + slideHelper = container.find(".sp-slider"), + alphaSliderInner = container.find(".sp-alpha-inner"), + alphaSlider = container.find(".sp-alpha"), + alphaSlideHelper = container.find(".sp-alpha-handle"), + textInput = container.find(".sp-input"), + paletteContainer = container.find(".sp-palette"), + initialColorContainer = container.find(".sp-initial"), + cancelButton = container.find(".sp-cancel"), + clearButton = container.find(".sp-clear"), + chooseButton = container.find(".sp-choose"), + toggleButton = container.find(".sp-palette-toggle"), + isInput = boundElement.is("input"), + isInputTypeColor = isInput && boundElement.attr("type") === "color" && inputTypeColorSupport(), + shouldReplace = isInput && !flat, + replacer = (shouldReplace) ? $(replaceInput).addClass(theme).addClass(opts.className).addClass(opts.replacerClassName) : $([]), + offsetElement = (shouldReplace) ? replacer : boundElement, + previewElement = replacer.find(".sp-preview-inner"), + initialColor = opts.color || (isInput && boundElement.val()), + colorOnShow = false, + preferredFormat = opts.preferredFormat, + currentPreferredFormat = preferredFormat, + clickoutFiresChange = !opts.showButtons || opts.clickoutFiresChange, + isEmpty = !initialColor, + allowEmpty = opts.allowEmpty && !isInputTypeColor; + + function applyOptions() { + + if (opts.showPaletteOnly) { + opts.showPalette = true; + } + + toggleButton.text(opts.showPaletteOnly ? opts.togglePaletteMoreText : opts.togglePaletteLessText); + + if (opts.palette) { + palette = opts.palette.slice(0); + paletteArray = $.isArray(palette[0]) ? palette : [palette]; + paletteLookup = {}; + for (var i = 0; i < paletteArray.length; i++) { + for (var j = 0; j < paletteArray[i].length; j++) { + var rgb = tinycolor(paletteArray[i][j]).toRgbString(); + paletteLookup[rgb] = true; + } + } + } + + container.toggleClass("sp-flat", flat); + container.toggleClass("sp-input-disabled", !opts.showInput); + container.toggleClass("sp-alpha-enabled", opts.showAlpha); + container.toggleClass("sp-clear-enabled", allowEmpty); + container.toggleClass("sp-buttons-disabled", !opts.showButtons); + container.toggleClass("sp-palette-buttons-disabled", !opts.togglePaletteOnly); + container.toggleClass("sp-palette-disabled", !opts.showPalette); + container.toggleClass("sp-palette-only", opts.showPaletteOnly); + container.toggleClass("sp-initial-disabled", !opts.showInitial); + container.addClass(opts.className).addClass(opts.containerClassName); + + reflow(); + } + + function initialize() { + + if (IE) { + container.find("*:not(input)").attr("unselectable", "on"); + } + + applyOptions(); + + if (shouldReplace) { + boundElement.after(replacer).hide(); + } + + if (!allowEmpty) { + clearButton.hide(); + } + + if (flat) { + boundElement.after(container).hide(); + } + else { + + var appendTo = opts.appendTo === "parent" ? boundElement.parent() : $(opts.appendTo); + if (appendTo.length !== 1) { + appendTo = $("body"); + } + + appendTo.append(container); + } + + updateSelectionPaletteFromStorage(); + + offsetElement.bind("click.spectrum touchstart.spectrum", function (e) { + if (!disabled) { + toggle(); + } + + e.stopPropagation(); + + if (!$(e.target).is("input")) { + e.preventDefault(); + } + }); + + if(boundElement.is(":disabled") || (opts.disabled === true)) { + disable(); + } + + // Prevent clicks from bubbling up to document. This would cause it to be hidden. + container.click(stopPropagation); + + // Handle user typed input + textInput.change(setFromTextInput); + textInput.bind("paste", function () { + setTimeout(setFromTextInput, 1); + }); + textInput.keydown(function (e) { if (e.keyCode == 13) { setFromTextInput(); } }); + + cancelButton.text(opts.cancelText); + cancelButton.bind("click.spectrum", function (e) { + e.stopPropagation(); + e.preventDefault(); + revert(); + hide(); + }); + + clearButton.attr("title", opts.clearText); + clearButton.bind("click.spectrum", function (e) { + e.stopPropagation(); + e.preventDefault(); + isEmpty = true; + move(); + + if(flat) { + //for the flat style, this is a change event + updateOriginalInput(true); + } + }); + + chooseButton.text(opts.chooseText); + chooseButton.bind("click.spectrum", function (e) { + e.stopPropagation(); + e.preventDefault(); + + if (IE && textInput.is(":focus")) { + textInput.trigger('change'); + } + + if (isValid()) { + updateOriginalInput(true); + hide(); + } + }); + + toggleButton.text(opts.showPaletteOnly ? opts.togglePaletteMoreText : opts.togglePaletteLessText); + toggleButton.bind("click.spectrum", function (e) { + e.stopPropagation(); + e.preventDefault(); + + opts.showPaletteOnly = !opts.showPaletteOnly; + + // To make sure the Picker area is drawn on the right, next to the + // Palette area (and not below the palette), first move the Palette + // to the left to make space for the picker, plus 5px extra. + // The 'applyOptions' function puts the whole container back into place + // and takes care of the button-text and the sp-palette-only CSS class. + if (!opts.showPaletteOnly && !flat) { + container.css('left', '-=' + (pickerContainer.outerWidth(true) + 5)); + } + applyOptions(); + }); + + draggable(alphaSlider, function (dragX, dragY, e) { + currentAlpha = (dragX / alphaWidth); + isEmpty = false; + if (e.shiftKey) { + currentAlpha = Math.round(currentAlpha * 10) / 10; + } + + move(); + }, dragStart, dragStop); + + draggable(slider, function (dragX, dragY) { + currentHue = parseFloat(dragY / slideHeight); + isEmpty = false; + if (!opts.showAlpha) { + currentAlpha = 1; + } + move(); + }, dragStart, dragStop); + + draggable(dragger, function (dragX, dragY, e) { + + // shift+drag should snap the movement to either the x or y axis. + if (!e.shiftKey) { + shiftMovementDirection = null; + } + else if (!shiftMovementDirection) { + var oldDragX = currentSaturation * dragWidth; + var oldDragY = dragHeight - (currentValue * dragHeight); + var furtherFromX = Math.abs(dragX - oldDragX) > Math.abs(dragY - oldDragY); + + shiftMovementDirection = furtherFromX ? "x" : "y"; + } + + var setSaturation = !shiftMovementDirection || shiftMovementDirection === "x"; + var setValue = !shiftMovementDirection || shiftMovementDirection === "y"; + + if (setSaturation) { + currentSaturation = parseFloat(dragX / dragWidth); + } + if (setValue) { + currentValue = parseFloat((dragHeight - dragY) / dragHeight); + } + + isEmpty = false; + if (!opts.showAlpha) { + currentAlpha = 1; + } + + move(); + + }, dragStart, dragStop); + + if (!!initialColor) { + set(initialColor); + + // In case color was black - update the preview UI and set the format + // since the set function will not run (default color is black). + updateUI(); + currentPreferredFormat = preferredFormat || tinycolor(initialColor).format; + + addColorToSelectionPalette(initialColor); + } + else { + updateUI(); + } + + if (flat) { + show(); + } + + function paletteElementClick(e) { + if (e.data && e.data.ignore) { + set($(e.target).closest(".sp-thumb-el").data("color")); + move(); + } + else { + set($(e.target).closest(".sp-thumb-el").data("color")); + move(); + updateOriginalInput(true); + if (opts.hideAfterPaletteSelect) { + hide(); + } + } + + return false; + } + + var paletteEvent = IE ? "mousedown.spectrum" : "click.spectrum touchstart.spectrum"; + paletteContainer.delegate(".sp-thumb-el", paletteEvent, paletteElementClick); + initialColorContainer.delegate(".sp-thumb-el:nth-child(1)", paletteEvent, { ignore: true }, paletteElementClick); + } + + function updateSelectionPaletteFromStorage() { + + if (localStorageKey && window.localStorage) { + + // Migrate old palettes over to new format. May want to remove this eventually. + try { + var oldPalette = window.localStorage[localStorageKey].split(",#"); + if (oldPalette.length > 1) { + delete window.localStorage[localStorageKey]; + $.each(oldPalette, function(i, c) { + addColorToSelectionPalette(c); + }); + } + } + catch(e) { } + + try { + selectionPalette = window.localStorage[localStorageKey].split(";"); + } + catch (e) { } + } + } + + function addColorToSelectionPalette(color) { + if (showSelectionPalette) { + var rgb = tinycolor(color).toRgbString(); + if (!paletteLookup[rgb] && $.inArray(rgb, selectionPalette) === -1) { + selectionPalette.push(rgb); + while(selectionPalette.length > maxSelectionSize) { + selectionPalette.shift(); + } + } + + if (localStorageKey && window.localStorage) { + try { + window.localStorage[localStorageKey] = selectionPalette.join(";"); + } + catch(e) { } + } + } + } + + function getUniqueSelectionPalette() { + var unique = []; + if (opts.showPalette) { + for (var i = 0; i < selectionPalette.length; i++) { + var rgb = tinycolor(selectionPalette[i]).toRgbString(); + + if (!paletteLookup[rgb]) { + unique.push(selectionPalette[i]); + } + } + } + + return unique.reverse().slice(0, opts.maxSelectionSize); + } + + function drawPalette() { + + var currentColor = get(); + + var html = $.map(paletteArray, function (palette, i) { + return paletteTemplate(palette, currentColor, "sp-palette-row sp-palette-row-" + i, opts); + }); + + updateSelectionPaletteFromStorage(); + + if (selectionPalette) { + html.push(paletteTemplate(getUniqueSelectionPalette(), currentColor, "sp-palette-row sp-palette-row-selection", opts)); + } + + paletteContainer.html(html.join("")); + } + + function drawInitial() { + if (opts.showInitial) { + var initial = colorOnShow; + var current = get(); + initialColorContainer.html(paletteTemplate([initial, current], current, "sp-palette-row-initial", opts)); + } + } + + function dragStart() { + if (dragHeight <= 0 || dragWidth <= 0 || slideHeight <= 0) { + reflow(); + } + isDragging = true; + container.addClass(draggingClass); + shiftMovementDirection = null; + boundElement.trigger('dragstart.spectrum', [ get() ]); + } + + function dragStop() { + isDragging = false; + container.removeClass(draggingClass); + boundElement.trigger('dragstop.spectrum', [ get() ]); + } + + function setFromTextInput() { + + var value = textInput.val(); + + if ((value === null || value === "") && allowEmpty) { + set(null); + updateOriginalInput(true); + } + else { + var tiny = tinycolor(value); + if (tiny.isValid()) { + set(tiny); + updateOriginalInput(true); + } + else { + textInput.addClass("sp-validation-error"); + } + } + } + + function toggle() { + if (visible) { + hide(); + } + else { + show(); + } + } + + function show() { + var event = $.Event('beforeShow.spectrum'); + + if (visible) { + reflow(); + return; + } + + boundElement.trigger(event, [ get() ]); + + if (callbacks.beforeShow(get()) === false || event.isDefaultPrevented()) { + return; + } + + hideAll(); + visible = true; + + $(doc).bind("keydown.spectrum", onkeydown); + $(doc).bind("click.spectrum", clickout); + $(window).bind("resize.spectrum", resize); + replacer.addClass("sp-active"); + container.removeClass("sp-hidden"); + + reflow(); + updateUI(); + + colorOnShow = get(); + + drawInitial(); + callbacks.show(colorOnShow); + boundElement.trigger('show.spectrum', [ colorOnShow ]); + } + + function onkeydown(e) { + // Close on ESC + if (e.keyCode === 27) { + hide(); + } + } + + function clickout(e) { + // Return on right click. + if (e.button == 2) { return; } + + // If a drag event was happening during the mouseup, don't hide + // on click. + if (isDragging) { return; } + + if (clickoutFiresChange) { + updateOriginalInput(true); + } + else { + revert(); + } + hide(); + } + + function hide() { + // Return if hiding is unnecessary + if (!visible || flat) { return; } + visible = false; + + $(doc).unbind("keydown.spectrum", onkeydown); + $(doc).unbind("click.spectrum", clickout); + $(window).unbind("resize.spectrum", resize); + + replacer.removeClass("sp-active"); + container.addClass("sp-hidden"); + + callbacks.hide(get()); + boundElement.trigger('hide.spectrum', [ get() ]); + } + + function revert() { + set(colorOnShow, true); + } + + function set(color, ignoreFormatChange) { + if (tinycolor.equals(color, get())) { + // Update UI just in case a validation error needs + // to be cleared. + updateUI(); + return; + } + + var newColor, newHsv; + if (!color && allowEmpty) { + isEmpty = true; + } else { + isEmpty = false; + newColor = tinycolor(color); + newHsv = newColor.toHsv(); + + currentHue = (newHsv.h % 360) / 360; + currentSaturation = newHsv.s; + currentValue = newHsv.v; + currentAlpha = newHsv.a; + } + updateUI(); + + if (newColor && newColor.isValid() && !ignoreFormatChange) { + currentPreferredFormat = preferredFormat || newColor.getFormat(); + } + } + + function get(opts) { + opts = opts || { }; + + if (allowEmpty && isEmpty) { + return null; + } + + return tinycolor.fromRatio({ + h: currentHue, + s: currentSaturation, + v: currentValue, + a: Math.round(currentAlpha * 100) / 100 + }, { format: opts.format || currentPreferredFormat }); + } + + function isValid() { + return !textInput.hasClass("sp-validation-error"); + } + + function move() { + updateUI(); + + callbacks.move(get()); + boundElement.trigger('move.spectrum', [ get() ]); + } + + function updateUI() { + + textInput.removeClass("sp-validation-error"); + + updateHelperLocations(); + + // Update dragger background color (gradients take care of saturation and value). + var flatColor = tinycolor.fromRatio({ h: currentHue, s: 1, v: 1 }); + dragger.css("background-color", flatColor.toHexString()); + + // Get a format that alpha will be included in (hex and names ignore alpha) + var format = currentPreferredFormat; + if (currentAlpha < 1 && !(currentAlpha === 0 && format === "name")) { + if (format === "hex" || format === "hex3" || format === "hex6" || format === "name") { + format = "rgb"; + } + } + + var realColor = get({ format: format }), + displayColor = ''; + + //reset background info for preview element + previewElement.removeClass("sp-clear-display"); + previewElement.css('background-color', 'transparent'); + + if (!realColor && allowEmpty) { + // Update the replaced elements background with icon indicating no color selection + previewElement.addClass("sp-clear-display"); + } + else { + var realHex = realColor.toHexString(), + realRgb = realColor.toRgbString(); + + // Update the replaced elements background color (with actual selected color) + if (rgbaSupport || realColor.alpha === 1) { + previewElement.css("background-color", realRgb); + } + else { + previewElement.css("background-color", "transparent"); + previewElement.css("filter", realColor.toFilter()); + } + + if (opts.showAlpha) { + var rgb = realColor.toRgb(); + rgb.a = 0; + var realAlpha = tinycolor(rgb).toRgbString(); + var gradient = "linear-gradient(left, " + realAlpha + ", " + realHex + ")"; + + if (IE) { + alphaSliderInner.css("filter", tinycolor(realAlpha).toFilter({ gradientType: 1 }, realHex)); + } + else { + alphaSliderInner.css("background", "-webkit-" + gradient); + alphaSliderInner.css("background", "-moz-" + gradient); + alphaSliderInner.css("background", "-ms-" + gradient); + // Use current syntax gradient on unprefixed property. + alphaSliderInner.css("background", + "linear-gradient(to right, " + realAlpha + ", " + realHex + ")"); + } + } + + displayColor = realColor.toString(format); + } + + // Update the text entry input as it changes happen + if (opts.showInput) { + textInput.val(displayColor); + } + + if (opts.showPalette) { + drawPalette(); + } + + drawInitial(); + } + + function updateHelperLocations() { + var s = currentSaturation; + var v = currentValue; + + if(allowEmpty && isEmpty) { + //if selected color is empty, hide the helpers + alphaSlideHelper.hide(); + slideHelper.hide(); + dragHelper.hide(); + } + else { + //make sure helpers are visible + alphaSlideHelper.show(); + slideHelper.show(); + dragHelper.show(); + + // Where to show the little circle in that displays your current selected color + var dragX = s * dragWidth; + var dragY = dragHeight - (v * dragHeight); + dragX = Math.max( + -dragHelperHeight, + Math.min(dragWidth - dragHelperHeight, dragX - dragHelperHeight) + ); + dragY = Math.max( + -dragHelperHeight, + Math.min(dragHeight - dragHelperHeight, dragY - dragHelperHeight) + ); + dragHelper.css({ + "top": dragY + "px", + "left": dragX + "px" + }); + + var alphaX = currentAlpha * alphaWidth; + alphaSlideHelper.css({ + "left": (alphaX - (alphaSlideHelperWidth / 2)) + "px" + }); + + // Where to show the bar that displays your current selected hue + var slideY = (currentHue) * slideHeight; + slideHelper.css({ + "top": (slideY - slideHelperHeight) + "px" + }); + } + } + + function updateOriginalInput(fireCallback) { + var color = get(), + displayColor = '', + hasChanged = !tinycolor.equals(color, colorOnShow); + + if (color) { + displayColor = color.toString(currentPreferredFormat); + // Update the selection palette with the current color + addColorToSelectionPalette(color); + } + + if (isInput) { + boundElement.val(displayColor); + } + + if (fireCallback && hasChanged) { + callbacks.change(color); + boundElement.trigger('change', [ color ]); + } + } + + function reflow() { + dragWidth = dragger.width(); + dragHeight = dragger.height(); + dragHelperHeight = dragHelper.height(); + slideWidth = slider.width(); + slideHeight = slider.height(); + slideHelperHeight = slideHelper.height(); + alphaWidth = alphaSlider.width(); + alphaSlideHelperWidth = alphaSlideHelper.width(); + + if (!flat) { + container.css("position", "absolute"); + if (opts.offset) { + container.offset(opts.offset); + } else { + container.offset(getOffset(container, offsetElement)); + } + } + + updateHelperLocations(); + + if (opts.showPalette) { + drawPalette(); + } + + boundElement.trigger('reflow.spectrum'); + } + + function destroy() { + boundElement.show(); + offsetElement.unbind("click.spectrum touchstart.spectrum"); + container.remove(); + replacer.remove(); + spectrums[spect.id] = null; + } + + function option(optionName, optionValue) { + if (optionName === undefined) { + return $.extend({}, opts); + } + if (optionValue === undefined) { + return opts[optionName]; + } + + opts[optionName] = optionValue; + applyOptions(); + } + + function enable() { + disabled = false; + boundElement.attr("disabled", false); + offsetElement.removeClass("sp-disabled"); + } + + function disable() { + hide(); + disabled = true; + boundElement.attr("disabled", true); + offsetElement.addClass("sp-disabled"); + } + + function setOffset(coord) { + opts.offset = coord; + reflow(); + } + + initialize(); + + var spect = { + show: show, + hide: hide, + toggle: toggle, + reflow: reflow, + option: option, + enable: enable, + disable: disable, + offset: setOffset, + set: function (c) { + set(c); + updateOriginalInput(); + }, + get: get, + destroy: destroy, + container: container + }; + + spect.id = spectrums.push(spect) - 1; + + return spect; + } + + /** + * checkOffset - get the offset below/above and left/right element depending on screen position + * Thanks https://github.com/jquery/jquery-ui/blob/master/ui/jquery.ui.datepicker.js + */ + function getOffset(picker, input) { + var extraY = 0; + var dpWidth = picker.outerWidth(); + var dpHeight = picker.outerHeight(); + var inputHeight = input.outerHeight(); + var doc = picker[0].ownerDocument; + var docElem = doc.documentElement; + var viewWidth = docElem.clientWidth + $(doc).scrollLeft(); + var viewHeight = docElem.clientHeight + $(doc).scrollTop(); + var offset = input.offset(); + offset.top += inputHeight; + + offset.left -= + Math.min(offset.left, (offset.left + dpWidth > viewWidth && viewWidth > dpWidth) ? + Math.abs(offset.left + dpWidth - viewWidth) : 0); + + offset.top -= + Math.min(offset.top, ((offset.top + dpHeight > viewHeight && viewHeight > dpHeight) ? + Math.abs(dpHeight + inputHeight - extraY) : extraY)); + + return offset; + } + + /** + * noop - do nothing + */ + function noop() { + + } + + /** + * stopPropagation - makes the code only doing this a little easier to read in line + */ + function stopPropagation(e) { + e.stopPropagation(); + } + + /** + * Create a function bound to a given object + * Thanks to underscore.js + */ + function bind(func, obj) { + var slice = Array.prototype.slice; + var args = slice.call(arguments, 2); + return function () { + return func.apply(obj, args.concat(slice.call(arguments))); + }; + } + + /** + * Lightweight drag helper. Handles containment within the element, so that + * when dragging, the x is within [0,element.width] and y is within [0,element.height] + */ + function draggable(element, onmove, onstart, onstop) { + onmove = onmove || function () { }; + onstart = onstart || function () { }; + onstop = onstop || function () { }; + var doc = document; + var dragging = false; + var offset = {}; + var maxHeight = 0; + var maxWidth = 0; + var hasTouch = ('ontouchstart' in window); + + var duringDragEvents = {}; + duringDragEvents["selectstart"] = prevent; + duringDragEvents["dragstart"] = prevent; + duringDragEvents["touchmove mousemove"] = move; + duringDragEvents["touchend mouseup"] = stop; + + function prevent(e) { + if (e.stopPropagation) { + e.stopPropagation(); + } + if (e.preventDefault) { + e.preventDefault(); + } + e.returnValue = false; + } + + function move(e) { + if (dragging) { + // Mouseup happened outside of window + if (IE && doc.documentMode < 9 && !e.button) { + return stop(); + } + + var t0 = e.originalEvent && e.originalEvent.touches && e.originalEvent.touches[0]; + var pageX = t0 && t0.pageX || e.pageX; + var pageY = t0 && t0.pageY || e.pageY; + + var dragX = Math.max(0, Math.min(pageX - offset.left, maxWidth)); + var dragY = Math.max(0, Math.min(pageY - offset.top, maxHeight)); + + if (hasTouch) { + // Stop scrolling in iOS + prevent(e); + } + + onmove.apply(element, [dragX, dragY, e]); + } + } + + function start(e) { + var rightclick = (e.which) ? (e.which == 3) : (e.button == 2); + + if (!rightclick && !dragging) { + if (onstart.apply(element, arguments) !== false) { + dragging = true; + maxHeight = $(element).height(); + maxWidth = $(element).width(); + offset = $(element).offset(); + + $(doc).bind(duringDragEvents); + $(doc.body).addClass("sp-dragging"); + + move(e); + + prevent(e); + } + } + } + + function stop() { + if (dragging) { + $(doc).unbind(duringDragEvents); + $(doc.body).removeClass("sp-dragging"); + + // Wait a tick before notifying observers to allow the click event + // to fire in Chrome. + setTimeout(function() { + onstop.apply(element, arguments); + }, 0); + } + dragging = false; + } + + $(element).bind("touchstart mousedown", start); + } + + function throttle(func, wait, debounce) { + var timeout; + return function () { + var context = this, args = arguments; + var throttler = function () { + timeout = null; + func.apply(context, args); + }; + if (debounce) clearTimeout(timeout); + if (debounce || !timeout) timeout = setTimeout(throttler, wait); + }; + } + + function inputTypeColorSupport() { + return $.fn.spectrum.inputTypeColorSupport(); + } + + /** + * Define a jQuery plugin + */ + var dataID = "spectrum.id"; + $.fn.spectrum = function (opts, extra) { + + if (typeof opts == "string") { + + var returnValue = this; + var args = Array.prototype.slice.call( arguments, 1 ); + + this.each(function () { + var spect = spectrums[$(this).data(dataID)]; + if (spect) { + var method = spect[opts]; + if (!method) { + throw new Error( "Spectrum: no such method: '" + opts + "'" ); + } + + if (opts == "get") { + returnValue = spect.get(); + } + else if (opts == "container") { + returnValue = spect.container; + } + else if (opts == "option") { + returnValue = spect.option.apply(spect, args); + } + else if (opts == "destroy") { + spect.destroy(); + $(this).removeData(dataID); + } + else { + method.apply(spect, args); + } + } + }); + + return returnValue; + } + + // Initializing a new instance of spectrum + return this.spectrum("destroy").each(function () { + var options = $.extend({}, opts, $(this).data()); + var spect = spectrum(this, options); + $(this).data(dataID, spect.id); + }); + }; + + $.fn.spectrum.load = true; + $.fn.spectrum.loadOpts = {}; + $.fn.spectrum.draggable = draggable; + $.fn.spectrum.defaults = defaultOpts; + $.fn.spectrum.inputTypeColorSupport = function inputTypeColorSupport() { + if (typeof inputTypeColorSupport._cachedResult === "undefined") { + var colorInput = $("<input type='color'/>")[0]; // if color element is supported, value will default to not null + inputTypeColorSupport._cachedResult = colorInput.type === "color" && colorInput.value !== ""; + } + return inputTypeColorSupport._cachedResult; + }; + + $.spectrum = { }; + $.spectrum.localization = { }; + $.spectrum.palettes = { }; + + $.fn.spectrum.processNativeColorInputs = function () { + var colorInputs = $("input[type=color]"); + if (colorInputs.length && !inputTypeColorSupport()) { + colorInputs.spectrum({ + preferredFormat: "hex6" + }); + } + }; + + // TinyColor v1.1.2 + // https://github.com/bgrins/TinyColor + // Brian Grinstead, MIT License + + (function() { + + var trimLeft = /^[\s,#]+/, + trimRight = /\s+$/, + tinyCounter = 0, + math = Math, + mathRound = math.round, + mathMin = math.min, + mathMax = math.max, + mathRandom = math.random; + + var tinycolor = function(color, opts) { + + color = (color) ? color : ''; + opts = opts || { }; + + // If input is already a tinycolor, return itself + if (color instanceof tinycolor) { + return color; + } + // If we are called as a function, call using new instead + if (!(this instanceof tinycolor)) { + return new tinycolor(color, opts); + } + + var rgb = inputToRGB(color); + this._originalInput = color, + this._r = rgb.r, + this._g = rgb.g, + this._b = rgb.b, + this._a = rgb.a, + this._roundA = mathRound(100*this._a) / 100, + this._format = opts.format || rgb.format; + this._gradientType = opts.gradientType; + + // Don't let the range of [0,255] come back in [0,1]. + // Potentially lose a little bit of precision here, but will fix issues where + // .5 gets interpreted as half of the total, instead of half of 1 + // If it was supposed to be 128, this was already taken care of by `inputToRgb` + if (this._r < 1) { this._r = mathRound(this._r); } + if (this._g < 1) { this._g = mathRound(this._g); } + if (this._b < 1) { this._b = mathRound(this._b); } + + this._ok = rgb.ok; + this._tc_id = tinyCounter++; + }; + + tinycolor.prototype = { + isDark: function() { + return this.getBrightness() < 128; + }, + isLight: function() { + return !this.isDark(); + }, + isValid: function() { + return this._ok; + }, + getOriginalInput: function() { + return this._originalInput; + }, + getFormat: function() { + return this._format; + }, + getAlpha: function() { + return this._a; + }, + getBrightness: function() { + var rgb = this.toRgb(); + return (rgb.r * 299 + rgb.g * 587 + rgb.b * 114) / 1000; + }, + setAlpha: function(value) { + this._a = boundAlpha(value); + this._roundA = mathRound(100*this._a) / 100; + return this; + }, + toHsv: function() { + var hsv = rgbToHsv(this._r, this._g, this._b); + return { h: hsv.h * 360, s: hsv.s, v: hsv.v, a: this._a }; + }, + toHsvString: function() { + var hsv = rgbToHsv(this._r, this._g, this._b); + var h = mathRound(hsv.h * 360), s = mathRound(hsv.s * 100), v = mathRound(hsv.v * 100); + return (this._a == 1) ? + "hsv(" + h + ", " + s + "%, " + v + "%)" : + "hsva(" + h + ", " + s + "%, " + v + "%, "+ this._roundA + ")"; + }, + toHsl: function() { + var hsl = rgbToHsl(this._r, this._g, this._b); + return { h: hsl.h * 360, s: hsl.s, l: hsl.l, a: this._a }; + }, + toHslString: function() { + var hsl = rgbToHsl(this._r, this._g, this._b); + var h = mathRound(hsl.h * 360), s = mathRound(hsl.s * 100), l = mathRound(hsl.l * 100); + return (this._a == 1) ? + "hsl(" + h + ", " + s + "%, " + l + "%)" : + "hsla(" + h + ", " + s + "%, " + l + "%, "+ this._roundA + ")"; + }, + toHex: function(allow3Char) { + return rgbToHex(this._r, this._g, this._b, allow3Char); + }, + toHexString: function(allow3Char) { + return '#' + this.toHex(allow3Char); + }, + toHex8: function() { + return rgbaToHex(this._r, this._g, this._b, this._a); + }, + toHex8String: function() { + return '#' + this.toHex8(); + }, + toRgb: function() { + return { r: mathRound(this._r), g: mathRound(this._g), b: mathRound(this._b), a: this._a }; + }, + toRgbString: function() { + return (this._a == 1) ? + "rgb(" + mathRound(this._r) + ", " + mathRound(this._g) + ", " + mathRound(this._b) + ")" : + "rgba(" + mathRound(this._r) + ", " + mathRound(this._g) + ", " + mathRound(this._b) + ", " + this._roundA + ")"; + }, + toPercentageRgb: function() { + return { r: mathRound(bound01(this._r, 255) * 100) + "%", g: mathRound(bound01(this._g, 255) * 100) + "%", b: mathRound(bound01(this._b, 255) * 100) + "%", a: this._a }; + }, + toPercentageRgbString: function() { + return (this._a == 1) ? + "rgb(" + mathRound(bound01(this._r, 255) * 100) + "%, " + mathRound(bound01(this._g, 255) * 100) + "%, " + mathRound(bound01(this._b, 255) * 100) + "%)" : + "rgba(" + mathRound(bound01(this._r, 255) * 100) + "%, " + mathRound(bound01(this._g, 255) * 100) + "%, " + mathRound(bound01(this._b, 255) * 100) + "%, " + this._roundA + ")"; + }, + toName: function() { + if (this._a === 0) { + return "transparent"; + } + + if (this._a < 1) { + return false; + } + + return hexNames[rgbToHex(this._r, this._g, this._b, true)] || false; + }, + toFilter: function(secondColor) { + var hex8String = '#' + rgbaToHex(this._r, this._g, this._b, this._a); + var secondHex8String = hex8String; + var gradientType = this._gradientType ? "GradientType = 1, " : ""; + + if (secondColor) { + var s = tinycolor(secondColor); + secondHex8String = s.toHex8String(); + } + + return "progid:DXImageTransform.Microsoft.gradient("+gradientType+"startColorstr="+hex8String+",endColorstr="+secondHex8String+")"; + }, + toString: function(format) { + var formatSet = !!format; + format = format || this._format; + + var formattedString = false; + var hasAlpha = this._a < 1 && this._a >= 0; + var needsAlphaFormat = !formatSet && hasAlpha && (format === "hex" || format === "hex6" || format === "hex3" || format === "name"); + + if (needsAlphaFormat) { + // Special case for "transparent", all other non-alpha formats + // will return rgba when there is transparency. + if (format === "name" && this._a === 0) { + return this.toName(); + } + return this.toRgbString(); + } + if (format === "rgb") { + formattedString = this.toRgbString(); + } + if (format === "prgb") { + formattedString = this.toPercentageRgbString(); + } + if (format === "hex" || format === "hex6") { + formattedString = this.toHexString(); + } + if (format === "hex3") { + formattedString = this.toHexString(true); + } + if (format === "hex8") { + formattedString = this.toHex8String(); + } + if (format === "name") { + formattedString = this.toName(); + } + if (format === "hsl") { + formattedString = this.toHslString(); + } + if (format === "hsv") { + formattedString = this.toHsvString(); + } + + return formattedString || this.toHexString(); + }, + + _applyModification: function(fn, args) { + var color = fn.apply(null, [this].concat([].slice.call(args))); + this._r = color._r; + this._g = color._g; + this._b = color._b; + this.setAlpha(color._a); + return this; + }, + lighten: function() { + return this._applyModification(lighten, arguments); + }, + brighten: function() { + return this._applyModification(brighten, arguments); + }, + darken: function() { + return this._applyModification(darken, arguments); + }, + desaturate: function() { + return this._applyModification(desaturate, arguments); + }, + saturate: function() { + return this._applyModification(saturate, arguments); + }, + greyscale: function() { + return this._applyModification(greyscale, arguments); + }, + spin: function() { + return this._applyModification(spin, arguments); + }, + + _applyCombination: function(fn, args) { + return fn.apply(null, [this].concat([].slice.call(args))); + }, + analogous: function() { + return this._applyCombination(analogous, arguments); + }, + complement: function() { + return this._applyCombination(complement, arguments); + }, + monochromatic: function() { + return this._applyCombination(monochromatic, arguments); + }, + splitcomplement: function() { + return this._applyCombination(splitcomplement, arguments); + }, + triad: function() { + return this._applyCombination(triad, arguments); + }, + tetrad: function() { + return this._applyCombination(tetrad, arguments); + } + }; + + // If input is an object, force 1 into "1.0" to handle ratios properly + // String input requires "1.0" as input, so 1 will be treated as 1 + tinycolor.fromRatio = function(color, opts) { + if (typeof color == "object") { + var newColor = {}; + for (var i in color) { + if (color.hasOwnProperty(i)) { + if (i === "a") { + newColor[i] = color[i]; + } + else { + newColor[i] = convertToPercentage(color[i]); + } + } + } + color = newColor; + } + + return tinycolor(color, opts); + }; + + // Given a string or object, convert that input to RGB + // Possible string inputs: + // + // "red" + // "#f00" or "f00" + // "#ff0000" or "ff0000" + // "#ff000000" or "ff000000" + // "rgb 255 0 0" or "rgb (255, 0, 0)" + // "rgb 1.0 0 0" or "rgb (1, 0, 0)" + // "rgba (255, 0, 0, 1)" or "rgba 255, 0, 0, 1" + // "rgba (1.0, 0, 0, 1)" or "rgba 1.0, 0, 0, 1" + // "hsl(0, 100%, 50%)" or "hsl 0 100% 50%" + // "hsla(0, 100%, 50%, 1)" or "hsla 0 100% 50%, 1" + // "hsv(0, 100%, 100%)" or "hsv 0 100% 100%" + // + function inputToRGB(color) { + + var rgb = { r: 0, g: 0, b: 0 }; + var a = 1; + var ok = false; + var format = false; + + if (typeof color == "string") { + color = stringInputToObject(color); + } + + if (typeof color == "object") { + if (color.hasOwnProperty("r") && color.hasOwnProperty("g") && color.hasOwnProperty("b")) { + rgb = rgbToRgb(color.r, color.g, color.b); + ok = true; + format = String(color.r).substr(-1) === "%" ? "prgb" : "rgb"; + } + else if (color.hasOwnProperty("h") && color.hasOwnProperty("s") && color.hasOwnProperty("v")) { + color.s = convertToPercentage(color.s); + color.v = convertToPercentage(color.v); + rgb = hsvToRgb(color.h, color.s, color.v); + ok = true; + format = "hsv"; + } + else if (color.hasOwnProperty("h") && color.hasOwnProperty("s") && color.hasOwnProperty("l")) { + color.s = convertToPercentage(color.s); + color.l = convertToPercentage(color.l); + rgb = hslToRgb(color.h, color.s, color.l); + ok = true; + format = "hsl"; + } + + if (color.hasOwnProperty("a")) { + a = color.a; + } + } + + a = boundAlpha(a); + + return { + ok: ok, + format: color.format || format, + r: mathMin(255, mathMax(rgb.r, 0)), + g: mathMin(255, mathMax(rgb.g, 0)), + b: mathMin(255, mathMax(rgb.b, 0)), + a: a + }; + } + + + // Conversion Functions + // -------------------- + + // `rgbToHsl`, `rgbToHsv`, `hslToRgb`, `hsvToRgb` modified from: + // <http://mjijackson.com/2008/02/rgb-to-hsl-and-rgb-to-hsv-color-model-conversion-algorithms-in-javascript> + + // `rgbToRgb` + // Handle bounds / percentage checking to conform to CSS color spec + // <http://www.w3.org/TR/css3-color/> + // *Assumes:* r, g, b in [0, 255] or [0, 1] + // *Returns:* { r, g, b } in [0, 255] + function rgbToRgb(r, g, b){ + return { + r: bound01(r, 255) * 255, + g: bound01(g, 255) * 255, + b: bound01(b, 255) * 255 + }; + } + + // `rgbToHsl` + // Converts an RGB color value to HSL. + // *Assumes:* r, g, and b are contained in [0, 255] or [0, 1] + // *Returns:* { h, s, l } in [0,1] + function rgbToHsl(r, g, b) { + + r = bound01(r, 255); + g = bound01(g, 255); + b = bound01(b, 255); + + var max = mathMax(r, g, b), min = mathMin(r, g, b); + var h, s, l = (max + min) / 2; + + if(max == min) { + h = s = 0; // achromatic + } + else { + var d = max - min; + s = l > 0.5 ? d / (2 - max - min) : d / (max + min); + switch(max) { + case r: h = (g - b) / d + (g < b ? 6 : 0); break; + case g: h = (b - r) / d + 2; break; + case b: h = (r - g) / d + 4; break; + } + + h /= 6; + } + + return { h: h, s: s, l: l }; + } + + // `hslToRgb` + // Converts an HSL color value to RGB. + // *Assumes:* h is contained in [0, 1] or [0, 360] and s and l are contained [0, 1] or [0, 100] + // *Returns:* { r, g, b } in the set [0, 255] + function hslToRgb(h, s, l) { + var r, g, b; + + h = bound01(h, 360); + s = bound01(s, 100); + l = bound01(l, 100); + + function hue2rgb(p, q, t) { + if(t < 0) t += 1; + if(t > 1) t -= 1; + if(t < 1/6) return p + (q - p) * 6 * t; + if(t < 1/2) return q; + if(t < 2/3) return p + (q - p) * (2/3 - t) * 6; + return p; + } + + if(s === 0) { + r = g = b = l; // achromatic + } + else { + var q = l < 0.5 ? l * (1 + s) : l + s - l * s; + var p = 2 * l - q; + r = hue2rgb(p, q, h + 1/3); + g = hue2rgb(p, q, h); + b = hue2rgb(p, q, h - 1/3); + } + + return { r: r * 255, g: g * 255, b: b * 255 }; + } + + // `rgbToHsv` + // Converts an RGB color value to HSV + // *Assumes:* r, g, and b are contained in the set [0, 255] or [0, 1] + // *Returns:* { h, s, v } in [0,1] + function rgbToHsv(r, g, b) { + + r = bound01(r, 255); + g = bound01(g, 255); + b = bound01(b, 255); + + var max = mathMax(r, g, b), min = mathMin(r, g, b); + var h, s, v = max; + + var d = max - min; + s = max === 0 ? 0 : d / max; + + if(max == min) { + h = 0; // achromatic + } + else { + switch(max) { + case r: h = (g - b) / d + (g < b ? 6 : 0); break; + case g: h = (b - r) / d + 2; break; + case b: h = (r - g) / d + 4; break; + } + h /= 6; + } + return { h: h, s: s, v: v }; + } + + // `hsvToRgb` + // Converts an HSV color value to RGB. + // *Assumes:* h is contained in [0, 1] or [0, 360] and s and v are contained in [0, 1] or [0, 100] + // *Returns:* { r, g, b } in the set [0, 255] + function hsvToRgb(h, s, v) { + + h = bound01(h, 360) * 6; + s = bound01(s, 100); + v = bound01(v, 100); + + var i = math.floor(h), + f = h - i, + p = v * (1 - s), + q = v * (1 - f * s), + t = v * (1 - (1 - f) * s), + mod = i % 6, + r = [v, q, p, p, t, v][mod], + g = [t, v, v, q, p, p][mod], + b = [p, p, t, v, v, q][mod]; + + return { r: r * 255, g: g * 255, b: b * 255 }; + } + + // `rgbToHex` + // Converts an RGB color to hex + // Assumes r, g, and b are contained in the set [0, 255] + // Returns a 3 or 6 character hex + function rgbToHex(r, g, b, allow3Char) { + + var hex = [ + pad2(mathRound(r).toString(16)), + pad2(mathRound(g).toString(16)), + pad2(mathRound(b).toString(16)) + ]; + + // Return a 3 character hex if possible + if (allow3Char && hex[0].charAt(0) == hex[0].charAt(1) && hex[1].charAt(0) == hex[1].charAt(1) && hex[2].charAt(0) == hex[2].charAt(1)) { + return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0); + } + + return hex.join(""); + } + // `rgbaToHex` + // Converts an RGBA color plus alpha transparency to hex + // Assumes r, g, b and a are contained in the set [0, 255] + // Returns an 8 character hex + function rgbaToHex(r, g, b, a) { + + var hex = [ + pad2(convertDecimalToHex(a)), + pad2(mathRound(r).toString(16)), + pad2(mathRound(g).toString(16)), + pad2(mathRound(b).toString(16)) + ]; + + return hex.join(""); + } + + // `equals` + // Can be called with any tinycolor input + tinycolor.equals = function (color1, color2) { + if (!color1 || !color2) { return false; } + return tinycolor(color1).toRgbString() == tinycolor(color2).toRgbString(); + }; + tinycolor.random = function() { + return tinycolor.fromRatio({ + r: mathRandom(), + g: mathRandom(), + b: mathRandom() + }); + }; + + + // Modification Functions + // ---------------------- + // Thanks to less.js for some of the basics here + // <https://github.com/cloudhead/less.js/blob/master/lib/less/functions.js> + + function desaturate(color, amount) { + amount = (amount === 0) ? 0 : (amount || 10); + var hsl = tinycolor(color).toHsl(); + hsl.s -= amount / 100; + hsl.s = clamp01(hsl.s); + return tinycolor(hsl); + } + + function saturate(color, amount) { + amount = (amount === 0) ? 0 : (amount || 10); + var hsl = tinycolor(color).toHsl(); + hsl.s += amount / 100; + hsl.s = clamp01(hsl.s); + return tinycolor(hsl); + } + + function greyscale(color) { + return tinycolor(color).desaturate(100); + } + + function lighten (color, amount) { + amount = (amount === 0) ? 0 : (amount || 10); + var hsl = tinycolor(color).toHsl(); + hsl.l += amount / 100; + hsl.l = clamp01(hsl.l); + return tinycolor(hsl); + } + + function brighten(color, amount) { + amount = (amount === 0) ? 0 : (amount || 10); + var rgb = tinycolor(color).toRgb(); + rgb.r = mathMax(0, mathMin(255, rgb.r - mathRound(255 * - (amount / 100)))); + rgb.g = mathMax(0, mathMin(255, rgb.g - mathRound(255 * - (amount / 100)))); + rgb.b = mathMax(0, mathMin(255, rgb.b - mathRound(255 * - (amount / 100)))); + return tinycolor(rgb); + } + + function darken (color, amount) { + amount = (amount === 0) ? 0 : (amount || 10); + var hsl = tinycolor(color).toHsl(); + hsl.l -= amount / 100; + hsl.l = clamp01(hsl.l); + return tinycolor(hsl); + } + + // Spin takes a positive or negative amount within [-360, 360] indicating the change of hue. + // Values outside of this range will be wrapped into this range. + function spin(color, amount) { + var hsl = tinycolor(color).toHsl(); + var hue = (mathRound(hsl.h) + amount) % 360; + hsl.h = hue < 0 ? 360 + hue : hue; + return tinycolor(hsl); + } + + // Combination Functions + // --------------------- + // Thanks to jQuery xColor for some of the ideas behind these + // <https://github.com/infusion/jQuery-xcolor/blob/master/jquery.xcolor.js> + + function complement(color) { + var hsl = tinycolor(color).toHsl(); + hsl.h = (hsl.h + 180) % 360; + return tinycolor(hsl); + } + + function triad(color) { + var hsl = tinycolor(color).toHsl(); + var h = hsl.h; + return [ + tinycolor(color), + tinycolor({ h: (h + 120) % 360, s: hsl.s, l: hsl.l }), + tinycolor({ h: (h + 240) % 360, s: hsl.s, l: hsl.l }) + ]; + } + + function tetrad(color) { + var hsl = tinycolor(color).toHsl(); + var h = hsl.h; + return [ + tinycolor(color), + tinycolor({ h: (h + 90) % 360, s: hsl.s, l: hsl.l }), + tinycolor({ h: (h + 180) % 360, s: hsl.s, l: hsl.l }), + tinycolor({ h: (h + 270) % 360, s: hsl.s, l: hsl.l }) + ]; + } + + function splitcomplement(color) { + var hsl = tinycolor(color).toHsl(); + var h = hsl.h; + return [ + tinycolor(color), + tinycolor({ h: (h + 72) % 360, s: hsl.s, l: hsl.l}), + tinycolor({ h: (h + 216) % 360, s: hsl.s, l: hsl.l}) + ]; + } + + function analogous(color, results, slices) { + results = results || 6; + slices = slices || 30; + + var hsl = tinycolor(color).toHsl(); + var part = 360 / slices; + var ret = [tinycolor(color)]; + + for (hsl.h = ((hsl.h - (part * results >> 1)) + 720) % 360; --results; ) { + hsl.h = (hsl.h + part) % 360; + ret.push(tinycolor(hsl)); + } + return ret; + } + + function monochromatic(color, results) { + results = results || 6; + var hsv = tinycolor(color).toHsv(); + var h = hsv.h, s = hsv.s, v = hsv.v; + var ret = []; + var modification = 1 / results; + + while (results--) { + ret.push(tinycolor({ h: h, s: s, v: v})); + v = (v + modification) % 1; + } + + return ret; + } + + // Utility Functions + // --------------------- + + tinycolor.mix = function(color1, color2, amount) { + amount = (amount === 0) ? 0 : (amount || 50); + + var rgb1 = tinycolor(color1).toRgb(); + var rgb2 = tinycolor(color2).toRgb(); + + var p = amount / 100; + var w = p * 2 - 1; + var a = rgb2.a - rgb1.a; + + var w1; + + if (w * a == -1) { + w1 = w; + } else { + w1 = (w + a) / (1 + w * a); + } + + w1 = (w1 + 1) / 2; + + var w2 = 1 - w1; + + var rgba = { + r: rgb2.r * w1 + rgb1.r * w2, + g: rgb2.g * w1 + rgb1.g * w2, + b: rgb2.b * w1 + rgb1.b * w2, + a: rgb2.a * p + rgb1.a * (1 - p) + }; + + return tinycolor(rgba); + }; + + + // Readability Functions + // --------------------- + // <http://www.w3.org/TR/AERT#color-contrast> + + // `readability` + // Analyze the 2 colors and returns an object with the following properties: + // `brightness`: difference in brightness between the two colors + // `color`: difference in color/hue between the two colors + tinycolor.readability = function(color1, color2) { + var c1 = tinycolor(color1); + var c2 = tinycolor(color2); + var rgb1 = c1.toRgb(); + var rgb2 = c2.toRgb(); + var brightnessA = c1.getBrightness(); + var brightnessB = c2.getBrightness(); + var colorDiff = ( + Math.max(rgb1.r, rgb2.r) - Math.min(rgb1.r, rgb2.r) + + Math.max(rgb1.g, rgb2.g) - Math.min(rgb1.g, rgb2.g) + + Math.max(rgb1.b, rgb2.b) - Math.min(rgb1.b, rgb2.b) + ); + + return { + brightness: Math.abs(brightnessA - brightnessB), + color: colorDiff + }; + }; + + // `readable` + // http://www.w3.org/TR/AERT#color-contrast + // Ensure that foreground and background color combinations provide sufficient contrast. + // *Example* + // tinycolor.isReadable("#000", "#111") => false + tinycolor.isReadable = function(color1, color2) { + var readability = tinycolor.readability(color1, color2); + return readability.brightness > 125 && readability.color > 500; + }; + + // `mostReadable` + // Given a base color and a list of possible foreground or background + // colors for that base, returns the most readable color. + // *Example* + // tinycolor.mostReadable("#123", ["#fff", "#000"]) => "#000" + tinycolor.mostReadable = function(baseColor, colorList) { + var bestColor = null; + var bestScore = 0; + var bestIsReadable = false; + for (var i=0; i < colorList.length; i++) { + + // We normalize both around the "acceptable" breaking point, + // but rank brightness constrast higher than hue. + + var readability = tinycolor.readability(baseColor, colorList[i]); + var readable = readability.brightness > 125 && readability.color > 500; + var score = 3 * (readability.brightness / 125) + (readability.color / 500); + + if ((readable && ! bestIsReadable) || + (readable && bestIsReadable && score > bestScore) || + ((! readable) && (! bestIsReadable) && score > bestScore)) { + bestIsReadable = readable; + bestScore = score; + bestColor = tinycolor(colorList[i]); + } + } + return bestColor; + }; + + + // Big List of Colors + // ------------------ + // <http://www.w3.org/TR/css3-color/#svg-color> + var names = tinycolor.names = { + aliceblue: "f0f8ff", + antiquewhite: "faebd7", + aqua: "0ff", + aquamarine: "7fffd4", + azure: "f0ffff", + beige: "f5f5dc", + bisque: "ffe4c4", + black: "000", + blanchedalmond: "ffebcd", + blue: "00f", + blueviolet: "8a2be2", + brown: "a52a2a", + burlywood: "deb887", + burntsienna: "ea7e5d", + cadetblue: "5f9ea0", + chartreuse: "7fff00", + chocolate: "d2691e", + coral: "ff7f50", + cornflowerblue: "6495ed", + cornsilk: "fff8dc", + crimson: "dc143c", + cyan: "0ff", + darkblue: "00008b", + darkcyan: "008b8b", + darkgoldenrod: "b8860b", + darkgray: "a9a9a9", + darkgreen: "006400", + darkgrey: "a9a9a9", + darkkhaki: "bdb76b", + darkmagenta: "8b008b", + darkolivegreen: "556b2f", + darkorange: "ff8c00", + darkorchid: "9932cc", + darkred: "8b0000", + darksalmon: "e9967a", + darkseagreen: "8fbc8f", + darkslateblue: "483d8b", + darkslategray: "2f4f4f", + darkslategrey: "2f4f4f", + darkturquoise: "00ced1", + darkviolet: "9400d3", + deeppink: "ff1493", + deepskyblue: "00bfff", + dimgray: "696969", + dimgrey: "696969", + dodgerblue: "1e90ff", + firebrick: "b22222", + floralwhite: "fffaf0", + forestgreen: "228b22", + fuchsia: "f0f", + gainsboro: "dcdcdc", + ghostwhite: "f8f8ff", + gold: "ffd700", + goldenrod: "daa520", + gray: "808080", + green: "008000", + greenyellow: "adff2f", + grey: "808080", + honeydew: "f0fff0", + hotpink: "ff69b4", + indianred: "cd5c5c", + indigo: "4b0082", + ivory: "fffff0", + khaki: "f0e68c", + lavender: "e6e6fa", + lavenderblush: "fff0f5", + lawngreen: "7cfc00", + lemonchiffon: "fffacd", + lightblue: "add8e6", + lightcoral: "f08080", + lightcyan: "e0ffff", + lightgoldenrodyellow: "fafad2", + lightgray: "d3d3d3", + lightgreen: "90ee90", + lightgrey: "d3d3d3", + lightpink: "ffb6c1", + lightsalmon: "ffa07a", + lightseagreen: "20b2aa", + lightskyblue: "87cefa", + lightslategray: "789", + lightslategrey: "789", + lightsteelblue: "b0c4de", + lightyellow: "ffffe0", + lime: "0f0", + limegreen: "32cd32", + linen: "faf0e6", + magenta: "f0f", + maroon: "800000", + mediumaquamarine: "66cdaa", + mediumblue: "0000cd", + mediumorchid: "ba55d3", + mediumpurple: "9370db", + mediumseagreen: "3cb371", + mediumslateblue: "7b68ee", + mediumspringgreen: "00fa9a", + mediumturquoise: "48d1cc", + mediumvioletred: "c71585", + midnightblue: "191970", + mintcream: "f5fffa", + mistyrose: "ffe4e1", + moccasin: "ffe4b5", + navajowhite: "ffdead", + navy: "000080", + oldlace: "fdf5e6", + olive: "808000", + olivedrab: "6b8e23", + orange: "ffa500", + orangered: "ff4500", + orchid: "da70d6", + palegoldenrod: "eee8aa", + palegreen: "98fb98", + paleturquoise: "afeeee", + palevioletred: "db7093", + papayawhip: "ffefd5", + peachpuff: "ffdab9", + peru: "cd853f", + pink: "ffc0cb", + plum: "dda0dd", + powderblue: "b0e0e6", + purple: "800080", + rebeccapurple: "663399", + red: "f00", + rosybrown: "bc8f8f", + royalblue: "4169e1", + saddlebrown: "8b4513", + salmon: "fa8072", + sandybrown: "f4a460", + seagreen: "2e8b57", + seashell: "fff5ee", + sienna: "a0522d", + silver: "c0c0c0", + skyblue: "87ceeb", + slateblue: "6a5acd", + slategray: "708090", + slategrey: "708090", + snow: "fffafa", + springgreen: "00ff7f", + steelblue: "4682b4", + tan: "d2b48c", + teal: "008080", + thistle: "d8bfd8", + tomato: "ff6347", + turquoise: "40e0d0", + violet: "ee82ee", + wheat: "f5deb3", + white: "fff", + whitesmoke: "f5f5f5", + yellow: "ff0", + yellowgreen: "9acd32" + }; + + // Make it easy to access colors via `hexNames[hex]` + var hexNames = tinycolor.hexNames = flip(names); + + + // Utilities + // --------- + + // `{ 'name1': 'val1' }` becomes `{ 'val1': 'name1' }` + function flip(o) { + var flipped = { }; + for (var i in o) { + if (o.hasOwnProperty(i)) { + flipped[o[i]] = i; + } + } + return flipped; + } + + // Return a valid alpha value [0,1] with all invalid values being set to 1 + function boundAlpha(a) { + a = parseFloat(a); + + if (isNaN(a) || a < 0 || a > 1) { + a = 1; + } + + return a; + } + + // Take input from [0, n] and return it as [0, 1] + function bound01(n, max) { + if (isOnePointZero(n)) { n = "100%"; } + + var processPercent = isPercentage(n); + n = mathMin(max, mathMax(0, parseFloat(n))); + + // Automatically convert percentage into number + if (processPercent) { + n = parseInt(n * max, 10) / 100; + } + + // Handle floating point rounding errors + if ((math.abs(n - max) < 0.000001)) { + return 1; + } + + // Convert into [0, 1] range if it isn't already + return (n % max) / parseFloat(max); + } + + // Force a number between 0 and 1 + function clamp01(val) { + return mathMin(1, mathMax(0, val)); + } + + // Parse a base-16 hex value into a base-10 integer + function parseIntFromHex(val) { + return parseInt(val, 16); + } + + // Need to handle 1.0 as 100%, since once it is a number, there is no difference between it and 1 + // <http://stackoverflow.com/questions/7422072/javascript-how-to-detect-number-as-a-decimal-including-1-0> + function isOnePointZero(n) { + return typeof n == "string" && n.indexOf('.') != -1 && parseFloat(n) === 1; + } + + // Check to see if string passed in is a percentage + function isPercentage(n) { + return typeof n === "string" && n.indexOf('%') != -1; + } + + // Force a hex value to have 2 characters + function pad2(c) { + return c.length == 1 ? '0' + c : '' + c; + } + + // Replace a decimal with it's percentage value + function convertToPercentage(n) { + if (n <= 1) { + n = (n * 100) + "%"; + } + + return n; + } + + // Converts a decimal to a hex value + function convertDecimalToHex(d) { + return Math.round(parseFloat(d) * 255).toString(16); + } + // Converts a hex value to a decimal + function convertHexToDecimal(h) { + return (parseIntFromHex(h) / 255); + } + + var matchers = (function() { + + // <http://www.w3.org/TR/css3-values/#integers> + var CSS_INTEGER = "[-\\+]?\\d+%?"; + + // <http://www.w3.org/TR/css3-values/#number-value> + var CSS_NUMBER = "[-\\+]?\\d*\\.\\d+%?"; + + // Allow positive/negative integer/number. Don't capture the either/or, just the entire outcome. + var CSS_UNIT = "(?:" + CSS_NUMBER + ")|(?:" + CSS_INTEGER + ")"; + + // Actual matching. + // Parentheses and commas are optional, but not required. + // Whitespace can take the place of commas or opening paren + var PERMISSIVE_MATCH3 = "[\\s|\\(]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")\\s*\\)?"; + var PERMISSIVE_MATCH4 = "[\\s|\\(]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")[,|\\s]+(" + CSS_UNIT + ")\\s*\\)?"; + + return { + rgb: new RegExp("rgb" + PERMISSIVE_MATCH3), + rgba: new RegExp("rgba" + PERMISSIVE_MATCH4), + hsl: new RegExp("hsl" + PERMISSIVE_MATCH3), + hsla: new RegExp("hsla" + PERMISSIVE_MATCH4), + hsv: new RegExp("hsv" + PERMISSIVE_MATCH3), + hsva: new RegExp("hsva" + PERMISSIVE_MATCH4), + hex3: /^([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/, + hex6: /^([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/, + hex8: /^([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/ + }; + })(); + + // `stringInputToObject` + // Permissive string parsing. Take in a number of formats, and output an object + // based on detected format. Returns `{ r, g, b }` or `{ h, s, l }` or `{ h, s, v}` + function stringInputToObject(color) { + + color = color.replace(trimLeft,'').replace(trimRight, '').toLowerCase(); + var named = false; + if (names[color]) { + color = names[color]; + named = true; + } + else if (color == 'transparent') { + return { r: 0, g: 0, b: 0, a: 0, format: "name" }; + } + + // Try to match string input using regular expressions. + // Keep most of the number bounding out of this function - don't worry about [0,1] or [0,100] or [0,360] + // Just return an object and let the conversion functions handle that. + // This way the result will be the same whether the tinycolor is initialized with string or object. + var match; + if ((match = matchers.rgb.exec(color))) { + return { r: match[1], g: match[2], b: match[3] }; + } + if ((match = matchers.rgba.exec(color))) { + return { r: match[1], g: match[2], b: match[3], a: match[4] }; + } + if ((match = matchers.hsl.exec(color))) { + return { h: match[1], s: match[2], l: match[3] }; + } + if ((match = matchers.hsla.exec(color))) { + return { h: match[1], s: match[2], l: match[3], a: match[4] }; + } + if ((match = matchers.hsv.exec(color))) { + return { h: match[1], s: match[2], v: match[3] }; + } + if ((match = matchers.hsva.exec(color))) { + return { h: match[1], s: match[2], v: match[3], a: match[4] }; + } + if ((match = matchers.hex8.exec(color))) { + return { + a: convertHexToDecimal(match[1]), + r: parseIntFromHex(match[2]), + g: parseIntFromHex(match[3]), + b: parseIntFromHex(match[4]), + format: named ? "name" : "hex8" + }; + } + if ((match = matchers.hex6.exec(color))) { + return { + r: parseIntFromHex(match[1]), + g: parseIntFromHex(match[2]), + b: parseIntFromHex(match[3]), + format: named ? "name" : "hex" + }; + } + if ((match = matchers.hex3.exec(color))) { + return { + r: parseIntFromHex(match[1] + '' + match[1]), + g: parseIntFromHex(match[2] + '' + match[2]), + b: parseIntFromHex(match[3] + '' + match[3]), + format: named ? "name" : "hex" + }; + } + + return false; + } + + window.tinycolor = tinycolor; + })(); + + $(function () { + if ($.fn.spectrum.load) { + $.fn.spectrum.processNativeColorInputs(); + } + }); + +}); diff --git a/scp/queues.php b/scp/queues.php new file mode 100644 index 0000000000000000000000000000000000000000..d348d570503e54c51dd17358dc0421645ceb7e5c --- /dev/null +++ b/scp/queues.php @@ -0,0 +1,26 @@ +<?php +/********************************************************************* + queues.php + + Handles management of custom queues + + Jared Hancock <jared@osticket.com> + Peter Rotich <peter@osticket.com> + Copyright (c) 2006-2015 osTicket + http://www.osticket.com + + Released under the GNU General Public License WITHOUT ANY WARRANTY. + See LICENSE.TXT for details. + + vim: expandtab sw=4 ts=4 sts=4: +**********************************************************************/ + +require('admin.inc.php'); + +require_once INCLUDE_DIR . 'class.queue.php'; + +$nav->setTabActive('settings', 'settings.php?t='.urlencode($_GET['t'])); + +require_once(STAFFINC_DIR.'header.inc.php'); +include_once(STAFFINC_DIR."queue.inc.php"); +include_once(STAFFINC_DIR.'footer.inc.php');