]> CyberLeo.Net >> Repos - Github/sugarcrm.git/blob - include/ListView/ListView.php
Release 6.5.3
[Github/sugarcrm.git] / include / ListView / ListView.php
1 <?php
2 if(!defined('sugarEntry') || !sugarEntry) die('Not A Valid Entry Point');
3 /*********************************************************************************
4  * SugarCRM Community Edition is a customer relationship management program developed by
5  * SugarCRM, Inc. Copyright (C) 2004-2012 SugarCRM Inc.
6  * 
7  * This program is free software; you can redistribute it and/or modify it under
8  * the terms of the GNU Affero General Public License version 3 as published by the
9  * Free Software Foundation with the addition of the following permission added
10  * to Section 15 as permitted in Section 7(a): FOR ANY PART OF THE COVERED WORK
11  * IN WHICH THE COPYRIGHT IS OWNED BY SUGARCRM, SUGARCRM DISCLAIMS THE WARRANTY
12  * OF NON INFRINGEMENT OF THIRD PARTY RIGHTS.
13  * 
14  * This program is distributed in the hope that it will be useful, but WITHOUT
15  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
16  * FOR A PARTICULAR PURPOSE.  See the GNU Affero General Public License for more
17  * details.
18  * 
19  * You should have received a copy of the GNU Affero General Public License along with
20  * this program; if not, see http://www.gnu.org/licenses or write to the Free
21  * Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
22  * 02110-1301 USA.
23  * 
24  * You can contact SugarCRM, Inc. headquarters at 10050 North Wolfe Road,
25  * SW2-130, Cupertino, CA 95014, USA. or at email address contact@sugarcrm.com.
26  * 
27  * The interactive user interfaces in modified source and object code versions
28  * of this program must display Appropriate Legal Notices, as required under
29  * Section 5 of the GNU Affero General Public License version 3.
30  * 
31  * In accordance with Section 7(b) of the GNU Affero General Public License version 3,
32  * these Appropriate Legal Notices must retain the display of the "Powered by
33  * SugarCRM" logo. If the display of the logo is not reasonably feasible for
34  * technical reasons, the Appropriate Legal Notices must display the words
35  * "Powered by SugarCRM".
36  ********************************************************************************/
37
38 require_once('include/EditView/SugarVCR.php');
39 /**
40  * ListView - list of many objects
41  * @api
42  */
43 class ListView
44 {
45     var $local_theme= null;
46     var $local_app_strings= null;
47     var $local_image_path = null;
48     var $local_current_module = null;
49     var $local_mod_strings = null;
50     var $records_per_page = 20;
51     var $xTemplate = null;
52     var $xTemplatePath = null;
53     var $seed_data = null;
54     var $query_where = null;
55     var $query_limit = -1;
56     var $query_orderby = null;
57     var $header_title = '';
58     var $header_text = '';
59     var $initialized = false;
60     var $show_export_button = true;
61     var $show_delete_button = true;
62     var $show_select_menu = true;
63     var $show_paging = true;
64     var $show_mass_update = true;
65     var $show_mass_update_form = true;
66     var $query_where_has_changed = false;
67     var $display_header_and_footer = true;
68     var $baseURL = '';
69     var $is_dynamic = false;
70     var $inline = false;
71     var $start_link_wrapper = '';
72     var $end_link_wrapper = '';
73     var $list_field_defs = array();
74
75     var $shouldProcess = false;
76     var $data_array;
77     var $related_field_name = '';
78     var $child_focus = '';
79     var $layout_manager = null;
80     var $process_for_popups = false;
81     var $multi_select_popup=false;
82     var $_additionalDetails = false;
83     var $additionalDetailsFunction = null;
84     var $sort_order = '';
85     var $force_mass_update=false;
86     var $keep_mass_update_form_open=false;
87     var $ignorePopulateOnly = false;
88
89 function setDataArray($value) {
90     $this->data_array = $value;
91 }
92
93 function processListViewMulti($seed, $xTemplateSection, $html_varName) {
94
95     $this->shouldProcess = true;
96
97     echo "<form name='MassUpdate' method='post' action='index.php'>";
98     $this->processListViewTwo($seed, $xTemplateSection, $html_varName);
99
100     echo "<a href='javascript:" . ((!$this->multi_select_popup) ? 'sListView.' : ''). "check_all(document.MassUpdate, \"mass[]\", true)'>".translate('LBL_CHECKALL')."</a> - <a href='javascript:sListView.check_all(document.MassUpdate, \"mass[]\", false);'>".translate('LBL_CLEARALL')."</a>";
101     echo '<br><br>';
102 }
103
104
105 function processListView($seed, $xTemplateSection, $html_varName)
106 {
107     global $sugar_config;
108
109     $populateOnly = $this->ignorePopulateOnly ? FALSE : (!empty($sugar_config['save_query']) && $sugar_config['save_query'] == 'populate_only');
110     if(isset($seed->module_dir) && $populateOnly) {
111         if(empty($GLOBALS['displayListView']) && strcmp(strtolower($_REQUEST['action']), 'popup') != 0 && (!empty($_REQUEST['clear_query']) || $_REQUEST['module'] == $seed->module_dir && ((empty($_REQUEST['query']) || $_REQUEST['query'] == 'MSI')&& (empty($_SESSION['last_search_mod']) || $_SESSION['last_search_mod'] != $seed->module_dir)))) {
112             $_SESSION['last_search_mod'] = $_REQUEST['module'] ;
113             return;
114         }
115     }
116     if(strcmp(strtolower($_REQUEST['action']), 'popup') != 0){
117         $_SESSION['last_search_mod'] = $_REQUEST['module'] ;
118     }
119     //following session variable will track the detail view navigation history.
120     //needs to the reset after each search.
121     $this->setLocalSessionVariable($html_varName,"DETAIL_NAV_HISTORY",false);
122
123     require_once('include/MassUpdate.php');
124     $mass = new MassUpdate();
125     $add_acl_javascript = false;
126     if(!isset($_REQUEST['action'])) {
127         $this->shouldProcess=false;
128     } else {
129     $this->shouldProcess = is_subclass_of($seed, "SugarBean")
130         && (($_REQUEST['action'] == 'index') || ('ListView' == substr($_REQUEST['action'],0,8)) /* cn: to include all ListViewXXX.php type views */)
131         && ($_REQUEST['module'] == $seed->module_dir);
132     }
133
134     //when processing a multi-select popup.
135     if($this->process_for_popups && $this->multi_select_popup)  $this->shouldProcess =true;
136     //mass update turned off?
137     if(!$this->show_mass_update) $this->shouldProcess = false;
138     if(is_subclass_of($seed, "SugarBean")) {
139         if($seed->bean_implements('ACL')) {
140             if(!ACLController::checkAccess($seed->module_dir,'list',true)) {
141                 if($_REQUEST['module'] != 'Home') {
142                     ACLController::displayNoAccess();
143                 }
144                 return;
145             }
146             if(!ACLController::checkAccess($seed->module_dir,'export',true)) {
147                 $sugar_config['disable_export']= true;
148             }
149
150         }
151     }
152
153     //force mass update form if requested.
154     if($this->force_mass_update) {
155         $this->shouldProcess = true;
156     }
157
158     if($this->shouldProcess) {
159         echo $mass->getDisplayMassUpdateForm(true, $this->multi_select_popup);
160         echo $mass->getMassUpdateFormHeader($this->multi_select_popup);
161         $mass->setSugarBean($seed);
162
163         //C.L. Fix for 10048, do not process handleMassUpdate for multi select popups
164         if(!$this->multi_select_popup) {
165             $mass->handleMassUpdate();
166         }
167     }
168
169     $this->processListViewTwo($seed,$xTemplateSection, $html_varName);
170
171     if($this->shouldProcess && empty($this->process_for_popups)) {
172         //echo "<a href='javascript:sListView.clear_all(document.MassUpdate, \"mass[]\");'>".translate('LBL_CLEARALL')."</a>";
173         // cn: preserves current functionality, exception is InboundEmail
174         if($this->show_mass_update_form) {
175             echo $mass->getMassUpdateForm();
176         }
177         if(!$this->keep_mass_update_form_open) {
178             echo $mass->endMassUpdateForm();
179         }
180     }
181 }
182
183
184 function process_dynamic_listview($source_module, $sugarbean,$subpanel_def)
185 {
186         $this->source_module = $source_module;
187         $this->subpanel_module = $subpanel_def->name;
188         if(!isset($this->xTemplate))
189             $this->createXTemplate();
190
191         $html_var = $this->subpanel_module . "_CELL";
192
193         $list_data = $this->processUnionBeans($sugarbean,$subpanel_def, $html_var);
194
195         $list = $list_data['list'];
196         $parent_data = $list_data['parent_data'];
197
198         if($subpanel_def->isCollection()) {
199             $thepanel=$subpanel_def->get_header_panel_def();
200         } else {
201             $thepanel=$subpanel_def;
202         }
203
204
205
206         $this->process_dynamic_listview_header($thepanel->get_module_name(), $thepanel, $html_var);
207         $this->process_dynamic_listview_rows($list,$parent_data, 'dyn_list_view', $html_var,$subpanel_def);
208
209         if($this->display_header_and_footer)
210         {
211             $this->getAdditionalHeader();
212             if(!empty($this->header_title))
213             {
214                 echo get_form_header($this->header_title, $this->header_text, false);
215             }
216         }
217
218         $this->xTemplate->out('dyn_list_view');
219
220         if(isset($_SESSION['validation']))
221         {
222             print base64_decode('PGEgaHJlZj0naHR0cDovL3d3dy5zdWdhcmNybS5jb20nPlBPV0VSRUQmbmJzcDtCWSZuYnNwO1NVR0FSQ1JNPC9hPg==');
223         }
224         if(isset($list_data['query'])) {
225             return ($list_data['query']);
226         }
227     }
228
229 /**
230  * @return void
231  * @param unknown $data
232  * @param unknown $xTemplateSection
233  * @param unknown $html_varName
234  * @desc INTERNAL FUNCTION handles the rows
235  */
236  function process_dynamic_listview_rows($data,$parent_data, $xtemplateSection, $html_varName, $subpanel_def)
237  {
238     global $subpanel_item_count;
239     global $odd_bg;
240     global $even_bg;
241     global $hilite_bg;
242     global $click_bg;
243
244     $this->xTemplate->assign("BG_HILITE", $hilite_bg);
245     $this->xTemplate->assign('CHECKALL', SugarThemeRegistry::current()->getImage('blank', '', 1, 1, ".gif", ''));
246     //$this->xTemplate->assign("BG_CLICK", $click_bg);
247     $subpanel_item_count = 0;
248     $oddRow = true;
249     $count = 0;
250     reset($data);
251
252     //GETTING OFFSET
253     $offset = $this->getOffset($html_varName);
254     //$totaltime = 0;
255     $processed_ids = array();
256
257     $fill_additional_fields = array();
258     //Either retrieve the is_fill_in_additional_fields property from the lone
259     //subpanel or visit each subpanel's subpanels to retrieve the is_fill_in_addition_fields
260     //property
261     $subpanel_list=array();
262     if($subpanel_def->isCollection()) {
263         $subpanel_list=$subpanel_def->sub_subpanels;
264     } else {
265         $subpanel_list[]= $subpanel_def;
266     }
267
268     foreach($subpanel_list as $this_subpanel)
269     {
270         if($this_subpanel->is_fill_in_additional_fields())
271         {
272             $fill_additional_fields[] = $this_subpanel->bean_name;
273             $fill_additional_fields[$this_subpanel->bean_name] = true;
274         }
275     }
276
277     if ( empty($data) ) {
278         $this->xTemplate->assign("ROW_COLOR", 'oddListRow');
279         $thepanel=$subpanel_def;
280         if($subpanel_def->isCollection())
281             $thepanel=$subpanel_def->get_header_panel_def();
282         $this->xTemplate->assign("COL_COUNT", count($thepanel->get_list_fields()));
283         $this->xTemplate->parse($xtemplateSection.".nodata");
284     }
285     while(list($aVal, $aItem) = each($data))
286     {
287         $subpanel_item_count++;
288         $aItem->check_date_relationships_load();
289         // TODO: expensive and needs to be removed and done better elsewhere
290
291         if(!empty($fill_additional_fields[$aItem->object_name])
292         || ($aItem->object_name == 'Case' && !empty($fill_additional_fields['aCase']))
293         )
294         {
295             $aItem->fill_in_additional_list_fields();
296             //$aItem->fill_in_additional_detail_fields();
297         }
298         //rrs bug: 25343
299         $aItem->call_custom_logic("process_record");
300
301         if(isset($parent_data[$aItem->id])) {
302
303             $aItem->parent_name = $parent_data[$aItem->id]['parent_name'];
304             if(!empty($parent_data[$aItem->id]['parent_name_owner'])) {
305             $aItem->parent_name_owner =  $parent_data[$aItem->id]['parent_name_owner'];
306             $aItem->parent_name_mod =  $parent_data[$aItem->id]['parent_name_mod'];
307         }}
308         $fields = $aItem->get_list_view_data();
309         if(isset($processed_ids[$aItem->id])) {
310             continue;
311
312         } else {
313             $processed_ids[$aItem->id] = 1;
314         }
315
316
317         //ADD OFFSET TO ARRAY
318         $fields['OFFSET'] = ($offset + $count + 1);
319
320         if($this->shouldProcess) {
321             if($aItem->ACLAccess('EditView')) {
322             $this->xTemplate->assign('PREROW', "<input type='checkbox' class='checkbox' name='mass[]' value='". $fields['ID']. "' />");
323             } else {
324                 $this->xTemplate->assign('PREROW', '');
325
326             }
327             if($aItem->ACLAccess('DetailView')) {
328                 $this->xTemplate->assign('TAG_NAME','a');
329             } else {
330                 $this->xTemplate->assign('TAG_NAME','span');
331             }
332             $this->xTemplate->assign('CHECKALL', "<input type='checkbox'  title='".$GLOBALS['app_strings']['LBL_SELECT_ALL_TITLE']."' class='checkbox' name='massall' id='massall' value='' onclick='sListView.check_all(document.MassUpdate, \"mass[]\", this.checked);' />");
333         }
334
335         if($oddRow)
336         {
337             $ROW_COLOR = 'oddListRow';
338             $BG_COLOR =  $odd_bg;
339         }
340         else
341         {
342             $ROW_COLOR = 'evenListRow';
343             $BG_COLOR =  $even_bg;
344         }
345         $oddRow = !$oddRow;
346                 $button_contents = array();
347         $this->xTemplate->assign("ROW_COLOR", $ROW_COLOR);
348         $this->xTemplate->assign("BG_COLOR", $BG_COLOR);
349         $layout_manager = $this->getLayoutManager();
350         $layout_manager->setAttribute('context','List');
351         $layout_manager->setAttribute('image_path',$this->local_image_path);
352         $layout_manager->setAttribute('module_name', $subpanel_def->_instance_properties['module']);
353         if(!empty($this->child_focus))
354             $layout_manager->setAttribute('related_module_name',$this->child_focus->module_dir);
355
356         //AG$subpanel_data = $this->list_field_defs;
357         //$bla = array_pop($subpanel_data);
358         //select which sub-panel to display here, the decision will be made based on the type of
359         //the sub-panel and panel in the bean being processed.
360         if($subpanel_def->isCollection()) {
361             $thepanel=$subpanel_def->sub_subpanels[$aItem->panel_name];
362         } else {
363             $thepanel=$subpanel_def;
364         }
365         //get data source name
366         $linked_field=$thepanel->get_data_source_name();
367         $linked_field_set=$thepanel->get_data_source_name(true);
368         static $count;
369         if(!isset($count))$count = 0;
370
371         $field_acl['DetailView'] = $aItem->ACLAccess('DetailView');
372         $field_acl['ListView'] = $aItem->ACLAccess('ListView');
373         $field_acl['EditView'] = $aItem->ACLAccess('EditView');
374         $field_acl['Delete'] = $aItem->ACLAccess('Delete');
375         foreach($thepanel->get_list_fields() as $field_name=>$list_field)
376         {
377             //add linked field attribute to the array.
378             $list_field['linked_field']=$linked_field;
379             $list_field['linked_field_set']=$linked_field_set;
380
381             $usage = empty($list_field['usage']) ? '' : $list_field['usage'];
382             if($usage != 'query_only')
383             {
384                 $list_field['name']=$field_name;
385
386                 $module_field = $field_name.'_mod';
387                 $owner_field = $field_name.'_owner';
388                 if(!empty($aItem->$module_field)) {
389
390                     $list_field['owner_id'] = $aItem->$owner_field;
391                     $list_field['owner_module'] = $aItem->$module_field;
392
393                 } else {
394                     $list_field['owner_id'] = false;
395                     $list_field['owner_module'] = false;
396                 }
397                 if(isset($list_field['alias'])) $list_field['name'] = $list_field['alias'];
398                 else $list_field['name']=$field_name;
399                 $list_field['fields'] = $fields;
400                 $list_field['module'] = $aItem->module_dir;
401                 $list_field['start_link_wrapper'] = $this->start_link_wrapper;
402                 $list_field['end_link_wrapper'] = $this->end_link_wrapper;
403                 $list_field['subpanel_id'] = $this->subpanel_id;
404                 $list_field += $field_acl;
405                 if ( isset($aItem->field_defs[strtolower($list_field['name'])])) {
406                     require_once('include/SugarFields/SugarFieldHandler.php');
407                     // We need to see if a sugar field exists for this field type first,
408                     // if it doesn't, toss it at the old sugarWidgets. This is for
409                     // backwards compatibility and will be removed in a future release
410                     $vardef = $aItem->field_defs[strtolower($list_field['name'])];
411                     if ( isset($vardef['type']) ) {
412                         $fieldType = isset($vardef['custom_type'])?$vardef['custom_type']:$vardef['type'];
413                         $tmpField = SugarFieldHandler::getSugarField($fieldType,true);
414                     } else {
415                         $tmpField = NULL;
416                     }
417
418                     if ( $tmpField != NULL ) {
419                         $widget_contents = SugarFieldHandler::displaySmarty($list_field['fields'],$vardef,'ListView',$list_field);
420                     } else {
421                         // No SugarField for this particular type
422                         // Use the old, icky, SugarWidget for now
423                         $widget_contents = $layout_manager->widgetDisplay($list_field);
424                     }
425
426                     if ( isset($list_field['widget_class']) && $list_field['widget_class'] == 'SubPanelDetailViewLink' ) {
427                         // We need to call into the old SugarWidgets for the time being, so it can generate a proper link with all the various corner-cases handled
428                         // So we'll populate the field data with the pre-rendered display for the field
429                         $list_field['fields'][$field_name] = $widget_contents;
430                         if('full_name' == $field_name){//bug #32465
431                            $list_field['fields'][strtoupper($field_name)] = $widget_contents;
432                         }
433
434                         //vardef source is non db, assign the field name to varname for processing of column.
435                         if(!empty($vardef['source']) && $vardef['source']=='non-db'){
436                             $list_field['varname'] = $field_name;
437
438                         }
439                         $widget_contents = $layout_manager->widgetDisplay($list_field);
440                     } else if(isset($list_field['widget_class']) && $list_field['widget_class'] == 'SubPanelEmailLink' ) {
441                         $widget_contents = $layout_manager->widgetDisplay($list_field);
442                     }
443
444                  $count++;
445                 $this->xTemplate->assign('CELL_COUNT', $count);
446                 $this->xTemplate->assign('CLASS', "");
447                 if ( empty($widget_contents) ) $widget_contents = '&nbsp;';
448                 $this->xTemplate->assign('CELL', $widget_contents);
449                 $this->xTemplate->parse($xtemplateSection.".row.cell");
450                 } else {
451                     // This handles the edit and remove buttons and icon widget
452                         if( isset($list_field['widget_class']) && $list_field['widget_class'] == "SubPanelIcon") {
453                                 $count++;
454                                 $widget_contents = $layout_manager->widgetDisplay($list_field);
455                                 $this->xTemplate->assign('CELL_COUNT', $count);
456                                 $this->xTemplate->assign('CLASS', "");
457                                 if ( empty($widget_contents) ) $widget_contents = '&nbsp;';
458                                 $this->xTemplate->assign('CELL', $widget_contents);
459                                 $this->xTemplate->parse($xtemplateSection.".row.cell");
460                         } elseif (preg_match("/button/i", $list_field['name'])) {
461                                 if($layout_manager->widgetDisplay($list_field) != "")
462                                 $button_contents[] = $layout_manager->widgetDisplay($list_field);
463                         } else {
464                                 $count++;
465                                 $this->xTemplate->assign('CLASS', "");
466                                 $widget_contents = $layout_manager->widgetDisplay($list_field);
467                                 $this->xTemplate->assign('CELL_COUNT', $count);
468                                 if ( empty($widget_contents) ) $widget_contents = '&nbsp;';
469                                 $this->xTemplate->assign('CELL', $widget_contents);
470                                 $this->xTemplate->parse($xtemplateSection.".row.cell");
471                         }
472                 }
473
474             }
475         }
476
477
478         // Make sure we have at least one button before rendering a column for
479         // the action buttons in a list view. Relevant bugs: #51647 and #51640.
480         if(isset($button_contents[0])) {
481             // this is for inline buttons on listviews
482             // bug#51275: smarty widget to help provide the action menu functionality as it is currently sprinkled throughout the app with html
483             require_once('include/Smarty/plugins/function.sugar_action_menu.php');
484             $tempid = create_guid();
485             $button_contents[0] = "<div style='display: inline' id='$tempid'>".$button_contents[0]."</div>";
486             $action_button = smarty_function_sugar_action_menu(array(
487                 'id' => $tempid,
488                 'buttons' => $button_contents,
489                 'class' => 'clickMenu subpanel records fancymenu button',
490                 'flat' => false //assign flat value as false to display dropdown menu at any other preferences.
491             ), $this->xTemplate);
492             $this->xTemplate->assign('CLASS', "inlineButtons");
493             $this->xTemplate->assign('CELL_COUNT', ++$count);
494             //Bug#51275 for beta3 pre_script is not required any more
495             $this->xTemplate->assign('CELL', $action_button);
496             $this->xTemplate->parse($xtemplateSection.".row.cell");
497         }
498
499
500         $aItem->setupCustomFields($aItem->module_dir);
501         $aItem->custom_fields->populateAllXTPL($this->xTemplate, 'detail', $html_varName, $fields);
502
503         $count++;
504
505         $this->xTemplate->parse($xtemplateSection.".row");
506     }
507
508     $this->xTemplate->parse($xtemplateSection);
509 }
510
511 /**sets whether or not to display the xtemplate header and footer
512  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
513  * All Rights Reserved.
514  * Contributor(s): ______________________________________.
515 */
516 function setDisplayHeaderAndFooter($bool) {
517         $this->display_header_and_footer = $bool;
518 }
519
520 /**initializes ListView
521  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
522  * All Rights Reserved.
523  * Contributor(s): ______________________________________.
524 */
525  function ListView() {
526
527
528     if(!$this->initialized) {
529         global $sugar_config;
530         $this->records_per_page = $sugar_config['list_max_entries_per_page'] + 0;
531         $this->initialized = true;
532         global $app_strings, $currentModule;
533         $this->local_theme = SugarThemeRegistry::current()->__toString();
534         $this->local_app_strings =$app_strings;
535         $this->local_image_path = SugarThemeRegistry::current()->getImagePath();
536         $this->local_current_module = $currentModule;
537     }
538 }
539 /**sets how many records should be displayed per page in the list view
540  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
541  * All Rights Reserved.
542  * Contributor(s): ______________________________________.
543 */
544  function setRecordsPerPage($count) {
545     $this->records_per_page = $count;
546 }
547 /**sets the header title */
548  function setHeaderTitle($value) {
549     $this->header_title = $value;
550 }
551 /**sets the header text this is text that's appended to the header table and is usually used for the creation of buttons
552  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
553  * All Rights Reserved.
554  * Contributor(s): ______________________________________.
555 */
556  function setHeaderText($value) {
557     $this->header_text = $value;
558 }
559 /**sets the path for the XTemplate HTML file to be used this is only needed to be set if you are allowing ListView to create the XTemplate
560  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
561  * All Rights Reserved.
562  * Contributor(s): ______________________________________.
563 */
564  function setXTemplatePath($value) {
565     $this->xTemplatePath= $value;
566 }
567
568 /**this is a helper function for allowing ListView to create a new XTemplate it groups parameters that should be set into a single function
569  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
570  * All Rights Reserved.
571  * Contributor(s): ______________________________________.
572 */
573  function initNewXTemplate($XTemplatePath, $modString, $imagePath = null) {
574     $this->setXTemplatePath($XTemplatePath);
575     if(isset($modString))
576         $this->setModStrings($modString);
577     if(isset($imagePath))
578         $this->setImagePath($imagePath);
579 }
580
581
582 function getOrderBy($varName, $defaultOrderBy='', $force_sortorder='') {
583     $sortBy = $this->getSessionVariable($varName, "ORDER_BY") ;
584
585     $orderByDirection = $this->getSessionVariableName($varName, "order_by_direction");
586     $orderByColumn = $this->getSessionVariableName($varName, "ORDER_BY");
587     $lastEqualsSortBy = false;
588     $defaultOrder = false; //ascending
589
590     if(empty($sortBy)) {
591         $this->setUserVariable($varName, "ORDER_BY", $defaultOrderBy);
592         $sortBy = $defaultOrderBy;
593     } else {
594         $this->setUserVariable($varName, "ORDER_BY", $sortBy);
595     }
596
597     $desc = $this->getSessionVariable($varName, $sortBy."S");
598
599     if (empty($desc))
600         {
601             $desc = $defaultOrder;
602         }
603         $defaultOrder = $desc ? 'desc' : 'asc';
604         $orderByValue = $defaultOrder;
605         if (isset($_REQUEST[$orderByDirection]))
606         {
607             $possibleRequestOrderBy = $_REQUEST[$orderByDirection];
608             if ($possibleRequestOrderBy == 'asc' || $possibleRequestOrderBy == 'desc')
609             {
610                 $orderByValue = $possibleRequestOrderBy;
611             }
612         }
613
614         if (isset($_REQUEST[$orderByColumn]))
615         {
616             $last = $this->getSessionVariable($varName, "OBL");
617         }
618         if (!empty($last) && $last == $sortBy)
619         {
620             $lastEqualsSortBy = true;
621         } else
622         {
623             $orderByValue = $defaultOrder;
624             $this->setSessionVariable($varName, "OBL", $sortBy);
625         }
626         $desc = $orderByValue == 'desc';
627         $orderByDirectionValue = false;
628         $this->setSessionVariable($varName, $sortBy . "S", $desc);
629         if (!empty($sortBy))
630         {
631             if (empty($force_sortorder))
632             {
633                 if (substr_count(strtolower($sortBy), ' desc') == 0 && substr_count(strtolower($sortBy), ' asc') == 0)
634                 {
635                     if ($sortBy)
636                     {
637                         $orderByDirectionValue = $desc ? 'asc' : 'desc';
638                     }
639                     $this->query_orderby = $sortBy . ' ' . $orderByValue;
640                 }
641             } else
642             {
643                 $this->query_orderby = $sortBy . ' ' . $force_sortorder;
644             }
645             if (!isset($this->appendToBaseUrl))
646             {
647                 $this->appendToBaseUrl = array();
648             }
649             if ($orderByDirectionValue)
650             {
651                 $this->appendToBaseUrl[$orderByDirection] = $orderByDirectionValue;
652             }
653             $offsetVar = $this->getSessionVariableName($varName, "offset");
654             if (isset($_REQUEST[$offsetVar]))
655             {
656                 $this->appendToBaseUrl[$offsetVar] = $_REQUEST[$offsetVar];
657             }
658             //Just clear from url...
659             $this->appendToBaseUrl[$orderByColumn] = false;
660     }else {
661         $this->query_orderby = "";
662     }
663     $this->sortby = $sortBy;
664     return $this->query_orderby;
665
666 }
667
668
669 /**sets the parameters dealing with the db
670  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
671  * All Rights Reserved.
672  * Contributor(s): ______________________________________.
673 */
674  function setQuery($where, $limit, $orderBy, $varName, $allowOrderByOveride=true) {
675     $this->query_where = $where;
676     if($this->getSessionVariable("query", "where") != $where) {
677         $this->query_where_has_changed = true;
678         $this->setSessionVariable("query", "where", $where);
679     }
680
681     $this->query_limit = $limit;
682     if(!$allowOrderByOveride) {
683         $this->query_orderby = $orderBy;
684         return;
685     }
686     $this->getOrderBy($varName, $orderBy);
687
688     $this->setLocalSessionVariable($varName, "QUERY_WHERE", $where);
689
690     //SETTING ORDER_BY FOR USE IN DETAILVIEW
691     $this->setLocalSessionVariable($varName, "ORDER_BY_DETAIL", $this->query_orderby);
692 }
693
694 function displayArrow() {
695
696 }
697
698 /**sets the theme used only use if it is different from the global
699  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
700  * All Rights Reserved.
701  * Contributor(s): ______________________________________.
702 */
703  function setTheme($theme) {
704     $this->local_theme = $theme;
705     if(isset($this->xTemplate))$this->xTemplate->assign("THEME", $this->local_theme);
706 }
707
708 /**sets the AppStrings used only use if it is different from the global
709  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
710  * All Rights Reserved.
711  * Contributor(s): ______________________________________.
712 */
713  function setAppStrings($app_strings) {
714     unset($this->local_app_strings);
715     $this->local_app_strings = $app_strings;
716     if(isset($this->xTemplate))$this->xTemplate->assign("APP", $this->local_app_strings);
717 }
718
719 /**sets the ModStrings used
720  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
721  * All Rights Reserved.
722  * Contributor(s): ______________________________________.
723 */
724  function setModStrings($mod_strings) {
725     unset($this->local_module_strings);
726     $this->local_mod_strings = $mod_strings;
727     if(isset($this->xTemplate))$this->xTemplate->assign("MOD", $this->local_mod_strings);
728 }
729
730 /**sets the ImagePath used
731  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
732  * All Rights Reserved.
733  * Contributor(s): ______________________________________.
734 */
735  function setImagePath($image_path) {
736     $this->local_image_path = $image_path;
737     if(empty($this->local_image_path)) {
738         $this->local_image_path = SugarThemeRegistry::get($this->local_theme)->getImagePath();
739     }
740     if(isset($this->xTemplate))$this->xTemplate->assign("IMAGE_PATH", $this->local_image_path);
741 }
742
743 /**sets the currentModule only use if this is different from the global
744  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
745  * All Rights Reserved.
746  * Contributor(s): ______________________________________.
747 */
748  function setCurrentModule($currentModule) {
749     unset($this->local_current_module);
750     $this->local_current_module = $currentModule;
751     if(isset($this->xTemplate))$this->xTemplate->assign("MODULE_NAME", $this->local_current_module);
752 }
753
754 /**INTERNAL FUNCTION creates an XTemplate DO NOT CALL THIS THIS IS AN INTERNAL FUNCTION
755  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
756  * All Rights Reserved.
757  * Contributor(s): ______________________________________.
758 */
759  function createXTemplate() {
760     if(!isset($this->xTemplate)) {
761         if(isset($this->xTemplatePath)) {
762
763             $this->xTemplate = new XTemplate($this->xTemplatePath);
764             $this->xTemplate->assign("APP", $this->local_app_strings);
765             if(isset($this->local_mod_strings))$this->xTemplate->assign("MOD", $this->local_mod_strings);
766             $this->xTemplate->assign("THEME", $this->local_theme);
767             $this->xTemplate->assign("IMAGE_PATH", $this->local_image_path);
768             $this->xTemplate->assign("MODULE_NAME", $this->local_current_module);
769         } else {
770             $GLOBALS['log']->error("NO XTEMPLATEPATH DEFINED CANNOT CREATE XTEMPLATE");
771         }
772     }
773 }
774
775 /**sets the XTemplate telling ListView to use newXTemplate as its current XTemplate
776  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
777  * All Rights Reserved.
778  * Contributor(s): ______________________________________.
779 */
780  function setXTemplate($newXTemplate) {
781     $this->xTemplate = $newXTemplate;
782 }
783
784 /**returns the XTemplate
785  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
786  * All Rights Reserved.
787  * Contributor(s): ______________________________________.
788 */
789  function getXTemplate() {
790     return $this->xTemplate;
791 }
792
793 /**assigns a name value pair to the XTemplate
794  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
795  * All Rights Reserved.
796  * Contributor(s): ______________________________________.
797 */
798  function xTemplateAssign($name, $value) {
799
800         if(!isset($this->xTemplate)) {
801             $this->createXTemplate();
802         }
803         $this->xTemplate->assign($name, $value);
804
805 }
806
807 /**INTERNAL FUNCTION returns the offset first checking the query then checking the session if the where clause has changed from the last time it returns 0
808  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
809  * All Rights Reserved.
810  * Contributor(s): ______________________________________.
811 */
812  function getOffset($localVarName) {
813         if($this->query_where_has_changed || isset($GLOBALS['record_has_changed'])) {
814                 $this->setSessionVariable($localVarName,"offset", 0);
815         }
816         $offset = $this->getSessionVariable($localVarName,"offset");
817         if(isset($offset)) {
818                 return $offset;
819         }
820         return 0;
821 }
822
823 /**INTERNAL FUNCTION sets the offset in the session
824  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
825  * All Rights Reserved.
826  * Contributor(s): ______________________________________.
827 */
828  function setOffset($localVarName, $value) {
829         $this->setSessionVariable($localVarName, "offset", $value);
830 }
831
832 /**INTERNAL FUNCTION sets a session variable
833  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
834  * All Rights Reserved.
835  * Contributor(s): ______________________________________.
836 */
837  function setSessionVariable($localVarName,$varName, $value) {
838     $_SESSION[$this->local_current_module."_".$localVarName."_".$varName] = $value;
839 }
840
841 function setUserVariable($localVarName,$varName, $value) {
842         if($this->is_dynamic ||  $localVarName == 'CELL')return;
843         global $current_user;
844         $current_user->setPreference($this->local_current_module."_".$localVarName."_".$varName, $value);
845 }
846
847 /**INTERNAL FUNCTION returns a session variable first checking the query for it then checking the session
848  * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
849  * All Rights Reserved.
850  * Contributor(s): ______________________________________.
851 */
852  function getSessionVariable($localVarName,$varName) {
853     //Set any variables pass in through request first
854     if(isset($_REQUEST[$this->getSessionVariableName($localVarName, $varName)])) {
855         $this->setSessionVariable($localVarName,$varName,$_REQUEST[$this->getSessionVariableName($localVarName, $varName)]);
856     }
857
858     if(isset($_SESSION[$this->getSessionVariableName($localVarName, $varName)])) {
859         return $_SESSION[$this->getSessionVariableName($localVarName, $varName)];
860     }
861     return "";
862 }
863
864 function getUserVariable($localVarName, $varName) {
865     global $current_user;
866     if($this->is_dynamic ||  $localVarName == 'CELL')return;
867     if(isset($_REQUEST[$this->getSessionVariableName($localVarName, $varName)])) {
868
869             $this->setUserVariable($localVarName,$varName,$_REQUEST[$this->getSessionVariableName($localVarName, $varName)]);
870     }
871     return $current_user->getPreference($this->getSessionVariableName($localVarName, $varName));
872 }
873
874
875     /**
876      * helper method to determine sort order by priority of source
877      * 1. explicit in request object
878      * 2. in session variable
879      * 3. subpaneldefs metadata
880      * 4. default 'asc'
881      * @param array $sortOrderList - contains options
882      * @return string 'asc' | 'desc'
883      */
884     function calculateSortOrder($sortOrderList)
885     {
886         $priority_map = array(
887           'request',
888           'session',
889           'subpaneldefs',
890           'default',
891         );
892
893         foreach($priority_map as $p) {
894             if (key_exists($p, $sortOrderList)) {
895                 $order = strtolower($sortOrderList[$p]);
896                 if (in_array($order, array('asc', 'desc'))) {
897                     return $order;
898                 }
899             }
900         }
901         return 'asc';
902     }
903
904
905     /**
906
907     * @return void
908     * @param unknown $localVarName
909     * @param unknown $varName
910     * @desc INTERNAL FUNCTION returns the session/query variable name
911     * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
912     * All Rights Reserved.
913     * Contributor(s): ______________________________________..
914     */
915     function getSessionVariableName($localVarName,$varName) {
916         return $this->local_current_module."_".$localVarName."_".$varName;
917     }
918
919     /**
920
921     * @return void
922     * @param unknown $seed
923     * @param unknown $xTemplateSection
924     * @param unknown $html_varName
925     * @desc INTERNAL FUNCTION Handles List Views using seeds that extend SugarBean
926         $XTemplateSection is the section in the XTemplate file that should be parsed usually main
927         $html_VarName is the variable name used in the XTemplateFile e.g. TASK
928         $seed is a seed that extends SugarBean
929         * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc..
930         * All Rights Reserved..
931         * Contributor(s): ______________________________________..
932     */
933     function processSugarBean($xtemplateSection, $html_varName, $seed) {
934         global $list_view_row_count;
935
936         $current_offset = $this->getOffset($html_varName);
937         $response = array();
938
939         //ADDING VCR CONTROL
940         SugarVCR::erase($seed->module_dir);
941         $params = array();
942         //$filter = array('id', 'full_name');
943         $filter=array();
944         $ret_array = $seed->create_new_list_query($this->query_orderby, $this->query_where, $filter, $params, 0, '', true, $seed, true);
945         if(!is_array($params)) $params = array();
946         if(!isset($params['custom_select'])) $params['custom_select'] = '';
947         if(!isset($params['custom_from'])) $params['custom_from'] = '';
948         if(!isset($params['custom_where'])) $params['custom_where'] = '';
949         if(!isset($params['custom_order_by'])) $params['custom_order_by'] = '';
950         $main_query = $ret_array['select'] . $params['custom_select'] . $ret_array['from'] . $params['custom_from'] . $ret_array['where'] . $params['custom_where'] . $ret_array['order_by'] . $params['custom_order_by'];
951         SugarVCR::store($seed->module_dir,  $main_query);
952         //ADDING VCR CONTROL
953
954         if(empty($this->related_field_name)) {
955             $response = $seed->get_list($this->query_orderby, $this->query_where, $current_offset, $this->query_limit);
956         } else {
957             $related_field_name = $this->related_field_name;
958             $response = $seed->get_related_list($this->child_focus,$related_field_name, $this->query_orderby,
959             $this->query_where, $current_offset, $this->query_limit);
960         }
961
962         $list = $response['list'];
963         $row_count = $response['row_count'];
964         $next_offset = $response['next_offset'];
965         $previous_offset = $response['previous_offset'];
966
967         if(!empty($response['current_offset'])) {
968             $current_offset = $response['current_offset'];
969         }
970
971         $list_view_row_count = $row_count;
972         $this->processListNavigation($xtemplateSection,$html_varName, $current_offset, $next_offset, $previous_offset, $row_count, null, null, empty($seed->column_fields) ? null : count($seed->column_fields));
973
974         return $list;
975     }
976
977
978
979     function processUnionBeans($sugarbean, $subpanel_def, $html_var = 'CELL') {
980
981                 $last_detailview_record = $this->getSessionVariable("detailview", "record");
982                 if(!empty($last_detailview_record) && $last_detailview_record != $sugarbean->id){
983                         $GLOBALS['record_has_changed'] = true;
984                 }
985                 $this->setSessionVariable("detailview", "record", $sugarbean->id);
986
987                 $current_offset = $this->getOffset($html_var);
988                 $module = isset($_REQUEST['module']) ? $_REQUEST['module'] : '';
989                 $response = array();
990
991         // choose sort order
992         $sort_order = array();
993         $sort_order['default'] = 'asc';
994
995         // explicit request parameter gets priority over all
996         $sort_order['request'] = isset($_REQUEST['sort_order']) ? $_REQUEST['sort_order'] : null;
997
998         // see if the session data has a sort order
999         if (isset($_SESSION['last_sub' . $this->subpanel_module . '_order']))
1000         {
1001             $sort_order['session'] = $_SESSION['last_sub' . $this->subpanel_module . '_order'];
1002
1003             // We swap the order when the request contains an offset (indicating a column sort issued);
1004             // otherwise we do not sort.  If we don't make this check, then the subpanel listview will
1005             // swap ordering each time a new record is entered via quick create forms
1006             if (isset($_REQUEST[$module . '_' . $html_var . '_offset']))
1007             {
1008                 $sort_order['session'] = $sort_order['session'] == 'asc' ? 'desc' : 'asc';
1009             }
1010         }
1011         else
1012         {
1013             $sort_order['session'] = null;
1014         }
1015
1016         // does the metadata have a default sort order?
1017         $sort_order['subpaneldefs'] = isset($subpanel_def->_instance_properties['sort_order']) ?
1018             $subpanel_def->_instance_properties['sort_order'] : null;
1019
1020         $this->sort_order = $this->calculateSortOrder($sort_order);
1021
1022
1023         if (isset($subpanel_def->_instance_properties['sort_by'])) {
1024             $this->query_orderby = $subpanel_def->_instance_properties['sort_by'];
1025         } else {
1026             $this->query_orderby = 'id';
1027         }
1028
1029         $this->getOrderBy($html_var,$this->query_orderby, $this->sort_order);
1030
1031         $_SESSION['last_sub' .$this->subpanel_module. '_order'] = $this->sort_order;
1032         $_SESSION['last_sub' .$this->subpanel_module. '_url'] = $this->getBaseURL($html_var);
1033
1034                 // Bug 8139 - Correct Subpanel sorting on 'name', when subpanel sorting default is 'last_name, first_name'
1035                 if (($this->sortby == 'name' || $this->sortby == 'last_name') &&
1036                         str_replace(' ', '', trim($subpanel_def->_instance_properties['sort_by'])) == 'last_name,first_name') {
1037                         $this->sortby = 'last_name '.$this->sort_order.', first_name ';
1038                 }
1039
1040         if(!empty($this->response)){
1041             $response =& $this->response;
1042             echo 'cached';
1043         }else{
1044             $response = SugarBean::get_union_related_list($sugarbean,$this->sortby, $this->sort_order, $this->query_where, $current_offset, -1,-1,$this->query_limit,$subpanel_def);
1045             $this->response =& $response;
1046         }
1047         $list = $response['list'];
1048         $row_count = $response['row_count'];
1049         $next_offset = $response['next_offset'];
1050         $previous_offset = $response['previous_offset'];
1051         if(!empty($response['current_offset']))$current_offset = $response['current_offset'];
1052         global $list_view_row_count;
1053         $list_view_row_count = $row_count;
1054         $this->processListNavigation('dyn_list_view', $html_var, $current_offset, $next_offset, $previous_offset, $row_count, $sugarbean,$subpanel_def);
1055
1056         return array('list'=>$list, 'parent_data'=>$response['parent_data'], 'query'=>$response['query']);
1057     }
1058
1059     function getBaseURL($html_varName) {
1060         static $cache = array();
1061
1062         if(!empty($cache[$html_varName]))return $cache[$html_varName];
1063         $blockVariables = array('mass', 'uid', 'massupdate', 'delete', 'merge', 'selectCount','current_query_by_page');
1064         if(!empty($this->base_URL)) {
1065             return $this->base_URL;
1066         }
1067
1068             $baseurl = $_SERVER['PHP_SELF'];
1069             if(empty($baseurl)) {
1070                 $baseurl = 'index.php';
1071             }
1072
1073             /*fixes an issue with deletes when doing a search*/
1074             foreach(array_merge($_GET, $_POST) as $name=>$value) {
1075                 //echo ("$name = $value <br/>");
1076                 if(!empty($value) && $name != 'sort_order' //&& $name != ListView::getSessionVariableName($html_varName,"ORDER_BY")
1077                         && $name != ListView::getSessionVariableName($html_varName,"offset")
1078                         /*&& substr_count($name, "ORDER_BY")==0*/ && !in_array($name, $blockVariables))
1079                 {
1080                     if(is_array($value)) {
1081                         foreach($value as $valuename=>$valuevalue) {
1082                             if(substr_count($baseurl, '?') > 0)
1083                                 $baseurl        .= "&{$name}[]=".$valuevalue;
1084                             else
1085                                 $baseurl        .= "?{$name}[]=".$valuevalue;
1086                         }
1087                     } else {
1088                         $value = urlencode($value);
1089                         if(substr_count($baseurl, '?') > 0) {
1090                             $baseurl    .= "&$name=$value";
1091                         } else {
1092                             $baseurl    .= "?$name=$value";
1093                         }
1094                     }
1095                 }
1096             }
1097
1098
1099             if($_SERVER['REQUEST_METHOD'] == 'POST') {
1100                 // at this point it is possible that the above foreach already executed resulting in double ?'s in the url
1101                 if(substr_count($baseurl, '?') == 0) {
1102                     $baseurl .= '?';
1103                 }
1104                 if(isset($_REQUEST['action'])) $baseurl.= '&action='.$_REQUEST['action'];
1105                 if(isset($_REQUEST['record'])) $baseurl .= '&record='.$_REQUEST['record'];
1106                 if(isset($_REQUEST['module'])) $baseurl .= '&module='.$_REQUEST['module'];
1107             }
1108
1109             $baseurl .= "&".ListView::getSessionVariableName($html_varName,"offset")."=";
1110             $cache[$html_varName] = $baseurl;
1111             return $baseurl;
1112     }
1113     /**
1114     * @return void
1115     * @param unknown $data
1116     * @param unknown $xTemplateSection
1117     * @param unknown $html_varName
1118     * @desc INTERNAL FUNCTION process the List Navigation
1119     * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
1120     * All Rights Reserved.
1121     * Contributor(s): ______________________________________..
1122     */
1123     function processListNavigation($xtemplateSection, $html_varName, $current_offset, $next_offset, $previous_offset, $row_count, $sugarbean=null, $subpanel_def=null, $col_count = 20) {
1124
1125         global $export_module;
1126         global $sugar_config;
1127         global $current_user;
1128         global $currentModule;
1129         global $app_strings;
1130
1131         $start_record = $current_offset + 1;
1132
1133         if(!is_numeric($col_count))
1134             $col_count = 20;
1135
1136         if($row_count == 0)
1137             $start_record = 0;
1138
1139         $end_record = $start_record + $this->records_per_page;
1140         // back up the the last page.
1141         if($end_record > $row_count+1) {
1142             $end_record = $row_count+1;
1143         }
1144         // Determine the start location of the last page
1145         if($row_count == 0)
1146             $number_pages = 0;
1147         else
1148             $number_pages = floor(($row_count - 1) / $this->records_per_page);
1149
1150         $last_offset = $number_pages * $this->records_per_page;
1151
1152         if(empty($this->query_limit)  || $this->query_limit > $this->records_per_page) {
1153             $this->base_URL = $this->getBaseURL($html_varName);
1154             $dynamic_url = '';
1155
1156             if($this->is_dynamic) {
1157                 $dynamic_url .='&'. $this->getSessionVariableName($html_varName,'ORDER_BY') . '='. $this->getSessionVariable($html_varName,'ORDER_BY').'&sort_order='.$this->sort_order.'&to_pdf=true&action=SubPanelViewer&subpanel=' . $this->subpanel_module;
1158             }
1159
1160             $current_URL = htmlentities($this->base_URL.$current_offset.$dynamic_url);
1161             $start_URL = htmlentities($this->base_URL."0".$dynamic_url);
1162             $previous_URL  = htmlentities($this->base_URL.$previous_offset.$dynamic_url);
1163             $next_URL  = htmlentities($this->base_URL.$next_offset.$dynamic_url);
1164             $end_URL  = htmlentities($this->base_URL.'end'.$dynamic_url);
1165
1166             if(!empty($this->start_link_wrapper)) {
1167                 $current_URL = $this->start_link_wrapper.$current_URL.$this->end_link_wrapper;
1168                 $start_URL = $this->start_link_wrapper.$start_URL.$this->end_link_wrapper;
1169                 $previous_URL = $this->start_link_wrapper.$previous_URL.$this->end_link_wrapper;
1170                 $next_URL = $this->start_link_wrapper.$next_URL.$this->end_link_wrapper;
1171                 $end_URL = $this->start_link_wrapper.$end_URL.$this->end_link_wrapper;
1172             }
1173
1174             $moduleString = "{$currentModule}_{$html_varName}_offset";
1175             $moduleStringOrder = "{$currentModule}_{$html_varName}_ORDER_BY";
1176             if($this->shouldProcess && !$this->multi_select_popup) {
1177                 // check the checkboxes onload
1178                 echo "<script>YAHOO.util.Event.addListener(window, \"load\", sListView.check_boxes);</script>\n";
1179
1180                 $massUpdateRun = isset($_REQUEST['massupdate']) && $_REQUEST['massupdate'] == 'true';
1181                 $uids = empty($_REQUEST['uid']) || $massUpdateRun ? '' : $_REQUEST['uid'];
1182                 $select_entire_list = ($massUpdateRun) ? 0 : (isset($_POST['select_entire_list']) ? $_POST['select_entire_list'] : (isset($_REQUEST['select_entire_list']) ? $_REQUEST['select_entire_list'] : 0));
1183
1184                 echo "<textarea style='display: none' name='uid'>{$uids}</textarea>\n" .
1185                     "<input type='hidden' name='select_entire_list' value='{$select_entire_list}'>\n".
1186                     "<input type='hidden' name='{$moduleString}' value='0'>\n".
1187                     "<input type='hidden' name='{$moduleStringOrder}' value='0'>\n";
1188
1189             }
1190
1191
1192             $GLOBALS['log']->debug("Offsets: (start, previous, next, last)(0, $previous_offset, $next_offset, $last_offset)");
1193
1194             if(0 == $current_offset) {
1195                 $start_link = "<button type='button' name='listViewStartButton' title='{$this->local_app_strings['LNK_LIST_START']}' class='button' disabled>".SugarThemeRegistry::current()->getImage("start_off","aborder='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_START'])."</button>";
1196                 $previous_link = "<button type='button' name='listViewPrevButton' title='{$this->local_app_strings['LNK_LIST_PREVIOUS']}' class='button' disabled>".SugarThemeRegistry::current()->getImage("previous_off","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_PREVIOUS'])."</button>";
1197             } else {
1198                 if($this->multi_select_popup) {// nav links for multiselect popup, submit form to save checks.
1199                     $start_link = "<button type='button' class='button' name='listViewStartButton' title='{$this->local_app_strings['LNK_LIST_START']}' onClick='javascript:save_checks(0, \"{$moduleString}\");'>".SugarThemeRegistry::current()->getImage("start","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_START'])."</button>";
1200                     $previous_link = "<button type='button' class='button' name='listViewPrevButton' title='{$this->local_app_strings['LNK_LIST_PREVIOUS']}' onClick='javascript:save_checks($previous_offset, \"{$moduleString}\");'>".SugarThemeRegistry::current()->getImage("previous","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_PREVIOUS'])."</button>";
1201                 } elseif($this->shouldProcess) {
1202                     $start_link = "<button type='button' class='button' name='listViewStartButton' title='{$this->local_app_strings['LNK_LIST_START']}' onClick='location.href=\"$start_URL\"; sListView.save_checks(0, \"{$moduleString}\");'>".SugarThemeRegistry::current()->getImage("start","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_START'])."</button>";
1203                     $previous_link = "<button type='button' class='button' name='listViewPrevButton' title='{$this->local_app_strings['LNK_LIST_PREVIOUS']}' onClick='location.href=\"$previous_URL\"; sListView.save_checks($previous_offset, \"{$moduleString}\");'>".SugarThemeRegistry::current()->getImage("previous","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_PREVIOUS'])."</button>";
1204                 } else {
1205                     $onClick = '';
1206                     if(0 != preg_match('/javascript.*/', $start_URL)){
1207                         $onClick = "\"$start_URL;\"";
1208                     }else{
1209                         $onClick ="'location.href=\"$start_URL\";'";
1210                     }
1211                     $start_link = "<button type='button' class='button' name='listViewStartButton' title='{$this->local_app_strings['LNK_LIST_START']}' onClick=".$onClick.">".SugarThemeRegistry::current()->getImage("start","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_START'])."</button>";
1212
1213                     $onClick = '';
1214                     if(0 != preg_match('/javascript.*/', $previous_URL)){
1215                         $onClick = "\"$previous_URL;\"";
1216                     }else{
1217                         $onClick = "'location.href=\"$previous_URL\";'";
1218                     }
1219                     $previous_link = "<button type='button' class='button' name='listViewPrevButton' title='{$this->local_app_strings['LNK_LIST_PREVIOUS']}' onClick=".$onClick.">".SugarThemeRegistry::current()->getImage("previous","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_PREVIOUS'])."</button>";
1220                 }
1221             }
1222
1223             if($last_offset <= $current_offset) {
1224                 $end_link = "<button type='button' name='listViewEndButton' title='{$this->local_app_strings['LNK_LIST_END']}' class='button' disabled>".SugarThemeRegistry::current()->getImage("end_off","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_END'])."</button>";
1225                 $next_link = "<button type='button' name='listViewNextButton' title='{$this->local_app_strings['LNK_LIST_NEXT']}' class='button' disabled>".SugarThemeRegistry::current()->getImage("next_off","aborder='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_NEXT'])."</button>";
1226             } else {
1227                 if($this->multi_select_popup) { // nav links for multiselect popup, submit form to save checks.
1228                     $end_link = "<button type='button' name='listViewEndButton' class='button' title='{$this->local_app_strings['LNK_LIST_END']}' onClick='javascript:save_checks($last_offset, \"{$moduleString}\");'>".SugarThemeRegistry::current()->getImage("end","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_END'])."</button>";
1229                     if(!empty($sugar_config['disable_count_query'])) {
1230                         $end_link = '';
1231                     }
1232                     $next_link = "<button type='button' name='listViewNextButton' title='{$this->local_app_strings['LNK_LIST_NEXT']}' class='button' onClick='javascript:save_checks($next_offset, \"{$moduleString}\");'>".SugarThemeRegistry::current()->getImage("next","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_NEXT'])."</button>";
1233                 } elseif($this->shouldProcess) {
1234                     $end_link = "<button type='button' name='listViewEndButton' class='button' title='{$this->local_app_strings['LNK_LIST_END']}' onClick='location.href=\"$end_URL\"; sListView.save_checks(\"end\", \"{$moduleString}\");'>".SugarThemeRegistry::current()->getImage("end","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_END'])."</button>";
1235                     $next_link = "<button type='button' name='listViewNextButton' class='button' title='{$this->local_app_strings['LNK_LIST_NEXT']}' onClick='location.href=\"$next_URL\"; sListView.save_checks($next_offset, \"{$moduleString}\");'>".SugarThemeRegistry::current()->getImage("next","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_NEXT'])."</button>";
1236                 } else {
1237                     $onClick = '';
1238                     if(0 != preg_match('/javascript.*/', $next_URL)){
1239                         $onClick = "\"$next_URL;\"";
1240                     }else{
1241                         $onClick ="'location.href=\"$next_URL\";'";
1242                     }
1243                     $next_link = "<button type='button' name='listViewNextButton' class='button' title='{$this->local_app_strings['LNK_LIST_NEXT']}' onClick=".$onClick.">".SugarThemeRegistry::current()->getImage("next","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_NEXT'])."</button>";
1244
1245                     $onClick = '';
1246                     if(0 != preg_match('/javascript.*/', $end_URL)){
1247                         $onClick = "\"$end_URL;\"";
1248                     }else{
1249                         $onClick = "'location.href=\"$end_URL\";'";
1250                     }
1251                     $end_link = "<button type='button' name='listViewEndButton' class='button' title='{$this->local_app_strings['LNK_LIST_END']}' onClick=".$onClick.">".SugarThemeRegistry::current()->getImage("end","border='0' align='absmiddle'",null,null,'.gif',$this->local_app_strings['LNK_LIST_END'])."</button>";
1252
1253                 }
1254             }
1255
1256             $GLOBALS['log']->info("Offset (next, current, prev)($next_offset, $current_offset, $previous_offset)");
1257             $GLOBALS['log']->info("Start/end records ($start_record, $end_record)");
1258
1259             $end_record = $end_record-1;
1260
1261 $script_href = "<a style=\'width: 150px\' name=\"thispage\" class=\'menuItem\' onmouseover=\'hiliteItem(this,\"yes\");\' onmouseout=\'unhiliteItem(this);\' onclick=\'if (document.MassUpdate.select_entire_list.value==1){document.MassUpdate.select_entire_list.value=0;sListView.check_all(document.MassUpdate, \"mass[]\", true, $this->records_per_page)}else {sListView.check_all(document.MassUpdate, \"mass[]\", true)};\' href=\'#\'>{$this->local_app_strings['LBL_LISTVIEW_OPTION_CURRENT']}&nbsp;&#x28;{$this->records_per_page}&#x29;&#x200E;</a>"
1262  . "<a style=\'width: 150px\' name=\"selectall\" class=\'menuItem\' onmouseover=\'hiliteItem(this,\"yes\");\' onmouseout=\'unhiliteItem(this);\' onclick=\'sListView.check_entire_list(document.MassUpdate, \"mass[]\",true,{$row_count});\' href=\'#\'>{$this->local_app_strings['LBL_LISTVIEW_OPTION_ENTIRE']}&nbsp;&#x28;{$row_count}&#x29;&#x200E;</a>"
1263  . "<a style=\'width: 150px\' name=\"deselect\" class=\'menuItem\' onmouseover=\'hiliteItem(this,\"yes\");\' onmouseout=\'unhiliteItem(this);\' onclick=\'sListView.clear_all(document.MassUpdate, \"mass[]\", false);\' href=\'#\'>{$this->local_app_strings['LBL_LISTVIEW_NONE']}</a>";
1264
1265 $close_inline_img = SugarThemeRegistry::current()->getImage('close_inline', 'border=0', null, null, ".gif", $app_strings['LBL_CLOSEINLINE']);
1266
1267             echo "<script>
1268                 function select_dialog() {
1269                         var \$dialog = \$('<div></div>')
1270                                         .html('<a style=\'width: 150px\' name=\"thispage\" class=\'menuItem\' onmouseover=\'hiliteItem(this,\"yes\");\' onmouseout=\'unhiliteItem(this);\' onclick=\'if (document.MassUpdate.select_entire_list.value==1){document.MassUpdate.select_entire_list.value=0;sListView.check_all(document.MassUpdate, \"mass[]\", true, $this->records_per_page)}else {sListView.check_all(document.MassUpdate, \"mass[]\", true)};\' href=\'javascript:void(0)\'>{$this->local_app_strings['LBL_LISTVIEW_OPTION_CURRENT']}&nbsp;&#x28;{$this->records_per_page}&#x29;&#x200E;</a>"
1271                 . "<a style=\'width: 150px\' name=\"selectall\" class=\'menuItem\' onmouseover=\'hiliteItem(this,\"yes\");\' onmouseout=\'unhiliteItem(this);\' onclick=\'sListView.check_entire_list(document.MassUpdate, \"mass[]\",true,{$row_count});\' href=\'javascript:void(0)\'>{$this->local_app_strings['LBL_LISTVIEW_OPTION_ENTIRE']}&nbsp;&#x28;{$row_count}&#x29;&#x200E;</a>"
1272                 . "<a style=\'width: 150px\' name=\"deselect\" class=\'menuItem\' onmouseover=\'hiliteItem(this,\"yes\");\' onmouseout=\'unhiliteItem(this);\' onclick=\'sListView.clear_all(document.MassUpdate, \"mass[]\", false);\' href=\'javascript:void(0)\'>{$this->local_app_strings['LBL_LISTVIEW_NONE']}</a>')
1273                                         .dialog({
1274                                                 autoOpen: false,
1275                                                 width: 150
1276                                         });
1277                                         \$dialog.dialog('open');
1278
1279                 }
1280                 </script>";
1281
1282             if($this->show_select_menu)
1283             {
1284                 $total_label = "";
1285                 $total = $row_count;
1286                 $pageTotal = ($row_count > 0) ? $end_record - $start_record + 1 : 0;
1287                 if (!empty($GLOBALS['sugar_config']['disable_count_query']) && $GLOBALS['sugar_config']['disable_count_query'] === true && $total > $pageTotal) {
1288                     $this->show_plus = true;
1289                     $total =  $pageTotal;
1290                     $total_label = $total.'+';
1291                 } else {
1292                     $this->show_plus = false;
1293                     $total_label = $total;
1294                 }
1295                 echo "<input type='hidden' name='show_plus' value='{$this->show_plus}'>\n";
1296
1297                 //Bug#52931: Replace with actionMenu
1298                 //$select_link = "<a id='select_link' onclick='return select_dialog();' href=\"javascript:void(0)\">".$this->local_app_strings['LBL_LINK_SELECT']."&nbsp;".SugarThemeRegistry::current()->getImage('MoreDetail', 'border=0', 11, 7, '.png', $app_strings['LBL_MOREDETAIL'])."</a>";
1299                 $menuItems = array(
1300                     "<input title=\"".$app_strings['LBL_SELECT_ALL_TITLE']."\" type='checkbox' class='checkbox massall' name='massall' id='massall' value='' onclick='sListView.check_all(document.MassUpdate, \"mass[]\", this.checked);' /><a href='javascript: void(0);'></a>",
1301                     "<a  name='thispage' id='button_select_this_page' class='menuItem' onmouseover='hiliteItem(this,\"yes\");' onmouseout='unhiliteItem(this);' onclick='if (document.MassUpdate.select_entire_list.value==1){document.MassUpdate.select_entire_list.value=0;sListView.check_all(document.MassUpdate, \"mass[]\", true, $pageTotal)}else {sListView.check_all(document.MassUpdate, \"mass[]\", true)};' href='#'>{$app_strings['LBL_LISTVIEW_OPTION_CURRENT']}&nbsp;&#x28;{$pageTotal}&#x29;&#x200E;</a>",
1302                     "<a  name='selectall' id='button_select_all' class='menuItem' onmouseover='hiliteItem(this,\"yes\");' onmouseout='unhiliteItem(this);' onclick='sListView.check_entire_list(document.MassUpdate, \"mass[]\",true,{$total});' href='#'>{$app_strings['LBL_LISTVIEW_OPTION_ENTIRE']}&nbsp;&#x28;{$total_label}&#x29;&#x200E;</a>",
1303                     "<a name='deselect' id='button_deselect' class='menuItem' onmouseover='hiliteItem(this,\"yes\");' onmouseout='unhiliteItem(this);' onclick='sListView.clear_all(document.MassUpdate, \"mass[]\", false);' href='#'>{$app_strings['LBL_LISTVIEW_NONE']}</a>",
1304                 );
1305                 require_once('include/Smarty/plugins/function.sugar_action_menu.php');
1306                 $select_link = smarty_function_sugar_action_menu(array(
1307                     'class' => 'clickMenu selectmenu',
1308                     'id' => 'selectLink',
1309                     'buttons' => $menuItems,
1310                     'flat' => false,
1311                 ),$this->xTemplate);
1312
1313             } else {
1314                 $select_link = "&nbsp;";
1315             }
1316
1317             $export_link = '<input class="button" type="button" value="'.$this->local_app_strings['LBL_EXPORT'].'" ' .
1318                     'onclick="return sListView.send_form(true, \''.$_REQUEST['module'].'\', \'index.php?entryPoint=export\',\''.$this->local_app_strings['LBL_LISTVIEW_NO_SELECTED'].'\')">';
1319
1320             if($this->show_delete_button) {
1321                 $delete_link = '<input class="button" type="button" id="delete_button" name="Delete" value="'.$this->local_app_strings['LBL_DELETE_BUTTON_LABEL'].'" onclick="return sListView.send_mass_update(\'selected\',\''.$this->local_app_strings['LBL_LISTVIEW_NO_SELECTED'].'\', 1)">';
1322             } else {
1323                 $delete_link = '&nbsp;';
1324             }
1325
1326             $admin = new Administration();
1327             $admin->retrieveSettings('system');
1328
1329             $user_merge = $current_user->getPreference('mailmerge_on');
1330             if($user_merge == 'on' && isset($admin->settings['system_mailmerge_on']) && $admin->settings['system_mailmerge_on']) {
1331                 echo "<script>
1332                 function mailmerge_dialog(el) {
1333                         var \$dialog = \$('<div></div>')
1334                                         .html('<a style=\'width: 150px\' class=\'menuItem\' onmouseover=\'hiliteItem(this,\"yes\");\' onmouseout=\'unhiliteItem(this);\' onclick=\'return sListView.send_form(true, \"MailMerge\", \"index.php\", \"{$this->local_app_strings['LBL_LISTVIEW_NO_SELECTED']}\")\' href=\'javascript:void(0)\'>{$this->local_app_strings['LBL_LISTVIEW_OPTION_SELECTED']}</a>"
1335                         . "<a style=\'width: 150px\' class=\'menuItem\' onmouseover=\'hiliteItem(this,\"yes\");\' onmouseout=\'unhiliteItem(this);\' href=\'index.php?action=index&module=MailMerge\'>{$this->local_app_strings['LBL_LISTVIEW_OPTION_CURRENT']}</a>"
1336                         . "<a style=\'width: 150px\' class=\'menuItem\' onmouseover=\'hiliteItem(this,\"yes\");\' onmouseout=\'unhiliteItem(this);\' href=\'index.php?action=index&module=MailMerge&entire=true\'>{$this->local_app_strings['LBL_LISTVIEW_OPTION_ENTIRE']}</a>')
1337                                         .dialog({
1338                                                 autoOpen: false,
1339                                                 title: '". $this->local_app_strings['LBL_MAILMERGE']."',
1340                                                 width: 150,
1341                                                 position: {
1342                                                     my: myPos,
1343                                                     at: atPos,
1344                                                     of: \$(el)
1345                                                 }
1346                                         });
1347
1348                 }
1349             </script>";
1350                 $merge_link = "&nbsp;|&nbsp;<a id='mailmerge_link' onclick='return mailmerge_dialog(this)'; href=\"javascript:void(0)\">".$this->local_app_strings['LBL_MAILMERGE']."</a>";
1351             } else {
1352                 $merge_link = "&nbsp;";
1353             }
1354
1355             $selected_objects_span = "&nbsp;|&nbsp;{$this->local_app_strings['LBL_LISTVIEW_SELECTED_OBJECTS']}<input  style='border: 0px; background: transparent; font-size: inherit; color: inherit' type='text' readonly name='selectCount[]' value='" . ((isset($_POST['mass'])) ? count($_POST['mass']): 0) . "' />";
1356
1357             if($_REQUEST['module'] == 'Home' || $this->local_current_module == 'Import'
1358                 || $this->show_export_button == false
1359                 || (!empty($sugar_config['disable_export']))
1360                 || (!empty($sugar_config['admin_export_only'])
1361                 && !(
1362                         is_admin($current_user)
1363                         || (ACLController::moduleSupportsACL($_REQUEST['module'])
1364                             && ACLAction::getUserAccessLevel($current_user->id,$_REQUEST['module'], 'access') == ACL_ALLOW_ENABLED
1365                             && (ACLAction::getUserAccessLevel($current_user->id, $_REQUEST['module'], 'admin') == ACL_ALLOW_ADMIN ||
1366                                 ACLAction::getUserAccessLevel($current_user->id, $_REQUEST['module'], 'admin') == ACL_ALLOW_ADMIN_DEV)))))
1367             {
1368                 if ($_REQUEST['module'] != 'InboundEmail' && $_REQUEST['module'] != 'EmailMan' && $_REQUEST['module'] != 'iFrames') {
1369                     $selected_objects_span = '';
1370                 }
1371                 $export_link = "&nbsp;";
1372                 $merge_link = "&nbsp;";
1373             } elseif($_REQUEST['module'] != "Accounts" && $_REQUEST['module'] != "Cases" && $_REQUEST['module'] != "Contacts" && $_REQUEST['module'] != "Leads" && $_REQUEST['module'] != "Opportunities") {
1374                 $merge_link = "&nbsp;";
1375             }
1376
1377             if($this->show_paging == true) {
1378                 if(!empty($sugar_config['disable_count_query'])) {
1379                     if($row_count > $end_record) {
1380                         $row_count .= '+';
1381                     }
1382                 }
1383
1384                 $html_text = '';
1385                 $html_text .= "<tr class='pagination' role='presentation'>\n";
1386                 $html_text .= "<td COLSPAN=\"$col_count\" align=\"right\">\n";
1387                 //$html_text .= "<table border=\"0\" cellpadding=\"0\" cellspacing=\"0\" width=\"100%\"><tr><td align=\"left\"  >$export_link$merge_link$selected_objects_span</td>\n";
1388                 //$html_text .= "<table border=\"0\" cellpadding=\"0\" cellspacing=\"0\" width=\"100%\"><tr><td align=\"left\"  >";
1389                 if ($subpanel_def != null) {
1390                     include_once('include/SubPanel/SubPanelTiles.php');
1391                     $subpanelTiles = new SubPanelTiles($sugarbean);
1392                     $html_text .= "<table border=\"0\" cellpadding=\"0\" cellspacing=\"0\" width=\"100%\"><tr><td align=\"left\"  >";
1393
1394                     //attempt to get the query to recreate this subpanel
1395                     if(!empty($this->response)){
1396                         $response =& $this->response;
1397                     }else{
1398                         $response = SugarBean::get_union_related_list($sugarbean,$this->sortby, $this->sort_order, $this->query_where, $current_offset, -1,-1,$this->query_limit,$subpanel_def);
1399                         $this->response = $response;
1400                     }
1401                     //if query is present, then pass it in as parameter
1402                     if (isset($response['query']) && !empty($response['query'])){
1403                         $html_text .= $subpanelTiles->get_buttons($subpanel_def, $response['query']);
1404                     }else{
1405                         $html_text .= $subpanelTiles->get_buttons($subpanel_def);
1406                     }
1407                 }
1408                 else {
1409                     $html_text .= "<table border=\"0\" cellpadding=\"0\" cellspacing=\"0\" width=\"100%\"><tr><td align=\"left\"  nowrap>$select_link&nbsp;$export_link&nbsp;$delete_link&nbsp;$selected_objects_span";
1410                 }
1411                 $html_text .= "</td>\n<td nowrap align=\"right\">".$start_link."&nbsp;&nbsp;".$previous_link."&nbsp;&nbsp;<span class='pageNumbers'>(".$start_record." - ".$end_record." ".$this->local_app_strings['LBL_LIST_OF']." ".$row_count.")</span>&nbsp;&nbsp;".$next_link."&nbsp;&nbsp;".$end_link."</td></tr></table>\n";
1412                 $html_text .= "</td>\n";
1413                 $html_text .= "</tr>\n";
1414                 $this->xTemplate->assign("PAGINATION",$html_text);
1415             }
1416
1417             //C.L. - Fix for 23461
1418             if(empty($_REQUEST['action']) || $_REQUEST['action'] != 'Popup') {
1419                 $_SESSION['export_where'] = $this->query_where;
1420             }
1421             $this->xTemplate->parse($xtemplateSection.".list_nav_row");
1422         }
1423     } // end processListNavigation
1424
1425     function processOrderBy($html_varName) {
1426
1427         if(!isset($this->base_URL)) {
1428             $this->base_URL = $_SERVER['PHP_SELF'];
1429
1430             if(isset($_SERVER['QUERY_STRING'])) {
1431                 $this->base_URL = preg_replace("/\&".$this->getSessionVariableName($html_varName,"ORDER_BY")."=[0-9a-zA-Z\_\.]*/","",$this->base_URL .'?'.$_SERVER['QUERY_STRING']);
1432                 $this->base_URL = preg_replace("/\&".$this->getSessionVariableName($html_varName,"offset")."=[0-9]*/","",$this->base_URL);
1433             }
1434             if($_SERVER['REQUEST_METHOD'] == 'POST') {
1435                 $this->base_URL .= '?';
1436                 if(isset($_REQUEST['action'])) $this->base_URL .= '&action='.$_REQUEST['action'];
1437                 if(isset($_REQUEST['record'])) $this->base_URL .= '&record='.$_REQUEST['record'];
1438                 if(isset($_REQUEST['module'])) $this->base_URL .= '&module='.$_REQUEST['module'];
1439             }
1440             $this->base_URL .= "&".$this->getSessionVariableName($html_varName,"offset")."=";
1441         }
1442
1443         if($this->is_dynamic) {
1444             $this->base_URL.='&to_pdf=true&action=SubPanelViewer&subpanel=' . $this->source_module;
1445         }
1446
1447         //bug43465 start
1448         if (isset($this->appendToBaseUrl) && is_array($this->appendToBaseUrl))
1449         {
1450             foreach ($this->appendToBaseUrl as $key => $value)
1451             {
1452                 $fullRequestString = $key . '=' . $value;
1453
1454                 if ($this->base_URL == "/index.php")
1455                 {
1456                     $this->base_URL .= "?";
1457                 } else
1458                 {
1459                     if ($fullRequestString == substr($this->baseURL, '-' . strlen($fullRequestString)))
1460                     {
1461                         $this->base_URL = preg_replace("/&" . $key . "\=.*/", "", $this->base_URL);
1462                     } else
1463                     {
1464                         $this->base_URL = preg_replace("/&" . $key . "\=.*?&/", "&", $this->base_URL);
1465                     }
1466                     $this->base_URL .= "&";
1467                 }
1468                 if (!empty($value))
1469                 {
1470                     $this->base_URL .= "{$key}={$value}";
1471                 }
1472             }
1473         }
1474         //bug43465 end
1475
1476         $sort_URL_base = $this->base_URL. "&".$this->getSessionVariableName($html_varName,"ORDER_BY")."=";
1477
1478         if($sort_URL_base !== "")
1479         {
1480             $this->xTemplate->assign("ORDER_BY", $sort_URL_base);
1481             return $sort_URL_base;
1482         } else {
1483             return '';
1484         }
1485     }
1486
1487
1488     function getAdditionalHeader() {
1489
1490     }
1491
1492
1493     /**
1494     * @return void
1495     * @param unknown $data
1496     * @param unknown $xTemplateSection
1497     * @param unknown $html_varName
1498     * @desc INTERNAL FUNCTION handles the rows
1499     * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
1500     * All Rights Reserved.
1501     * Contributor(s): ______________________________________..
1502     */
1503     function processListRows($data, $xtemplateSection, $html_varName)
1504     {
1505         global $odd_bg;
1506         global $even_bg;
1507         global $hilite_bg;
1508         global $app_strings, $sugar_version, $sugar_config;
1509         global $currentModule;
1510
1511         $this->xTemplate->assign('BG_HILITE', $hilite_bg);
1512         $this->xTemplate->assign('CHECKALL', SugarThemeRegistry::current()->getImage('blank', '', 1, 1, ".gif", ''));
1513     //$this->xTemplate->assign("BG_CLICK", $click_bg);
1514         $oddRow = true;
1515         $count = 0;
1516         reset($data);
1517
1518         //GETTING OFFSET
1519         $offset = $this->getOffset($html_varName);
1520         $timeStamp = $this->unique_id();
1521         $_SESSION[$html_varName."_FROM_LIST_VIEW"] = $timeStamp;
1522
1523         $associated_row_data = array();
1524
1525         //mail merge list
1526         $mergeList = array();
1527         $module = '';
1528         //todo what is this?  It is using an array as a boolean
1529         while(list($aVal, $aItem) = each($data))
1530         {
1531             if(isset($this->data_array)) {
1532                 $fields = $this->data_array;
1533             } else {
1534                 $aItem->check_date_relationships_load();
1535                 $fields = $aItem->get_list_view_data();
1536             }
1537
1538             if(is_object($aItem)) { // cn: bug 5349
1539                 //add item id to merge list, if the button is clicked
1540                 $mergeList[] = $aItem->id;
1541                 if(empty($module)) {
1542                     $module = $aItem->module_dir;
1543                 }
1544             }
1545             //ADD OFFSET TO ARRAY
1546
1547                 $fields['OFFSET'] = ($offset + $count + 1);
1548
1549             $fields['STAMP'] = $timeStamp;
1550             if($this->shouldProcess) {
1551
1552             $prerow = '';
1553             if(!isset($this->data_array)) {
1554                 $prerow .= "<input onclick='sListView.check_item(this, document.MassUpdate)' type='checkbox' class='checkbox' name='mass[]' value='". $fields['ID']. "'>";
1555             }
1556             $this->xTemplate->assign('PREROW', $prerow);
1557
1558             $this->xTemplate->assign('CHECKALL', "<input type='checkbox' class='checkbox'  title='".$GLOBALS['app_strings']['LBL_SELECT_ALL_TITLE']."'  name='massall' id='massall' value='' onclick='sListView.check_all(document.MassUpdate, \"mass[]\", this.checked)'>");
1559             }
1560             if(!isset($this->data_array)) {
1561                 $tag = $aItem->listviewACLHelper();
1562                 $this->xTemplate->assign('TAG',$tag) ;
1563             }
1564
1565             if($oddRow)
1566             {
1567                 $ROW_COLOR = 'oddListRow';
1568                 $BG_COLOR =  $odd_bg;
1569             }
1570             else
1571             {
1572                 $ROW_COLOR = 'evenListRow';
1573                 $BG_COLOR =  $even_bg;
1574             }
1575             $oddRow = !$oddRow;
1576
1577             $this->xTemplate->assign('ROW_COLOR', $ROW_COLOR);
1578             $this->xTemplate->assign('BG_COLOR', $BG_COLOR);
1579
1580             if(isset($this->data_array))
1581             {
1582                 $this->xTemplate->assign('KEY', $aVal);
1583                 $this->xTemplate->assign('VALUE', $aItem);
1584                 $this->xTemplate->assign('INDEX', $count);
1585
1586             }
1587             else
1588             {
1589     //AED -- some modules do not have their additionalDetails.php established. Add a check to ensure require_once does not fail
1590     // Bug #2786
1591                 if($this->_additionalDetails && $aItem->ACLAccess('DetailView') && (file_exists('modules/' . $aItem->module_dir . '/metadata/additionalDetails.php') || file_exists('custom/modules/' . $aItem->module_dir . '/metadata/additionalDetails.php'))) {
1592
1593                     $additionalDetailsFile = 'modules/' . $aItem->module_dir . '/metadata/additionalDetails.php';
1594                     if(file_exists('custom/modules/' . $aItem->module_dir . '/metadata/additionalDetails.php')){
1595                         $additionalDetailsFile = 'custom/modules/' . $aItem->module_dir . '/metadata/additionalDetails.php';
1596                     }
1597
1598                     require_once($additionalDetailsFile);
1599                     $ad_function = (empty($this->additionalDetailsFunction) ? 'additionalDetails' : $this->additionalDetailsFunction) . $aItem->object_name;
1600                     $results = $ad_function($fields);
1601                     $results['string'] = str_replace(array("&#039", "'"), '\&#039', $results['string']); // no xss!
1602
1603                     if(trim($results['string']) == '') $results['string'] = $app_strings['LBL_NONE'];
1604                     $fields[$results['fieldToAddTo']] = $fields[$results['fieldToAddTo']].'</a>';
1605                 }
1606
1607                 if($aItem->ACLAccess('Delete')) {
1608                     $delete = '<a class="listViewTdToolsS1" onclick="return confirm(\''.$this->local_app_strings['NTC_DELETE_CONFIRMATION'].'\')" href="'.'index.php?action=Delete&module='.$aItem->module_dir.'&record='.$fields['ID'].'&return_module='.$aItem->module_dir.'&return_action=index&return_id=">'.$this->local_app_strings['LBL_DELETE_INLINE'].'</a>';
1609                     require_once('include/Smarty/plugins/function.sugar_action_menu.php');
1610                     $fields['DELETE_BUTTON'] = smarty_function_sugar_action_menu(array(
1611                         'id' => $aItem->module_dir.'_'.$fields['ID'].'_create_button',
1612                         'buttons' => array($delete),
1613                     ), $this);
1614
1615                 }
1616
1617                 $this->xTemplate->assign($html_varName, $fields);
1618                 $aItem->setupCustomFields($aItem->module_dir);
1619                 $aItem->custom_fields->populateAllXTPL($this->xTemplate, 'detail', $html_varName, $fields);
1620             }
1621             if(!isset($this->data_array) && $aItem->ACLAccess('DetailView')) {
1622                 $count++;
1623             }
1624             if(isset($this->data_array)) {
1625                 $count++;
1626             }
1627             if(!isset($this->data_array)) {
1628                 $aItem->list_view_parse_additional_sections($this->xTemplate, $xtemplateSection);
1629
1630                 if($this->xTemplate->exists($xtemplateSection.'.row.pro')) {
1631                     $this->xTemplate->parse($xtemplateSection.'.row.pro');
1632                 }
1633             }
1634             $this->xTemplate->parse($xtemplateSection . '.row');
1635
1636             if(isset($fields['ID'])) {
1637                 $associated_row_data[$fields['ID']] = $fields;
1638                 // Bug 38908: cleanup data for JS to avoid having &nbsp; shuffled around
1639                 foreach($fields as $key => $value) {
1640                     if($value == '&nbsp;') {
1641                         $associated_row_data[$fields['ID']][$key] = '';
1642                     }
1643                 }
1644             }
1645         }
1646
1647         $_SESSION['MAILMERGE_RECORDS'] = $mergeList;
1648         $_SESSION['MAILMERGE_MODULE_FROM_LISTVIEW'] = $module;
1649         if(empty($_REQUEST['action']) || $_REQUEST['action'] != 'Popup') {
1650             $_SESSION['MAILMERGE_MODULE'] = $module;
1651         }
1652
1653         if($this->process_for_popups)
1654         {
1655             $json = getJSONobj();
1656             $is_show_fullname = showFullName() ? 1 : 0;
1657             $associated_javascript_data = '<script type="text/javascript">' . "\n"
1658                 //. '<!-- // associated javascript data generated by ListView' . "\n"
1659                 . 'var associated_javascript_data = '
1660                 . $json->encode($associated_row_data) . ";\n"
1661                 //. '-->' . "\n"
1662                 . 'var is_show_fullname = '
1663                 . $is_show_fullname . ";\n"
1664                 . '</script>';
1665             $this->xTemplate->assign('ASSOCIATED_JAVASCRIPT_DATA', $associated_javascript_data);
1666         }
1667
1668         $this->xTemplate->parse($xtemplateSection);
1669     }
1670
1671
1672     function getLayoutManager()
1673     {
1674         require_once('include/generic/LayoutManager.php');
1675         if($this->layout_manager == null)
1676         {
1677             $this->layout_manager = new LayoutManager();
1678         }
1679         return $this->layout_manager;
1680     }
1681
1682
1683     function process_dynamic_listview_header($source_module, $subpanel_def, $html_var = 'CELL')
1684     {
1685
1686
1687         $layout_manager = $this->getLayoutManager();
1688         $layout_manager->setAttribute('order_by_link',$this->processOrderBy($html_var));
1689         $layout_manager->setAttribute('context','HeaderCell');
1690         $layout_manager->setAttribute('image_path',$this->local_image_path);
1691         $layout_manager->setAttribute('html_varName',$html_var);
1692         $layout_manager->setAttribute('module_name', $source_module);
1693         list($orderBy,$desc) = $this->getOrderByInfo($html_var);
1694
1695         if($orderBy == 'amount*1')
1696         {
1697             $orderBy=  'amount';
1698         }
1699                 $buttons = false;
1700         foreach($subpanel_def->get_list_fields() as $column_name=>$widget_args)
1701         {
1702             $usage = empty($widget_args['usage']) ? '' : $widget_args['usage'];
1703             if($usage != 'query_only')
1704             {
1705                 $imgArrow = '';
1706
1707                 if($orderBy == $column_name || (isset($widget_args['sort_by']) && str_replace('.','_',$widget_args['sort_by']) == $orderBy))
1708                 {
1709                     $imgArrow = "_down";
1710                     if($this->sort_order == 'asc') {
1711                         $imgArrow = "_up";
1712                     }
1713                 }
1714
1715                 if (!preg_match("/_button/i", $column_name)) {
1716                         $widget_args['name']=$column_name;
1717                         $widget_args['sort'] = $imgArrow;
1718                         $widget_args['start_link_wrapper'] = $this->start_link_wrapper;
1719                         $widget_args['end_link_wrapper'] = $this->end_link_wrapper;
1720                         $widget_args['subpanel_module'] = $this->subpanel_module;
1721
1722                         $widget_contents = $layout_manager->widgetDisplay($widget_args);
1723                         $cell_width = empty($widget_args['width']) ? '' : $widget_args['width'];
1724                         $this->xTemplate->assign('HEADER_CELL', $widget_contents);
1725                         static $count;
1726                     if(!isset($count))$count = 0; else $count++;
1727                         $this->xTemplate->assign('CELL_COUNT', $count);
1728                         $this->xTemplate->assign('CELL_WIDTH', $cell_width);
1729                         $this->xTemplate->parse('dyn_list_view.header_cell');
1730                 } else {
1731                         $buttons = true;
1732                 }
1733             }
1734         }
1735
1736         if($buttons) {
1737                                 $this->xTemplate->assign('HEADER_CELL', "&nbsp;");
1738                                 $this->xTemplate->assign('CELL_COUNT', $count);
1739                         $this->xTemplate->assign('CELL_WIDTH', $cell_width);
1740                         $this->xTemplate->parse('dyn_list_view.header_cell');
1741         }
1742
1743     }
1744
1745
1746     /**
1747     * @return void
1748     * @param unknown $seed
1749     * @param unknown $xTemplateSection
1750     * @param unknown $html_varName
1751     * @desc PUBLIC FUNCTION Handles List Views using seeds that extend SugarBean
1752         $XTemplateSection is the section in the XTemplate file that should be parsed usually main
1753         $html_VarName is the variable name used in the XTemplateFile e.g. TASK
1754         $seed is a seed there are two types of seeds one is a subclass of SugarBean, the other is a list usually created from a sugar bean using get_list
1755         if no XTemplate is set it will create  a new XTemplate
1756         * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc..
1757         * All Rights Reserved..
1758         * Contributor(s): ______________________________________..
1759     */
1760
1761     function processListViewTwo($seed, $xTemplateSection, $html_varName) {
1762         global $current_user;
1763         if(!isset($this->xTemplate)) {
1764             $this->createXTemplate();
1765         }
1766
1767         $isSugarBean = is_subclass_of($seed, "SugarBean");
1768         $list = null;
1769
1770         if($isSugarBean) {
1771             $list = $this->processSugarBean($xTemplateSection, $html_varName, $seed);
1772         } else {
1773             $list = $seed;
1774         }
1775
1776         if (is_object($seed) && isset($seed->object_name) && $seed->object_name == 'WorkFlow') {
1777             $tab=array();
1778             $access = get_workflow_admin_modules_for_user($current_user);
1779             for ($i = 0; $i < count($list); $i++) {
1780                 if(!empty($access[$list[$i]->base_module])){
1781                     $tab[]=$list[$i];
1782                 }
1783             }
1784             $list = $tab;
1785         }
1786
1787         if($this->is_dynamic) {
1788             $this->processHeaderDynamic($xTemplateSection,$html_varName);
1789             $this->processListRows($list,$xTemplateSection, $html_varName);
1790         } else {
1791             $this->processSortArrows($html_varName);
1792
1793             if($isSugarBean) {
1794                 $seed->parse_additional_headers($this->xTemplate, $xTemplateSection);
1795             }
1796             $this->xTemplateAssign('CHECKALL', SugarThemeRegistry::current()->getImage('blank', '', 1, 1, ".gif", ''));
1797
1798             // Process the  order by before processing the pro_nav.  The pro_nav requires the order by values to be set
1799             $this->processOrderBy($html_varName);
1800
1801
1802             $this->processListRows($list,$xTemplateSection, $html_varName);
1803         }
1804
1805         if($this->display_header_and_footer) {
1806             $this->getAdditionalHeader();
1807             if(!empty($this->header_title)) {
1808                 echo get_form_header($this->header_title, $this->header_text, false);
1809             }
1810         }
1811
1812         $this->xTemplate->out($xTemplateSection);
1813
1814         if(isset($_SESSION['validation'])) {
1815             print base64_decode('PGEgaHJlZj0naHR0cDovL3d3dy5zdWdhcmNybS5jb20nPlBPV0VSRUQmbmJzcDtCWSZuYnNwO1NVR0FSQ1JNPC9hPg==');
1816         }
1817     }
1818
1819     function getArrowStart() {
1820         $imgFileParts = pathinfo(SugarThemeRegistry::current()->getImageURL("arrow.gif"));
1821
1822         return "&nbsp;<!--not_in_theme!--><img border='0' src='".$imgFileParts['dirname']."/".$imgFileParts['filename']."";
1823     }
1824
1825     function getArrowUpDownStart($upDown) {
1826         $ext = ( SugarThemeRegistry::current()->pngSupport ? "png" : "gif" );
1827
1828         if (!isset($upDown) || empty($upDown)) {
1829             $upDown = "";
1830         }
1831         return "&nbsp;<img border='0' src='".SugarThemeRegistry::current()->getImageURL("arrow{$upDown}.{$ext}")."' ";
1832     }
1833
1834         function getArrowEnd() {
1835                 $imgFileParts = pathinfo(SugarThemeRegistry::current()->getImageURL("arrow.gif"));
1836
1837         list($width,$height) = ListView::getArrowImageSize();
1838
1839                 return '.'.$imgFileParts['extension']."' width='$width' height='$height' align='absmiddle' alt=".translate('LBL_SORT').">";
1840     }
1841
1842     function getArrowUpDownEnd($upDown) {
1843         if (!isset($upDown) || empty($upDown)) {
1844             $upDown = "";
1845         }
1846         $imgFileParts = pathinfo(SugarThemeRegistry::current()->getImageURL("arrow{$upDown}.gif"));
1847
1848         list($width,$height) = ListView::getArrowUpDownImageSize($upDown);
1849
1850         //get the right alt tag for the sort
1851         $sortStr = translate('LBL_ALT_SORT');
1852         if($upDown == '_down'){
1853             $sortStr = translate('LBL_ALT_SORT_DESC');
1854         }elseif($upDown == '_up'){
1855             $sortStr = translate('LBL_ALT_SORT_ASC');
1856         }
1857         return " width='$width' height='$height' align='absmiddle' alt='$sortStr'>";
1858     }
1859
1860         function getArrowImageSize() {
1861             // jbasicChartDashletsExpColust get the non-sort image's size.. the up and down have be the same.
1862                 $image = SugarThemeRegistry::current()->getImageURL("arrow.gif",false);
1863
1864         $cache_key = 'arrow_size.'.$image;
1865
1866         // Check the cache
1867         $result = sugar_cache_retrieve($cache_key);
1868         if(!empty($result))
1869         return $result;
1870
1871         // No cache hit.  Calculate the value and return.
1872         $result = getimagesize($image);
1873         sugar_cache_put($cache_key, $result);
1874         return $result;
1875     }
1876
1877     function getArrowUpDownImageSize($upDown) {
1878         // just get the non-sort image's size.. the up and down have be the same.
1879         $image = SugarThemeRegistry::current()->getImageURL("arrow{$upDown}.gif",false);
1880
1881         $cache_key = 'arrowupdown_size.'.$image;
1882
1883         // Check the cache
1884         $result = sugar_cache_retrieve($cache_key);
1885         if(!empty($result))
1886         return $result;
1887
1888         // No cache hit.  Calculate the value and return.
1889         $result = getimagesize($image);
1890         sugar_cache_put($cache_key, $result);
1891         return $result;
1892     }
1893
1894         function getOrderByInfo($html_varName)
1895         {
1896                 $orderBy = $this->getSessionVariable($html_varName, "OBL");
1897                 $desc = $this->getSessionVariable($html_varName, $orderBy.'S');
1898                 $orderBy = str_replace('.', '_', $orderBy);
1899                 return array($orderBy,$desc);
1900         }
1901
1902     function processSortArrows($html_varName)
1903     {
1904
1905         $this->xTemplateAssign("arrow_start", $this->getArrowStart());
1906
1907         list($orderBy,$desc) = $this->getOrderByInfo($html_varName);
1908
1909                 $imgArrow = "_up";
1910                 if($desc) {
1911                         $imgArrow = "_down";
1912                 }
1913                 /**
1914                  * @deprecated only used by legacy opportunites listview, nothing current. Leaving for BC
1915                  */
1916                 if($orderBy == 'amount')
1917                 {
1918                         $this->xTemplateAssign('amount_arrow', $imgArrow);
1919                 }
1920                 else if($orderBy == 'amount_usdollar')
1921                 {
1922                         $this->xTemplateAssign('amount_usdollar_arrow', $imgArrow);
1923                 }
1924                 else
1925                 {
1926                         $this->xTemplateAssign($orderBy.'_arrow', $imgArrow);
1927                 }
1928
1929         $this->xTemplateAssign('arrow_end', $this->getArrowEnd());
1930     }
1931
1932     // this is where translation happens for dynamic list views
1933     function loadListFieldDefs(&$subpanel_fields,&$child_focus)
1934     {
1935         $this->list_field_defs = $subpanel_fields;
1936
1937         for($i=0;$i < count($this->list_field_defs);$i++)
1938         {
1939             $list_field = $this->list_field_defs[$i];
1940             $field_def = null;
1941             $key = '';
1942             if(!empty($list_field['vname']))
1943             {
1944                 $key = $list_field['vname'];
1945             } else if(isset($list_field['name']) &&  isset($child_focus->field_defs[$list_field['name']]))
1946             {
1947                     $field_def = $child_focus->field_defs[$list_field['name']];
1948                     $key = $field_def['vname'];
1949             }
1950             if(!empty($key))
1951             {
1952                 $list_field['label'] = translate($key,$child_focus->module_dir);
1953                 $this->list_field_defs[$i]['label'] = preg_replace('/:$/','',$list_field['label']);
1954             }
1955             else
1956             {
1957                 $this->list_field_defs[$i]['label'] ='&nbsp;';
1958             }
1959         }
1960     }
1961
1962     function unique_id() {
1963         return sugar_microtime();
1964     }
1965
1966      /**INTERNAL FUNCTION sets a session variable keeping it local to the listview
1967      not the current_module
1968      * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
1969      * All Rights Reserved.
1970      * Contributor(s): ______________________________________.
1971      */
1972      function setLocalSessionVariable($localVarName,$varName, $value) {
1973         $_SESSION[$localVarName."_".$varName] = $value;
1974      }
1975
1976      /**INTERNAL FUNCTION returns a session variable that is local to the listview,
1977      not the current_module
1978      * Portions created by SugarCRM are Copyright (C) SugarCRM, Inc.
1979      * All Rights Reserved.
1980      * Contributor(s): ______________________________________.
1981      */
1982  function getLocalSessionVariable($localVarName,$varName) {
1983     if(isset($_SESSION[$localVarName."_".$varName])) {
1984         return $_SESSION[$localVarName."_".$varName];
1985     }
1986     else{
1987         return "";
1988     }
1989  }
1990
1991  /* Set to true if you want Additional Details to appear in the listview
1992   */
1993  function setAdditionalDetails($value = true, $function = '') {
1994     if(!empty($function)) $this->additionalDetailsFunction = $function;
1995     $this->_additionalDetails = $value;
1996  }
1997
1998 }
1999 ?>