diff options
| author | Nicolai Tellefsen <niccofyren@gmail.com> | 2016-03-04 21:56:46 +0100 | 
|---|---|---|
| committer | Nicolai Tellefsen <niccofyren@gmail.com> | 2016-03-04 21:56:46 +0100 | 
| commit | 7227a97b6a6a867b0b370657fe164c3151cccdeb (patch) | |
| tree | c34d3a50eefdd4418b8f21bdc28f275c44a27058 | |
| parent | f3b5042fddf1ceedd89f72527685b191df6bc6fa (diff) | |
| parent | 82128b9b97fdc5df611b42b1ebd119e1650d36aa (diff) | |
Merge branch 'timetravel-gui'
| -rw-r--r-- | web/nms.gathering.org/css/jquery.datetimepicker.css | 568 | ||||
| -rw-r--r-- | web/nms.gathering.org/index.html | 60 | ||||
| -rw-r--r-- | web/nms.gathering.org/js/jquery.datetimepicker.full.js | 3073 | ||||
| -rw-r--r-- | web/nms.gathering.org/js/nms.js | 59 | 
4 files changed, 3721 insertions, 39 deletions
| diff --git a/web/nms.gathering.org/css/jquery.datetimepicker.css b/web/nms.gathering.org/css/jquery.datetimepicker.css new file mode 100644 index 0000000..beda145 --- /dev/null +++ b/web/nms.gathering.org/css/jquery.datetimepicker.css @@ -0,0 +1,568 @@ +.xdsoft_datetimepicker { +	box-shadow: 0 5px 15px -5px rgba(0, 0, 0, 0.506); +	background: #fff; +	border-bottom: 1px solid #bbb; +	border-left: 1px solid #ccc; +	border-right: 1px solid #ccc; +	border-top: 1px solid #ccc; +	color: #333; +	font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; +	padding: 8px; +	padding-left: 0; +	padding-top: 2px; +	position: absolute; +	z-index: 9999; +	-moz-box-sizing: border-box; +	box-sizing: border-box; +	display: none; +} +.xdsoft_datetimepicker.xdsoft_rtl { +	padding: 8px 0 8px 8px; +} + +.xdsoft_datetimepicker iframe { +	position: absolute; +	left: 0; +	top: 0; +	width: 75px; +	height: 210px; +	background: transparent; +	border: none; +} + +/*For IE8 or lower*/ +.xdsoft_datetimepicker button { +	border: none !important; +} + +.xdsoft_noselect { +	-webkit-touch-callout: none; +	-webkit-user-select: none; +	-khtml-user-select: none; +	-moz-user-select: none; +	-ms-user-select: none; +	-o-user-select: none; +	user-select: none; +} + +.xdsoft_noselect::selection { background: transparent } +.xdsoft_noselect::-moz-selection { background: transparent } + +.xdsoft_datetimepicker.xdsoft_inline { +	display: inline-block; +	position: static; +	box-shadow: none; +} + +.xdsoft_datetimepicker * { +	-moz-box-sizing: border-box; +	box-sizing: border-box; +	padding: 0; +	margin: 0; +} + +.xdsoft_datetimepicker .xdsoft_datepicker, .xdsoft_datetimepicker .xdsoft_timepicker { +	display: none; +} + +.xdsoft_datetimepicker .xdsoft_datepicker.active, .xdsoft_datetimepicker .xdsoft_timepicker.active { +	display: block; +} + +.xdsoft_datetimepicker .xdsoft_datepicker { +	width: 224px; +	float: left; +	margin-left: 8px; +} +.xdsoft_datetimepicker.xdsoft_rtl .xdsoft_datepicker { +	float: right; +	margin-right: 8px; +	margin-left: 0; +} + +.xdsoft_datetimepicker.xdsoft_showweeks .xdsoft_datepicker { +	width: 256px; +} + +.xdsoft_datetimepicker .xdsoft_timepicker { +	width: 58px; +	float: left; +	text-align: center; +	margin-left: 8px; +	margin-top: 0; +} +.xdsoft_datetimepicker.xdsoft_rtl .xdsoft_timepicker { +	float: right; +	margin-right: 8px; +	margin-left: 0; +} + +.xdsoft_datetimepicker .xdsoft_datepicker.active+.xdsoft_timepicker { +	margin-top: 8px; +	margin-bottom: 3px +} + +.xdsoft_datetimepicker .xdsoft_mounthpicker { +	position: relative; +	text-align: center; +} + +.xdsoft_datetimepicker .xdsoft_label i, +.xdsoft_datetimepicker .xdsoft_prev, +.xdsoft_datetimepicker .xdsoft_next, +.xdsoft_datetimepicker .xdsoft_today_button { +	background-image: url(); +} + +.xdsoft_datetimepicker .xdsoft_label i { +	opacity: 0.5; +	background-position: -92px -19px; +	display: inline-block; +	width: 9px; +	height: 20px; +	vertical-align: middle; +} + +.xdsoft_datetimepicker .xdsoft_prev { +	float: left; +	background-position: -20px 0; +} +.xdsoft_datetimepicker .xdsoft_today_button { +	float: left; +	background-position: -70px 0; +	margin-left: 5px; +} + +.xdsoft_datetimepicker .xdsoft_next { +	float: right; +	background-position: 0 0; +} + +.xdsoft_datetimepicker .xdsoft_next, +.xdsoft_datetimepicker .xdsoft_prev , +.xdsoft_datetimepicker .xdsoft_today_button { +	background-color: transparent; +	background-repeat: no-repeat; +	border: 0 none; +	cursor: pointer; +	display: block; +	height: 30px; +	opacity: 0.5; +	-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=50)"; +	outline: medium none; +	overflow: hidden; +	padding: 0; +	position: relative; +	text-indent: 100%; +	white-space: nowrap; +	width: 20px; +	min-width: 0; +} + +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_prev, +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_next { +	float: none; +	background-position: -40px -15px; +	height: 15px; +	width: 30px; +	display: block; +	margin-left: 14px; +	margin-top: 7px; +} +.xdsoft_datetimepicker.xdsoft_rtl .xdsoft_timepicker .xdsoft_prev, +.xdsoft_datetimepicker.xdsoft_rtl .xdsoft_timepicker .xdsoft_next { +	float: none; +	margin-left: 0; +	margin-right: 14px; +} + +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_prev { +	background-position: -40px 0; +	margin-bottom: 7px; +	margin-top: 0; +} + +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_time_box { +	height: 151px; +	overflow: hidden; +	border-bottom: 1px solid #ddd; +} + +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_time_box >div >div { +	background: #f5f5f5; +	border-top: 1px solid #ddd; +	color: #666; +	font-size: 12px; +	text-align: center; +	border-collapse: collapse; +	cursor: pointer; +	border-bottom-width: 0; +	height: 25px; +	line-height: 25px; +} + +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_time_box >div > div:first-child { +	border-top-width: 0; +} + +.xdsoft_datetimepicker .xdsoft_today_button:hover, +.xdsoft_datetimepicker .xdsoft_next:hover, +.xdsoft_datetimepicker .xdsoft_prev:hover { +	opacity: 1; +	-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=100)"; +} + +.xdsoft_datetimepicker .xdsoft_label { +	display: inline; +	position: relative; +	z-index: 9999; +	margin: 0; +	padding: 5px 3px; +	font-size: 14px; +	line-height: 20px; +	font-weight: bold; +	background-color: #fff; +	float: left; +	width: 182px; +	text-align: center; +	cursor: pointer; +} + +.xdsoft_datetimepicker .xdsoft_label:hover>span { +	text-decoration: underline; +} + +.xdsoft_datetimepicker .xdsoft_label:hover i { +	opacity: 1.0; +} + +.xdsoft_datetimepicker .xdsoft_label > .xdsoft_select { +	border: 1px solid #ccc; +	position: absolute; +	right: 0; +	top: 30px; +	z-index: 101; +	display: none; +	background: #fff; +	max-height: 160px; +	overflow-y: hidden; +} + +.xdsoft_datetimepicker .xdsoft_label > .xdsoft_select.xdsoft_monthselect{ right: -7px } +.xdsoft_datetimepicker .xdsoft_label > .xdsoft_select.xdsoft_yearselect{ right: 2px } +.xdsoft_datetimepicker .xdsoft_label > .xdsoft_select > div > .xdsoft_option:hover { +	color: #fff; +	background: #ff8000; +} + +.xdsoft_datetimepicker .xdsoft_label > .xdsoft_select > div > .xdsoft_option { +	padding: 2px 10px 2px 5px; +	text-decoration: none !important; +} + +.xdsoft_datetimepicker .xdsoft_label > .xdsoft_select > div > .xdsoft_option.xdsoft_current { +	background: #33aaff; +	box-shadow: #178fe5 0 1px 3px 0 inset; +	color: #fff; +	font-weight: 700; +} + +.xdsoft_datetimepicker .xdsoft_month { +	width: 100px; +	text-align: right; +} + +.xdsoft_datetimepicker .xdsoft_calendar { +	clear: both; +} + +.xdsoft_datetimepicker .xdsoft_year{ +	width: 48px; +	margin-left: 5px; +} + +.xdsoft_datetimepicker .xdsoft_calendar table { +	border-collapse: collapse; +	width: 100%; + +} + +.xdsoft_datetimepicker .xdsoft_calendar td > div { +	padding-right: 5px; +} + +.xdsoft_datetimepicker .xdsoft_calendar th { +	height: 25px; +} + +.xdsoft_datetimepicker .xdsoft_calendar td,.xdsoft_datetimepicker .xdsoft_calendar th { +	width: 14.2857142%; +	background: #f5f5f5; +	border: 1px solid #ddd; +	color: #666; +	font-size: 12px; +	text-align: right; +	vertical-align: middle; +	padding: 0; +	border-collapse: collapse; +	cursor: pointer; +	height: 25px; +} +.xdsoft_datetimepicker.xdsoft_showweeks .xdsoft_calendar td,.xdsoft_datetimepicker.xdsoft_showweeks .xdsoft_calendar th { +	width: 12.5%; +} + +.xdsoft_datetimepicker .xdsoft_calendar th { +	background: #f1f1f1; +} + +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_today { +	color: #33aaff; +} + +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_highlighted_default { +	background: #ffe9d2; +	box-shadow: #ffb871 0 1px 4px 0 inset; +	color: #000; +} +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_highlighted_mint { +	background: #c1ffc9; +	box-shadow: #00dd1c 0 1px 4px 0 inset; +	color: #000; +} + +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_default, +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_current, +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_time_box >div >div.xdsoft_current { +	background: #33aaff; +	box-shadow: #178fe5 0 1px 3px 0 inset; +	color: #fff; +	font-weight: 700; +} + +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_other_month, +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_disabled, +.xdsoft_datetimepicker .xdsoft_time_box >div >div.xdsoft_disabled { +	opacity: 0.5; +	-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=50)"; +	cursor: default; +} + +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_other_month.xdsoft_disabled { +	opacity: 0.2; +	-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=20)"; +} + +.xdsoft_datetimepicker .xdsoft_calendar td:hover, +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_time_box >div >div:hover { +	color: #fff !important; +	background: #ff8000 !important; +	box-shadow: none !important; +} + +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_current.xdsoft_disabled:hover, +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_time_box>div>div.xdsoft_current.xdsoft_disabled:hover { +	background: #33aaff !important; +	box-shadow: #178fe5 0 1px 3px 0 inset !important; +	color: #fff !important; +} + +.xdsoft_datetimepicker .xdsoft_calendar td.xdsoft_disabled:hover, +.xdsoft_datetimepicker .xdsoft_timepicker .xdsoft_time_box >div >div.xdsoft_disabled:hover { +	color: inherit	!important; +	background: inherit !important; +	box-shadow: inherit !important; +} + +.xdsoft_datetimepicker .xdsoft_calendar th { +	font-weight: 700; +	text-align: center; +	color: #999; +	cursor: default; +} + +.xdsoft_datetimepicker .xdsoft_copyright { +	color: #ccc !important; +	font-size: 10px; +	clear: both; +	float: none; +	margin-left: 8px; +} + +.xdsoft_datetimepicker .xdsoft_copyright a { color: #eee !important } +.xdsoft_datetimepicker .xdsoft_copyright a:hover { color: #aaa !important } + +.xdsoft_time_box { +	position: relative; +	border: 1px solid #ccc; +} +.xdsoft_scrollbar >.xdsoft_scroller { +	background: #ccc !important; +	height: 20px; +	border-radius: 3px; +} +.xdsoft_scrollbar { +	position: absolute; +	width: 7px; +	right: 0; +	top: 0; +	bottom: 0; +	cursor: pointer; +} +.xdsoft_datetimepicker.xdsoft_rtl .xdsoft_scrollbar { +	left: 0; +	right: auto; +} +.xdsoft_scroller_box { +	position: relative; +} + +.xdsoft_datetimepicker.xdsoft_dark { +	box-shadow: 0 5px 15px -5px rgba(255, 255, 255, 0.506); +	background: #000; +	border-bottom: 1px solid #444; +	border-left: 1px solid #333; +	border-right: 1px solid #333; +	border-top: 1px solid #333; +	color: #ccc; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_timepicker .xdsoft_time_box { +	border-bottom: 1px solid #222; +} +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_timepicker .xdsoft_time_box >div >div { +	background: #0a0a0a; +	border-top: 1px solid #222; +	color: #999; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_label { +	background-color: #000; +} +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_label > .xdsoft_select { +	border: 1px solid #333; +	background: #000; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_label > .xdsoft_select > div > .xdsoft_option:hover { +	color: #000; +	background: #007fff; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_label > .xdsoft_select > div > .xdsoft_option.xdsoft_current { +	background: #cc5500; +	box-shadow: #b03e00 0 1px 3px 0 inset; +	color: #000; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_label i, +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_prev, +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_next, +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_today_button { +	background-image: url(); +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar td, +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar th { +	background: #0a0a0a; +	border: 1px solid #222; +	color: #999; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar th { +	background: #0e0e0e; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar td.xdsoft_today { +	color: #cc5500; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar td.xdsoft_highlighted_default { +	background: #ffe9d2; +	box-shadow: #ffb871 0 1px 4px 0 inset; +	color:#000; +} +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar td.xdsoft_highlighted_mint { +	background: #c1ffc9; +	box-shadow: #00dd1c 0 1px 4px 0 inset; +	color:#000; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar td.xdsoft_default, +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar td.xdsoft_current, +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_timepicker .xdsoft_time_box >div >div.xdsoft_current { +	background: #cc5500; +	box-shadow: #b03e00 0 1px 3px 0 inset; +	color: #000; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar td:hover, +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_timepicker .xdsoft_time_box >div >div:hover { +	color: #000 !important; +	background: #007fff !important; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_calendar th { +	color: #666; +} + +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_copyright { color: #333 !important } +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_copyright a { color: #111 !important } +.xdsoft_datetimepicker.xdsoft_dark .xdsoft_copyright a:hover { color: #555 !important } + +.xdsoft_dark .xdsoft_time_box { +	border: 1px solid #333; +} + +.xdsoft_dark .xdsoft_scrollbar >.xdsoft_scroller { +	background: #333 !important; +} +.xdsoft_datetimepicker .xdsoft_save_selected { +    display: block; +    border: 1px solid #dddddd !important; +    margin-top: 5px; +    width: 100%; +    color: #454551; +    font-size: 13px; +} +.xdsoft_datetimepicker .blue-gradient-button { +	font-family: "museo-sans", "Book Antiqua", sans-serif; +	font-size: 12px; +	font-weight: 300; +	color: #82878c; +	height: 28px; +	position: relative; +	padding: 4px 17px 4px 33px; +	border: 1px solid #d7d8da; +	background: -moz-linear-gradient(top, #fff 0%, #f4f8fa 73%); +	/* FF3.6+ */ +	background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #fff), color-stop(73%, #f4f8fa)); +	/* Chrome,Safari4+ */ +	background: -webkit-linear-gradient(top, #fff 0%, #f4f8fa 73%); +	/* Chrome10+,Safari5.1+ */ +	background: -o-linear-gradient(top, #fff 0%, #f4f8fa 73%); +	/* Opera 11.10+ */ +	background: -ms-linear-gradient(top, #fff 0%, #f4f8fa 73%); +	/* IE10+ */ +	background: linear-gradient(to bottom, #fff 0%, #f4f8fa 73%); +	/* W3C */ +	filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#fff', endColorstr='#f4f8fa',GradientType=0 ); +/* IE6-9 */ +} +.xdsoft_datetimepicker .blue-gradient-button:hover, .xdsoft_datetimepicker .blue-gradient-button:focus, .xdsoft_datetimepicker .blue-gradient-button:hover span, .xdsoft_datetimepicker .blue-gradient-button:focus span { +  color: #454551; +  background: -moz-linear-gradient(top, #f4f8fa 0%, #FFF 73%); +  /* FF3.6+ */ +  background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #f4f8fa), color-stop(73%, #FFF)); +  /* Chrome,Safari4+ */ +  background: -webkit-linear-gradient(top, #f4f8fa 0%, #FFF 73%); +  /* Chrome10+,Safari5.1+ */ +  background: -o-linear-gradient(top, #f4f8fa 0%, #FFF 73%); +  /* Opera 11.10+ */ +  background: -ms-linear-gradient(top, #f4f8fa 0%, #FFF 73%); +  /* IE10+ */ +  background: linear-gradient(to bottom, #f4f8fa 0%, #FFF 73%); +  /* W3C */ +  filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#f4f8fa', endColorstr='#FFF',GradientType=0 ); +  /* IE6-9 */ +} diff --git a/web/nms.gathering.org/index.html b/web/nms.gathering.org/index.html index 111dd52..4cece1a 100644 --- a/web/nms.gathering.org/index.html +++ b/web/nms.gathering.org/index.html @@ -17,6 +17,9 @@      <!-- Custom styles for this template -->      <link href="css/navbar-static-top.css" rel="stylesheet"> +    <!-- Date-picker styles --> +    <link href="css/jquery.datetimepicker.css" rel="stylesheet"> +      <!-- HTML5 shim and Respond.js for IE8 support of HTML5 elements and media queries -->      <!--[if lt IE 9]>        <script src="https://oss.maxcdn.com/html5shiv/3.7.2/html5shiv.min.js"></script> @@ -62,7 +65,7 @@  		<li><a href="#traffictot" onclick="setUpdater(handler_traffic_tot)">Total switch traffic</a></li>  		<li><a href="#disco" onclick="setUpdater(handler_disco)">DISCO</a></li>  		<li class="divider"> </li> -		<li><a href="#" onclick="toggleLayer('nowPickerBox');document.getElementById('nowPicker').focus();">Travel in time</a></li> +		<li><a href="#" onclick="toggleLayer('nowPickerBox');startNowPicker();">Travel in time</a></li>  		<li><a href="#" onclick="startReplay();" title="Replay from opening 120 minutes per second">Replay TG</a></li>  		<li class="divider"> </li>  		<li class="dropdown-header">View</li> @@ -288,33 +291,39 @@  	      </table>  	    </div>  	  </div> -	  <div id="nowPickerBox" style="position: absolute; display: none; z-index: 130;" class="col-md-4"> +	  <div id="nowPickerBox" style="position: absolute; display: none; z-index: 130;" class="col-sm-6 col-md-5">  	    <div class="panel panel-default">   	      <div class="panel-heading"> -		<h3 class="panel-title">Time travel -		  <button type="button" class="close" aria-labe="Close" onclick="document.getElementById('nowPickerBox').style.display = 'none';" style="float: right;"> -		    <span aria-hidden="true">×</span> -		  </button> -		</h3> +          <h3 class="panel-title">Time travel +            <button type="button" class="close" aria-labe="Close" onclick="document.getElementById('nowPickerBox').style.display = 'none';" style="float: right;"> +              <span aria-hidden="true">×</span> +            </button> +          </h3>  	      </div> -	      <div class="panel-body"> -		<p>Some features do not have time travel support (comment -		spotting and DHCP map at the moment). We also lack -		compatible SNMP data for the first day or so, so you'll -		only have ping data for the first day of TG15.</p> -		<p>It could take some time to load a specific point in time -		for the first time. See "About performance" under the help -		menu for more information.</p> -		<p>You can also step backwards and forwards in time, stop -		and start replay and go back to real time using keyboard -		shortcuts. See the help menu for an overview of keyboard -		shortcuts.</p> -		<div class="input-group"> -		  <input type="text" class="form-control" placeholder="YYYY-MM-DDThh:mm:ss" id="nowPicker"> -		  <span class="input-group-btn"> -		    <button class="btn btn-default" onclick="changeNow();">Travel</button> -		  </span> -		</div> +	      <div class="panel-body row"> +          <div class="col-sm-12"> +            <div class="form-group"> +              <input type="text" class="form-control" placeholder="YYYY-MM-DDThh:mm:ss" id="nowPicker"> +              <div class="button-group"> +                <button class="btn btn-primary" onclick="changeNow(document.getElementById('nowPicker').dataset.iso,false);">Travel</button> +                <button class="btn btn-danger" onclick="startNowPicker(Date.now());changeNow(false,true);">Back to reality</button> +                <button class="btn btn-info" data-toggle="button" onclick="toggleLayer('nowPickerInfo');">Info</button> +              </div> +            </div> +          </div> +          <div id="nowPickerInfo" class="col-sm-12" style="display:none;"> +            <p>Some features do not have time travel support (comment +            spotting and DHCP map at the moment). We also lack +            compatible SNMP data for the first day or so, so you'll +            only have ping data for the first day of TG15.</p> +            <p>It could take some time to load a specific point in time +            for the first time. See "About performance" under the help +            menu for more information.</p> +            <p>You can also step backwards and forwards in time, stop +            and start replay and go back to real time using keyboard +            shortcuts. See the help menu for an overview of keyboard +            shortcuts.</p> +          </div>  	      </div>  	    </div>  	  </div> @@ -520,6 +529,7 @@      <script type="text/javascript" src="js/nms.js"></script>      <script type="text/javascript" src="js/nms-color-util.js"></script>      <script type="text/javascript" src="js/nms-map-handlers.js"></script> +    <script src="js/jquery.datetimepicker.full.js" type="text/javascript"></script>      <script type="text/javascript">  initNMS();      </script> diff --git a/web/nms.gathering.org/js/jquery.datetimepicker.full.js b/web/nms.gathering.org/js/jquery.datetimepicker.full.js new file mode 100644 index 0000000..417037c --- /dev/null +++ b/web/nms.gathering.org/js/jquery.datetimepicker.full.js @@ -0,0 +1,3073 @@ +/*! + * @copyright Copyright © Kartik Visweswaran, Krajee.com, 2014 - 2015 + * @version 1.3.3 + * + * Date formatter utility library that allows formatting date/time variables or Date objects using PHP DateTime format. + * @see http://php.net/manual/en/function.date.php + * + * For more JQuery plugins visit http://plugins.krajee.com + * For more Yii related demos visit http://demos.krajee.com + */ +var DateFormatter; +(function () { +    "use strict"; + +    var _compare, _lpad, _extend, defaultSettings, DAY, HOUR; +    DAY = 1000 * 60 * 60 * 24; +    HOUR = 3600; + +    _compare = function (str1, str2) { +        return typeof(str1) === 'string' && typeof(str2) === 'string' && str1.toLowerCase() === str2.toLowerCase(); +    }; +    _lpad = function (value, length, char) { +        var chr = char || '0', val = value.toString(); +        return val.length < length ? _lpad(chr + val, length) : val; +    }; +    _extend = function (out) { +        var i, obj; +        out = out || {}; +        for (i = 1; i < arguments.length; i++) { +            obj = arguments[i]; +            if (!obj) { +                continue; +            } +            for (var key in obj) { +                if (obj.hasOwnProperty(key)) { +                    if (typeof obj[key] === 'object') { +                        _extend(out[key], obj[key]); +                    } else { +                        out[key] = obj[key]; +                    } +                } +            } +        } +        return out; +    }; +    defaultSettings = { +        dateSettings: { +            days: ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'], +            daysShort: ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'], +            months: [ +                'January', 'February', 'March', 'April', 'May', 'June', 'July', +                'August', 'September', 'October', 'November', 'December' +            ], +            monthsShort: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'], +            meridiem: ['AM', 'PM'], +            ordinal: function (number) { +                var n = number % 10, suffixes = {1: 'st', 2: 'nd', 3: 'rd'}; +                return Math.floor(number % 100 / 10) === 1 || !suffixes[n] ? 'th' : suffixes[n]; +            } +        }, +        separators: /[ \-+\/\.T:@]/g, +        validParts: /[dDjlNSwzWFmMntLoYyaABgGhHisueTIOPZcrU]/g, +        intParts: /[djwNzmnyYhHgGis]/g, +        tzParts: /\b(?:[PMCEA][SDP]T|(?:Pacific|Mountain|Central|Eastern|Atlantic) (?:Standard|Daylight|Prevailing) Time|(?:GMT|UTC)(?:[-+]\d{4})?)\b/g, +        tzClip: /[^-+\dA-Z]/g +    }; + +    DateFormatter = function (options) { +        var self = this, config = _extend(defaultSettings, options); +        self.dateSettings = config.dateSettings; +        self.separators = config.separators; +        self.validParts = config.validParts; +        self.intParts = config.intParts; +        self.tzParts = config.tzParts; +        self.tzClip = config.tzClip; +    }; + +    DateFormatter.prototype = { +        constructor: DateFormatter, +        parseDate: function (vDate, vFormat) { +            var self = this, vFormatParts, vDateParts, i, vDateFlag = false, vTimeFlag = false, vDatePart, iDatePart, +                vSettings = self.dateSettings, vMonth, vMeriIndex, vMeriOffset, len, mer, +                out = {date: null, year: null, month: null, day: null, hour: 0, min: 0, sec: 0}; +            if (!vDate) { +                return undefined; +            } +            if (vDate instanceof Date) { +                return vDate; +            } +            if (typeof vDate === 'number') { +                return new Date(vDate); +            } +            if (vFormat === 'U') { +                i = parseInt(vDate); +                return i ? new Date(i * 1000) : vDate; +            } +            if (typeof vDate !== 'string') { +                return ''; +            } +            vFormatParts = vFormat.match(self.validParts); +            if (!vFormatParts || vFormatParts.length === 0) { +                throw new Error("Invalid date format definition."); +            } +            vDateParts = vDate.replace(self.separators, '\0').split('\0'); +            for (i = 0; i < vDateParts.length; i++) { +                vDatePart = vDateParts[i]; +                iDatePart = parseInt(vDatePart); +                switch (vFormatParts[i]) { +                    case 'y': +                    case 'Y': +                        len = vDatePart.length; +                        if (len === 2) { +                            out.year = parseInt((iDatePart < 70 ? '20' : '19') + vDatePart); +                        } else if (len === 4) { +                            out.year = iDatePart; +                        } +                        vDateFlag = true; +                        break; +                    case 'm': +                    case 'n': +                    case 'M': +                    case 'F': +                        if (isNaN(vDatePart)) { +                            vMonth = vSettings.monthsShort.indexOf(vDatePart); +                            if (vMonth > -1) { +                                out.month = vMonth + 1; +                            } +                            vMonth = vSettings.months.indexOf(vDatePart); +                            if (vMonth > -1) { +                                out.month = vMonth + 1; +                            } +                        } else { +                            if (iDatePart >= 1 && iDatePart <= 12) { +                                out.month = iDatePart; +                            } +                        } +                        vDateFlag = true; +                        break; +                    case 'd': +                    case 'j': +                        if (iDatePart >= 1 && iDatePart <= 31) { +                            out.day = iDatePart; +                        } +                        vDateFlag = true; +                        break; +                    case 'g': +                    case 'h': +                        vMeriIndex = (vFormatParts.indexOf('a') > -1) ? vFormatParts.indexOf('a') : +                            (vFormatParts.indexOf('A') > -1) ? vFormatParts.indexOf('A') : -1; +                        mer = vDateParts[vMeriIndex]; +                        if (vMeriIndex > -1) { +                            vMeriOffset = _compare(mer, vSettings.meridiem[0]) ? 0 : +                                (_compare(mer, vSettings.meridiem[1]) ? 12 : -1); +                            if (iDatePart >= 1 && iDatePart <= 12 && vMeriOffset > -1) { +                                out.hour = iDatePart + vMeriOffset - 1; +                            } else if (iDatePart >= 0 && iDatePart <= 23) { +                                out.hour = iDatePart; +                            } +                        } else if (iDatePart >= 0 && iDatePart <= 23) { +                            out.hour = iDatePart; +                        } +                        vTimeFlag = true; +                        break; +                    case 'G': +                    case 'H': +                        if (iDatePart >= 0 && iDatePart <= 23) { +                            out.hour = iDatePart; +                        } +                        vTimeFlag = true; +                        break; +                    case 'i': +                        if (iDatePart >= 0 && iDatePart <= 59) { +                            out.min = iDatePart; +                        } +                        vTimeFlag = true; +                        break; +                    case 's': +                        if (iDatePart >= 0 && iDatePart <= 59) { +                            out.sec = iDatePart; +                        } +                        vTimeFlag = true; +                        break; +                } +            } +            if (vDateFlag === true && out.year && out.month && out.day) { +                out.date = new Date(out.year, out.month - 1, out.day, out.hour, out.min, out.sec, 0); +            } else { +                if (vTimeFlag !== true) { +                    return false; +                } +                out.date = new Date(0, 0, 0, out.hour, out.min, out.sec, 0); +            } +            return out.date; +        }, +        guessDate: function (vDateStr, vFormat) { +            if (typeof vDateStr !== 'string') { +                return vDateStr; +            } +            var self = this, vParts = vDateStr.replace(self.separators, '\0').split('\0'), vPattern = /^[djmn]/g, +                vFormatParts = vFormat.match(self.validParts), vDate = new Date(), vDigit = 0, vYear, i, iPart, iSec; + +            if (!vPattern.test(vFormatParts[0])) { +                return vDateStr; +            } + +            for (i = 0; i < vParts.length; i++) { +                vDigit = 2; +                iPart = vParts[i]; +                iSec = parseInt(iPart.substr(0, 2)); +                switch (i) { +                    case 0: +                        if (vFormatParts[0] === 'm' || vFormatParts[0] === 'n') { +                            vDate.setMonth(iSec - 1); +                        } else { +                            vDate.setDate(iSec); +                        } +                        break; +                    case 1: +                        if (vFormatParts[0] === 'm' || vFormatParts[0] === 'n') { +                            vDate.setDate(iSec); +                        } else { +                            vDate.setMonth(iSec - 1); +                        } +                        break; +                    case 2: +                        vYear = vDate.getFullYear(); +                        if (iPart.length < 4) { +                            vDate.setFullYear(parseInt(vYear.toString().substr(0, 4 - iPart.length) + iPart)); +                            vDigit = iPart.length; +                        } else { +                            vDate.setFullYear = parseInt(iPart.substr(0, 4)); +                            vDigit = 4; +                        } +                        break; +                    case 3: +                        vDate.setHours(iSec); +                        break; +                    case 4: +                        vDate.setMinutes(iSec); +                        break; +                    case 5: +                        vDate.setSeconds(iSec); +                        break; +                } +                if (iPart.substr(vDigit).length > 0) { +                    vParts.splice(i + 1, 0, iPart.substr(vDigit)); +                } +            } +            return vDate; +        }, +        parseFormat: function (vChar, vDate) { +            var self = this, vSettings = self.dateSettings, fmt, backspace = /\\?(.?)/gi, doFormat = function (t, s) { +                return fmt[t] ? fmt[t]() : s; +            }; +            fmt = { +                ///////// +                // DAY // +                ///////// +                /** +                 * Day of month with leading 0: `01..31` +                 * @return {string} +                 */ +                d: function () { +                    return _lpad(fmt.j(), 2); +                }, +                /** +                 * Shorthand day name: `Mon...Sun` +                 * @return {string} +                 */ +                D: function () { +                    return vSettings.daysShort[fmt.w()]; +                }, +                /** +                 * Day of month: `1..31` +                 * @return {number} +                 */ +                j: function () { +                    return vDate.getDate(); +                }, +                /** +                 * Full day name: `Monday...Sunday` +                 * @return {number} +                 */ +                l: function () { +                    return vSettings.days[fmt.w()]; +                }, +                /** +                 * ISO-8601 day of week: `1[Mon]..7[Sun]` +                 * @return {number} +                 */ +                N: function () { +                    return fmt.w() || 7; +                }, +                /** +                 * Day of week: `0[Sun]..6[Sat]` +                 * @return {number} +                 */ +                w: function () { +                    return vDate.getDay(); +                }, +                /** +                 * Day of year: `0..365` +                 * @return {number} +                 */ +                z: function () { +                    var a = new Date(fmt.Y(), fmt.n() - 1, fmt.j()), b = new Date(fmt.Y(), 0, 1); +                    return Math.round((a - b) / DAY); +                }, + +                ////////// +                // WEEK // +                ////////// +                /** +                 * ISO-8601 week number +                 * @return {number} +                 */ +                W: function () { +                    var a = new Date(fmt.Y(), fmt.n() - 1, fmt.j() - fmt.N() + 3), b = new Date(a.getFullYear(), 0, 4); +                    return _lpad(1 + Math.round((a - b) / DAY / 7), 2); +                }, + +                /////////// +                // MONTH // +                /////////// +                /** +                 * Full month name: `January...December` +                 * @return {string} +                 */ +                F: function () { +                    return vSettings.months[vDate.getMonth()]; +                }, +                /** +                 * Month w/leading 0: `01..12` +                 * @return {string} +                 */ +                m: function () { +                    return _lpad(fmt.n(), 2); +                }, +                /** +                 * Shorthand month name; `Jan...Dec` +                 * @return {string} +                 */ +                M: function () { +                    return vSettings.monthsShort[vDate.getMonth()]; +                }, +                /** +                 * Month: `1...12` +                 * @return {number} +                 */ +                n: function () { +                    return vDate.getMonth() + 1; +                }, +                /** +                 * Days in month: `28...31` +                 * @return {number} +                 */ +                t: function () { +                    return (new Date(fmt.Y(), fmt.n(), 0)).getDate(); +                }, + +                ////////// +                // YEAR // +                ////////// +                /** +                 * Is leap year? `0 or 1` +                 * @return {number} +                 */ +                L: function () { +                    var Y = fmt.Y(); +                    return (Y % 4 === 0 && Y % 100 !== 0 || Y % 400 === 0) ? 1 : 0; +                }, +                /** +                 * ISO-8601 year +                 * @return {number} +                 */ +                o: function () { +                    var n = fmt.n(), W = fmt.W(), Y = fmt.Y(); +                    return Y + (n === 12 && W < 9 ? 1 : n === 1 && W > 9 ? -1 : 0); +                }, +                /** +                 * Full year: `e.g. 1980...2010` +                 * @return {number} +                 */ +                Y: function () { +                    return vDate.getFullYear(); +                }, +                /** +                 * Last two digits of year: `00...99` +                 * @return {string} +                 */ +                y: function () { +                    return fmt.Y().toString().slice(-2); +                }, + +                ////////// +                // TIME // +                ////////// +                /** +                 * Meridian lower: `am or pm` +                 * @return {string} +                 */ +                a: function () { +                    return fmt.A().toLowerCase(); +                }, +                /** +                 * Meridian upper: `AM or PM` +                 * @return {string} +                 */ +                A: function () { +                    var n = fmt.G() < 12 ? 0 : 1; +                    return vSettings.meridiem[n]; +                }, +                /** +                 * Swatch Internet time: `000..999` +                 * @return {string} +                 */ +                B: function () { +                    var H = vDate.getUTCHours() * HOUR, i = vDate.getUTCMinutes() * 60, s = vDate.getUTCSeconds(); +                    return _lpad(Math.floor((H + i + s + HOUR) / 86.4) % 1000, 3); +                }, +                /** +                 * 12-Hours: `1..12` +                 * @return {number} +                 */ +                g: function () { +                    return fmt.G() % 12 || 12; +                }, +                /** +                 * 24-Hours: `0..23` +                 * @return {number} +                 */ +                G: function () { +                    return vDate.getHours(); +                }, +                /** +                 * 12-Hours with leading 0: `01..12` +                 * @return {string} +                 */ +                h: function () { +                    return _lpad(fmt.g(), 2); +                }, +                /** +                 * 24-Hours w/leading 0: `00..23` +                 * @return {string} +                 */ +                H: function () { +                    return _lpad(fmt.G(), 2); +                }, +                /** +                 * Minutes w/leading 0: `00..59` +                 * @return {string} +                 */ +                i: function () { +                    return _lpad(vDate.getMinutes(), 2); +                }, +                /** +                 * Seconds w/leading 0: `00..59` +                 * @return {string} +                 */ +                s: function () { +                    return _lpad(vDate.getSeconds(), 2); +                }, +                /** +                 * Microseconds: `000000-999000` +                 * @return {string} +                 */ +                u: function () { +                    return _lpad(vDate.getMilliseconds() * 1000, 6); +                }, + +                ////////////// +                // TIMEZONE // +                ////////////// +                /** +                 * Timezone identifier: `e.g. Atlantic/Azores, ...` +                 * @return {string} +                 */ +                e: function () { +                    var str = /\((.*)\)/.exec(String(vDate))[1]; +                    return str || 'Coordinated Universal Time'; +                }, +                /** +                 * Timezone abbreviation: `e.g. EST, MDT, ...` +                 * @return {string} +                 */ +                T: function () { +                    var str = (String(vDate).match(self.tzParts) || [""]).pop().replace(self.tzClip, ""); +                    return str || 'UTC'; +                }, +                /** +                 * DST observed? `0 or 1` +                 * @return {number} +                 */ +                I: function () { +                    var a = new Date(fmt.Y(), 0), c = Date.UTC(fmt.Y(), 0), +                        b = new Date(fmt.Y(), 6), d = Date.UTC(fmt.Y(), 6); +                    return ((a - c) !== (b - d)) ? 1 : 0; +                }, +                /** +                 * Difference to GMT in hour format: `e.g. +0200` +                 * @return {string} +                 */ +                O: function () { +                    var tzo = vDate.getTimezoneOffset(), a = Math.abs(tzo); +                    return (tzo > 0 ? '-' : '+') + _lpad(Math.floor(a / 60) * 100 + a % 60, 4); +                }, +                /** +                 * Difference to GMT with colon: `e.g. +02:00` +                 * @return {string} +                 */ +                P: function () { +                    var O = fmt.O(); +                    return (O.substr(0, 3) + ':' + O.substr(3, 2)); +                }, +                /** +                 * Timezone offset in seconds: `-43200...50400` +                 * @return {number} +                 */ +                Z: function () { +                    return -vDate.getTimezoneOffset() * 60; +                }, + +                //////////////////// +                // FULL DATE TIME // +                //////////////////// +                /** +                 * ISO-8601 date +                 * @return {string} +                 */ +                c: function () { +                    return 'Y-m-d\\TH:i:s'.replace(backspace, doFormat); +                }, +                /** +                 * RFC 2822 date +                 * @return {string} +                 */ +                r: function () { +                    return 'D, d M Y H:i:s O'.replace(backspace, doFormat); +                }, +                /** +                 * Seconds since UNIX epoch +                 * @return {number} +                 */ +                U: function () { +                    return vDate.getTime() / 1000 || 0; +                } +            }; +            return doFormat(vChar, vChar); +        }, +        formatDate: function (vDate, vFormat) { +            var self = this, i, n, len, str, vChar, vDateStr = ''; +            if (typeof vDate === 'string') { +                vDate = self.parseDate(vDate, vFormat); +                if (vDate === false) { +                    return false; +                } +            } +            if (vDate instanceof Date) { +                len = vFormat.length; +                for (i = 0; i < len; i++) { +                    vChar = vFormat.charAt(i); +                    if (vChar === 'S') { +                        continue; +                    } +                    str = self.parseFormat(vChar, vDate); +                    if (i !== (len - 1) && self.intParts.test(vChar) && vFormat.charAt(i + 1) === 'S') { +                        n = parseInt(str); +                        str += self.dateSettings.ordinal(n); +                    } +                    vDateStr += str; +                } +                return vDateStr; +            } +            return ''; +        } +    }; +})();/** + * @preserve jQuery DateTimePicker plugin v2.5.1 + * @homepage http://xdsoft.net/jqplugins/datetimepicker/ + * @author Chupurnov Valeriy (<chupurnov@gmail.com>) + */ +/*global DateFormatter, document,window,jQuery,setTimeout,clearTimeout,HighlightedDate,getCurrentValue*/ +;(function (factory) { +	if ( typeof define === 'function' && define.amd ) { +		// AMD. Register as an anonymous module. +		define(['jquery', 'jquery-mousewheel'], factory); +	} else if (typeof exports === 'object') { +		// Node/CommonJS style for Browserify +		module.exports = factory; +	} else { +		// Browser globals +		factory(jQuery); +	} +}(function ($) { +	'use strict'; +	var default_options  = { +		i18n: { +			ar: { // Arabic +				months: [ +					"كانون الثاني", "شباط", "آذار", "نيسان", "مايو", "حزيران", "تموز", "آب", "أيلول", "تشرين الأول", "تشرين الثاني", "كانون الأول" +				], +				dayOfWeekShort: [ +					"ن", "ث", "ع", "خ", "ج", "س", "ح" +				], +				dayOfWeek: ["الأحد", "الاثنين", "الثلاثاء", "الأربعاء", "الخميس", "الجمعة", "السبت", "الأحد"] +			}, +			ro: { // Romanian +				months: [ +					"Ianuarie", "Februarie", "Martie", "Aprilie", "Mai", "Iunie", "Iulie", "August", "Septembrie", "Octombrie", "Noiembrie", "Decembrie" +				], +				dayOfWeekShort: [ +					"Du", "Lu", "Ma", "Mi", "Jo", "Vi", "Sâ" +				], +				dayOfWeek: ["Duminică", "Luni", "Marţi", "Miercuri", "Joi", "Vineri", "Sâmbătă"] +			}, +			id: { // Indonesian +				months: [ +					"Januari", "Februari", "Maret", "April", "Mei", "Juni", "Juli", "Agustus", "September", "Oktober", "November", "Desember" +				], +				dayOfWeekShort: [ +					"Min", "Sen", "Sel", "Rab", "Kam", "Jum", "Sab" +				], +				dayOfWeek: ["Minggu", "Senin", "Selasa", "Rabu", "Kamis", "Jumat", "Sabtu"] +			}, +			is: { // Icelandic +				months: [ +					"Janúar", "Febrúar", "Mars", "Apríl", "Maí", "Júní", "Júlí", "Ágúst", "September", "Október", "Nóvember", "Desember" +				], +				dayOfWeekShort: [ +					"Sun", "Mán", "Þrið", "Mið", "Fim", "Fös", "Lau" +				], +				dayOfWeek: ["Sunnudagur", "Mánudagur", "Þriðjudagur", "Miðvikudagur", "Fimmtudagur", "Föstudagur", "Laugardagur"] +			}, +			bg: { // Bulgarian +				months: [ +					"Януари", "Февруари", "Март", "Април", "Май", "Юни", "Юли", "Август", "Септември", "Октомври", "Ноември", "Декември" +				], +				dayOfWeekShort: [ +					"Нд", "Пн", "Вт", "Ср", "Чт", "Пт", "Сб" +				], +				dayOfWeek: ["Неделя", "Понеделник", "Вторник", "Сряда", "Четвъртък", "Петък", "Събота"] +			}, +			fa: { // Persian/Farsi +				months: [ +					'فروردین', 'اردیبهشت', 'خرداد', 'تیر', 'مرداد', 'شهریور', 'مهر', 'آبان', 'آذر', 'دی', 'بهمن', 'اسفند' +				], +				dayOfWeekShort: [ +					'یکشنبه', 'دوشنبه', 'سه شنبه', 'چهارشنبه', 'پنجشنبه', 'جمعه', 'شنبه' +				], +				dayOfWeek: ["یکشنبه", "دوشنبه", "سهشنبه", "چهارشنبه", "پنجشنبه", "جمعه", "شنبه", "یکشنبه"] +			}, +			ru: { // Russian +				months: [ +					'Январь', 'Февраль', 'Март', 'Апрель', 'Май', 'Июнь', 'Июль', 'Август', 'Сентябрь', 'Октябрь', 'Ноябрь', 'Декабрь' +				], +				dayOfWeekShort: [ +					"Вс", "Пн", "Вт", "Ср", "Чт", "Пт", "Сб" +				], +				dayOfWeek: ["Воскресенье", "Понедельник", "Вторник", "Среда", "Четверг", "Пятница", "Суббота"] +			}, +			uk: { // Ukrainian +				months: [ +					'Січень', 'Лютий', 'Березень', 'Квітень', 'Травень', 'Червень', 'Липень', 'Серпень', 'Вересень', 'Жовтень', 'Листопад', 'Грудень' +				], +				dayOfWeekShort: [ +					"Ндл", "Пнд", "Втр", "Срд", "Чтв", "Птн", "Сбт" +				], +				dayOfWeek: ["Неділя", "Понеділок", "Вівторок", "Середа", "Четвер", "П'ятниця", "Субота"] +			}, +			en: { // English +				months: [ +					"January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December" +				], +				dayOfWeekShort: [ +					"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" +				], +				dayOfWeek: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"] +			}, +			el: { // Ελληνικά +				months: [ +					"Ιανουάριος", "Φεβρουάριος", "Μάρτιος", "Απρίλιος", "Μάιος", "Ιούνιος", "Ιούλιος", "Αύγουστος", "Σεπτέμβριος", "Οκτώβριος", "Νοέμβριος", "Δεκέμβριος" +				], +				dayOfWeekShort: [ +					"Κυρ", "Δευ", "Τρι", "Τετ", "Πεμ", "Παρ", "Σαβ" +				], +				dayOfWeek: ["Κυριακή", "Δευτέρα", "Τρίτη", "Τετάρτη", "Πέμπτη", "Παρασκευή", "Σάββατο"] +			}, +			de: { // German +				months: [ +					'Januar', 'Februar', 'März', 'April', 'Mai', 'Juni', 'Juli', 'August', 'September', 'Oktober', 'November', 'Dezember' +				], +				dayOfWeekShort: [ +					"So", "Mo", "Di", "Mi", "Do", "Fr", "Sa" +				], +				dayOfWeek: ["Sonntag", "Montag", "Dienstag", "Mittwoch", "Donnerstag", "Freitag", "Samstag"] +			}, +			nl: { // Dutch +				months: [ +					"januari", "februari", "maart", "april", "mei", "juni", "juli", "augustus", "september", "oktober", "november", "december" +				], +				dayOfWeekShort: [ +					"zo", "ma", "di", "wo", "do", "vr", "za" +				], +				dayOfWeek: ["zondag", "maandag", "dinsdag", "woensdag", "donderdag", "vrijdag", "zaterdag"] +			}, +			tr: { // Turkish +				months: [ +					"Ocak", "Şubat", "Mart", "Nisan", "Mayıs", "Haziran", "Temmuz", "Ağustos", "Eylül", "Ekim", "Kasım", "Aralık" +				], +				dayOfWeekShort: [ +					"Paz", "Pts", "Sal", "Çar", "Per", "Cum", "Cts" +				], +				dayOfWeek: ["Pazar", "Pazartesi", "Salı", "Çarşamba", "Perşembe", "Cuma", "Cumartesi"] +			}, +			fr: { //French +				months: [ +					"Janvier", "Février", "Mars", "Avril", "Mai", "Juin", "Juillet", "Août", "Septembre", "Octobre", "Novembre", "Décembre" +				], +				dayOfWeekShort: [ +					"Dim", "Lun", "Mar", "Mer", "Jeu", "Ven", "Sam" +				], +				dayOfWeek: ["dimanche", "lundi", "mardi", "mercredi", "jeudi", "vendredi", "samedi"] +			}, +			es: { // Spanish +				months: [ +					"Enero", "Febrero", "Marzo", "Abril", "Mayo", "Junio", "Julio", "Agosto", "Septiembre", "Octubre", "Noviembre", "Diciembre" +				], +				dayOfWeekShort: [ +					"Dom", "Lun", "Mar", "Mié", "Jue", "Vie", "Sáb" +				], +				dayOfWeek: ["Domingo", "Lunes", "Martes", "Miércoles", "Jueves", "Viernes", "Sábado"] +			}, +			th: { // Thai +				months: [ +					'มกราคม', 'กุมภาพันธ์', 'มีนาคม', 'เมษายน', 'พฤษภาคม', 'มิถุนายน', 'กรกฎาคม', 'สิงหาคม', 'กันยายน', 'ตุลาคม', 'พฤศจิกายน', 'ธันวาคม' +				], +				dayOfWeekShort: [ +					'อา.', 'จ.', 'อ.', 'พ.', 'พฤ.', 'ศ.', 'ส.' +				], +				dayOfWeek: ["อาทิตย์", "จันทร์", "อังคาร", "พุธ", "พฤหัส", "ศุกร์", "เสาร์", "อาทิตย์"] +			}, +			pl: { // Polish +				months: [ +					"styczeń", "luty", "marzec", "kwiecień", "maj", "czerwiec", "lipiec", "sierpień", "wrzesień", "październik", "listopad", "grudzień" +				], +				dayOfWeekShort: [ +					"nd", "pn", "wt", "śr", "cz", "pt", "sb" +				], +				dayOfWeek: ["niedziela", "poniedziałek", "wtorek", "środa", "czwartek", "piątek", "sobota"] +			}, +			pt: { // Portuguese +				months: [ +					"Janeiro", "Fevereiro", "Março", "Abril", "Maio", "Junho", "Julho", "Agosto", "Setembro", "Outubro", "Novembro", "Dezembro" +				], +				dayOfWeekShort: [ +					"Dom", "Seg", "Ter", "Qua", "Qui", "Sex", "Sab" +				], +				dayOfWeek: ["Domingo", "Segunda", "Terça", "Quarta", "Quinta", "Sexta", "Sábado"] +			}, +			ch: { // Simplified Chinese +				months: [ +					"一月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "十二月" +				], +				dayOfWeekShort: [ +					"日", "一", "二", "三", "四", "五", "六" +				] +			}, +			se: { // Swedish +				months: [ +					"Januari", "Februari", "Mars", "April", "Maj", "Juni", "Juli", "Augusti", "September",  "Oktober", "November", "December" +				], +				dayOfWeekShort: [ +					"Sön", "Mån", "Tis", "Ons", "Tor", "Fre", "Lör" +				] +			}, +			kr: { // Korean +				months: [ +					"1월", "2월", "3월", "4월", "5월", "6월", "7월", "8월", "9월", "10월", "11월", "12월" +				], +				dayOfWeekShort: [ +					"일", "월", "화", "수", "목", "금", "토" +				], +				dayOfWeek: ["일요일", "월요일", "화요일", "수요일", "목요일", "금요일", "토요일"] +			}, +			it: { // Italian +				months: [ +					"Gennaio", "Febbraio", "Marzo", "Aprile", "Maggio", "Giugno", "Luglio", "Agosto", "Settembre", "Ottobre", "Novembre", "Dicembre" +				], +				dayOfWeekShort: [ +					"Dom", "Lun", "Mar", "Mer", "Gio", "Ven", "Sab" +				], +				dayOfWeek: ["Domenica", "Lunedì", "Martedì", "Mercoledì", "Giovedì", "Venerdì", "Sabato"] +			}, +			da: { // Dansk +				months: [ +					"January", "Februar", "Marts", "April", "Maj", "Juni", "July", "August", "September", "Oktober", "November", "December" +				], +				dayOfWeekShort: [ +					"Søn", "Man", "Tir", "Ons", "Tor", "Fre", "Lør" +				], +				dayOfWeek: ["søndag", "mandag", "tirsdag", "onsdag", "torsdag", "fredag", "lørdag"] +			}, +			no: { // Norwegian +				months: [ +					"Januar", "Februar", "Mars", "April", "Mai", "Juni", "Juli", "August", "September", "Oktober", "November", "Desember" +				], +				dayOfWeekShort: [ +					"Søn", "Man", "Tir", "Ons", "Tor", "Fre", "Lør" +				], +				dayOfWeek: ['Søndag', 'Mandag', 'Tirsdag', 'Onsdag', 'Torsdag', 'Fredag', 'Lørdag'] +			}, +			ja: { // Japanese +				months: [ +					"1月", "2月", "3月", "4月", "5月", "6月", "7月", "8月", "9月", "10月", "11月", "12月" +				], +				dayOfWeekShort: [ +					"日", "月", "火", "水", "木", "金", "土" +				], +				dayOfWeek: ["日曜", "月曜", "火曜", "水曜", "木曜", "金曜", "土曜"] +			}, +			vi: { // Vietnamese +				months: [ +					"Tháng 1", "Tháng 2", "Tháng 3", "Tháng 4", "Tháng 5", "Tháng 6", "Tháng 7", "Tháng 8", "Tháng 9", "Tháng 10", "Tháng 11", "Tháng 12" +				], +				dayOfWeekShort: [ +					"CN", "T2", "T3", "T4", "T5", "T6", "T7" +				], +				dayOfWeek: ["Chủ nhật", "Thứ hai", "Thứ ba", "Thứ tư", "Thứ năm", "Thứ sáu", "Thứ bảy"] +			}, +			sl: { // Slovenščina +				months: [ +					"Januar", "Februar", "Marec", "April", "Maj", "Junij", "Julij", "Avgust", "September", "Oktober", "November", "December" +				], +				dayOfWeekShort: [ +					"Ned", "Pon", "Tor", "Sre", "Čet", "Pet", "Sob" +				], +				dayOfWeek: ["Nedelja", "Ponedeljek", "Torek", "Sreda", "Četrtek", "Petek", "Sobota"] +			}, +			cs: { // Čeština +				months: [ +					"Leden", "Únor", "Březen", "Duben", "Květen", "Červen", "Červenec", "Srpen", "Září", "Říjen", "Listopad", "Prosinec" +				], +				dayOfWeekShort: [ +					"Ne", "Po", "Út", "St", "Čt", "Pá", "So" +				] +			}, +			hu: { // Hungarian +				months: [ +					"Január", "Február", "Március", "Április", "Május", "Június", "Július", "Augusztus", "Szeptember", "Október", "November", "December" +				], +				dayOfWeekShort: [ +					"Va", "Hé", "Ke", "Sze", "Cs", "Pé", "Szo" +				], +				dayOfWeek: ["vasárnap", "hétfő", "kedd", "szerda", "csütörtök", "péntek", "szombat"] +			}, +			az: { //Azerbaijanian (Azeri) +				months: [ +					"Yanvar", "Fevral", "Mart", "Aprel", "May", "Iyun", "Iyul", "Avqust", "Sentyabr", "Oktyabr", "Noyabr", "Dekabr" +				], +				dayOfWeekShort: [ +					"B", "Be", "Ça", "Ç", "Ca", "C", "Ş" +				], +				dayOfWeek: ["Bazar", "Bazar ertəsi", "Çərşənbə axşamı", "Çərşənbə", "Cümə axşamı", "Cümə", "Şənbə"] +			}, +			bs: { //Bosanski +				months: [ +					"Januar", "Februar", "Mart", "April", "Maj", "Jun", "Jul", "Avgust", "Septembar", "Oktobar", "Novembar", "Decembar" +				], +				dayOfWeekShort: [ +					"Ned", "Pon", "Uto", "Sri", "Čet", "Pet", "Sub" +				], +				dayOfWeek: ["Nedjelja","Ponedjeljak", "Utorak", "Srijeda", "Četvrtak", "Petak", "Subota"] +			}, +			ca: { //Català +				months: [ +					"Gener", "Febrer", "Març", "Abril", "Maig", "Juny", "Juliol", "Agost", "Setembre", "Octubre", "Novembre", "Desembre" +				], +				dayOfWeekShort: [ +					"Dg", "Dl", "Dt", "Dc", "Dj", "Dv", "Ds" +				], +				dayOfWeek: ["Diumenge", "Dilluns", "Dimarts", "Dimecres", "Dijous", "Divendres", "Dissabte"] +			}, +			'en-GB': { //English (British) +				months: [ +					"January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December" +				], +				dayOfWeekShort: [ +					"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" +				], +				dayOfWeek: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"] +			}, +			et: { //"Eesti" +				months: [ +					"Jaanuar", "Veebruar", "Märts", "Aprill", "Mai", "Juuni", "Juuli", "August", "September", "Oktoober", "November", "Detsember" +				], +				dayOfWeekShort: [ +					"P", "E", "T", "K", "N", "R", "L" +				], +				dayOfWeek: ["Pühapäev", "Esmaspäev", "Teisipäev", "Kolmapäev", "Neljapäev", "Reede", "Laupäev"] +			}, +			eu: { //Euskara +				months: [ +					"Urtarrila", "Otsaila", "Martxoa", "Apirila", "Maiatza", "Ekaina", "Uztaila", "Abuztua", "Iraila", "Urria", "Azaroa", "Abendua" +				], +				dayOfWeekShort: [ +					"Ig.", "Al.", "Ar.", "Az.", "Og.", "Or.", "La." +				], +				dayOfWeek: ['Igandea', 'Astelehena', 'Asteartea', 'Asteazkena', 'Osteguna', 'Ostirala', 'Larunbata'] +			}, +			fi: { //Finnish (Suomi) +				months: [ +					"Tammikuu", "Helmikuu", "Maaliskuu", "Huhtikuu", "Toukokuu", "Kesäkuu", "Heinäkuu", "Elokuu", "Syyskuu", "Lokakuu", "Marraskuu", "Joulukuu" +				], +				dayOfWeekShort: [ +					"Su", "Ma", "Ti", "Ke", "To", "Pe", "La" +				], +				dayOfWeek: ["sunnuntai", "maanantai", "tiistai", "keskiviikko", "torstai", "perjantai", "lauantai"] +			}, +			gl: { //Galego +				months: [ +					"Xan", "Feb", "Maz", "Abr", "Mai", "Xun", "Xul", "Ago", "Set", "Out", "Nov", "Dec" +				], +				dayOfWeekShort: [ +					"Dom", "Lun", "Mar", "Mer", "Xov", "Ven", "Sab" +				], +				dayOfWeek: ["Domingo", "Luns", "Martes", "Mércores", "Xoves", "Venres", "Sábado"] +			}, +			hr: { //Hrvatski +				months: [ +					"Siječanj", "Veljača", "Ožujak", "Travanj", "Svibanj", "Lipanj", "Srpanj", "Kolovoz", "Rujan", "Listopad", "Studeni", "Prosinac" +				], +				dayOfWeekShort: [ +					"Ned", "Pon", "Uto", "Sri", "Čet", "Pet", "Sub" +				], +				dayOfWeek: ["Nedjelja", "Ponedjeljak", "Utorak", "Srijeda", "Četvrtak", "Petak", "Subota"] +			}, +			ko: { //Korean (한국어) +				months: [ +					"1월", "2월", "3월", "4월", "5월", "6월", "7월", "8월", "9월", "10월", "11월", "12월" +				], +				dayOfWeekShort: [ +					"일", "월", "화", "수", "목", "금", "토" +				], +				dayOfWeek: ["일요일", "월요일", "화요일", "수요일", "목요일", "금요일", "토요일"] +			}, +			lt: { //Lithuanian (lietuvių) +				months: [ +					"Sausio", "Vasario", "Kovo", "Balandžio", "Gegužės", "Birželio", "Liepos", "Rugpjūčio", "Rugsėjo", "Spalio", "Lapkričio", "Gruodžio" +				], +				dayOfWeekShort: [ +					"Sek", "Pir", "Ant", "Tre", "Ket", "Pen", "Šeš" +				], +				dayOfWeek: ["Sekmadienis", "Pirmadienis", "Antradienis", "Trečiadienis", "Ketvirtadienis", "Penktadienis", "Šeštadienis"] +			}, +			lv: { //Latvian (Latviešu) +				months: [ +					"Janvāris", "Februāris", "Marts", "Aprīlis ", "Maijs", "Jūnijs", "Jūlijs", "Augusts", "Septembris", "Oktobris", "Novembris", "Decembris" +				], +				dayOfWeekShort: [ +					"Sv", "Pr", "Ot", "Tr", "Ct", "Pk", "St" +				], +				dayOfWeek: ["Svētdiena", "Pirmdiena", "Otrdiena", "Trešdiena", "Ceturtdiena", "Piektdiena", "Sestdiena"] +			}, +			mk: { //Macedonian (Македонски) +				months: [ +					"јануари", "февруари", "март", "април", "мај", "јуни", "јули", "август", "септември", "октомври", "ноември", "декември" +				], +				dayOfWeekShort: [ +					"нед", "пон", "вто", "сре", "чет", "пет", "саб" +				], +				dayOfWeek: ["Недела", "Понеделник", "Вторник", "Среда", "Четврток", "Петок", "Сабота"] +			}, +			mn: { //Mongolian (Монгол) +				months: [ +					"1-р сар", "2-р сар", "3-р сар", "4-р сар", "5-р сар", "6-р сар", "7-р сар", "8-р сар", "9-р сар", "10-р сар", "11-р сар", "12-р сар" +				], +				dayOfWeekShort: [ +					"Дав", "Мяг", "Лха", "Пүр", "Бсн", "Бям", "Ням" +				], +				dayOfWeek: ["Даваа", "Мягмар", "Лхагва", "Пүрэв", "Баасан", "Бямба", "Ням"] +			}, +			'pt-BR': { //Português(Brasil) +				months: [ +					"Janeiro", "Fevereiro", "Março", "Abril", "Maio", "Junho", "Julho", "Agosto", "Setembro", "Outubro", "Novembro", "Dezembro" +				], +				dayOfWeekShort: [ +					"Dom", "Seg", "Ter", "Qua", "Qui", "Sex", "Sáb" +				], +				dayOfWeek: ["Domingo", "Segunda", "Terça", "Quarta", "Quinta", "Sexta", "Sábado"] +			}, +			sk: { //Slovenčina +				months: [ +					"Január", "Február", "Marec", "Apríl", "Máj", "Jún", "Júl", "August", "September", "Október", "November", "December" +				], +				dayOfWeekShort: [ +					"Ne", "Po", "Ut", "St", "Št", "Pi", "So" +				], +				dayOfWeek: ["Nedeľa", "Pondelok", "Utorok", "Streda", "Štvrtok", "Piatok", "Sobota"] +			}, +			sq: { //Albanian (Shqip) +				months: [ +					"Janar", "Shkurt", "Mars", "Prill", "Maj", "Qershor", "Korrik", "Gusht", "Shtator", "Tetor", "Nëntor", "Dhjetor" +				], +				dayOfWeekShort: [ +					"Die", "Hën", "Mar", "Mër", "Enj", "Pre", "Shtu" +				], +				dayOfWeek: ["E Diel", "E Hënë", "E Martē", "E Mërkurë", "E Enjte", "E Premte", "E Shtunë"] +			}, +			'sr-YU': { //Serbian (Srpski) +				months: [ +					"Januar", "Februar", "Mart", "April", "Maj", "Jun", "Jul", "Avgust", "Septembar", "Oktobar", "Novembar", "Decembar" +				], +				dayOfWeekShort: [ +					"Ned", "Pon", "Uto", "Sre", "čet", "Pet", "Sub" +				], +				dayOfWeek: ["Nedelja","Ponedeljak", "Utorak", "Sreda", "Četvrtak", "Petak", "Subota"] +			}, +			sr: { //Serbian Cyrillic (Српски) +				months: [ +					"јануар", "фебруар", "март", "април", "мај", "јун", "јул", "август", "септембар", "октобар", "новембар", "децембар" +				], +				dayOfWeekShort: [ +					"нед", "пон", "уто", "сре", "чет", "пет", "суб" +				], +				dayOfWeek: ["Недеља","Понедељак", "Уторак", "Среда", "Четвртак", "Петак", "Субота"] +			}, +			sv: { //Svenska +				months: [ +					"Januari", "Februari", "Mars", "April", "Maj", "Juni", "Juli", "Augusti", "September", "Oktober", "November", "December" +				], +				dayOfWeekShort: [ +					"Sön", "Mån", "Tis", "Ons", "Tor", "Fre", "Lör" +				], +				dayOfWeek: ["Söndag", "Måndag", "Tisdag", "Onsdag", "Torsdag", "Fredag", "Lördag"] +			}, +			'zh-TW': { //Traditional Chinese (繁體中文) +				months: [ +					"一月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "十二月" +				], +				dayOfWeekShort: [ +					"日", "一", "二", "三", "四", "五", "六" +				], +				dayOfWeek: ["星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六"] +			}, +			zh: { //Simplified Chinese (简体中文) +				months: [ +					"一月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "十二月" +				], +				dayOfWeekShort: [ +					"日", "一", "二", "三", "四", "五", "六" +				], +				dayOfWeek: ["星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六"] +			}, +			he: { //Hebrew (עברית) +				months: [ +					'ינואר', 'פברואר', 'מרץ', 'אפריל', 'מאי', 'יוני', 'יולי', 'אוגוסט', 'ספטמבר', 'אוקטובר', 'נובמבר', 'דצמבר' +				], +				dayOfWeekShort: [ +					'א\'', 'ב\'', 'ג\'', 'ד\'', 'ה\'', 'ו\'', 'שבת' +				], +				dayOfWeek: ["ראשון", "שני", "שלישי", "רביעי", "חמישי", "שישי", "שבת", "ראשון"] +			}, +			hy: { // Armenian +				months: [ +					"Հունվար", "Փետրվար", "Մարտ", "Ապրիլ", "Մայիս", "Հունիս", "Հուլիս", "Օգոստոս", "Սեպտեմբեր", "Հոկտեմբեր", "Նոյեմբեր", "Դեկտեմբեր" +				], +				dayOfWeekShort: [ +					"Կի", "Երկ", "Երք", "Չոր", "Հնգ", "Ուրբ", "Շբթ" +				], +				dayOfWeek: ["Կիրակի", "Երկուշաբթի", "Երեքշաբթի", "Չորեքշաբթի", "Հինգշաբթի", "Ուրբաթ", "Շաբաթ"] +			}, +			kg: { // Kyrgyz +				months: [ +					'Үчтүн айы', 'Бирдин айы', 'Жалган Куран', 'Чын Куран', 'Бугу', 'Кулжа', 'Теке', 'Баш Оона', 'Аяк Оона', 'Тогуздун айы', 'Жетинин айы', 'Бештин айы' +				], +				dayOfWeekShort: [ +					"Жек", "Дүй", "Шей", "Шар", "Бей", "Жум", "Ише" +				], +				dayOfWeek: [ +					"Жекшемб", "Дүйшөмб", "Шейшемб", "Шаршемб", "Бейшемби", "Жума", "Ишенб" +				] +			}, +			rm: { // Romansh +				months: [ +					"Schaner", "Favrer", "Mars", "Avrigl", "Matg", "Zercladur", "Fanadur", "Avust", "Settember", "October", "November", "December" +				], +				dayOfWeekShort: [ +					"Du", "Gli", "Ma", "Me", "Gie", "Ve", "So" +				], +				dayOfWeek: [ +					"Dumengia", "Glindesdi", "Mardi", "Mesemna", "Gievgia", "Venderdi", "Sonda" +				] +			}, +		}, +		value: '', +		rtl: false, + +		format:	'Y/m/d H:i', +		formatTime:	'H:i', +		formatDate:	'Y/m/d', + +		startDate:	false, // new Date(), '1986/12/08', '-1970/01/05','-1970/01/05', +		step: 60, +		monthChangeSpinner: true, + +		closeOnDateSelect: false, +		closeOnTimeSelect: true, +		closeOnWithoutClick: true, +		closeOnInputClick: true, + +		timepicker: true, +		datepicker: true, +		weeks: false, + +		defaultTime: false,	// use formatTime format (ex. '10:00' for formatTime:	'H:i') +		defaultDate: false,	// use formatDate format (ex new Date() or '1986/12/08' or '-1970/01/05' or '-1970/01/05') + +		minDate: false, +		maxDate: false, +		minTime: false, +		maxTime: false, +		disabledMinTime: false, +		disabledMaxTime: false, + +		allowTimes: [], +		opened: false, +		initTime: true, +		inline: false, +		theme: '', + +		onSelectDate: function () {}, +		onSelectTime: function () {}, +		onChangeMonth: function () {}, +		onGetWeekOfYear: function () {}, +		onChangeYear: function () {}, +		onChangeDateTime: function () {}, +		onShow: function () {}, +		onClose: function () {}, +		onGenerate: function () {}, + +		withoutCopyright: true, +		inverseButton: false, +		hours12: false, +		next: 'xdsoft_next', +		prev : 'xdsoft_prev', +		dayOfWeekStart: 0, +		parentID: 'body', +		timeHeightInTimePicker: 25, +		timepickerScrollbar: true, +		todayButton: true, +		prevButton: true, +		nextButton: true, +		defaultSelect: true, + +		scrollMonth: true, +		scrollTime: true, +		scrollInput: true, + +		lazyInit: false, +		mask: false, +		validateOnBlur: true, +		allowBlank: true, +		yearStart: 1950, +		yearEnd: 2050, +		monthStart: 0, +		monthEnd: 11, +		style: '', +		id: '', +		fixed: false, +		roundTime: 'round', // ceil, floor +		className: '', +		weekends: [], +		highlightedDates: [], +		highlightedPeriods: [], +		allowDates : [], +		allowDateRe : null, +		disabledDates : [], +		disabledWeekDays: [], +		yearOffset: 0, +		beforeShowDay: null, + +		enterLikeTab: true, +        showApplyButton: false +	}; + +	var dateHelper = null, +		globalLocaleDefault = 'en', +		globalLocale = 'en'; + +	var dateFormatterOptionsDefault = { +		meridiem: ['AM', 'PM'] +	}; + +	var initDateFormatter = function(){ +		var locale = default_options.i18n[globalLocale], +			opts = { +				days: locale.dayOfWeek, +				daysShort: locale.dayOfWeekShort, +				months: locale.months, +				monthsShort: $.map(locale.months, function(n){ return n.substring(0, 3) }), +			}; + +	 	dateHelper = new DateFormatter({ +			dateSettings: $.extend({}, dateFormatterOptionsDefault, opts) +		}); +	}; + +	// for locale settings +	$.datetimepicker = { +		setLocale: function(locale){ +			var newLocale = default_options.i18n[locale]?locale:globalLocaleDefault; +			if(globalLocale != newLocale){ +				globalLocale = newLocale; +				// reinit date formatter +				initDateFormatter(); +			} +		}, +		RFC_2822: 'D, d M Y H:i:s O', +		ATOM: 'Y-m-d\TH:i:sP', +		ISO_8601: 'Y-m-d\TH:i:sO', +		RFC_822: 'D, d M y H:i:s O', +		RFC_850: 'l, d-M-y H:i:s T', +		RFC_1036: 'D, d M y H:i:s O', +		RFC_1123: 'D, d M Y H:i:s O', +		RSS: 'D, d M Y H:i:s O', +		W3C: 'Y-m-d\TH:i:sP' +	}; + +	// first init date formatter +	initDateFormatter(); + +	// fix for ie8 +	if (!window.getComputedStyle) { +		window.getComputedStyle = function (el, pseudo) { +			this.el = el; +			this.getPropertyValue = function (prop) { +				var re = /(\-([a-z]){1})/g; +				if (prop === 'float') { +					prop = 'styleFloat'; +				} +				if (re.test(prop)) { +					prop = prop.replace(re, function (a, b, c) { +						return c.toUpperCase(); +					}); +				} +				return el.currentStyle[prop] || null; +			}; +			return this; +		}; +	} +	if (!Array.prototype.indexOf) { +		Array.prototype.indexOf = function (obj, start) { +			var i, j; +			for (i = (start || 0), j = this.length; i < j; i += 1) { +				if (this[i] === obj) { return i; } +			} +			return -1; +		}; +	} +	Date.prototype.countDaysInMonth = function () { +		return new Date(this.getFullYear(), this.getMonth() + 1, 0).getDate(); +	}; +	$.fn.xdsoftScroller = function (percent) { +		return this.each(function () { +			var timeboxparent = $(this), +				pointerEventToXY = function (e) { +					var out = {x: 0, y: 0}, +						touch; +					if (e.type === 'touchstart' || e.type === 'touchmove' || e.type === 'touchend' || e.type === 'touchcancel') { +						touch  = e.originalEvent.touches[0] || e.originalEvent.changedTouches[0]; +						out.x = touch.clientX; +						out.y = touch.clientY; +					} else if (e.type === 'mousedown' || e.type === 'mouseup' || e.type === 'mousemove' || e.type === 'mouseover' || e.type === 'mouseout' || e.type === 'mouseenter' || e.type === 'mouseleave') { +						out.x = e.clientX; +						out.y = e.clientY; +					} +					return out; +				}, +				move = 0, +				timebox, +				parentHeight, +				height, +				scrollbar, +				scroller, +				maximumOffset = 100, +				start = false, +				startY = 0, +				startTop = 0, +				h1 = 0, +				touchStart = false, +				startTopScroll = 0, +				calcOffset = function () {}; +			if (percent === 'hide') { +				timeboxparent.find('.xdsoft_scrollbar').hide(); +				return; +			} +			if (!$(this).hasClass('xdsoft_scroller_box')) { +				timebox = timeboxparent.children().eq(0); +				parentHeight = timeboxparent[0].clientHeight; +				height = timebox[0].offsetHeight; +				scrollbar = $('<div class="xdsoft_scrollbar"></div>'); +				scroller = $('<div class="xdsoft_scroller"></div>'); +				scrollbar.append(scroller); + +				timeboxparent.addClass('xdsoft_scroller_box').append(scrollbar); +				calcOffset = function calcOffset(event) { +					var offset = pointerEventToXY(event).y - startY + startTopScroll; +					if (offset < 0) { +						offset = 0; +					} +					if (offset + scroller[0].offsetHeight > h1) { +						offset = h1 - scroller[0].offsetHeight; +					} +					timeboxparent.trigger('scroll_element.xdsoft_scroller', [maximumOffset ? offset / maximumOffset : 0]); +				}; + +				scroller +					.on('touchstart.xdsoft_scroller mousedown.xdsoft_scroller', function (event) { +						if (!parentHeight) { +							timeboxparent.trigger('resize_scroll.xdsoft_scroller', [percent]); +						} + +						startY = pointerEventToXY(event).y; +						startTopScroll = parseInt(scroller.css('margin-top'), 10); +						h1 = scrollbar[0].offsetHeight; + +						if (event.type === 'mousedown' || event.type === 'touchstart') { +							if (document) { +								$(document.body).addClass('xdsoft_noselect'); +							} +							$([document.body, window]).on('touchend mouseup.xdsoft_scroller', function arguments_callee() { +								$([document.body, window]).off('touchend mouseup.xdsoft_scroller', arguments_callee) +									.off('mousemove.xdsoft_scroller', calcOffset) +									.removeClass('xdsoft_noselect'); +							}); +							$(document.body).on('mousemove.xdsoft_scroller', calcOffset); +						} else { +							touchStart = true; +							event.stopPropagation(); +							event.preventDefault(); +						} +					}) +					.on('touchmove', function (event) { +						if (touchStart) { +							event.preventDefault(); +							calcOffset(event); +						} +					}) +					.on('touchend touchcancel', function (event) { +						touchStart =  false; +						startTopScroll = 0; +					}); + +				timeboxparent +					.on('scroll_element.xdsoft_scroller', function (event, percentage) { +						if (!parentHeight) { +							timeboxparent.trigger('resize_scroll.xdsoft_scroller', [percentage, true]); +						} +						percentage = percentage > 1 ? 1 : (percentage < 0 || isNaN(percentage)) ? 0 : percentage; + +						scroller.css('margin-top', maximumOffset * percentage); + +						setTimeout(function () { +							timebox.css('marginTop', -parseInt((timebox[0].offsetHeight - parentHeight) * percentage, 10)); +						}, 10); +					}) +					.on('resize_scroll.xdsoft_scroller', function (event, percentage, noTriggerScroll) { +						var percent, sh; +						parentHeight = timeboxparent[0].clientHeight; +						height = timebox[0].offsetHeight; +						percent = parentHeight / height; +						sh = percent * scrollbar[0].offsetHeight; +						if (percent > 1) { +							scroller.hide(); +						} else { +							scroller.show(); +							scroller.css('height', parseInt(sh > 10 ? sh : 10, 10)); +							maximumOffset = scrollbar[0].offsetHeight - scroller[0].offsetHeight; +							if (noTriggerScroll !== true) { +								timeboxparent.trigger('scroll_element.xdsoft_scroller', [percentage || Math.abs(parseInt(timebox.css('marginTop'), 10)) / (height - parentHeight)]); +							} +						} +					}); + +				timeboxparent.on('mousewheel', function (event) { +					var top = Math.abs(parseInt(timebox.css('marginTop'), 10)); + +					top = top - (event.deltaY * 20); +					if (top < 0) { +						top = 0; +					} + +					timeboxparent.trigger('scroll_element.xdsoft_scroller', [top / (height - parentHeight)]); +					event.stopPropagation(); +					return false; +				}); + +				timeboxparent.on('touchstart', function (event) { +					start = pointerEventToXY(event); +					startTop = Math.abs(parseInt(timebox.css('marginTop'), 10)); +				}); + +				timeboxparent.on('touchmove', function (event) { +					if (start) { +						event.preventDefault(); +						var coord = pointerEventToXY(event); +						timeboxparent.trigger('scroll_element.xdsoft_scroller', [(startTop - (coord.y - start.y)) / (height - parentHeight)]); +					} +				}); + +				timeboxparent.on('touchend touchcancel', function (event) { +					start = false; +					startTop = 0; +				}); +			} +			timeboxparent.trigger('resize_scroll.xdsoft_scroller', [percent]); +		}); +	}; + +	$.fn.datetimepicker = function (opt, opt2) { +		var result = this, +            KEY0 = 48, +			KEY9 = 57, +			_KEY0 = 96, +			_KEY9 = 105, +			CTRLKEY = 17, +			DEL = 46, +			ENTER = 13, +			ESC = 27, +			BACKSPACE = 8, +			ARROWLEFT = 37, +			ARROWUP = 38, +			ARROWRIGHT = 39, +			ARROWDOWN = 40, +			TAB = 9, +			F5 = 116, +			AKEY = 65, +			CKEY = 67, +			VKEY = 86, +			ZKEY = 90, +			YKEY = 89, +			ctrlDown	=	false, +			options = ($.isPlainObject(opt) || !opt) ? $.extend(true, {}, default_options, opt) : $.extend(true, {}, default_options), + +			lazyInitTimer = 0, +			createDateTimePicker, +			destroyDateTimePicker, + +			lazyInit = function (input) { +				input +					.on('open.xdsoft focusin.xdsoft mousedown.xdsoft touchstart', function initOnActionCallback(event) { +						if (input.is(':disabled') || input.data('xdsoft_datetimepicker')) { +							return; +						} +						clearTimeout(lazyInitTimer); +						lazyInitTimer = setTimeout(function () { + +							if (!input.data('xdsoft_datetimepicker')) { +								createDateTimePicker(input); +							} +							input +								.off('open.xdsoft focusin.xdsoft mousedown.xdsoft touchstart', initOnActionCallback) +								.trigger('open.xdsoft'); +						}, 100); +					}); +			}; + +		createDateTimePicker = function (input) { +			var datetimepicker = $('<div class="xdsoft_datetimepicker xdsoft_noselect"></div>'), +				xdsoft_copyright = $('<div class="xdsoft_copyright"><a target="_blank" href="http://xdsoft.net/jqplugins/datetimepicker/">xdsoft.net</a></div>'), +				datepicker = $('<div class="xdsoft_datepicker active"></div>'), +				mounth_picker = $('<div class="xdsoft_mounthpicker"><button type="button" class="xdsoft_prev"></button><button type="button" class="xdsoft_today_button"></button>' + +					'<div class="xdsoft_label xdsoft_month"><span></span><i></i></div>' + +					'<div class="xdsoft_label xdsoft_year"><span></span><i></i></div>' + +					'<button type="button" class="xdsoft_next"></button></div>'), +				calendar = $('<div class="xdsoft_calendar"></div>'), +				timepicker = $('<div class="xdsoft_timepicker active"><button type="button" class="xdsoft_prev"></button><div class="xdsoft_time_box"></div><button type="button" class="xdsoft_next"></button></div>'), +				timeboxparent = timepicker.find('.xdsoft_time_box').eq(0), +				timebox = $('<div class="xdsoft_time_variant"></div>'), +                applyButton = $('<button type="button" class="xdsoft_save_selected blue-gradient-button">Save Selected</button>'), + +				monthselect = $('<div class="xdsoft_select xdsoft_monthselect"><div></div></div>'), +				yearselect = $('<div class="xdsoft_select xdsoft_yearselect"><div></div></div>'), +				triggerAfterOpen = false, +				XDSoft_datetime, + +				xchangeTimer, +				timerclick, +				current_time_index, +				setPos, +				timer = 0, +				timer1 = 0, +				_xdsoft_datetime; + +			if (options.id) { +				datetimepicker.attr('id', options.id); +			} +			if (options.style) { +				datetimepicker.attr('style', options.style); +			} +			if (options.weeks) { +				datetimepicker.addClass('xdsoft_showweeks'); +			} +			if (options.rtl) { +				datetimepicker.addClass('xdsoft_rtl'); +			} + +			datetimepicker.addClass('xdsoft_' + options.theme); +			datetimepicker.addClass(options.className); + +			mounth_picker +				.find('.xdsoft_month span') +					.after(monthselect); +			mounth_picker +				.find('.xdsoft_year span') +					.after(yearselect); + +			mounth_picker +				.find('.xdsoft_month,.xdsoft_year') +					.on('touchstart mousedown.xdsoft', function (event) { +					var select = $(this).find('.xdsoft_select').eq(0), +						val = 0, +						top = 0, +						visible = select.is(':visible'), +						items, +						i; + +					mounth_picker +						.find('.xdsoft_select') +							.hide(); +					if (_xdsoft_datetime.currentTime) { +						val = _xdsoft_datetime.currentTime[$(this).hasClass('xdsoft_month') ? 'getMonth' : 'getFullYear'](); +					} + +					select[visible ? 'hide' : 'show'](); +					for (items = select.find('div.xdsoft_option'), i = 0; i < items.length; i += 1) { +						if (items.eq(i).data('value') === val) { +							break; +						} else { +							top += items[0].offsetHeight; +						} +					} + +					select.xdsoftScroller(top / (select.children()[0].offsetHeight - (select[0].clientHeight))); +					event.stopPropagation(); +					return false; +				}); + +			mounth_picker +				.find('.xdsoft_select') +					.xdsoftScroller() +				.on('touchstart mousedown.xdsoft', function (event) { +					event.stopPropagation(); +					event.preventDefault(); +				}) +				.on('touchstart mousedown.xdsoft', '.xdsoft_option', function (event) { +					if (_xdsoft_datetime.currentTime === undefined || _xdsoft_datetime.currentTime === null) { +						_xdsoft_datetime.currentTime = _xdsoft_datetime.now(); +					} + +					var year = _xdsoft_datetime.currentTime.getFullYear(); +					if (_xdsoft_datetime && _xdsoft_datetime.currentTime) { +						_xdsoft_datetime.currentTime[$(this).parent().parent().hasClass('xdsoft_monthselect') ? 'setMonth' : 'setFullYear']($(this).data('value')); +					} + +					$(this).parent().parent().hide(); + +					datetimepicker.trigger('xchange.xdsoft'); +					if (options.onChangeMonth && $.isFunction(options.onChangeMonth)) { +						options.onChangeMonth.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); +					} + +					if (year !== _xdsoft_datetime.currentTime.getFullYear() && $.isFunction(options.onChangeYear)) { +						options.onChangeYear.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); +					} +				}); + +			datetimepicker.getValue = function () { +                return _xdsoft_datetime.getCurrentTime(); +            }; + +			datetimepicker.setOptions = function (_options) { +				var highlightedDates = {}; + + +				options = $.extend(true, {}, options, _options); + +				if (_options.allowTimes && $.isArray(_options.allowTimes) && _options.allowTimes.length) { +					options.allowTimes = $.extend(true, [], _options.allowTimes); +				} + +				if (_options.weekends && $.isArray(_options.weekends) && _options.weekends.length) { +					options.weekends = $.extend(true, [], _options.weekends); +				} + +				if (_options.allowDates && $.isArray(_options.allowDates) && _options.allowDates.length) { +					options.allowDates = $.extend(true, [], _options.allowDates); +				} + +				if (_options.allowDateRe && Object.prototype.toString.call(_options.allowDateRe)==="[object String]") { +					options.allowDateRe = new RegExp(_options.allowDateRe); +				} + +				if (_options.highlightedDates && $.isArray(_options.highlightedDates) && _options.highlightedDates.length) { +					$.each(_options.highlightedDates, function (index, value) { +						var splitData = $.map(value.split(','), $.trim), +							exDesc, +							hDate = new HighlightedDate(dateHelper.parseDate(splitData[0], options.formatDate), splitData[1], splitData[2]), // date, desc, style +							keyDate = dateHelper.formatDate(hDate.date, options.formatDate); +						if (highlightedDates[keyDate] !== undefined) { +							exDesc = highlightedDates[keyDate].desc; +							if (exDesc && exDesc.length && hDate.desc && hDate.desc.length) { +								highlightedDates[keyDate].desc = exDesc + "\n" + hDate.desc; +							} +						} else { +							highlightedDates[keyDate] = hDate; +						} +					}); + +					options.highlightedDates = $.extend(true, [], highlightedDates); +				} + +				if (_options.highlightedPeriods && $.isArray(_options.highlightedPeriods) && _options.highlightedPeriods.length) { +					highlightedDates = $.extend(true, [], options.highlightedDates); +					$.each(_options.highlightedPeriods, function (index, value) { +						var dateTest, // start date +							dateEnd, +							desc, +							hDate, +							keyDate, +							exDesc, +							style; +						if ($.isArray(value)) { +							dateTest = value[0]; +							dateEnd = value[1]; +							desc = value[2]; +							style = value[3]; +						} +						else { +							var splitData = $.map(value.split(','), $.trim); +							dateTest = dateHelper.parseDate(splitData[0], options.formatDate); +							dateEnd = dateHelper.parseDate(splitData[1], options.formatDate); +							desc = splitData[2]; +							style = splitData[3]; +						} + +						while (dateTest <= dateEnd) { +							hDate = new HighlightedDate(dateTest, desc, style); +							keyDate = dateHelper.formatDate(dateTest, options.formatDate); +							dateTest.setDate(dateTest.getDate() + 1); +							if (highlightedDates[keyDate] !== undefined) { +								exDesc = highlightedDates[keyDate].desc; +								if (exDesc && exDesc.length && hDate.desc && hDate.desc.length) { +									highlightedDates[keyDate].desc = exDesc + "\n" + hDate.desc; +								} +							} else { +								highlightedDates[keyDate] = hDate; +							} +						} +					}); + +					options.highlightedDates = $.extend(true, [], highlightedDates); +				} + +				if (_options.disabledDates && $.isArray(_options.disabledDates) && _options.disabledDates.length) { +					options.disabledDates = $.extend(true, [], _options.disabledDates); +				} + +				if (_options.disabledWeekDays && $.isArray(_options.disabledWeekDays) && _options.disabledWeekDays.length) { +				    options.disabledWeekDays = $.extend(true, [], _options.disabledWeekDays); +				} + +				if ((options.open || options.opened) && (!options.inline)) { +					input.trigger('open.xdsoft'); +				} + +				if (options.inline) { +					triggerAfterOpen = true; +					datetimepicker.addClass('xdsoft_inline'); +					input.after(datetimepicker).hide(); +				} + +				if (options.inverseButton) { +					options.next = 'xdsoft_prev'; +					options.prev = 'xdsoft_next'; +				} + +				if (options.datepicker) { +					datepicker.addClass('active'); +				} else { +					datepicker.removeClass('active'); +				} + +				if (options.timepicker) { +					timepicker.addClass('active'); +				} else { +					timepicker.removeClass('active'); +				} + +				if (options.value) { +					_xdsoft_datetime.setCurrentTime(options.value); +					if (input && input.val) { +						input.val(_xdsoft_datetime.str); +					} +				} + +				if (isNaN(options.dayOfWeekStart)) { +					options.dayOfWeekStart = 0; +				} else { +					options.dayOfWeekStart = parseInt(options.dayOfWeekStart, 10) % 7; +				} + +				if (!options.timepickerScrollbar) { +					timeboxparent.xdsoftScroller('hide'); +				} + +				if (options.minDate && /^[\+\-](.*)$/.test(options.minDate)) { +					options.minDate = dateHelper.formatDate(_xdsoft_datetime.strToDateTime(options.minDate), options.formatDate); +				} + +				if (options.maxDate &&  /^[\+\-](.*)$/.test(options.maxDate)) { +					options.maxDate = dateHelper.formatDate(_xdsoft_datetime.strToDateTime(options.maxDate), options.formatDate); +				} + +				applyButton.toggle(options.showApplyButton); + +				mounth_picker +					.find('.xdsoft_today_button') +						.css('visibility', !options.todayButton ? 'hidden' : 'visible'); + +				mounth_picker +					.find('.' + options.prev) +						.css('visibility', !options.prevButton ? 'hidden' : 'visible'); + +				mounth_picker +					.find('.' + options.next) +						.css('visibility', !options.nextButton ? 'hidden' : 'visible'); + +				setMask(options); + +				if (options.validateOnBlur) { +					input +						.off('blur.xdsoft') +						.on('blur.xdsoft', function () { +							if (options.allowBlank && (!$.trim($(this).val()).length || $.trim($(this).val()) === options.mask.replace(/[0-9]/g, '_'))) { +								$(this).val(null); +								datetimepicker.data('xdsoft_datetime').empty(); +							} else if (!dateHelper.parseDate($(this).val(), options.format)) { +								var splittedHours   = +([$(this).val()[0], $(this).val()[1]].join('')), +									splittedMinutes = +([$(this).val()[2], $(this).val()[3]].join('')); + +								// parse the numbers as 0312 => 03:12 +								if (!options.datepicker && options.timepicker && splittedHours >= 0 && splittedHours < 24 && splittedMinutes >= 0 && splittedMinutes < 60) { +									$(this).val([splittedHours, splittedMinutes].map(function (item) { +										return item > 9 ? item : '0' + item; +									}).join(':')); +								} else { +									$(this).val(dateHelper.formatDate(_xdsoft_datetime.now(), options.format)); +								} + +								datetimepicker.data('xdsoft_datetime').setCurrentTime($(this).val()); +							} else { +								datetimepicker.data('xdsoft_datetime').setCurrentTime($(this).val()); +							} + +							datetimepicker.trigger('changedatetime.xdsoft'); +							datetimepicker.trigger('close.xdsoft'); +						}); +				} +				options.dayOfWeekStartPrev = (options.dayOfWeekStart === 0) ? 6 : options.dayOfWeekStart - 1; + +				datetimepicker +					.trigger('xchange.xdsoft') +					.trigger('afterOpen.xdsoft'); +			}; + +			datetimepicker +				.data('options', options) +				.on('touchstart mousedown.xdsoft', function (event) { +					event.stopPropagation(); +					event.preventDefault(); +					yearselect.hide(); +					monthselect.hide(); +					return false; +				}); + +			//scroll_element = timepicker.find('.xdsoft_time_box'); +			timeboxparent.append(timebox); +			timeboxparent.xdsoftScroller(); + +			datetimepicker.on('afterOpen.xdsoft', function () { +				timeboxparent.xdsoftScroller(); +			}); + +			datetimepicker +				.append(datepicker) +				.append(timepicker); + +			if (options.withoutCopyright !== true) { +				datetimepicker +					.append(xdsoft_copyright); +			} + +			datepicker +				.append(mounth_picker) +				.append(calendar) +				.append(applyButton); + +			$(options.parentID) +				.append(datetimepicker); + +			XDSoft_datetime = function () { +				var _this = this; +				_this.now = function (norecursion) { +					var d = new Date(), +						date, +						time; + +					if (!norecursion && options.defaultDate) { +						date = _this.strToDateTime(options.defaultDate); +						d.setFullYear(date.getFullYear()); +						d.setMonth(date.getMonth()); +						d.setDate(date.getDate()); +					} + +					if (options.yearOffset) { +						d.setFullYear(d.getFullYear() + options.yearOffset); +					} + +					if (!norecursion && options.defaultTime) { +						time = _this.strtotime(options.defaultTime); +						d.setHours(time.getHours()); +						d.setMinutes(time.getMinutes()); +					} +					return d; +				}; + +				_this.isValidDate = function (d) { +					if (Object.prototype.toString.call(d) !== "[object Date]") { +						return false; +					} +					return !isNaN(d.getTime()); +				}; + +				_this.setCurrentTime = function (dTime) { +					_this.currentTime = (typeof dTime === 'string') ? _this.strToDateTime(dTime) : _this.isValidDate(dTime) ? dTime : _this.now(); +					datetimepicker.trigger('xchange.xdsoft'); +				}; + +				_this.empty = function () { +					_this.currentTime = null; +				}; + +				_this.getCurrentTime = function (dTime) { +					return _this.currentTime; +				}; + +				_this.nextMonth = function () { + +					if (_this.currentTime === undefined || _this.currentTime === null) { +						_this.currentTime = _this.now(); +					} + +					var month = _this.currentTime.getMonth() + 1, +						year; +					if (month === 12) { +						_this.currentTime.setFullYear(_this.currentTime.getFullYear() + 1); +						month = 0; +					} + +					year = _this.currentTime.getFullYear(); + +					_this.currentTime.setDate( +						Math.min( +							new Date(_this.currentTime.getFullYear(), month + 1, 0).getDate(), +							_this.currentTime.getDate() +						) +					); +					_this.currentTime.setMonth(month); + +					if (options.onChangeMonth && $.isFunction(options.onChangeMonth)) { +						options.onChangeMonth.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); +					} + +					if (year !== _this.currentTime.getFullYear() && $.isFunction(options.onChangeYear)) { +						options.onChangeYear.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); +					} + +					datetimepicker.trigger('xchange.xdsoft'); +					return month; +				}; + +				_this.prevMonth = function () { + +					if (_this.currentTime === undefined || _this.currentTime === null) { +						_this.currentTime = _this.now(); +					} + +					var month = _this.currentTime.getMonth() - 1; +					if (month === -1) { +						_this.currentTime.setFullYear(_this.currentTime.getFullYear() - 1); +						month = 11; +					} +					_this.currentTime.setDate( +						Math.min( +							new Date(_this.currentTime.getFullYear(), month + 1, 0).getDate(), +							_this.currentTime.getDate() +						) +					); +					_this.currentTime.setMonth(month); +					if (options.onChangeMonth && $.isFunction(options.onChangeMonth)) { +						options.onChangeMonth.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); +					} +					datetimepicker.trigger('xchange.xdsoft'); +					return month; +				}; + +				_this.getWeekOfYear = function (datetime) { +					if (options.onGetWeekOfYear && $.isFunction(options.onGetWeekOfYear)) { +						var week = options.onGetWeekOfYear.call(datetimepicker, datetime); +						if (typeof week !== 'undefined') { +							return week; +						} +					} +					var onejan = new Date(datetime.getFullYear(), 0, 1); +					//First week of the year is th one with the first Thursday according to ISO8601 +					if(onejan.getDay()!=4) +						onejan.setMonth(0, 1 + ((4 - onejan.getDay()+ 7) % 7)); +					return Math.ceil((((datetime - onejan) / 86400000) + onejan.getDay() + 1) / 7); +				}; + +				_this.strToDateTime = function (sDateTime) { +					var tmpDate = [], timeOffset, currentTime; + +					if (sDateTime && sDateTime instanceof Date && _this.isValidDate(sDateTime)) { +						return sDateTime; +					} + +					tmpDate = /^(\+|\-)(.*)$/.exec(sDateTime); +					if (tmpDate) { +						tmpDate[2] = dateHelper.parseDate(tmpDate[2], options.formatDate); +					} +					if (tmpDate  && tmpDate[2]) { +						timeOffset = tmpDate[2].getTime() - (tmpDate[2].getTimezoneOffset()) * 60000; +						currentTime = new Date((_this.now(true)).getTime() + parseInt(tmpDate[1] + '1', 10) * timeOffset); +					} else { +						currentTime = sDateTime ? dateHelper.parseDate(sDateTime, options.format) : _this.now(); +					} + +					if (!_this.isValidDate(currentTime)) { +						currentTime = _this.now(); +					} + +					return currentTime; +				}; + +				_this.strToDate = function (sDate) { +					if (sDate && sDate instanceof Date && _this.isValidDate(sDate)) { +						return sDate; +					} + +					var currentTime = sDate ? dateHelper.parseDate(sDate, options.formatDate) : _this.now(true); +					if (!_this.isValidDate(currentTime)) { +						currentTime = _this.now(true); +					} +					return currentTime; +				}; + +				_this.strtotime = function (sTime) { +					if (sTime && sTime instanceof Date && _this.isValidDate(sTime)) { +						return sTime; +					} +					var currentTime = sTime ? dateHelper.parseDate(sTime, options.formatTime) : _this.now(true); +					if (!_this.isValidDate(currentTime)) { +						currentTime = _this.now(true); +					} +					return currentTime; +				}; + +				_this.str = function () { +					return dateHelper.formatDate(_this.currentTime, options.format); +				}; +				_this.currentTime = this.now(); +			}; + +			_xdsoft_datetime = new XDSoft_datetime(); + +			applyButton.on('touchend click', function (e) {//pathbrite +                e.preventDefault(); +                datetimepicker.data('changed', true); +                _xdsoft_datetime.setCurrentTime(getCurrentValue()); +                input.val(_xdsoft_datetime.str()); +                datetimepicker.trigger('close.xdsoft'); +            }); +			mounth_picker +				.find('.xdsoft_today_button') +				.on('touchend mousedown.xdsoft', function () { +					datetimepicker.data('changed', true); +					_xdsoft_datetime.setCurrentTime(0); +					datetimepicker.trigger('afterOpen.xdsoft'); +				}).on('dblclick.xdsoft', function () { +					var currentDate = _xdsoft_datetime.getCurrentTime(), minDate, maxDate; +					currentDate = new Date(currentDate.getFullYear(), currentDate.getMonth(), currentDate.getDate()); +					minDate = _xdsoft_datetime.strToDate(options.minDate); +					minDate = new Date(minDate.getFullYear(), minDate.getMonth(), minDate.getDate()); +					if (currentDate < minDate) { +						return; +					} +					maxDate = _xdsoft_datetime.strToDate(options.maxDate); +					maxDate = new Date(maxDate.getFullYear(), maxDate.getMonth(), maxDate.getDate()); +					if (currentDate > maxDate) { +						return; +					} +					input.val(_xdsoft_datetime.str()); +					input.trigger('change'); +					datetimepicker.trigger('close.xdsoft'); +				}); +			mounth_picker +				.find('.xdsoft_prev,.xdsoft_next') +				.on('touchend mousedown.xdsoft', function () { +					var $this = $(this), +						timer = 0, +						stop = false; + +					(function arguments_callee1(v) { +						if ($this.hasClass(options.next)) { +							_xdsoft_datetime.nextMonth(); +						} else if ($this.hasClass(options.prev)) { +							_xdsoft_datetime.prevMonth(); +						} +						if (options.monthChangeSpinner) { +							if (!stop) { +								timer = setTimeout(arguments_callee1, v || 100); +							} +						} +					}(500)); + +					$([document.body, window]).on('touchend mouseup.xdsoft', function arguments_callee2() { +						clearTimeout(timer); +						stop = true; +						$([document.body, window]).off('touchend mouseup.xdsoft', arguments_callee2); +					}); +				}); + +			timepicker +				.find('.xdsoft_prev,.xdsoft_next') +				.on('touchend mousedown.xdsoft', function () { +					var $this = $(this), +						timer = 0, +						stop = false, +						period = 110; +					(function arguments_callee4(v) { +						var pheight = timeboxparent[0].clientHeight, +							height = timebox[0].offsetHeight, +							top = Math.abs(parseInt(timebox.css('marginTop'), 10)); +						if ($this.hasClass(options.next) && (height - pheight) - options.timeHeightInTimePicker >= top) { +							timebox.css('marginTop', '-' + (top + options.timeHeightInTimePicker) + 'px'); +						} else if ($this.hasClass(options.prev) && top - options.timeHeightInTimePicker >= 0) { +							timebox.css('marginTop', '-' + (top - options.timeHeightInTimePicker) + 'px'); +						} +						timeboxparent.trigger('scroll_element.xdsoft_scroller', [Math.abs(parseInt(timebox.css('marginTop'), 10) / (height - pheight))]); +						period = (period > 10) ? 10 : period - 10; +						if (!stop) { +							timer = setTimeout(arguments_callee4, v || period); +						} +					}(500)); +					$([document.body, window]).on('touchend mouseup.xdsoft', function arguments_callee5() { +						clearTimeout(timer); +						stop = true; +						$([document.body, window]) +							.off('touchend mouseup.xdsoft', arguments_callee5); +					}); +				}); + +			xchangeTimer = 0; +			// base handler - generating a calendar and timepicker +			datetimepicker +				.on('xchange.xdsoft', function (event) { +					clearTimeout(xchangeTimer); +					xchangeTimer = setTimeout(function () { + +						if (_xdsoft_datetime.currentTime === undefined || _xdsoft_datetime.currentTime === null) { +							_xdsoft_datetime.currentTime = _xdsoft_datetime.now(); +						} + +						var table =	'', +							start = new Date(_xdsoft_datetime.currentTime.getFullYear(), _xdsoft_datetime.currentTime.getMonth(), 1, 12, 0, 0), +							i = 0, +							j, +							today = _xdsoft_datetime.now(), +							maxDate = false, +							minDate = false, +							hDate, +							day, +							d, +							y, +							m, +							w, +							classes = [], +							customDateSettings, +							newRow = true, +							time = '', +							h = '', +							line_time, +							description; + +						while (start.getDay() !== options.dayOfWeekStart) { +							start.setDate(start.getDate() - 1); +						} + +						table += '<table><thead><tr>'; + +						if (options.weeks) { +							table += '<th></th>'; +						} + +						for (j = 0; j < 7; j += 1) { +							table += '<th>' + options.i18n[globalLocale].dayOfWeekShort[(j + options.dayOfWeekStart) % 7] + '</th>'; +						} + +						table += '</tr></thead>'; +						table += '<tbody>'; + +						if (options.maxDate !== false) { +							maxDate = _xdsoft_datetime.strToDate(options.maxDate); +							maxDate = new Date(maxDate.getFullYear(), maxDate.getMonth(), maxDate.getDate(), 23, 59, 59, 999); +						} + +						if (options.minDate !== false) { +							minDate = _xdsoft_datetime.strToDate(options.minDate); +							minDate = new Date(minDate.getFullYear(), minDate.getMonth(), minDate.getDate()); +						} + +						while (i < _xdsoft_datetime.currentTime.countDaysInMonth() || start.getDay() !== options.dayOfWeekStart || _xdsoft_datetime.currentTime.getMonth() === start.getMonth()) { +							classes = []; +							i += 1; + +							day = start.getDay(); +							d = start.getDate(); +							y = start.getFullYear(); +							m = start.getMonth(); +							w = _xdsoft_datetime.getWeekOfYear(start); +							description = ''; + +							classes.push('xdsoft_date'); + +							if (options.beforeShowDay && $.isFunction(options.beforeShowDay.call)) { +								customDateSettings = options.beforeShowDay.call(datetimepicker, start); +							} else { +								customDateSettings = null; +							} + +							if(options.allowDateRe && Object.prototype.toString.call(options.allowDateRe) === "[object RegExp]"){ +								if(!options.allowDateRe.test(dateHelper.formatDate(start, options.formatDate))){ +									classes.push('xdsoft_disabled'); +								} +							} else if(options.allowDates && options.allowDates.length>0){ +								if(options.allowDates.indexOf(dateHelper.formatDate(start, options.formatDate)) === -1){ +									classes.push('xdsoft_disabled'); +								} +							} else if ((maxDate !== false && start > maxDate) || (minDate !== false && start < minDate) || (customDateSettings && customDateSettings[0] === false)) { +								classes.push('xdsoft_disabled'); +							} else if (options.disabledDates.indexOf(dateHelper.formatDate(start, options.formatDate)) !== -1) { +								classes.push('xdsoft_disabled'); +							} else if (options.disabledWeekDays.indexOf(day) !== -1) { +							    classes.push('xdsoft_disabled'); +							} + +							if (customDateSettings && customDateSettings[1] !== "") { +								classes.push(customDateSettings[1]); +							} + +							if (_xdsoft_datetime.currentTime.getMonth() !== m) { +								classes.push('xdsoft_other_month'); +							} + +							if ((options.defaultSelect || datetimepicker.data('changed')) && dateHelper.formatDate(_xdsoft_datetime.currentTime, options.formatDate) === dateHelper.formatDate(start, options.formatDate)) { +								classes.push('xdsoft_current'); +							} + +							if (dateHelper.formatDate(today, options.formatDate) === dateHelper.formatDate(start, options.formatDate)) { +								classes.push('xdsoft_today'); +							} + +							if (start.getDay() === 0 || start.getDay() === 6 || options.weekends.indexOf(dateHelper.formatDate(start, options.formatDate)) !== -1) { +								classes.push('xdsoft_weekend'); +							} + +							if (options.highlightedDates[dateHelper.formatDate(start, options.formatDate)] !== undefined) { +								hDate = options.highlightedDates[dateHelper.formatDate(start, options.formatDate)]; +								classes.push(hDate.style === undefined ? 'xdsoft_highlighted_default' : hDate.style); +								description = hDate.desc === undefined ? '' : hDate.desc; +							} + +							if (options.beforeShowDay && $.isFunction(options.beforeShowDay)) { +								classes.push(options.beforeShowDay(start)); +							} + +							if (newRow) { +								table += '<tr>'; +								newRow = false; +								if (options.weeks) { +									table += '<th>' + w + '</th>'; +								} +							} + +							table += '<td data-date="' + d + '" data-month="' + m + '" data-year="' + y + '"' + ' class="xdsoft_date xdsoft_day_of_week' + start.getDay() + ' ' + classes.join(' ') + '" title="' + description + '">' + +										'<div>' + d + '</div>' + +									'</td>'; + +							if (start.getDay() === options.dayOfWeekStartPrev) { +								table += '</tr>'; +								newRow = true; +							} + +							start.setDate(d + 1); +						} +						table += '</tbody></table>'; + +						calendar.html(table); + +						mounth_picker.find('.xdsoft_label span').eq(0).text(options.i18n[globalLocale].months[_xdsoft_datetime.currentTime.getMonth()]); +						mounth_picker.find('.xdsoft_label span').eq(1).text(_xdsoft_datetime.currentTime.getFullYear()); + +						// generate timebox +						time = ''; +						h = ''; +						m = ''; + +						line_time = function line_time(h, m) { +							var now = _xdsoft_datetime.now(), optionDateTime, current_time, +								isALlowTimesInit = options.allowTimes && $.isArray(options.allowTimes) && options.allowTimes.length; +							now.setHours(h); +							h = parseInt(now.getHours(), 10); +							now.setMinutes(m); +							m = parseInt(now.getMinutes(), 10); +							optionDateTime = new Date(_xdsoft_datetime.currentTime); +							optionDateTime.setHours(h); +							optionDateTime.setMinutes(m); +							classes = []; +							if ((options.minDateTime !== false && options.minDateTime > optionDateTime) || (options.maxTime !== false && _xdsoft_datetime.strtotime(options.maxTime).getTime() < now.getTime()) || (options.minTime !== false && _xdsoft_datetime.strtotime(options.minTime).getTime() > now.getTime())) { +								classes.push('xdsoft_disabled'); +							} +							if ((options.minDateTime !== false && options.minDateTime > optionDateTime) || ((options.disabledMinTime !== false && now.getTime() > _xdsoft_datetime.strtotime(options.disabledMinTime).getTime()) && (options.disabledMaxTime !== false && now.getTime() < _xdsoft_datetime.strtotime(options.disabledMaxTime).getTime()))) { +								classes.push('xdsoft_disabled'); +							} + +							current_time = new Date(_xdsoft_datetime.currentTime); +							current_time.setHours(parseInt(_xdsoft_datetime.currentTime.getHours(), 10)); + +							if (!isALlowTimesInit) { +								current_time.setMinutes(Math[options.roundTime](_xdsoft_datetime.currentTime.getMinutes() / options.step) * options.step); +							} + +							if ((options.initTime || options.defaultSelect || datetimepicker.data('changed')) && current_time.getHours() === parseInt(h, 10) && ((!isALlowTimesInit && options.step > 59) || current_time.getMinutes() === parseInt(m, 10))) { +								if (options.defaultSelect || datetimepicker.data('changed')) { +									classes.push('xdsoft_current'); +								} else if (options.initTime) { +									classes.push('xdsoft_init_time'); +								} +							} +							if (parseInt(today.getHours(), 10) === parseInt(h, 10) && parseInt(today.getMinutes(), 10) === parseInt(m, 10)) { +								classes.push('xdsoft_today'); +							} +							time += '<div class="xdsoft_time ' + classes.join(' ') + '" data-hour="' + h + '" data-minute="' + m + '">' + dateHelper.formatDate(now, options.formatTime) + '</div>'; +						}; + +						if (!options.allowTimes || !$.isArray(options.allowTimes) || !options.allowTimes.length) { +							for (i = 0, j = 0; i < (options.hours12 ? 12 : 24); i += 1) { +								for (j = 0; j < 60; j += options.step) { +									h = (i < 10 ? '0' : '') + i; +									m = (j < 10 ? '0' : '') + j; +									line_time(h, m); +								} +							} +						} else { +							for (i = 0; i < options.allowTimes.length; i += 1) { +								h = _xdsoft_datetime.strtotime(options.allowTimes[i]).getHours(); +								m = _xdsoft_datetime.strtotime(options.allowTimes[i]).getMinutes(); +								line_time(h, m); +							} +						} + +						timebox.html(time); + +						opt = ''; +						i = 0; + +						for (i = parseInt(options.yearStart, 10) + options.yearOffset; i <= parseInt(options.yearEnd, 10) + options.yearOffset; i += 1) { +							opt += '<div class="xdsoft_option ' + (_xdsoft_datetime.currentTime.getFullYear() === i ? 'xdsoft_current' : '') + '" data-value="' + i + '">' + i + '</div>'; +						} +						yearselect.children().eq(0) +												.html(opt); + +						for (i = parseInt(options.monthStart, 10), opt = ''; i <= parseInt(options.monthEnd, 10); i += 1) { +							opt += '<div class="xdsoft_option ' + (_xdsoft_datetime.currentTime.getMonth() === i ? 'xdsoft_current' : '') + '" data-value="' + i + '">' + options.i18n[globalLocale].months[i] + '</div>'; +						} +						monthselect.children().eq(0).html(opt); +						$(datetimepicker) +							.trigger('generate.xdsoft'); +					}, 10); +					event.stopPropagation(); +				}) +				.on('afterOpen.xdsoft', function () { +					if (options.timepicker) { +						var classType, pheight, height, top; +						if (timebox.find('.xdsoft_current').length) { +							classType = '.xdsoft_current'; +						} else if (timebox.find('.xdsoft_init_time').length) { +							classType = '.xdsoft_init_time'; +						} +						if (classType) { +							pheight = timeboxparent[0].clientHeight; +							height = timebox[0].offsetHeight; +							top = timebox.find(classType).index() * options.timeHeightInTimePicker + 1; +							if ((height - pheight) < top) { +								top = height - pheight; +							} +							timeboxparent.trigger('scroll_element.xdsoft_scroller', [parseInt(top, 10) / (height - pheight)]); +						} else { +							timeboxparent.trigger('scroll_element.xdsoft_scroller', [0]); +						} +					} +				}); + +			timerclick = 0; +			calendar +				.on('touchend click.xdsoft', 'td', function (xdevent) { +					xdevent.stopPropagation();  // Prevents closing of Pop-ups, Modals and Flyouts in Bootstrap +					timerclick += 1; +					var $this = $(this), +						currentTime = _xdsoft_datetime.currentTime; + +					if (currentTime === undefined || currentTime === null) { +						_xdsoft_datetime.currentTime = _xdsoft_datetime.now(); +						currentTime = _xdsoft_datetime.currentTime; +					} + +					if ($this.hasClass('xdsoft_disabled')) { +						return false; +					} + +					currentTime.setDate(1); +					currentTime.setFullYear($this.data('year')); +					currentTime.setMonth($this.data('month')); +					currentTime.setDate($this.data('date')); + +					datetimepicker.trigger('select.xdsoft', [currentTime]); + +					input.val(_xdsoft_datetime.str()); + +					if (options.onSelectDate &&	$.isFunction(options.onSelectDate)) { +						options.onSelectDate.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input'), xdevent); +					} + +					datetimepicker.data('changed', true); +					datetimepicker.trigger('xchange.xdsoft'); +					datetimepicker.trigger('changedatetime.xdsoft'); +					if ((timerclick > 1 || (options.closeOnDateSelect === true || (options.closeOnDateSelect === false && !options.timepicker))) && !options.inline) { +						datetimepicker.trigger('close.xdsoft'); +					} +					setTimeout(function () { +						timerclick = 0; +					}, 200); +				}); + +			timebox +				.on('touchend click.xdsoft', 'div', function (xdevent) { +					xdevent.stopPropagation(); +					var $this = $(this), +						currentTime = _xdsoft_datetime.currentTime; + +					if (currentTime === undefined || currentTime === null) { +						_xdsoft_datetime.currentTime = _xdsoft_datetime.now(); +						currentTime = _xdsoft_datetime.currentTime; +					} + +					if ($this.hasClass('xdsoft_disabled')) { +						return false; +					} +					currentTime.setHours($this.data('hour')); +					currentTime.setMinutes($this.data('minute')); +					datetimepicker.trigger('select.xdsoft', [currentTime]); + +					datetimepicker.data('input').val(_xdsoft_datetime.str()); + +					if (options.onSelectTime && $.isFunction(options.onSelectTime)) { +						options.onSelectTime.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input'), xdevent); +					} +					datetimepicker.data('changed', true); +					datetimepicker.trigger('xchange.xdsoft'); +					datetimepicker.trigger('changedatetime.xdsoft'); +					if (options.inline !== true && options.closeOnTimeSelect === true) { +						datetimepicker.trigger('close.xdsoft'); +					} +				}); + + +			datepicker +				.on('mousewheel.xdsoft', function (event) { +					if (!options.scrollMonth) { +						return true; +					} +					if (event.deltaY < 0) { +						_xdsoft_datetime.nextMonth(); +					} else { +						_xdsoft_datetime.prevMonth(); +					} +					return false; +				}); + +			input +				.on('mousewheel.xdsoft', function (event) { +					if (!options.scrollInput) { +						return true; +					} +					if (!options.datepicker && options.timepicker) { +						current_time_index = timebox.find('.xdsoft_current').length ? timebox.find('.xdsoft_current').eq(0).index() : 0; +						if (current_time_index + event.deltaY >= 0 && current_time_index + event.deltaY < timebox.children().length) { +							current_time_index += event.deltaY; +						} +						if (timebox.children().eq(current_time_index).length) { +							timebox.children().eq(current_time_index).trigger('mousedown'); +						} +						return false; +					} +					if (options.datepicker && !options.timepicker) { +						datepicker.trigger(event, [event.deltaY, event.deltaX, event.deltaY]); +						if (input.val) { +							input.val(_xdsoft_datetime.str()); +						} +						datetimepicker.trigger('changedatetime.xdsoft'); +						return false; +					} +				}); + +			datetimepicker +				.on('changedatetime.xdsoft', function (event) { +					if (options.onChangeDateTime && $.isFunction(options.onChangeDateTime)) { +						var $input = datetimepicker.data('input'); +						options.onChangeDateTime.call(datetimepicker, _xdsoft_datetime.currentTime, $input, event); +						delete options.value; +						$input.trigger('change'); +					} +				}) +				.on('generate.xdsoft', function () { +					if (options.onGenerate && $.isFunction(options.onGenerate)) { +						options.onGenerate.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); +					} +					if (triggerAfterOpen) { +						datetimepicker.trigger('afterOpen.xdsoft'); +						triggerAfterOpen = false; +					} +				}) +				.on('click.xdsoft', function (xdevent) { +					xdevent.stopPropagation(); +				}); + +			current_time_index = 0; + +			setPos = function () { +				/** +                 * 修复输入框在window最右边,且输入框的宽度小于日期控件宽度情况下,日期控件显示不全的bug。 +                 * Bug fixed - The datetimepicker will overflow-y when the width of the date input less than its, which +                 * could causes part of the datetimepicker being hidden. +                 * by Soon start +                 */ +                var offset = datetimepicker.data('input').offset(), +                    datetimepickerelement = datetimepicker.data('input')[0], +                    top = offset.top + datetimepickerelement.offsetHeight - 1, +                    left = offset.left, +                    position = "absolute", +                    node; + +                if ((document.documentElement.clientWidth - offset.left) < datepicker.parent().outerWidth(true)) { +                    var diff = datepicker.parent().outerWidth(true) - datetimepickerelement.offsetWidth; +                    left = left - diff; +                } +                /** +                 * by Soon end +                 */ +				if (datetimepicker.data('input').parent().css('direction') == 'rtl') +					left -= (datetimepicker.outerWidth() - datetimepicker.data('input').outerWidth()); +				if (options.fixed) { +					top -= $(window).scrollTop(); +					left -= $(window).scrollLeft(); +					position = "fixed"; +				} else { +					if (top + datetimepickerelement.offsetHeight > $(window).height() + $(window).scrollTop()) { +						top = offset.top - datetimepickerelement.offsetHeight + 1; +					} +					if (top < 0) { +						top = 0; +					} +					if (left + datetimepickerelement.offsetWidth > $(window).width()) { +						left = $(window).width() - datetimepickerelement.offsetWidth; +					} +				} + +				node = datetimepicker[0]; +				do { +					node = node.parentNode; +					if (window.getComputedStyle(node).getPropertyValue('position') === 'relative' && $(window).width() >= node.offsetWidth) { +						left = left - (($(window).width() - node.offsetWidth) / 2); +						break; +					} +				} while (node.nodeName !== 'HTML'); +				datetimepicker.css({ +					left: left, +					top: top, +					position: position +				}); +			}; +			datetimepicker +				.on('open.xdsoft', function (event) { +					var onShow = true; +					if (options.onShow && $.isFunction(options.onShow)) { +						onShow = options.onShow.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input'), event); +					} +					if (onShow !== false) { +						datetimepicker.show(); +						setPos(); +						$(window) +							.off('resize.xdsoft', setPos) +							.on('resize.xdsoft', setPos); + +						if (options.closeOnWithoutClick) { +							$([document.body, window]).on('touchstart mousedown.xdsoft', function arguments_callee6() { +								datetimepicker.trigger('close.xdsoft'); +								$([document.body, window]).off('touchstart mousedown.xdsoft', arguments_callee6); +							}); +						} +					} +				}) +				.on('close.xdsoft', function (event) { +					var onClose = true; +					mounth_picker +						.find('.xdsoft_month,.xdsoft_year') +							.find('.xdsoft_select') +								.hide(); +					if (options.onClose && $.isFunction(options.onClose)) { +						onClose = options.onClose.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input'), event); +					} +					if (onClose !== false && !options.opened && !options.inline) { +						datetimepicker.hide(); +					} +					event.stopPropagation(); +				}) +				.on('toggle.xdsoft', function (event) { +					if (datetimepicker.is(':visible')) { +						datetimepicker.trigger('close.xdsoft'); +					} else { +						datetimepicker.trigger('open.xdsoft'); +					} +				}) +				.data('input', input); + +			timer = 0; +			timer1 = 0; + +			datetimepicker.data('xdsoft_datetime', _xdsoft_datetime); +			datetimepicker.setOptions(options); + +			function getCurrentValue() { +				var ct = false, time; + +				if (options.startDate) { +					ct = _xdsoft_datetime.strToDate(options.startDate); +				} else { +					ct = options.value || ((input && input.val && input.val()) ? input.val() : ''); +					if (ct) { +						ct = _xdsoft_datetime.strToDateTime(ct); +					} else if (options.defaultDate) { +						ct = _xdsoft_datetime.strToDateTime(options.defaultDate); +						if (options.defaultTime) { +							time = _xdsoft_datetime.strtotime(options.defaultTime); +							ct.setHours(time.getHours()); +							ct.setMinutes(time.getMinutes()); +						} +					} +				} + +				if (ct && _xdsoft_datetime.isValidDate(ct)) { +					datetimepicker.data('changed', true); +				} else { +					ct = ''; +				} + +				return ct || 0; +			} + +			function setMask(options) { + +				var isValidValue = function (mask, value) { +					var reg = mask +						.replace(/([\[\]\/\{\}\(\)\-\.\+]{1})/g, '\\$1') +						.replace(/_/g, '{digit+}') +						.replace(/([0-9]{1})/g, '{digit$1}') +						.replace(/\{digit([0-9]{1})\}/g, '[0-$1_]{1}') +						.replace(/\{digit[\+]\}/g, '[0-9_]{1}'); +					return (new RegExp(reg)).test(value); +				}, +				getCaretPos = function (input) { +					try { +						if (document.selection && document.selection.createRange) { +							var range = document.selection.createRange(); +							return range.getBookmark().charCodeAt(2) - 2; +						} +						if (input.setSelectionRange) { +							return input.selectionStart; +						} +					} catch (e) { +						return 0; +					} +				}, +				setCaretPos = function (node, pos) { +					node = (typeof node === "string" || node instanceof String) ? document.getElementById(node) : node; +					if (!node) { +						return false; +					} +					if (node.createTextRange) { +						var textRange = node.createTextRange(); +						textRange.collapse(true); +						textRange.moveEnd('character', pos); +						textRange.moveStart('character', pos); +						textRange.select(); +						return true; +					} +					if (node.setSelectionRange) { +						node.setSelectionRange(pos, pos); +						return true; +					} +					return false; +				}; +				if(options.mask) { +					input.off('keydown.xdsoft'); +				} +				if (options.mask === true) { +														if (typeof moment != 'undefined') { +																	options.mask = options.format +																			.replace(/Y{4}/g, '9999') +																			.replace(/Y{2}/g, '99') +																			.replace(/M{2}/g, '19') +																			.replace(/D{2}/g, '39') +																			.replace(/H{2}/g, '29') +																			.replace(/m{2}/g, '59') +																			.replace(/s{2}/g, '59'); +														} else { +																	options.mask = options.format +																			.replace(/Y/g, '9999') +																			.replace(/F/g, '9999') +																			.replace(/m/g, '19') +																			.replace(/d/g, '39') +																			.replace(/H/g, '29') +																			.replace(/i/g, '59') +																			.replace(/s/g, '59'); +														} +				} + +				if ($.type(options.mask) === 'string') { +					if (!isValidValue(options.mask, input.val())) { +						input.val(options.mask.replace(/[0-9]/g, '_')); +						setCaretPos(input[0], 0); +					} + +					input.on('keydown.xdsoft', function (event) { +						var val = this.value, +							key = event.which, +							pos, +							digit; + +						if (((key >= KEY0 && key <= KEY9) || (key >= _KEY0 && key <= _KEY9)) || (key === BACKSPACE || key === DEL)) { +							pos = getCaretPos(this); +							digit = (key !== BACKSPACE && key !== DEL) ? String.fromCharCode((_KEY0 <= key && key <= _KEY9) ? key - KEY0 : key) : '_'; + +							if ((key === BACKSPACE || key === DEL) && pos) { +								pos -= 1; +								digit = '_'; +							} + +							while (/[^0-9_]/.test(options.mask.substr(pos, 1)) && pos < options.mask.length && pos > 0) { +								pos += (key === BACKSPACE || key === DEL) ? -1 : 1; +							} + +							val = val.substr(0, pos) + digit + val.substr(pos + 1); +							if ($.trim(val) === '') { +								val = options.mask.replace(/[0-9]/g, '_'); +							} else { +								if (pos === options.mask.length) { +									event.preventDefault(); +									return false; +								} +							} + +							pos += (key === BACKSPACE || key === DEL) ? 0 : 1; +							while (/[^0-9_]/.test(options.mask.substr(pos, 1)) && pos < options.mask.length && pos > 0) { +								pos += (key === BACKSPACE || key === DEL) ? -1 : 1; +							} + +							if (isValidValue(options.mask, val)) { +								this.value = val; +								setCaretPos(this, pos); +							} else if ($.trim(val) === '') { +								this.value = options.mask.replace(/[0-9]/g, '_'); +							} else { +								input.trigger('error_input.xdsoft'); +							} +						} else { +							if (([AKEY, CKEY, VKEY, ZKEY, YKEY].indexOf(key) !== -1 && ctrlDown) || [ESC, ARROWUP, ARROWDOWN, ARROWLEFT, ARROWRIGHT, F5, CTRLKEY, TAB, ENTER].indexOf(key) !== -1) { +								return true; +							} +						} + +						event.preventDefault(); +						return false; +					}); +				} +			} + +			_xdsoft_datetime.setCurrentTime(getCurrentValue()); + +			input +				.data('xdsoft_datetimepicker', datetimepicker) +				.on('open.xdsoft focusin.xdsoft mousedown.xdsoft touchstart', function (event) { +					if (input.is(':disabled') || (input.data('xdsoft_datetimepicker').is(':visible') && options.closeOnInputClick)) { +						return; +					} +					clearTimeout(timer); +					timer = setTimeout(function () { +						if (input.is(':disabled')) { +							return; +						} + +						triggerAfterOpen = true; +						_xdsoft_datetime.setCurrentTime(getCurrentValue()); +						if(options.mask) { +							setMask(options); +						} +						datetimepicker.trigger('open.xdsoft'); +					}, 100); +				}) +				.on('keydown.xdsoft', function (event) { +					var val = this.value, elementSelector, +						key = event.which; +					if ([ENTER].indexOf(key) !== -1 && options.enterLikeTab) { +						elementSelector = $("input:visible,textarea:visible,button:visible,a:visible"); +						datetimepicker.trigger('close.xdsoft'); +						elementSelector.eq(elementSelector.index(this) + 1).focus(); +						return false; +					} +					if ([TAB].indexOf(key) !== -1) { +						datetimepicker.trigger('close.xdsoft'); +						return true; +					} +				}) +				.on('blur.xdsoft', function () { +					datetimepicker.trigger('close.xdsoft'); +				}); +		}; +		destroyDateTimePicker = function (input) { +			var datetimepicker = input.data('xdsoft_datetimepicker'); +			if (datetimepicker) { +				datetimepicker.data('xdsoft_datetime', null); +				datetimepicker.remove(); +				input +					.data('xdsoft_datetimepicker', null) +					.off('.xdsoft'); +				$(window).off('resize.xdsoft'); +				$([window, document.body]).off('mousedown.xdsoft touchstart'); +				if (input.unmousewheel) { +					input.unmousewheel(); +				} +			} +		}; +		$(document) +			.off('keydown.xdsoftctrl keyup.xdsoftctrl') +			.on('keydown.xdsoftctrl', function (e) { +				if (e.keyCode === CTRLKEY) { +					ctrlDown = true; +				} +			}) +			.on('keyup.xdsoftctrl', function (e) { +				if (e.keyCode === CTRLKEY) { +					ctrlDown = false; +				} +			}); + +        this.each(function () { +			var datetimepicker = $(this).data('xdsoft_datetimepicker'), $input; +			if (datetimepicker) { +				if ($.type(opt) === 'string') { +					switch (opt) { +					case 'show': +						$(this).select().focus(); +						datetimepicker.trigger('open.xdsoft'); +						break; +					case 'hide': +						datetimepicker.trigger('close.xdsoft'); +						break; +					case 'toggle': +						datetimepicker.trigger('toggle.xdsoft'); +						break; +					case 'destroy': +						destroyDateTimePicker($(this)); +						break; +					case 'reset': +						this.value = this.defaultValue; +						if (!this.value || !datetimepicker.data('xdsoft_datetime').isValidDate(dateHelper.parseDate(this.value, options.format))) { +							datetimepicker.data('changed', false); +						} +						datetimepicker.data('xdsoft_datetime').setCurrentTime(this.value); +						break; +					case 'validate': +						$input = datetimepicker.data('input'); +						$input.trigger('blur.xdsoft'); +						break; +                    default: +                        if (datetimepicker[opt] && $.isFunction(datetimepicker[opt])) { +                            result = datetimepicker[opt](opt2); +                        } +					} +				} else { +					datetimepicker +						.setOptions(opt); +				} +				return 0; +			} +			if ($.type(opt) !== 'string') { +				if (!options.lazyInit || options.open || options.inline) { +					createDateTimePicker($(this)); +				} else { +					lazyInit($(this)); +				} +			} +		}); + +        return result; +	}; +	$.fn.datetimepicker.defaults = default_options; + +	function HighlightedDate(date, desc, style) { +		"use strict"; +		this.date = date; +		this.desc = desc; +		this.style = style; +	} + +})); +/*! + * jQuery Mousewheel 3.1.13 + * + * Copyright jQuery Foundation and other contributors + * Released under the MIT license + * http://jquery.org/license + */ + +(function (factory) { +    if ( typeof define === 'function' && define.amd ) { +        // AMD. Register as an anonymous module. +        define(['jquery'], factory); +    } else if (typeof exports === 'object') { +        // Node/CommonJS style for Browserify +        module.exports = factory; +    } else { +        // Browser globals +        factory(jQuery); +    } +}(function ($) { + +    var toFix  = ['wheel', 'mousewheel', 'DOMMouseScroll', 'MozMousePixelScroll'], +        toBind = ( 'onwheel' in document || document.documentMode >= 9 ) ? +                    ['wheel'] : ['mousewheel', 'DomMouseScroll', 'MozMousePixelScroll'], +        slice  = Array.prototype.slice, +        nullLowestDeltaTimeout, lowestDelta; + +    if ( $.event.fixHooks ) { +        for ( var i = toFix.length; i; ) { +            $.event.fixHooks[ toFix[--i] ] = $.event.mouseHooks; +        } +    } + +    var special = $.event.special.mousewheel = { +        version: '3.1.12', + +        setup: function() { +            if ( this.addEventListener ) { +                for ( var i = toBind.length; i; ) { +                    this.addEventListener( toBind[--i], handler, false ); +                } +            } else { +                this.onmousewheel = handler; +            } +            // Store the line height and page height for this particular element +            $.data(this, 'mousewheel-line-height', special.getLineHeight(this)); +            $.data(this, 'mousewheel-page-height', special.getPageHeight(this)); +        }, + +        teardown: function() { +            if ( this.removeEventListener ) { +                for ( var i = toBind.length; i; ) { +                    this.removeEventListener( toBind[--i], handler, false ); +                } +            } else { +                this.onmousewheel = null; +            } +            // Clean up the data we added to the element +            $.removeData(this, 'mousewheel-line-height'); +            $.removeData(this, 'mousewheel-page-height'); +        }, + +        getLineHeight: function(elem) { +            var $elem = $(elem), +                $parent = $elem['offsetParent' in $.fn ? 'offsetParent' : 'parent'](); +            if (!$parent.length) { +                $parent = $('body'); +            } +            return parseInt($parent.css('fontSize'), 10) || parseInt($elem.css('fontSize'), 10) || 16; +        }, + +        getPageHeight: function(elem) { +            return $(elem).height(); +        }, + +        settings: { +            adjustOldDeltas: true, // see shouldAdjustOldDeltas() below +            normalizeOffset: true  // calls getBoundingClientRect for each event +        } +    }; + +    $.fn.extend({ +        mousewheel: function(fn) { +            return fn ? this.bind('mousewheel', fn) : this.trigger('mousewheel'); +        }, + +        unmousewheel: function(fn) { +            return this.unbind('mousewheel', fn); +        } +    }); + + +    function handler(event) { +        var orgEvent   = event || window.event, +            args       = slice.call(arguments, 1), +            delta      = 0, +            deltaX     = 0, +            deltaY     = 0, +            absDelta   = 0, +            offsetX    = 0, +            offsetY    = 0; +        event = $.event.fix(orgEvent); +        event.type = 'mousewheel'; + +        // Old school scrollwheel delta +        if ( 'detail'      in orgEvent ) { deltaY = orgEvent.detail * -1;      } +        if ( 'wheelDelta'  in orgEvent ) { deltaY = orgEvent.wheelDelta;       } +        if ( 'wheelDeltaY' in orgEvent ) { deltaY = orgEvent.wheelDeltaY;      } +        if ( 'wheelDeltaX' in orgEvent ) { deltaX = orgEvent.wheelDeltaX * -1; } + +        // Firefox < 17 horizontal scrolling related to DOMMouseScroll event +        if ( 'axis' in orgEvent && orgEvent.axis === orgEvent.HORIZONTAL_AXIS ) { +            deltaX = deltaY * -1; +            deltaY = 0; +        } + +        // Set delta to be deltaY or deltaX if deltaY is 0 for backwards compatabilitiy +        delta = deltaY === 0 ? deltaX : deltaY; + +        // New school wheel delta (wheel event) +        if ( 'deltaY' in orgEvent ) { +            deltaY = orgEvent.deltaY * -1; +            delta  = deltaY; +        } +        if ( 'deltaX' in orgEvent ) { +            deltaX = orgEvent.deltaX; +            if ( deltaY === 0 ) { delta  = deltaX * -1; } +        } + +        // No change actually happened, no reason to go any further +        if ( deltaY === 0 && deltaX === 0 ) { return; } + +        // Need to convert lines and pages to pixels if we aren't already in pixels +        // There are three delta modes: +        //   * deltaMode 0 is by pixels, nothing to do +        //   * deltaMode 1 is by lines +        //   * deltaMode 2 is by pages +        if ( orgEvent.deltaMode === 1 ) { +            var lineHeight = $.data(this, 'mousewheel-line-height'); +            delta  *= lineHeight; +            deltaY *= lineHeight; +            deltaX *= lineHeight; +        } else if ( orgEvent.deltaMode === 2 ) { +            var pageHeight = $.data(this, 'mousewheel-page-height'); +            delta  *= pageHeight; +            deltaY *= pageHeight; +            deltaX *= pageHeight; +        } + +        // Store lowest absolute delta to normalize the delta values +        absDelta = Math.max( Math.abs(deltaY), Math.abs(deltaX) ); + +        if ( !lowestDelta || absDelta < lowestDelta ) { +            lowestDelta = absDelta; + +            // Adjust older deltas if necessary +            if ( shouldAdjustOldDeltas(orgEvent, absDelta) ) { +                lowestDelta /= 40; +            } +        } + +        // Adjust older deltas if necessary +        if ( shouldAdjustOldDeltas(orgEvent, absDelta) ) { +            // Divide all the things by 40! +            delta  /= 40; +            deltaX /= 40; +            deltaY /= 40; +        } + +        // Get a whole, normalized value for the deltas +        delta  = Math[ delta  >= 1 ? 'floor' : 'ceil' ](delta  / lowestDelta); +        deltaX = Math[ deltaX >= 1 ? 'floor' : 'ceil' ](deltaX / lowestDelta); +        deltaY = Math[ deltaY >= 1 ? 'floor' : 'ceil' ](deltaY / lowestDelta); + +        // Normalise offsetX and offsetY properties +        if ( special.settings.normalizeOffset && this.getBoundingClientRect ) { +            var boundingRect = this.getBoundingClientRect(); +            offsetX = event.clientX - boundingRect.left; +            offsetY = event.clientY - boundingRect.top; +        } + +        // Add information to the event object +        event.deltaX = deltaX; +        event.deltaY = deltaY; +        event.deltaFactor = lowestDelta; +        event.offsetX = offsetX; +        event.offsetY = offsetY; +        // Go ahead and set deltaMode to 0 since we converted to pixels +        // Although this is a little odd since we overwrite the deltaX/Y +        // properties with normalized deltas. +        event.deltaMode = 0; + +        // Add event and delta to the front of the arguments +        args.unshift(event, delta, deltaX, deltaY); + +        // Clearout lowestDelta after sometime to better +        // handle multiple device types that give different +        // a different lowestDelta +        // Ex: trackpad = 3 and mouse wheel = 120 +        if (nullLowestDeltaTimeout) { clearTimeout(nullLowestDeltaTimeout); } +        nullLowestDeltaTimeout = setTimeout(nullLowestDelta, 200); + +        return ($.event.dispatch || $.event.handle).apply(this, args); +    } + +    function nullLowestDelta() { +        lowestDelta = null; +    } + +    function shouldAdjustOldDeltas(orgEvent, absDelta) { +        // If this is an older event and the delta is divisable by 120, +        // then we are assuming that the browser is treating this as an +        // older mouse wheel event and that we should divide the deltas +        // by 40 to try and get a more usable deltaFactor. +        // Side note, this actually impacts the reported scroll distance +        // in older browsers and can cause scrolling to be slower than native. +        // Turn this off by setting $.event.special.mousewheel.settings.adjustOldDeltas to false. +        return special.settings.adjustOldDeltas && orgEvent.type === 'mousewheel' && absDelta % 120 === 0; +    } + +})); + +$(document).ready(function(){ +  $.datetimepicker.setLocale('en'); +}); diff --git a/web/nms.gathering.org/js/nms.js b/web/nms.gathering.org/js/nms.js index ace6e31..0aba266 100644 --- a/web/nms.gathering.org/js/nms.js +++ b/web/nms.gathering.org/js/nms.js @@ -249,12 +249,12 @@ function toggleNightMode()   *   * Should probably just use stringToEpoch() instead, but alas, not yet.   */ -function checkNow(now) +function parseNow(now)  {  	if (Date.parse(now)) {  		var d = new Date(Date.parse(now)); -		var str = d.getFullYear() + "-" + (parseInt(d.getMonth())+1) + "-" + d.getDate() + "T"; -		str += d.getHours() + ":" + d.getMinutes() + ":" + d.getSeconds(); +		var str = d.getFullYear() + "-" + ("00" + (parseInt(d.getMonth())+1)).slice(-2) + "-" + ("00" + d.getDate()).slice(-2) + "T"; +		str += ("00" + d.getHours()).slice(-2) + ":" + ("00" + d.getMinutes()).slice(-2) + ":" + ("00" + d.getSeconds()).slice(-2);  		return str;  	} @@ -348,22 +348,29 @@ function startReplay() {  }  /* - * Used to move to a specific time, but not replay. + * Used to move to a specific time   */ -function changeNow() { -	 var newnow = checkNow(document.getElementById("nowPicker").value); -	if (!newnow) { -		alert('Bad date-field in time travel field'); -		return; -	} +function changeNow(newnow,playing) {  	if (newnow == "")  		newnow = false; +  if (playing == "") +    playing = false; + +  if(newnow)  +    newnow = parseNow(newnow); +  nms.timers.replay.stop();  	nms.now = newnow; -	if (newnow) { -		nms.timers.ping.stop();; -		nms.timers.ports.stop();; -	} +	resetColors(); + +	if (!playing) { +		nms.timers.ping.stop(); +		nms.timers.ports.stop(); +	} else { +    nms.timers.ping.start(); +    nms.timers.ports.start(); +  } +  	updatePorts();  	updatePing();  	var boxHide = document.getElementById("nowPickerBox"); @@ -1752,3 +1759,27 @@ function forgetSettings()  {  	document.cookie = 'nms=' + btoa('{}');  } + +/* + * Time travel gui + */ +function startNowPicker(now) { +  $('#nowPicker').datetimepicker('destroy'); +  $('#nowPicker').datetimepicker({ +    value: now, +    mask:false, +    inline:true, +    todayButton: false, +    validateOnBlur:false, +    maxDate:'+1970/01/01', +    onSelectDate: function(ct,$i){ +      document.getElementById('nowPicker').dataset.iso = parseNow(ct); +    }, +    onSelectTime: function(ct,$i){ +      document.getElementById('nowPicker').dataset.iso = parseNow(ct); +    }, +    onGenerate: function(ct,$i){ +      document.getElementById('nowPicker').dataset.iso = parseNow(ct); +    } +  }); +} | 
