0byt3m1n1
Path:
/
data
/
applications
/
aps
/
tikiwiki
/
14.1-0
/
standard
/
htdocs
/
lib
/
wiki-plugins
/
[
Home
]
File: wikiplugin_customsearch.php
<?php // (c) Copyright 2002-2015 by authors of the Tiki Wiki CMS Groupware Project // // All Rights Reserved. See copyright.txt for details and a complete list of authors. // Licensed under the GNU LESSER GENERAL PUBLIC LICENSE. See license.txt for details. // $Id: wikiplugin_customsearch.php 56257 2015-09-20 03:05:57Z lindonb $ function wikiplugin_customsearch_info() { return array( 'name' => tra('Custom Search'), 'documentation' => 'PluginCustomSearch', 'description' => tra('Create a custom search form for searching or listing items on the site'), 'prefs' => array('wikiplugin_customsearch', 'wikiplugin_list', 'feature_ajax', 'feature_search'), 'body' => tra('LIST plugin configuration information'), 'filter' => 'wikicontent', 'profile_reference' => 'search_plugin_content', 'iconname' => 'search', 'introduced' => 8, 'tags' => array('advanced'), 'params' => array( 'wiki' => array( 'required' => false, 'name' => tra('Template wiki page'), 'description' => tra('Wiki page where search user interface template is found'), 'since' => '8.0', 'filter' => 'pagename', 'default' => '', 'profile_reference' => 'wiki_page', ), 'tpl' => array( 'required' => false, 'name' => tra('Template file'), 'description' => tra('TPL file where search user interface template is found'), 'since' => '12.2', 'default' => '', ), 'id' => array( 'required' => false, 'name' => tra('Search Id'), 'description' => tra('A unique identifier to distinguish custom searches for storing of previous search criteria entered by users'), 'since' => '8.0', 'filter' => 'alnum', 'default' => '0', ), 'autosearchdelay' => array( 'required' => false, 'name' => tra('Search Delay'), 'description' => tr('Delay in milliseconds before automatically triggering search after change (%00%1 disables and is the default)', '<code>', '</code>'), 'since' => '8.0', 'filter' => 'digits', 'default' => '0', ), 'searchfadediv' => array( 'required' => false, 'name' => tra('Fade DIV Id'), 'description' => tra('The specific ID of the specific div to fade out when AJAX search is in progress, if not set will attempt to fade the whole area or if failing simply show the spinner'), 'since' => '8.0', 'filter' => 'text', 'default' => '', ), 'recalllastsearch' => array( 'required' => false, 'name' => tra('Recall Last Search'), 'description' => tra('In the same session, return users to same search parameters on coming back to the search page after leaving'), 'since' => '8.0', 'options' => array( array('text' => tra('No'), 'value' => '0'), array('text' => tra('Yes'), 'value' => '1'), ), 'filter' => 'digits', 'default' => '0', ), 'callbackscript' => array( 'required' => false, 'name' => tra('Custom JavaScript Page'), 'description' => tra('The wiki page on which custom JavaScript that is to be executed on return of AJAX results'), 'since' => '8.0', 'filter' => 'pagename', 'default' => '', ), 'destdiv' => array( 'required' => false, 'name' => tra('Destination Div'), 'description' => tra('Id of a pre-existing div to contain the search results'), 'since' => '9.0', 'filter' => 'text', 'default' => '', ), 'searchonload' => array( 'required' => false, 'name' => tra('Search On Load'), 'description' => tra('Execute the search when the page loads (default: Yes)'), 'since' => '9.0', 'options' => array( array('text' => tra('No'), 'value' => '0'), array('text' => tra('Yes'), 'value' => '1'), ), 'filter' => 'digits', 'default' => '1', ), 'requireinput' => array( 'required' => false, 'name' => tra('Require Input'), 'description' => tra('Require first input field to be filled for search to trigger'), 'since' => '12.0', 'options' => array( array('text' => tra('No'), 'value' => '0'), array('text' => tra('Yes'), 'value' => '1'), ), 'filter' => 'digits', 'default' => '0', ), 'forcesortmode' => array( 'required' => false, 'name' => tra('Force Sort'), 'description' => tra('Force the use of specified sort mode in place of search relevance even when there is a text search query'), 'since' => '13.0', 'options' => array( array('text' => tra('No'), 'value' => '0'), array('text' => tra('Yes'), 'value' => '1'), ), 'filter' => 'digits', 'default' => '1', ), 'trimlinefeeds' => array( 'required' => false, 'name' => tra('Trim Linefeeds'), 'description' => tra('Remove the linefeeds added after each input which casues the wiki parser to add extra paragraphs.'), 'since' => '14.1', 'options' => array( array('text' => tra('No'), 'value' => '0'), array('text' => tra('Yes'), 'value' => '1'), ), 'filter' => 'digits', 'default' => '0', ), ), ); } function wikiplugin_customsearch($data, $params) { global $prefs; if (empty($params['wiki']) && empty($params['tpl'])) { return tra('Template is not specified'); } elseif (!empty($params['wiki']) && !TikiLib::lib('tiki')->page_exists($params['wiki'])) { return tra('Template page not found'); } if (isset($params['id'])) { $id = $params['id']; } else { $id = '0'; } if (isset($params['recalllastsearch']) && $params['recalllastsearch'] == 1 && (!isset($_REQUEST['forgetlastsearch']) || $_REQUEST['forgetlastsearch'] != 'y')) { $recalllastsearch = 1; } else { $recalllastsearch = 0; } if (isset($params['searchfadediv'])) { $searchfadediv = $params['searchfadediv']; } else { $searchfadediv = ''; } if (!isset($params['requireinput'])) { $params['requireinput'] = 0; } if (!isset($params['forcesortmode'])) { $params['forcesortmode'] = 1; } if (!isset($_REQUEST["offset"])) { $offset = 0; } else { $offset = (int) $_REQUEST["offset"]; } if (isset($_REQUEST['maxRecords'])) { $maxRecords = (int) $_REQUEST['maxRecords']; } elseif ($recalllastsearch && !empty($_SESSION["customsearch_$id"]['maxRecords'])) { $maxRecords = (int) $_SESSION["customsearch_$id"]['maxRecords']; } else { $maxRecords = (int) $prefs['maxRecords']; $maxDefault = true; } if (!empty($_REQUEST['sort_mode'])) { $sort_mode = $_REQUEST['sort_mode']; } elseif ($recalllastsearch && !empty($_SESSION["customsearch_$id"]['sort_mode'])) { $sort_mode = $_SESSION["customsearch_$id"]['sort_mode']; } else { $sort_mode = ''; } if (!isset($params['searchonload'])) { $params['searchonload'] = 1; } if (!isset($params['requireinput'])) { $params['requireinput'] = false; } $definitionKey = md5($data); $matches = WikiParser_PluginMatcher::match($data); $query = new Search_Query; $query->filterIdentifier('y', 'searchable'); $builder = new Search_Query_WikiBuilder($query); $builder->apply($matches); $paginationArguments = $builder->getPaginationArguments(); // Use maxRecords set in LIST parameters rather then global default if set. if (isset($maxDefault) && $maxDefault) { if (!empty($paginationArguments['max'])) { $maxRecords = $paginationArguments['max']; } } // setup AJAX pagination $paginationArguments['offset_jsvar'] = "customsearch_$id.offset"; $paginationArguments['sort_jsvar'] = "customsearch_$id.sort_mode"; $paginationArguments['_onclick'] = "$('#customsearch_$id').submit();return false;"; $builder = new Search_Formatter_Builder; $builder->setPaginationArguments($paginationArguments); $builder->apply($matches); $formatter = $builder->getFormatter(); $facets = new Search_Query_FacetWikiBuilder; $facets->apply($matches); $cachelib = TikiLib::lib('cache'); $cachelib->cacheItem( $definitionKey, serialize( array( 'query' => $query, 'formatter' => $formatter, 'facets' => $facets, ) ), 'customsearch' ); if (!empty($params['wiki'])) { $wikitpl = "tplwiki:" . $params['wiki']; } else { $wikitpl = $params['tpl']; } $wikicontent = TikiLib::lib('smarty')->fetch($wikitpl); TikiLib::lib('parser')->parse_wiki_argvariable($wikicontent); $matches = WikiParser_PluginMatcher::match($wikicontent); $fingerprint = md5($wikicontent); $sessionprint = "customsearch_{$id}_$fingerprint"; if (isset($_SESSION[$sessionprint]) && $_SESSION[$sessionprint] != $fingerprint) { unset($_SESSION["customsearch_$id"]); } $_SESSION[$sessionprint] = $fingerprint; // important that offset from session is set after fingerprint check otherwise blank page might show if ($recalllastsearch && !isset($_REQUEST['offset']) && !empty($_SESSION["customsearch_$id"]["offset"])) { $offset = (int) $_SESSION["customsearch_$id"]["offset"]; } $options = array( 'searchfadetext' => tr('Loading...'), 'searchfadediv' => $searchfadediv, 'results' => empty($params['destdiv']) ? "#customsearch_{$id}_results" : "#{$params['destdiv']}", 'autosearchdelay' => isset($params['autosearchdelay']) ? max(1500, (int) $params['autosearchdelay']) : 0, 'searchonload' => (int) $params['searchonload'], 'requireinput' => (bool) $params['requireinput'], 'origrequireinput' => (bool) $params['requireinput'], 'forcesortmode' => (bool) $params['forcesortmode'], ); /** * NOTES: Search Execution * * There is a global delay on execution of 1 second. This makes sure * multiple submissions will never trigger multiple requests. * * There is an additional autosearchdelay configuration that can trigger the search * on field change rather than explicit request. Explicit requests will still work. */ $script = " var customsearch = { options: " . json_encode($options) . ", id: " . json_encode($id) . ", offset: 0, searchdata: {}, definition: " . json_encode((string) $definitionKey) . ", autoTimeout: null, add: function (fieldId, filter) { this.searchdata[fieldId] = filter; this.auto(); }, remove: function (fieldId) { delete this.searchdata[fieldId]; this.auto(); }, load: function () { this._executor(this); }, auto: function () { }, _executor: delayedExecutor(1000, function (cs) { var selector = '#' + cs.options.searchfadediv; if (cs.options.searchfadediv.length <= 1 && $(selector).length === 0) { selector = '#customsearch_' + cs.id; } $(selector).tikiModal(cs.options.searchfadetext); cs._load(function (data) { $(selector).tikiModal(); $(cs.options.results).html(data); $(document).trigger('pageSearchReady'); }); cs.store_query = ''; }), init: function () { var that = this; if (that.options.searchonload) { that.load(); } if (that.options.autosearchdelay) { that.auto = delayedExecutor(that.options.autosearchdelay, function () { if (that.options.requireinput && (!$('#customsearch_$id').find(':text').val() || $('#customsearch_$id').find(':text').val().indexOf('...') > 0)) { return false; } that.load(); }); } } }; $('#customsearch_$id').click(function() { customsearch.offset = 0; }); $('#customsearch_$id').submit(function() { if (customsearch.options.requireinput && (!$(this).find(':text').val() || $(this).find(':text').val().indexOf('...') > 0)) { alert(tr('Please enter a search query')); return false; } if (customsearch.options.origrequireinput != customsearch.options.requireinput) { customsearch.options.requireinput = customsearch.options.origrequireinput; } customsearch.load(); return false; }); window.customsearch_$id = customsearch; "; $parser = new WikiParser_PluginArgumentParser; foreach ($matches as $match) { $name = $match->getName(); $arguments = $parser->parse($match->getArguments()); $key = $match->getInitialStart(); $fieldid = "customsearch_{$id}_$key"; if (isset($arguments['id'])) { $fieldid = $arguments['id']; } if ($name == 'sort' && !empty($arguments['mode']) && empty($sort_mode)) { $sort_mode = $arguments['mode']; $match->replaceWith(''); continue; } if (!empty($arguments['_field']) && !empty($arguments['_filter']) && $arguments['_filter'] == 'content') { $filter = $arguments['_field']; } elseif (empty($arguments['_field']) && !empty($arguments['_filter']) && $arguments['_filter'] == 'content') { $filter = 'content'; } else { $filter = ''; } if ( $filter && !empty($_REQUEST['default'][$filter]) ) { $default = $_REQUEST['default'][$filter]; } elseif ($recalllastsearch && isset($_SESSION["customsearch_$id"][$fieldid])) { $default = $_SESSION["customsearch_$id"][$fieldid]; } elseif (!empty($arguments['_default'])) { if (strpos($arguments['_default'], ',') !== false) { $default = explode(',', $arguments['_default']); } else { $default = $arguments['_default']; } } else { $default = ''; } if ( $name == 'categories' ) { $parent = $arguments['_parent']; if (!empty($_REQUEST['defaultcat'][$parent])) { $default = $_REQUEST['defaultcat'][$parent]; } } $function = "cs_design_{$name}"; if (function_exists($function)) { if (isset($arguments['_group'])) { $fieldname = "customsearch_{$id}_gr" . $arguments['_group']; } elseif (isset($arguments['_textrange'])) { $fieldname = "customsearch_{$id}_textrange" . $arguments['_textrange']; } elseif (isset($arguments['_daterange'])) { $fieldname = "customsearch_{$id}_daterange" . $arguments['_daterange']; } else { $fieldname = $fieldid; } $html = $function($id, $fieldname, $fieldid, $arguments, $default, $script); if ($params['trimlinefeeds']) { $html = trim($html); } $match->replaceWith($html); } } $callbackScript = null; if (!empty($params['callbackscript']) && TikiLib::lib('tiki')->page_exists($params['callbackscript'])) { $callbackscript_tpl = "wiki:" . $params['callbackscript']; $callbackScript = TikiLib::lib('smarty')->fetch($callbackscript_tpl); } global $page; $script .= " customsearch._load = function (receive) { var datamap = { definition: this.definition, adddata: $.toJSON(this.searchdata), searchid: this.id, offset: customsearch.offset, maxRecords: this.maxRecords, store_query: this.store_query, page: " . json_encode($page) . ", recalllastsearch: $recalllastsearch }; if (!customsearch.options.forcesortmode && $('#customsearch_$id').find(':text').val() && $('#customsearch_$id').find(':text').val().indexOf('...') <= 0) { customsearch.sort_mode = 'score_desc'; } if (customsearch.sort_mode) { // blank sort_mode is not allowed by Tiki input filter datamap.sort_mode = customsearch.sort_mode; } $.ajax({ type: 'POST', url: $.service('search_customsearch', 'customsearch'), data: datamap, dataType: 'html', success: function(data) { receive(data); $callbackScript; } }); }; customsearch.sort_mode = " . json_encode($sort_mode) . "; customsearch.offset = $offset; customsearch.maxRecords = $maxRecords; customsearch.store_query =''; customsearch.init(); "; TikiLib::lib('header')->add_jq_onready($script); $out = '<div id="customsearch_' . $id . '_form"><form id="customsearch_' . $id . '">' . $matches->getText() . '</form></div>'; if (empty($params['destdiv'])) { $out .= '<div id="customsearch_' . $id . '_results"></div>'; } if (!empty($params['wiki'])) { return $out; } else { // If using smarty tpl should assume it's all HTML $out = str_replace('~np~', '', $out); $out = str_replace('~/np~', '', $out); return '~np~' . $out . '~/np~'; } } function cs_design_setbasic($element, $fieldid, $fieldname, $arguments) { $element->setAttribute('id', $fieldid); $element->setAttribute('name', $fieldname); foreach ($arguments as $k => $v) { if (substr($k, 0, 1) != '_') { $element->setAttribute($k, $v); } } } function cs_design_input($id, $fieldname, $fieldid, $arguments, $default, &$script) { $document = new DOMDocument; $element = $document->createElement('input'); cs_design_setbasic($element, $fieldid, $fieldname, $arguments); if (!empty($default)) { $arguments['default'] = $default; } $script .= " (function (id, config, fieldname) { var field = $('#' + id); field.change(function() { var filter = { config: config, name: 'input', value: $(this).val() }; if ($(this).is(':checkbox, :radio')) { filter.value = $(this).is(':checked'); } if ($(this).is(':radio')) { $(this).closest('form').find(':radio') .filter(function () { return $(this).attr('name') == fieldname }) .each(function() { customsearch.remove($(this).attr('id')); }); } customsearch.add($(this).attr('id'), filter); }); if (config['default'] || $(field).attr('type') === 'hidden') { field.change(); } })('$fieldid', " . json_encode($arguments) . ", " . json_encode($fieldname) . "); "; $arguments = new JitFilter($arguments); $default = $arguments->default->text(); $type = $arguments->type->word(); if ($default && $type != "hidden") { if ((string) $default != 'n' && ($type == 'checkbox' || $type == 'radio')) { $element->setAttribute('checked', 'checked'); } else { $element->setAttribute('value', $default); } } $document->appendChild($element); return $document->saveHTML(); } function cs_design_categories($id, $fieldname, $fieldid, $arguments, $default, &$script) { $document = new DOMDocument; extract($arguments, EXTR_SKIP); if (!isset($_style)) { $_style = 'select'; } if (empty($_group) && ($_style == 'checkbox' || $_style == 'radio')) { return tr("_group is needed to be set if _style is checkbox or radio"); } $showSubcategories = isset($_showdeep) && $_showdeep != 'n'; if (isset($_parent) && ctype_digit($_parent) && $_parent > 0) { $filter = array('identifier'=>$_parent, 'type'=> $showSubcategories ? 'descendants' : 'children'); } else { $filter = array('type'=>$showSubcategories ? 'all' : 'roots'); } if (!isset($_categpath)) { $_categpath = false; } else { $_categpath = ($_categpath === 'y'); } $cats = TikiLib::lib('categ')->getCategories($filter); $element = $document->createElement('select'); cs_design_setbasic($element, $fieldid, $fieldname, $arguments); $document->appendChild($element); if ($_style == 'checkbox' || $_style == 'radio') { $currentlevel = 0; $orig_fieldid = $fieldid; foreach ($cats as $c) { $categId = $c['categId']; $fieldid = $orig_fieldid . "_cat$categId"; $level = count($c['tepath']); if ($level > $currentlevel) { $ul{$level} = $document->createElement('ul'); if ($currentlevel) { $ul{$currentlevel}->appendChild($ul{$level}); } else { $document->appendChild($ul{$level}); } $currentlevel = $level; } elseif ($level < $currentlevel) { $currentlevel = $level; } $li = $document->createElement('li'); $ul{$currentlevel}->appendChild($li); $input = $document->createElement('input'); $input->setAttribute('type', $_style); cs_design_setbasic($input, $fieldid, $fieldname, $arguments); $input->setAttribute('value', $categId); $li->appendChild($input); $label = $document->createTextNode($_categpath ? $c['relativePathString'] : $c['name']); $li->appendChild($label); if ($_style == 'radio') { $radioreset = "$('input[type=radio][name=$fieldname]').each(function() { customsearch.remove($(this).attr('id')); });" ; } else { $radioreset = ''; } $script .= " $('#$fieldid').change(function() { if ($(this).is(':checked')) { var filter = { config = " . json_encode($arguments) . ", name = 'categories', value = $(this).val() } $radioreset customsearch.add('$fieldid', filter); } else { customsearch.remove('$fieldid', filter); } }); "; if ($default && in_array($c['categId'], (array) $default)) { $element->setAttribute('checked', 'checked'); $script .= " $('#$fieldid').trigger('change'); "; } } } elseif ($_style == 'select') { // leave a blank one in the front if (!isset($arguments['multiple']) && !isset($arguments['size']) || isset($arguments['_firstlabel'])) { if (!empty($arguments['_firstlabel'])) { $label = $arguments['_firstlabel']; } else { $label = ''; } $option = $document->createElement('option', $label); $option->setAttribute('value', ''); $element->appendChild($option); } $script .= " $('#$fieldid').change(function() { customsearch.add('$fieldid', { config: " . json_encode($arguments) . ", name: 'categories', value: $(this).val() }); }); "; foreach ($cats as $c) { $option = $document->createElement('option', $_categpath ? $c['relativePathString'] : $c['name']); $option->setAttribute('value', $c['categId']); $element->appendChild($option); if ($default && in_array($c['categId'], (array) $default)) { $option->setAttribute('selected', 'selected'); $script .= " $('#$fieldid').trigger('change'); "; } } } return '~np~' . $document->saveHTML() . '~/np~'; } function cs_design_select($id, $fieldname, $fieldid, $arguments, $default, &$script) { $document = new DOMDocument; $element = $document->createElement('select'); cs_design_setbasic($element, $fieldid, $fieldname, $arguments); $document->appendChild($element); if (isset($arguments['_labels'])) { $labels = explode(',', $arguments['_labels']); } else { $labels = array(); } if (isset($arguments['_options'])) { $options = explode(',', $arguments['_options']); } else { $options = array(); } // get the options for an ItemLink field - needs _trackerId and _field set in the {select} plugin if (empty($options) && empty($labels) && isset($arguments['_field']) && strpos($arguments['_field'], 'tracker_field_') === 0 && !empty($arguments['_trackerId'])) { $definition = Tracker_Definition::get($arguments['_trackerId']); $field = $definition->getFieldFromPermName(str_replace('tracker_field_', '', $arguments['_field'])); $handler = TikiLib::lib('trk')->get_field_handler($field); if ($field['type'] === 'r') { // Item Link $labels = $handler->getItemList(); $options = array_keys($labels); $labels = array_values($labels); } else if ($field['type'] === 't') { // Text field so get all values up to a sensible(?) amount global $prefs; // turns out using a straight "old fashioned" DISTINCT MySQL query here is the most efficient $result = TikiLib::lib('tiki')->query( 'SELECT DISTINCT `value` FROM `tiki_tracker_item_fields` WHERE `fieldId` = ? ORDER BY `value` LIMIT 1000;', $field['fieldId'] ); while ($row = $result->fetchRow()) { $label = $row['value']; if (! in_array($label, $labels)) { $labels[] = $label; $label = explode(' ', $label); foreach($label as & $word) { if ($prefs['unified_engine'] !== 'mysql') { if (in_array($word, $prefs['unified_stopwords'])) { $word = ''; } } else { if (strlen($word) < 4) { // default mysql fulltext minimum word length TODO find current value for ft_min_word_len $word = ''; } } } $options[] = implode(' AND ', array_filter($label)); } } } else if ($field['type'] === 'u') { // User Selector (only when in dropdown list mode) $html = $handler->renderInput(); $html = preg_replace('/id=[\'"].*?[\'"]/', 'id="' . $fieldid . '"', $html); $script .= " $('#$fieldid').change(function() { customsearch.add('$fieldid', { config: " . json_encode($arguments) . ", name: 'select', value: $(this).val() }); }); "; return $html; } } if (isset($arguments['_mandatory']) && $arguments['_mandatory'] == 'y') { $mandatory = true; } else { $mandatory = false; } // leave a blank one in the front if (!$mandatory && !isset($arguments['multiple']) && !isset($arguments['size']) || isset($arguments['_firstlabel'])) { if (!empty($arguments['_firstlabel'])) { $label = $arguments['_firstlabel']; } else { $label = ''; } $option = $document->createElement('option', $label); $option->setAttribute('value', ''); $element->appendChild($option); } $script .= " $('#$fieldid').change(function() { customsearch.add('$fieldid', { config: " . json_encode($arguments) . ", name: 'select', value: $(this).val() }); }); "; foreach ($options as $k => $opt) { if (!empty($labels[$k])) { $body = $labels[$k]; } else { $body = $opt; } $option = $document->createElement('option', $body); $option->setAttribute('value', $opt); if ($default && in_array($opt, (array) $default)) { $option->setAttribute('selected', 'selected'); $script .= " $('#$fieldid').trigger('change'); "; } $element->appendChild($option); } return $document->saveHTML(); } function cs_design_daterange($id, $fieldname, $fieldid, $arguments, $default, &$script) { extract($arguments, EXTR_SKIP); $smarty = TikiLib::lib('smarty'); $smarty->loadPlugin('smarty_function_jscalendar'); $params_from = array(); $params_to = array(); if (!empty($_showtime) && $_showtime == 'y') { $params_from['showtime'] = 'y'; $params_to['showtime'] = 'y'; } else { $params_from['showtime'] = 'n'; $params_to['showtime'] = 'n'; } $params_from['fieldname'] = $fieldname . '_from'; $params_to['fieldname'] = $fieldname . '_to'; $params_from['id'] = $fieldid_from = $fieldid . '_from'; $params_to['id'] = $fieldid_to = $fieldid . '_to'; if (!empty($_from)) { if ($_from == 'now') { $params_from['date'] = TikiLib::lib('tiki')->now; } else { $params_from['date'] = $_from; } if (empty($_to)) { if (empty($_gap)) { $_gap = 365 * 24 * 3600; } $params_to['date'] = $params_from['date'] + $_gap; } } else { $params_from['date'] = TikiLib::lib('tiki')->now; } if (!empty($_to)) { if ($_to == 'now') { $params_to['date'] = TikiLib::lib('tiki')->now; } else { $params_to['date'] = $_to; } if (empty($_from)) { if (empty($_gap)) { $_gap = 365 * 24 * 3600; } $params_from['date'] = $params_to['date'] - $_gap; } } elseif (empty($params_to['date'])) { $params_to['date'] = TikiLib::lib('tiki')->now + 365 * 24 * 3600; } $picker = ''; $picker .= smarty_function_jscalendar($params_from, $smarty); $picker .= smarty_function_jscalendar($params_to, $smarty); $script .= " $('#{$fieldid_from}_dptxt,#{$fieldid_to}_dptxt').change(function() { var from = $('#$fieldid_from').val(); var to = $('#$fieldid_to').val(); from = from.substr(0,10);to = to.substr(0,10); // prevent trailing 000 from date picker customsearch.add('$fieldid', { config: " . json_encode($arguments) . ", name: 'daterange', value: from + ',' + to }); }); "; return $picker; } function cs_design_store($id, $fieldname, $fieldid, $arguments, $default, &$script) { global $prefs; if ($prefs['storedsearch_enabled'] != 'y') { return; } $document = new DOMDocument; $element = $document->createElement('input'); $element->setAttribute('type', 'submit'); cs_design_setbasic($element, $fieldid, $fieldname, $arguments); $document->appendChild($element); $script .= " $('#$fieldid').click(function() { $(this).serviceDialog({ title: $(this).val(), controller: 'search_stored', action: 'select', success: function (data) { customsearch.store_query = data.queryId; customsearch.load(); } }); return false; }); "; return $document->saveHTML(); }