Project

General

Profile

Download (22.8 KB) Statistics
| Branch: | Tag: | Revision:
1
<?php
2
/*
3
 * index.php
4
 *
5
 * part of pfSense (https://www.pfsense.org)
6
 * Copyright (c) 2004-2016 Rubicon Communications, LLC (Netgate)
7
 * All rights reserved.
8
 *
9
 * originally based on m0n0wall (http://m0n0.ch/wall)
10
 * Copyright (c) 2003-2004 Manuel Kasper <mk@neon1.net>.
11
 * All rights reserved.
12
 *
13
 * Licensed under the Apache License, Version 2.0 (the "License");
14
 * you may not use this file except in compliance with the License.
15
 * You may obtain a copy of the License at
16
 *
17
 * http://www.apache.org/licenses/LICENSE-2.0
18
 *
19
 * Unless required by applicable law or agreed to in writing, software
20
 * distributed under the License is distributed on an "AS IS" BASIS,
21
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
22
 * See the License for the specific language governing permissions and
23
 * limitations under the License.
24
 */
25

    
26
##|+PRIV
27
##|*IDENT=page-system-login-logout
28
##|*NAME=System: Login / Logout / Dashboard
29
##|*DESCR=Allow access to the 'System: Login / Logout' page and Dashboard.
30
##|*MATCH=index.php*
31
##|-PRIV
32

    
33
// Turn on buffering to speed up rendering
34
ini_set('output_buffering', 'true');
35

    
36
// Start buffering with a cache size of 100000
37
ob_start(null, "1000");
38

    
39
## Load Essential Includes
40
require_once('guiconfig.inc');
41
require_once('functions.inc');
42
require_once('notices.inc');
43
require_once("pkg-utils.inc");
44

    
45
if (isset($_POST['closenotice'])) {
46
	close_notice($_POST['closenotice']);
47
	sleep(1);
48
	exit;
49
}
50

    
51
if (isset($_REQUEST['closenotice'])) {
52
	close_notice($_REQUEST['closenotice']);
53
	sleep(1);
54
}
55

    
56
if ($g['disablecrashreporter'] != true) {
57
	// Check to see if we have a crash report
58
	$x = 0;
59
	if (file_exists("/tmp/PHP_errors.log")) {
60
		$total = `/bin/cat /tmp/PHP_errors.log | /usr/bin/wc -l | /usr/bin/awk '{ print $1 }'`;
61
		if ($total > 0) {
62
			$x++;
63
		}
64
	}
65

    
66
	$crash = glob("/var/crash/*");
67
	$skip_files = array(".", "..", "minfree", "");
68

    
69
	if (is_array($crash)) {
70
		foreach ($crash as $c) {
71
			if (!in_array(basename($c), $skip_files)) {
72
				$x++;
73
			}
74
		}
75

    
76
		if ($x > 0) {
77
			$savemsg = sprintf(gettext("%s has detected a crash report or programming bug."), $g['product_name']) . " ";
78
			if (isAllowedPage("/crash_reporter.php")) {
79
				$savemsg .= sprintf(gettext('Click %1$shere%2$s for more information.'), '<a href="crash_reporter.php">', '</a>');
80
			} else {
81
				$savemsg .= sprintf(gettext("Contact a firewall administrator for more information."));
82
			}
83
			$class = "warning";
84
		}
85
	}
86
}
87

    
88
## Include each widget php include file.
89
## These define vars that specify the widget title and title link.
90

    
91
$directory = "/usr/local/www/widgets/include/";
92
$dirhandle = opendir($directory);
93
$filename = "";
94

    
95
while (($filename = readdir($dirhandle)) !== false) {
96
	if (strtolower(substr($filename, -4)) == ".inc" && file_exists($directory . $filename)) {
97
		include_once($directory . $filename);
98
	}
99
}
100

    
101
##build list of widgets
102
foreach (glob("/usr/local/www/widgets/widgets/*.widget.php") as $file) {
103
	$basename = basename($file, '.widget.php');
104
	// Get the widget title that should be in a var defined in the widget's inc file.
105
	$widgettitle = ${$basename . '_title'};
106

    
107
	if (empty(trim($widgettitle))) {
108
		// Fall back to constructing a title from the file name of the widget.
109
		$widgettitle = ucwords(str_replace('_', ' ', $basename));
110
	}
111

    
112
	$known_widgets[$basename . '-0'] = array(
113
		'basename' => $basename,
114
		'title' => $widgettitle,
115
		'display' => 'none',
116
		'multicopy' => ${$basename . '_allow_multiple_widget_copies'}
117
	);
118
}
119

    
120
##if no config entry found, initialize config entry
121
if (!is_array($config['widgets'])) {
122
	$config['widgets'] = array();
123
}
124

    
125
if (!is_array($user_settings['widgets'])) {
126
	$user_settings['widgets'] = array();
127
}
128

    
129
if ($_POST && $_POST['sequence']) {
130

    
131
	// Start with the user's widget settings.
132
	$widget_settings = $user_settings['widgets'];
133

    
134
	$widget_sep = ',';
135
	$widget_seq_array = explode($widget_sep, rtrim($_POST['sequence'], $widget_sep));
136
	$widget_counter_array = array();
137
	$widget_sep = '';
138

    
139
	// Make a record of the counter of each widget that is in use.
140
	foreach ($widget_seq_array as $widget_seq_data) {
141
		list($basename, $col, $display, $widget_counter) = explode(':', $widget_seq_data);
142

    
143
		if ($widget_counter != 'next') {
144
			if (!is_numeric($widget_counter)) {
145
				continue;
146
			}
147
			$widget_counter_array[$basename][$widget_counter] = true;
148
			$widget_sequence .= $widget_sep . $widget_seq_data;
149
			$widget_sep = ',';
150
		}
151
	}
152

    
153
	// Find any new entry (and do not assume there is only 1 new entry)
154
	foreach ($widget_seq_array as $widget_seq_data) {
155
		list($basename, $col, $display, $widget_counter) = explode(':', $widget_seq_data);
156

    
157
		if ($widget_counter == 'next') {
158
			// Construct the widget counter of the new widget instance by finding
159
			// the first non-negative integer that is not in use.
160
			// The reasoning here is that if you just deleted a widget instance,
161
			// e.g. had System Information 0,1,2 and deleted 1,
162
			// then when you add System Information again it will become instance 1,
163
			// which will bring back whatever filter selections happened to be on
164
			// the previous instance 1.
165
			$instance_num = 0;
166

    
167
			while (isset($widget_counter_array[$basename][$instance_num])) {
168
				$instance_num++;
169
			}
170

    
171
			$widget_sequence .= $widget_sep . $basename . ':' . $col . ':' . $display . ':' . $instance_num;
172
			$widget_counter_array[$basename][$instance_num] = true;
173
			$widget_sep = ',';
174
		}
175
	}
176

    
177
	$widget_settings['sequence'] = $widget_sequence;
178

    
179
	foreach ($widget_counter_array as $basename => $instances) {
180
		foreach ($instances as $instance => $value) {
181
			$widgetconfigname = $basename . '-' . $instance . '-config';
182
			if ($_POST[$widgetconfigname]) {
183
				$widget_settings[$widgetconfigname] = $_POST[$widgetconfigname];
184
			}
185
		}
186
	}
187

    
188
	save_widget_settings($_SESSION['Username'], $widget_settings);
189
	header("Location: /");
190
	exit;
191
}
192

    
193
## Load Functions Files
194
require_once('includes/functions.inc.php');
195

    
196
## Check to see if we have a swap space,
197
## if true, display, if false, hide it ...
198
if (file_exists("/usr/sbin/swapinfo")) {
199
	$swapinfo = `/usr/sbin/swapinfo`;
200
	if (stristr($swapinfo, '%') == true) $showswap=true;
201
}
202

    
203
## User recently restored his config.
204
## If packages are installed lets resync
205
if (file_exists('/conf/needs_package_sync')) {
206
	if ($config['installedpackages'] <> '' && is_array($config['installedpackages']['package'])) {
207
		## If the user has logged into webGUI quickly while the system is booting then do not redirect them to
208
		## the package reinstall page. That is about to be done by the boot script anyway.
209
		## The code in head.inc will put up a notice to the user.
210
		if (!platform_booting()) {
211
			header('Location: pkg_mgr_install.php?mode=reinstallall');
212
			exit;
213
		}
214
	} else {
215
		@unlink('/conf/needs_package_sync');
216
	}
217
}
218

    
219
## If it is the first time webConfigurator has been
220
## accessed since initial install show this stuff.
221
if (file_exists('/conf/trigger_initial_wizard')) {
222
?>
223
<!DOCTYPE html>
224
<html lang="en">
225
	<head>
226
		<link rel="stylesheet" href="/css/pfSense.css" />
227
		<title><?=$g['product_name']?>.localdomain - <?=$g['product_name']?> first time setup</title>
228
		<meta http-equiv="refresh" content="1;url=wizard.php?xml=setup_wizard.xml" />
229
	</head>
230
	<body id="loading-wizard" class="no-menu">
231
		<div id="jumbotron">
232
			<div class="container">
233
				<div class="col-sm-offset-3 col-sm-6 col-xs-12">
234
					<font color="white">
235
					<p><h3><?=sprintf(gettext("Welcome to %s!") . "\n", $g['product_name'])?></h3></p>
236
					<p><?=gettext("One moment while the initial setup wizard starts.")?></p>
237
					<p><?=gettext("Embedded platform users: Please be patient, the wizard takes a little longer to run than the normal GUI.")?></p>
238
					<p><?=sprintf(gettext("To bypass the wizard, click on the %s logo on the initial page."), $g['product_name'])?></p>
239
					</font>
240
				</div>
241
			</div>
242
		</div>
243
	</body>
244
</html>
245
<?php
246
	exit;
247
}
248

    
249
## Find out whether there's hardware encryption or not
250
unset($hwcrypto);
251
$fd = @fopen("{$g['varlog_path']}/dmesg.boot", "r");
252
if ($fd) {
253
	while (!feof($fd)) {
254
		$dmesgl = fgets($fd);
255
		if (preg_match("/^hifn.: (.*?),/", $dmesgl, $matches)
256
			or preg_match("/.*(VIA Padlock)/", $dmesgl, $matches)
257
			or preg_match("/^safe.: (\w.*)/", $dmesgl, $matches)
258
			or preg_match("/^ubsec.: (.*?),/", $dmesgl, $matches)
259
			or preg_match("/^padlock.: <(.*?)>,/", $dmesgl, $matches)) {
260
			$hwcrypto = $matches[1];
261
			break;
262
		}
263
	}
264
	fclose($fd);
265
	if (!isset($hwcrypto) && get_single_sysctl("dev.aesni.0.%desc")) {
266
		$hwcrypto = get_single_sysctl("dev.aesni.0.%desc");
267
	}
268
}
269

    
270
##build widget saved list information
271
if ($user_settings['widgets']['sequence'] != "") {
272
	$dashboardcolumns = isset($user_settings['webgui']['dashboardcolumns']) ? $user_settings['webgui']['dashboardcolumns'] : 2;
273
	$pconfig['sequence'] = $user_settings['widgets']['sequence'];
274
	$widgetsfromconfig = array();
275

    
276
	foreach (explode(',', $pconfig['sequence']) as $line) {
277
		$line_items = explode(':', $line);
278
		if (count($line_items) == 3) {
279
			// There can be multiple copies of a widget on the dashboard.
280
			// Default the copy number if it is not present (e.g. from old configs)
281
			$line_items[] = 0;
282
		}
283

    
284
		list($basename, $col, $display, $copynum) = $line_items;
285
		if (!is_numeric($copynum)) {
286
			continue;
287
		}
288

    
289
		// be backwards compatible
290
		// If the display column information is missing, we will assign a temporary
291
		// column here. Next time the user saves the dashboard it will fix itself
292
		if ($col == "") {
293
			if ($basename == "system_information") {
294
				$col = "col1";
295
			} else {
296
				$col = "col2";
297
			}
298
		}
299

    
300
		// Limit the column to the current dashboard columns.
301
		if (substr($col, 3) > $dashboardcolumns) {
302
			$col = "col" . $dashboardcolumns;
303
		}
304

    
305
		$offset = strpos($basename, '-container');
306
		if (false !== $offset) {
307
			$basename = substr($basename, 0, $offset);
308
		}
309
		$widgetkey = $basename . '-' . $copynum;
310

    
311
		if (isset($user_settings['widgets'][$widgetkey]['descr'])) {
312
			$widgettitle = htmlentities($user_settings['widgets'][$widgetkey]['descr']);
313
		} else {
314
			// Get the widget title that should be in a var defined in the widget's inc file.
315
			$widgettitle = ${$basename . '_title'};
316

    
317
			if (empty(trim($widgettitle))) {
318
				// Fall back to constructing a title from the file name of the widget.
319
				$widgettitle = ucwords(str_replace('_', ' ', $basename));
320
			}
321
		}
322

    
323
		$widgetsfromconfig[$widgetkey] = array(
324
			'basename' => $basename,
325
			'title' => $widgettitle,
326
			'col' => $col,
327
			'display' => $display,
328
			'copynum' => $copynum,
329
			'multicopy' => ${$basename . '_allow_multiple_widget_copies'}
330
		);
331

    
332
		// Update the known_widgets entry so we know if any copy of the widget is being displayed
333
		$known_widgets[$basename . '-0']['display'] = $display;
334
	}
335

    
336
	// add widgets that may not be in the saved configuration, in case they are to be displayed later
337
	$widgets = $widgetsfromconfig + $known_widgets;
338

    
339
	##find custom configurations of a particular widget and load its info to $pconfig
340
	foreach ($widgets as $widgetname => $widgetconfig) {
341
		if ($config['widgets'][$widgetname . '-config']) {
342
			$pconfig[$widgetname . '-config'] = $config['widgets'][$widgetname . '-config'];
343
		}
344
	}
345
}
346

    
347
## Get the configured options for Show/Hide available widgets panel.
348
$dashboard_available_widgets_hidden = !$user_settings['webgui']['dashboardavailablewidgetspanel'];
349

    
350
if ($dashboard_available_widgets_hidden) {
351
	$panel_state = 'out';
352
	$panel_body_state = 'in';
353
} else {
354
	$panel_state = 'in';
355
	$panel_body_state = 'out';
356
}
357

    
358
## Set Page Title and Include Header
359
$pgtitle = array(gettext("Status"), gettext("Dashboard"));
360
include("head.inc");
361

    
362
if ($savemsg) {
363
	print_info_box($savemsg, $class);
364
}
365

    
366
pfSense_handle_custom_code("/usr/local/pkg/dashboard/pre_dashboard");
367

    
368
?>
369

    
370
<div class="panel panel-default collapse <?=$panel_state?>" id="widget-available">
371
	<div class="panel-heading">
372
		<h2 class="panel-title"><?=gettext("Available Widgets"); ?>
373
			<span class="widget-heading-icon">
374
				<a data-toggle="collapse" href="#widget-available_panel-body" id="widgets-available">
375
					<i class="fa fa-plus-circle"></i>
376
				</a>
377
			</span>
378
		</h2>
379
	</div>
380
	<div id="widget-available_panel-body" class="panel-body collapse <?=$panel_body_state?>">
381
		<div class="content">
382
			<div class="row">
383
<?php
384

    
385
// Build the Available Widgets table using a sorted copy of the $known_widgets array
386
$available = $known_widgets;
387
uasort($available, function($a, $b){ return strcasecmp($a['title'], $b['title']); });
388

    
389
foreach ($available as $widgetkey => $widgetconfig):
390
	// If the widget supports multiple copies, or no copies are displayed yet, then it is available to add
391
	if (($widgetconfig['multicopy']) || ($widgetconfig['display'] == 'none')):
392
?>
393
		<div class="col-sm-3"><a href="#" id="btnadd-<?=$widgetconfig['basename']?>"><i class="fa fa-plus"></i> <?=$widgetconfig['title']?></a></div>
394
	<?php endif; ?>
395
<?php
396
endforeach;
397
?>
398
			</div>
399
<p style="text-align:center"><?=sprintf(gettext('Other dashboard settings are available from the <a href="%s">General Setup</a> page.'), '/system.php')?></p>
400
		</div>
401
	</div>
402
</div>
403

    
404
<div class="hidden" id="widgetSequence">
405
	<form action="/" method="post" id="widgetSequence_form" name="widgetForm">
406
		<input type="hidden" name="sequence" value="" />
407
	</form>
408
</div>
409

    
410
<?php
411
$widgetColumns = array();
412
foreach ($widgets as $widgetkey => $widgetconfig) {
413
	if ($widgetconfig['display'] != 'none' && file_exists("/usr/local/www/widgets/widgets/{$widgetconfig['basename']}.widget.php")) {
414
		if (!isset($widgetColumns[$widgetconfig['col']])) {
415
			$widgetColumns[$widgetconfig['col']] = array();
416
		}
417
		$widgetColumns[$widgetconfig['col']][$widgetkey] = $widgetconfig;
418
	}
419
}
420
?>
421

    
422
<div class="row">
423
<?php
424
	$columnWidth = (int) (12 / $numColumns);
425

    
426
	for ($currentColumnNumber = 1; $currentColumnNumber <= $numColumns; $currentColumnNumber++) {
427

    
428

    
429
		//if col$currentColumnNumber exists
430
		if (isset($widgetColumns['col'.$currentColumnNumber])) {
431
			echo '<div class="col-md-' . $columnWidth . '" id="widgets-col' . $currentColumnNumber . '">';
432
			$columnWidgets = $widgetColumns['col'.$currentColumnNumber];
433

    
434
			foreach ($columnWidgets as $widgetkey => $widgetconfig) {
435
				// Construct some standard names for the ids this widget will use for its commonly-used elements.
436
				// Included widget.php code can rely on and use these, so the format does not have to be repeated in every widget.php
437
				$widget_panel_body_id = 'widget-' . $widgetkey . '_panel-body';
438
				$widget_panel_footer_id = 'widget-' . $widgetkey . '_panel-footer';
439
				$widget_showallnone_id = 'widget-' . $widgetkey . '_showallnone';
440

    
441
				// Compose the widget title and include the title link if available
442
				$widgetlink = ${$widgetconfig['basename'] . '_title_link'};
443

    
444
				if ((strlen($widgetlink) > 0)) {
445
					$wtitle = '<a href="' . $widgetlink . '"> ' . $widgetconfig['title'] . '</a>';
446
				} else {
447
					$wtitle = $widgetconfig['title'];
448
				}
449
				?>
450
				<div class="panel panel-default" id="widget-<?=$widgetkey?>">
451
					<div class="panel-heading">
452
						<h2 class="panel-title">
453
							<?=$wtitle?>
454
							<span class="widget-heading-icon">
455
								<a data-toggle="collapse" href="#<?=$widget_panel_footer_id?>" class="config hidden">
456
									<i class="fa fa-wrench"></i>
457
								</a>
458
								<a data-toggle="collapse" href="#<?=$widget_panel_body_id?>">
459
									<!--  actual icon is determined in css based on state of body -->
460
									<i class="fa fa-plus-circle"></i>
461
								</a>
462
								<a data-toggle="close" href="#widget-<?=$widgetkey?>">
463
									<i class="fa fa-times-circle"></i>
464
								</a>
465
							</span>
466
						</h2>
467
					</div>
468
					<div id="<?=$widget_panel_body_id?>" class="panel-body collapse<?=($widgetconfig['display'] == 'close' ? '' : ' in')?>">
469
						<?php
470
							// For backward compatibility, included *.widget.php code needs the var $widgetname
471
							$widgetname = $widgetkey;
472
							// Determine if this is the first instance of this particular widget.
473
							// Provide the $widget_first_instance var, to make it easy for the included widget code
474
							// to be able to know if it is being included for the first time.
475
							if ($widgets_found[$widgetconfig['basename']]) {
476
								$widget_first_instance = false;
477
							} else {
478
								$widget_first_instance = true;
479
								$widgets_found[$widgetconfig['basename']] = true;
480
							}
481
							include('/usr/local/www/widgets/widgets/' . $widgetconfig['basename'] . '.widget.php');
482
						?>
483
					</div>
484
				</div>
485
				<?php
486
			}
487
			echo "</div>";
488
		} else {
489
			echo '<div class="col-md-' . $columnWidth . '" id="widgets-col' . $currentColumnNumber . '"></div>';
490
		}
491

    
492
	}
493
?>
494

    
495
</div>
496

    
497
<?php
498
/*
499
 * Import the modal form used to display the copyright/usage information
500
 * when trigger file exists. Trigger file is created during upgrade process
501
 * when /etc/version changes
502
 */
503
if (file_exists("{$g['cf_conf_path']}/copynotice_display")) {
504
	require_once("{$g['www_path']}/copynotice.inc");
505
	@unlink("{$g['cf_conf_path']}/copynotice_display");
506
}
507
?>
508

    
509
<script type="text/javascript">
510
//<![CDATA[
511

    
512
dirty = false;
513
function updateWidgets(newWidget) {
514
	var sequence = '';
515

    
516
	$('.container .col-md-<?=$columnWidth?>').each(function(idx, col) {
517
		$('.panel', col).each(function(idx, widget) {
518
			var isOpen = $('.panel-body', widget).hasClass('in');
519
			var widget_basename = widget.id.split('-')[1];
520

    
521
			// Only save details for panels that have id's like'widget-*'
522
			// Some widgets create other panels, so ignore any of those.
523
			if ((widget.id.split('-')[0] == 'widget') && (typeof widget_basename !== 'undefined')) {
524
				sequence += widget_basename + ':' + col.id.split('-')[1] + ':' + (isOpen ? 'open' : 'close') + ':' + widget.id.split('-')[2] + ',';
525
			}
526
		});
527
	});
528

    
529
	if (typeof newWidget !== 'undefined') {
530
		// The system_information widget is always added to column one. Others go in column two
531
		if (newWidget == "system_information") {
532
			sequence += newWidget.split('-')[0] + ':' + 'col1:open:next';
533
		} else {
534
			sequence += newWidget.split('-')[0] + ':' + 'col2:open:next';
535
		}
536
	}
537

    
538
	$('input[name=sequence]', $('#widgetSequence_form')).val(sequence);
539
}
540

    
541
// Determine if all the checkboxes are checked
542
function are_all_checked(checkbox_panel_ref) {
543
	var allBoxesChecked = true;
544
	$(checkbox_panel_ref).each(function() {
545
		if ((this.type == 'checkbox') && !this.checked) {
546
			allBoxesChecked = false;
547
		}
548
	});
549
	return allBoxesChecked;
550
}
551

    
552
// If the checkboxes are all checked, then clear them all.
553
// Otherwise set them all.
554
function set_clear_checkboxes(checkbox_panel_ref) {
555
	checkTheBoxes = !are_all_checked(checkbox_panel_ref);
556

    
557
	$(checkbox_panel_ref).each(function() {
558
		$(this).prop("checked", checkTheBoxes);
559
	});
560
}
561

    
562
// Set the given id to All or None button depending if the checkboxes are all checked.
563
function set_all_none_button(checkbox_panel_ref, all_none_button_id) {
564
	if (are_all_checked(checkbox_panel_ref)) {
565
		text = "<?=gettext('None')?>";
566
	} else {
567
		text = "<?=gettext('All')?>";
568
	}
569

    
570
	$("#" + all_none_button_id).html('<i class="fa fa-undo icon-embed-btn"></i>' + text);
571
}
572

    
573
// Setup the necessary events to manage the All/None button and included checkboxes
574
// used for selecting the items to show on a widget.
575
function set_widget_checkbox_events(checkbox_panel_ref, all_none_button_id) {
576
		set_all_none_button(checkbox_panel_ref, all_none_button_id);
577

    
578
		$(checkbox_panel_ref).change(function() {
579
			set_all_none_button(checkbox_panel_ref, all_none_button_id);
580
		});
581

    
582
		$("#" + all_none_button_id).click(function() {
583
			set_clear_checkboxes(checkbox_panel_ref);
584
			set_all_none_button(checkbox_panel_ref, all_none_button_id);
585
		});
586
}
587

    
588
// ---------------------Centralized widget refresh system -------------------------------------------
589
// These need to live outsie of the events.push() function to enable the widgets to see them
590
var ajaxspecs = new Array();	// Array to hold widget refresh specifications (objects )
591
var ajaxidx = 0;
592
var ajaxmutex = false;
593
var ajaxcntr = 0;
594

    
595
// Add a widget refresh object to the array list
596
function register_ajax(ws) {
597
  ajaxspecs.push(ws);
598
}
599
// ---------------------------------------------------------------------------------------------------
600

    
601
events.push(function() {
602
	// Make panels destroyable
603
	$('.container .panel-heading a[data-toggle="close"]').each(function (idx, el) {
604
		$(el).on('click', function(e) {
605
			$(el).parents('.panel').remove();
606
			updateWidgets();
607
			// Submit the form save/display all selected widgets
608
			$('[name=widgetForm]').submit();
609
		})
610
	});
611

    
612
	// Make panels sortable
613
	$('.container .col-md-<?=$columnWidth?>').sortable({
614
		handle: '.panel-heading',
615
		cursor: 'grabbing',
616
		connectWith: '.container .col-md-<?=$columnWidth?>',
617
		update: function(){
618
			dirty = true;
619
			$('#btnstore').removeClass('invisible');
620
		}
621
	});
622

    
623
	// On clicking a widget to install . .
624
	$('[id^=btnadd-]').click(function(event) {
625
		// Add the widget name to the list of displayed widgets
626
		updateWidgets(this.id.replace('btnadd-', ''));
627

    
628
		// Submit the form save/display all selected widgets
629
		$('[name=widgetForm]').submit();
630
	});
631

    
632

    
633
	$('#btnstore').click(function() {
634
		updateWidgets();
635
		dirty = false;
636
		$(this).addClass('invisible');
637
		$('[name=widgetForm]').submit();
638
	});
639

    
640
	// provide a warning message if the user tries to change page before saving
641
	$(window).bind('beforeunload', function(){
642
		if (dirty) {
643
			return ("<?=gettext('One or more widgets have been moved but have not yet been saved')?>");
644
		} else {
645
			return undefined;
646
		}
647
	});
648

    
649
	// Show the fa-save icon in the breadcrumb bar if the user opens or closes a panel (In case he/she wants to save the new state)
650
	// (Sometimes this will cause us to see the icon when we don't need it, but better that than the other way round)
651
	$('.panel').on('hidden.bs.collapse shown.bs.collapse', function (e) {
652
	    if (e.currentTarget.id != 'widget-available') {
653
			$('#btnstore').removeClass("invisible");
654
		}
655
	});
656

    
657
	// --------------------- EXPERIMENTAL centralized widget refresh system ------------------------------
658
	function make_ajax_call(wd) {
659
		ajaxmutex = true;
660

    
661
		$.ajax({
662
			type: 'POST',
663
			url: wd.url,
664
			dataType: 'html',
665
			data: wd.parms,
666

    
667
			success: function(data){
668
				if (data.length > 0 ) {
669
					wd.callback(data);
670
				}
671

    
672
				ajaxmutex = false;
673
			},
674

    
675
			error: function(e){
676
//				alert("Error: " + e);
677
				ajaxmutex = false;
678
			}
679
		});
680
	}
681

    
682
	// Loop through each AJAX widget refresh object, make the AJAX call and pass the
683
	// results back to the widget's callback function
684
	function executewidget() {
685
		if (ajaxspecs.length > 0) {
686
			var freq = ajaxspecs[ajaxidx].freq;	// widget can specifify it should be called freq times around hte loop
687

    
688
			if (!ajaxmutex) {
689
				if (((ajaxcntr % freq) === 0) && (typeof ajaxspecs[ajaxidx].callback === "function" )) {
690
				    make_ajax_call(ajaxspecs[ajaxidx]);
691
				}
692

    
693
			    if (++ajaxidx >= ajaxspecs.length) {
694
					ajaxidx = 0;
695

    
696
					if (++ajaxcntr >= 4096) {
697
						ajaxcntr = 0;
698
					}
699
			    }
700
			}
701

    
702
		    setTimeout(function() { executewidget(); }, 1000);
703
	  	}
704
	}
705

    
706
	// Kick it off
707
	executewidget();
708

    
709
	//----------------------------------------------------------------------------------------------------
710
});
711
//]]>
712
</script>
713

    
714
<?php
715
//build list of javascript include files
716
foreach (glob('widgets/javascript/*.js') as $file) {
717
	$mtime = filemtime("/usr/local/www/{$file}");
718
	echo '<script src="'.$file.'?v='.$mtime.'"></script>';
719
}
720

    
721
include("foot.inc");
(70-70/230)