PHP 8.2.31
Preview: AdminTourBookingTab.js Size: 12.46 KB
/home/nshryvcy/himaltourism.com/wp-content/themes/adventure-tours/assets/admin/AdminTourBookingTab.js

/**
 * Namespace for the tour booking periods management widget processing.
 *
 * @type Object
 *
 * Version: 4.2.1
 */
var AdminTourBookingTab = {
	dateSelectorsFormat:"yy-mm-dd",

	dateSelectorsResetAttributes:{
		"name":"",
		"value":"",
		"placeholder":"YYYY-MM-DD",
		"autocomplete":"off"
	},

	init:function(){
		var self = this;

		this._errorsRenderer = new ThemeTools.ErrorsRendererSet({
			containerSelector:'#tour_booking_rows_cont',
			itemSelector:'.tour-booking-row',
			rowRendererConfig:{
				errorWraper:{
					className: 'field-error-msg'
				},
				_getErrorWraper:function( fieldKey, onlyIfExists ){
					var field = this._getField( fieldKey ),
						errorWrapperCont = field.parent();

					if ( 'days' == fieldKey) {
						errorWrapperCont = field.parents('.tour-booking-row__days');
					}

					var errorWraperEl = errorWrapperCont.find( '.' + this.errorWraper.className );
					if ( errorWraperEl.length < 1 && !onlyIfExists ) {
						var errorWraperTypeOutput = this.errorWraper.typeOutput,
							errorWraperTemplateHtml = '<' + this.errorWraper.tag + ' class="' + this.errorWraper.className + '"></' + this.errorWraper.tag + '>';

						errorWraperEl = jQuery( errorWraperTemplateHtml ).appendTo( errorWrapperCont );
					}

					return errorWraperEl;
				}
			}
		});

		this.getAddButton().click(function(event){
			event.preventDefault();
			self.addRow();
		});

		this.getSaveButton().click(function(event){
			event.preventDefault();
			self.saveRows();
		});

		this.getPreviewCalendarButton().click(function(event){
			event.preventDefault();
			this.is_on = !this.is_on;

			var btn = jQuery(this),
				alt_text = btn.data('togletext');
			if (alt_text) {
				var origin_text = btn.data('origintext');
				if ( ! origin_text && this.is_on ) {
					origin_text = btn.text();
					btn.data('origintext', origin_text);
				}
				if (origin_text) {
					btn.text(this.is_on ? alt_text : origin_text);
				}
			}

			self.previewCalendar(this.is_on);
		});

		this.getRows().each(function(){
			self._initRow(jQuery(this));
		});

		this.getCont().on('change', 'input,select,textarea', function(){
			jQuery(self).trigger('change');
		});
	},

	/**
	 * Adds new row into management form.
	 */
	addRow:function(){
		var btn = this.getAddButton();

		var newRow = jQuery(btn.data('row'));
		newRow.appendTo(this.getRowsCont());
		this._initRow(newRow);
		this._reworkFieldIndexes();
	},

	saveRows:function(){
		var self = this,
			render_errors = function(response) {
				if ( response && response.errors ) {
					self._errorsRenderer.render( response.errors );
				} else {
					alert( 'Unknown error. Please contact support.' );
				}
			};

		jQuery.ajax({
			url:ajaxurl + '?action=save_tour_booking_periods',
			method:'POST',
			data:this._getPeriodsQueryString(),
			dataType:'json',
			success:function(response){
				if (response && response.success) {
					// changes have been saved
					self._errorsRenderer.render();
				} else {
					render_errors(response);
				}
			},
			error:function(response){
				render_errors(response);
			}
		});
	},

	_getPeriodsQueryString:function(){
		var inputs = this.getCont().find('input,select,textarea'), //inputs = this.getRows().find('input,select,textarea'),
			dataRow = inputs.serialize();
		return dataRow;
	},

	/**
	 * Show/hide preview calendar.
	 *
	 * @param  boolean show
	 * @return void
	 */
	previewCalendar:function(show){
		if ( ! show ) {
			if ( this._dp) {
				this._dp.hide();
			}
			return;
		}

		if ( ! this._dp) {
			this._loadAvailableTickets();
			var self = this;
			this._dp = jQuery('<div class="tour-booking-preview-calendar"></div>').datepicker({
				dateFormat:'yy-mm-dd',
				changeMonth: true,
				changeYear: true,
				nextText:'',
				prevText:'',
				// firstDay: 1, //to start from Mon
				beforeShowDay: function(date){
					var tickets = self._previewGetAvailableTickets(date);
					return [tickets > 0, '', tickets > 0 ? tickets + ' ticket(s) left' : ''];
				}
			}).insertBefore(this.getPreviewCalendarButton());

			jQuery(this).on('change', function(){
				self._redrawPreviewCalendar();
			});
		} else {
			this._loadAvailableTickets();
			this._dp.datepicker('refresh');
			if (!this._dp.is(':visible')) {
				this._dp.show();
			}
		}
	},

	/**
	 * Removes particular row.
	 *
	 * @param  jQuery row
	 * @return void
	 */
	removeRow:function(row){
		//var row = this.getRows().filter(jQuery(btn).parents('tr'));
		if (row.length) {
			row.remove();
			this._reworkFieldIndexes();
		}
	},

	/**
	 * Returns button that should be used for new row creation.
	 *
	 * @return jQuery
	 */
	getAddButton:function(){
		return this.getCont().find('.add_row_btn');
	},

	/**
	 * Returns button that should be used saving information about tour booking periods.
	 *
	 * @return jQuery
	 */
	getSaveButton:function(){
		return this.getCont().find('.save_ranges_btn');
	},

	/**
	 * Returns button that should be used to show/hide calendar with available tickets information.
	 *
	 * @return jQuery
	 */
	getPreviewCalendarButton:function(){
		return this.getCont().find('.preview_btn');
	},

	/**
	 * Returns set of the row each of that contains fields for single period details management.
	 *
	 * @return jQuery
	 */
	getRows:function(){
		return this.getRowsCont().find('tr');
	},

	/**
	 * Returns rows container.
	 *
	 * @return jQuery
	 */
	getRowsCont:function(){
		return this.getCont().find('#tour_booking_rows_cont');
	},

	/**
	 * Returns global widget container.
	 *
	 * @return jQuery
	 */
	getCont:function(){
		return jQuery('#tour_booking_tab');
	},

	/**
	 * Inits all handlers related on the row.
	 *
	 * @param  jQuery row
	 * @return jQuery
	 */
	_initRow:function(row){
		var self = this;
		row.find('[data-role=remove-row]').click(function(event){
			event.preventDefault();
			if ( confirm('Are you sure want to remove this item?') ) {
				self.removeRow(row);
			}
		});

		row.find('[name$="[mode]"]').on('change', function(){
			self.changeRowMode( row, jQuery(this).val() );
		}).trigger('change');

		row.find('.add_exact_date_btn').on('click', function(ev){
			ev.preventDefault();

			var btn = jQuery(this),
				newRow = jQuery(btn.parent().find('[data-role=exact-date-template]').html());
				newRow.insertBefore(btn);
			self._initExactDates(newRow);
		});

		row.find('.add_time_btn').on('click', function(ev){
			ev.preventDefault();

			var btn = jQuery(this),
				newRow = jQuery(btn.parent().find('[data-role=time-template]').html());
				newRow.insertBefore(btn);
			self._initTimeRows(newRow);
		});

		this._initExactDates(row);
		this._initTimeRows(row);

		var datepickers = row.find('.dateselector');
		if ( datepickers.length ) {
			var start = datepickers.filter('[name$="[from]"]'),
				end = datepickers.filter('[name$="[to]"]');

			this._buildDatePickerElement(start,{
				beforeShow:function(el, ev){
					var end_date = end.val();
					jQuery(el).datepicker('option', 'maxDate',end_date ? new Date(end_date) : null);
				}
			});
			this._buildDatePickerElement(end,{
				beforeShow:function(el, ev){
					var start_date = start.val();
					jQuery(el).datepicker('option', 'minDate', start_date ? new Date(start_date) : null);
				}
			});
		}

		return row;
	},

	_buildDatePickerElement:function(originField, customDatepickerCfg){
		if (!originField || originField.length < 1) {
			return jQuery([]);
		}

		if (this.dateSelectorsFormat != 'yy-mm-dd') {
			var pickerCfg = {
				dateFormat:this.dateSelectorsFormat,
				altField: originField,
				altFormat:'yy-mm-dd',
				changeMonth: true,
				changeYear: true,
				nextText:'',
				prevText:''
			};

			var dateFieldClone = originField.clone()
					.attr(this.dateSelectorsResetAttributes)
					.insertBefore(originField)
					.datepicker(customDatepickerCfg ? jQuery.extend(pickerCfg, customDatepickerCfg) : pickerCfg);

			if("" != originField.val()){
				dateFieldClone.datepicker("setDate", new Date(originField.val()));
			}

			originField.hide();
			return dateFieldClone;
		} else {
			var pickerCfg = {
				dateFormat:'yy-mm-dd',
				changeMonth: true,
				changeYear: true,
				nextText:'',
				prevText:''
			};
			return originField.datepicker(customDatepickerCfg ? jQuery.extend(pickerCfg, customDatepickerCfg) : pickerCfg);
		}
	},

	/**
	 * Updates row related field names with the right row index.
	 * Should be called after any changes in row set (add/remove/reorder).
	 *
	 * @return void
	 */
	_reworkFieldIndexes:function(){
		var self = this,
			rows = this.getRows();

		rows.each(function(index, el){
			self._setFieldIndexTo(jQuery(this), index);
		});

		// creates flag for post meta processing action that tour booking periods should be saved
		var cont = this.getCont(),
			save_flag_field_name = 'tour-booking-save-action',
			flag_field = cont.find('input[name="'+save_flag_field_name+'"]');
		if ( flag_field.length ) {
			if ( rows.length > 0 ) {
				flag_field.remove();
			}
		} else if ( rows.length < 1 ) {
			jQuery('<input />')
				.attr('name', save_flag_field_name)
				.attr('type','hidden')
				.val(1)
				.appendTo(cont);
		}

	},

	/**
	 * Set row related inputs index to some particular value.
	 *
	 * @param jQuery  row
	 * @param integer newIndex
	 */
	_setFieldIndexTo:function(row, newIndex){
		row.find('[name^="tour-booking-row["]').each(function(){
			var input = jQuery(this),
				newName = input.attr('name').replace(/\[\d+\]/, '['+newIndex+']');
			input.attr('name', newName);
		});

		row.find('script[data-role$="-template"]').each(function(){
			var template = jQuery(this);
			template.text( template.text().replace(/\[\d+\]/, '['+newIndex+']') );
		})
	},

	/**
	 * Refresh preview calendar.
	 *
	 * @return void
	 */
	_redrawPreviewCalendar:function(){
		if (this._dp && this._dp.is(':visible')) {
			this._loadAvailableTickets();
			this._dp.datepicker('refresh');
		}
	},

	/**
	 * Returns number of tickets available for specific date.
	 *
	 * @param  string date
	 * @return int
	 */
	_previewGetAvailableTickets:function(date){
		if ( ! this._previewAvailableDates ) {
			return 0;
		}
		var formattedDate = jQuery.datepicker.formatDate('yy-mm-dd', date);
		return this._previewAvailableDates[formattedDate] ? this._previewAvailableDates[formattedDate].all : 0;
	},

	/**
	 * Loads information about available dates for preview calendar.
	 *
	 * @return void
	 */
	_loadAvailableTickets:function(){
		jQuery.ajax({
			async:false,
			url:ajaxurl + '?action=preview_booking_periods',
			method:'POST',
			data:this._getPeriodsQueryString(),
			dataType:'json',
			success:function(response){
				if (response && response.success && response.data) {
					this._previewAvailableDates = this._convertAvailableDates( response.data );
				} else {
					this._previewAvailableDates = null;
				}
			},
			error:function(response){
				this._previewAvailableDates = null;
			},
			context:this
		});
	},

	changeRowMode:function(row, mode){
		if ( ! mode ) {
			mode = 'default';
		}
		var mode_containers = row.find('[data-mode-box]').each(function( el, index ){
			var box = jQuery(this);

			if ( box.data('mode-box') != mode ) {
				box.hide();
				box.find('select,input').prop('disabled',true);
			} else {
				box.show();
				box.find('select,input').prop('disabled',false);
			}
		});
	},

	_initExactDates:function(cont){
		this._buildDatePickerElement(cont.find('[data-role="datepicker"]'));

		cont.find('.remove_exact_date_btn').on('click',function(ev){
			ev.preventDefault();
			jQuery(this).parents('[data-role=row]').remove();
		});
	},

	_initTimeRows:function(cont){
		cont.find('.remove_time_btn').on('click',function(ev){
			ev.preventDefault();
			jQuery(this).parents('[data-role=row]').remove();
		});
	},

	_convertAvailableDates:function( unconverted ) {
		var r = {};
		if ( ! unconverted || jQuery.isEmptyObject( unconverted ) ) {
			return unconverted;
		}

		var def_time = '00:00';
		for( var full_date in unconverted ) {
			var _val = parseInt( unconverted[ full_date ], 10 ),
				_cur_date = new Date( full_date ),
				_formatted = jQuery.datepicker.formatDate('yy-mm-dd', _cur_date),
				_time_formatted = _formatted == full_date 
					? def_time
					: ('0' + _cur_date.getHours() ).slice(-2) + ':' + ('0' + _cur_date.getMinutes()).slice(-2);

			if ( ! r[ _formatted ] ) {
				r[ _formatted ] = {
					'all': 0,
					'times': {}
				};
			}

			r[ _formatted ]['all'] += _val;
			if ( r[ _formatted ].times[_time_formatted] ) {
				r[ _formatted ].times[_time_formatted] += _val;
			} else {
				r[ _formatted ].times[_time_formatted] = _val;
			}
		}

		for(var d in r){
			if (r[d].times[def_time] == r[d].all ) {
				delete(r[d].times[def_time]);
			}
		}

		return r;
	}
};

jQuery(function(){
	AdminTourBookingTab.init();
});

Directory Contents

Dirs: 0 × Files: 5

Name Size Perms Modified Actions
12.46 KB lrw-r--r-- 2022-11-10 06:13:26
Edit Download
1.45 KB lrw-r--r-- 2022-11-10 06:13:26
Edit Download
1.41 KB lrw-r--r-- 2022-11-10 06:13:26
Edit Download
5.00 KB lrw-r--r-- 2022-11-10 06:13:26
Edit Download
2.23 KB lrw-r--r-- 2022-11-10 06:13:26
Edit Download

If ZipArchive is unavailable, a .tar will be created (no compression).