HEX
Server: nginx/1.18.0
System: Linux iZj6c1ieg2jrpk1z5tzi19Z 6.3.9-1.el7.elrepo.x86_64 #1 SMP PREEMPT_DYNAMIC Wed Jun 21 22:18:40 EDT 2023 x86_64
User: www (1001)
PHP: 8.2.4
Disabled: passthru,exec,system,putenv,chroot,chgrp,chown,shell_exec,popen,proc_open,pcntl_exec,ini_alter,ini_restore,dl,openlog,syslog,readlink,symlink,popepassthru,pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,imap_open,apache_setenv
Upload Files
File: /www/wwwroot/www.cytocare.cn/wp-content/plugins/woocommerce-advanced-bulk-edit/js/slick.editors.js
/***
 * Contains basic SlickGrid editors.
 * @module Editors
 * @namespace Slick
 */

(function ($) {
	// register namespace
	$.extend(true, window, {
		"Slick": {
			"Editors": {
				"Text": TextEditor,
				"Integer": IntegerEditor,
				"Date": DateEditor,
				"YesNoSelect": YesNoSelectEditor,
				"CustomDate": customDateEditor,
				"Checkbox": CheckboxEditor,
				"PercentComplete": PercentCompleteEditor,
				"LongText": LongTextEditor,
				"TextArea": TextAreaEditor,
				"WordPress": WordPressEditor,
				"Image": ImageEditor,
				"Gallery": ImageGalleryEditor,
				"Select": SelectCellEditor,
				"Category": CategoryEditor,
				"DefAttrs": DefAttrsEditor,
				"CustAttrs": CustAttrsEditor,
				"DateTime": DateTimeEditor,
				"SyncWith": SyncWithEditor,
				"AutocompleteProductSelector": AutocompleteProductSelectorEditor,
				"SerializedToCSV":SerializedToCSVEditor
			}
		}
	});

	function escapeRegExp(string) {
		return string.replace(/([.*+?^=!:${}()|\[\]\/\\])/g, "\\$1");
	}

	function replaceAll(string, find, replace) {
		return string.replace(new RegExp(escapeRegExp(find), 'g'), replace);
	}

	function ImageEditor(args) {
		var $input, $wrapper,$img,$aremove,$aset;
		var defaultValue;
		var scope = this;
		var file_frame;
		var set_to_post_id = -1;
		var attachment_id = -1;
		var bapplied = false;
		var W3Ex = window.W3Ex || {};
		W3Ex.imageeditor = true;
		this.init = function () {
			$('#imagepopup').remove();
			if(W3Ex._global_settings !== undefined && W3Ex._global_settings['openimage'] === true) {
				if( args.item['_thumbnail_id_val'] !== undefined && args.item['_thumbnail_id_val'] !== '') {
					attachment_id = parseInt(args.item['_thumbnail_id']);
					set_to_post_id =  parseInt(args.item['ID']);
				} else {
					attachment_id = -1;
					set_to_post_id = -1;
			 	}
				this.handleClick();
				return;
			}
			var $container = $('body');

			$wrapper = $('<DIV id="w3-imagewrapper" style="z-index:10000;position:absolute;background:white;padding:15px;padding-bottom:10px;border:3px solid gray; -moz-border-radius:5px; border-radius:5px;max-height: 300px;overflow: auto;"/>')
				.appendTo($container);
			$('<div id="w3exabe-note">Note! - image changes are saved immediately.</div>')
				.appendTo($wrapper);
			$input = $('<div class="imageholder"></div>')
				.appendTo($wrapper);
			$('<div class="imageinfo"></div>')
				.appendTo($wrapper);
			$img = $('<img src="" width="150px" height="150px" style="cursor:pointer;">').appendTo($input);
			$aset =  $('<a href="#" class="setimage" style="position:absolute;left:45px;top:100px;">Set product image</a>')
				.appendTo($input);

			$aremove = $('<a href="#" class="removeimage">Remove product image</a>')
				.appendTo($wrapper);
			var data = args.grid.getData();
			if(args.grid.getActiveCell() !== null) {
				var iRow = args.grid.getActiveCell().row;
				var selectedRows = args.grid.getSelectedRows();
				if(selectedRows.length > 0) {
					for(var irow=0; irow < selectedRows.length; irow++) {
						var rowid = selectedRows[irow];
						if(rowid === undefined) continue;
						if(data[rowid] === undefined) continue;
						if(rowid === iRow) {
							//clicked on selected
							$('<br/><br/><label><input id="ifapplyaction" type="checkbox">Apply action to all<br/>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; selected products</label>').appendTo($wrapper);
							break;
						}
					}
				}
			}

			$('<DIV style="text-align:right"><BUTTON>Close</BUTTON></DIV>')
				.appendTo($wrapper);

			$wrapper.find('button:first').bind('click', this.cancel);
			$aremove.bind('click', this.removeImage);
			$aset.find('a .setimage').bind('click', this.setImage);
			$input.bind('keydown', this.handleKeyDown);
			$input.bind('click', this.handleClick);
			scope.position(args.position);
			$input.focus().select();
		};

		this.setImage = function (e) {};

		this.removeImage = function (e) {
			try {
				e.preventDefault();
			} catch(err) {
				;
			}

			var prodids = '';
			if($('#ifapplyaction').length > 0) {
				if($('#ifapplyaction').is(':checked')) {
					var data = args.grid.getData();
					if(args.grid.getActiveCell() !== null) {
						var iRow = args.grid.getActiveCell().row;
						var selectedRows = args.grid.getSelectedRows();
						if(selectedRows.length > 0) {
							for(var irow=0; irow < selectedRows.length; irow++) {
								var rowid = selectedRows[irow];
								if(rowid === undefined) {
									continue;
								}
								if(data[rowid] === undefined) {
									continue;
								}
								data[rowid]._thumbnail_id = '';
								data[rowid]._thumbnail_id_val = '';
								if(prodids == "") {
									prodids = data[rowid].ID.toString() + ',';
								} else {
									prodids+= ',' + data[rowid].ID.toString();
								}
							}
						}
					}
				}
			}

			var ajaxarr = {};
			ajaxarr.action = 'wpmelon_adv_bulk_edit';
			ajaxarr.type = 'removethumb';
			ajaxarr.nonce = window.W3ExABE.nonce;
			var _arrData = [];
			if(prodids !== '') {
				_arrData[0] = prodids;
				bapplied = true;
			} else {
				_arrData[0] = args.item.ID;
			}
			ajaxarr.data = _arrData;
			$wrapper.append('<div class="showajax"></div>');
			$('.showajax').css({
				left:'200px',
				top:'100px'
			});
			jQuery.ajax({
				type : 'post',
				dataType : 'json',
				url : W3ExABE.ajaxurl,
				data : ajaxarr,
				success: function(response) {
					$wrapper.find('.imageinfo').html('');
					$img.css('display','none');
					$aremove.css('display','none');
					$aset.css('display','inline');
					set_to_post_id = -1;
					attachment_id = -1;
					args.item._thumbnail_id = '';
					args.item._thumbnail_id_val = '';
					var W3Ex = window.W3Ex || {};
					W3Ex._global_settings.invalidateimages = true;
					W3Ex._w3abe_refreshlibrary = true;
				},
				complete:function (args) {
					$('.showajax').remove();
				},
				async : false
			}) ;
		};

		this.handleClick = function (e) {
			var editor = this;
			var W3Ex = window.W3Ex || {};
			if ( file_frame ) {
				// Set the post ID to what we want
				if(set_to_post_id !== -1) {
					file_frame.uploader.uploader.param( 'post_id', set_to_post_id );
				}

				// Open frame
				file_frame.open();
				return;
			} else {
				// Set the wp.media post id so the uploader grabs the ID we want when initialised
				if(set_to_post_id !== -1) {
					wp.media.model.settings.post.id = set_to_post_id;
				}
			}

			// Create the media frame.
			file_frame = wp.media.frames.file_frame = wp.media({
				title: 'Set product image',
				button: {
					text:  'Set product image',
				},
				states : [
					new wp.media.controller.Library({
						library:   wp.media.query({ type: 'image' }),
						title: 'Set product image',
						filterable :	'uploaded',
						multiple: false
					})
				] ,
				attachment:true,
				multiple: false // Set to true to allow multiple files to be selected
			});
			file_frame.on( 'close', function() {
				var W3Ex = window.W3Ex || {};
				if(W3Ex._global_settings['openimage'] === true) {
					W3Ex.bdontgodown = true;
					editor.cancel();
				}
			});
			// When an image is selected, run a callback.
			file_frame.on( 'select', function() {
				// We set multiple to false so only get one image from the uploader
				attachment = file_frame.state().get('selection').first().toJSON();
				attachment_id = attachment.id;
				set_to_post_id = args.item.ID;
				args.item._thumbnail_id = attachment.id;
				args.item._thumbnail_id_val = attachment.url;
				args.item._thumbnail_id_original = attachment.url;
				//			var imageinfo = "";
				if(W3Ex.uploaddir !== undefined) {
					args.item._thumbnail_id_info = '<div class="fileinfo">'+attachment.url.replace(W3Ex.uploaddir+'/','')+'</div>';
				} else {
					args.item._thumbnail_id_info = "<div class='fileinfo'>"+attachment.url+"</div>";
				}
				if(attachment.width !== undefined && attachment.height) {
					args.item._thumbnail_id_info+= '<div class="dims">'+attachment.width+' x '+attachment.height+'</div>';
				}
				var ajaxarr = {};
				ajaxarr.action = 'wpmelon_adv_bulk_edit';
				ajaxarr.type = 'setthumb';
				ajaxarr.nonce = window.W3ExABE.nonce;
				var _arrData = [];
				var prodids = '';
				if($('#ifapplyaction').length > 0) {
					if($('#ifapplyaction').is(':checked')) {
						var data = args.grid.getData();
						if(args.grid.getActiveCell() !== null) {
							var iRow = args.grid.getActiveCell().row;
							var selectedRows = args.grid.getSelectedRows();
							if(selectedRows.length > 0) {
								for(var irow=0; irow < selectedRows.length; irow++) {
									var rowid = selectedRows[irow];
									if(rowid === undefined) {
										continue;
									}
									if(data[rowid] === undefined) {
										continue;
									}
									data[rowid]._thumbnail_id = attachment.id;
									data[rowid]._thumbnail_id_val = attachment.url;
									data[rowid]._thumbnail_id_original = attachment.url;
									if(prodids == '') {
										prodids = data[rowid].ID.toString() + ',';
									} else {
										prodids+= ',' + data[rowid].ID.toString();
									}
								}
							}
						}
					}
				}
				if(prodids !== '') {
					_arrData[0] = prodids;
					bapplied = true;
				} else {
					_arrData[0] = args.item.ID;
				}
				_arrData[1] = attachment.id;
				ajaxarr.data = _arrData;

				if(W3Ex._global_settings['openimage'] !== true) {
					$wrapper.append('<div class="showajax"></div>');
					$('.showajax').css({
						left:'200px',
						top:'100px'
					});
				}
				jQuery.ajax({
					type : 'post',
					dataType : 'json',
					url : W3ExABE.ajaxurl,
					data : ajaxarr,
					success: function(response) {
						var W3Ex = window.W3Ex || {};
						W3Ex._global_settings.invalidateimages = true;
						if(W3Ex._global_settings['openimage'] !== true) {
							$img.css('display','inline');
							$img.attr('src', attachment.url);
							$wrapper.find('.imageinfo').html(args.item._thumbnail_id_info);
							$aset.css('display','none');
							$aremove.css('display','inline');
						} else {
							W3Ex.bdontgodown = true;
							W3Ex.invalidateselected = true;
							editor.cancel();
						}

					},
					complete:function (args) {
						$('.showajax').remove();
					},
					async : false
				}) ;


				// Do something with attachment.id and/or attachment.url here
				// Restore the main post ID
				// alert('asd');
				// wp.media.model.settings.post.id = wp_media_post_id;
			});
			file_frame.on('open', function () {
				var attachment,
					selection = file_frame.state().get('selection');
				if(attachment_id !== -1) {
					attachment = wp.media.attachment(attachment_id);
					attachment.fetch();
					selection.add(attachment ? [ attachment ] : []);
				}
			});
			// Finally, open the modal
			file_frame.open();
		};


		this.handleKeyDown = function (e) {
			if (e.which == $.ui.keyCode.ENTER && e.ctrlKey) {
				scope.save();
			} else if (e.which == $.ui.keyCode.ESCAPE) {
				e.preventDefault();
				scope.cancel();
			} else if (e.which == $.ui.keyCode.TAB && e.shiftKey) {
				e.preventDefault();
				args.grid.navigatePrev();
			} else if (e.which == $.ui.keyCode.TAB) {
				e.preventDefault();
				args.grid.navigateNext();
			}
		};


		this.save = function () {
			var W3Ex = window.W3Ex || {};
			W3Ex.bdontgodown = true;
		};

		this.cancel = function () {
			args.cancelChanges();
		};

		this.hide = function () {
			var W3Ex = window.W3Ex || {};
			if(W3Ex._global_settings['openimage'] !== true)
			{
				$wrapper.hide();
			}
		};

		this.show = function () {
			var W3Ex = window.W3Ex || {};
			if(W3Ex._global_settings['openimage'] !== true)
			{
				$wrapper.show();
			}
		};

		this.position = function (position) {
			if(W3Ex._global_settings['openimage'] !== true)
			{
				$wrapper
					.css("top", position.top - 170)
					.css("left", position.left - 185)
			}
		};


		this.destroy = function () {
			var W3Ex = window.W3Ex || {};
			if(W3Ex._global_settings['openimage'] !== true) {
				$wrapper.remove();
			} else {
				W3Ex.invalidateselected = true;
			}

			W3Ex.imageeditor = false;

			if(bapplied) {
				W3Ex.invalidateselected = true;
			}
		};

		this.focus = function () {
			// $input.focus();
		};


		this.loadValue = function (item) {
			var W3Ex = window.W3Ex || {};
			if( item['_thumbnail_id_val'] !== undefined && item['_thumbnail_id_val'] !== '') {
				if(W3Ex._global_settings['openimage'] !== true) {
					$input.find('img').attr("src", item['_thumbnail_id_val']);
					$aset.css('display','none');
					if(item['_thumbnail_id_info'] !== undefined) {
						$wrapper.find('.imageinfo').append(item['_thumbnail_id_info']);
					}
				}

				attachment_id = parseInt(item['_thumbnail_id']);
				set_to_post_id =  parseInt(item['ID']);
			} else {
				if(W3Ex._global_settings['openimage'] !== true) {
					$img.css('display','none');
					$aremove.css('display','none');
				}
				attachment_id = -1;
				set_to_post_id = -1;
			}
		};

		this.serializeValue = function () {
			//return $input.val();
		};

		this.applyValue = function (item, state) {
			// if(bapplied)
			// {
			// 	var data = args.grid.getData();
			// 	var selectedRows = args.grid.getSelectedRows();
			// 	if(selectedRows.length > 0)
			// 	{
			// 		for(var irow=0; irow < selectedRows.length; irow++)
			// 		{
			// 			var rowid = selectedRows[irow];
			// 			if(rowid === undefined) continue;
			// 			if(data[rowid] === undefined) continue;
			// 			data[rowid][args.column.field] = state;
			// 		}
			// 	}
			// }else
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return false;
			//return (!($input.val() == "" && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function CustAttrsEditor(args) {
		var $input,$input_val, $wrapper, $table;
		var input_val;
		var defaultValue;
		var scope = this;
		var newtrvar;
		var bonlyvisible = false;
		var arrcust = [];
		var W3Ex = window.W3Ex;
		this.init = function () {
			var $container = $("body");
			if(args.item.post_type !== "product_variation") {
				$wrapper = $('<DIV style="z-index:10000;position:absolute;background:white;padding:15px;border:3px solid gray; -moz-border-radius:10px; border-radius:10px;width:400px;"/>')
					.appendTo($container);
			} else {
				$wrapper = $('<DIV style="z-index:10000;position:absolute;background:white;padding:15px;border:3px solid gray; -moz-border-radius:10px; border-radius:10px;width:300px;"/>')
					.appendTo($container);
			}

			$input = $('<TEXTAREA hidefocus rows=1 style="backround:white;width:0px;height:0px;border:0;outline:0;display:none;">')
				.appendTo($wrapper);
			newtrvar = '<tr> \
							<td> \
							Name:<input style="width:80px;" class="attrname" placeholder="Name" name="attrname" value="" type="text">\
							</td>\
							<td> \
							Value(s):<input style="width:190px;" class="attrvalue" placeholder="Value(s)" name="attrvalue" value="" type="text">\
							</td>\
							<td style="width:10%;"> \
							<a href="#" class="deleteattr">Del</a>\
							</td>\
						</tr> \
						<tr>\
							<td colspan="3"><label style="white-space: nowrap;"><input class="ifvisibleproductpage" type="checkbox">Visible on the product page</label>\
							<label style="white-space: nowrap;"><input class="ifusedforvars" type="checkbox">Used for variations</label>\
							</td>\
						</tr>';
			var newtrsimple = '<tr> \
									<td> \
									Name:<input style="width:80px;" class="attrname" placeholder="Name" name="attrname" value="" type="text">\
									</td>\
									<td> \
									Value(s):<input style="width:190px;" class="attrvalue" placeholder="Value(s)" name="attrvalue" value="" type="text">\
									</td>\
									<td style="width:10%;"> \
									<a href="#" class="deleteattr">Del</a>\
									</td>\
								</tr> \
								<tr>\
									<td colspan="3"><label style="white-space: nowrap;"><input class="ifvisibleproductpage" type="checkbox">Visible on the product page</label>\
									</td>\
								</tr>';
			$table = $('<div style="max-height:250px;overflow:auto;"><table id="tablecustattributes"></table></div>')
				.appendTo($wrapper);

			var selitem = args.item;
			if(selitem._custom_attributes === undefined) {
				selitem._custom_attributes = [];
			}
			var addsel = "";
			if(selitem.post_type === "product_variation") {
				var parent = {};
				if(W3Ex._w3ex_active_parent !== undefined) {
					parent = W3Ex._w3ex_active_parent;
				}
				W3Ex._w3ex_map_attributes = W3Ex._w3ex_map_attributes || {};
				if(parent._custom_attributes !== undefined) {
					for (var key in parent._custom_attributes) {
						if (parent._custom_attributes.hasOwnProperty(key)) {
							var attr = parent._custom_attributes[key];
							var attrslug = W3Ex._w3ex_map_attributes[attr.name];
							if(attrslug === undefined) {
								attrslug = "";
							}
							addsel+= '<tr><td>'+attr.name+'</td><td><select data-slug="'+attrslug+'" data_type="customattr"><option value="" selected="selected">Any</option>';
							var values = attr.value.split(W3Ex._w3ex_wc_delimiter);
							for(var i = 0; i < values.length; i++) {
								var value = $.trim(values[i]);
								var valueslug = W3Ex._w3ex_map_attributes[value];
								if(valueslug === undefined) {
									valueslug = "";
								}
								addsel+= '<option value="'+valueslug+'">'+value+'</option>';
							}
							addsel+= "</select></td></tr>";
						}
					}
				}

			} else {
				$table = $table.find('table');
				for (var key in selitem._custom_attributes) {
					if (selitem._custom_attributes.hasOwnProperty(key)) {
						var attr = selitem._custom_attributes[key];
						if(selitem.product_type === 'simple') {
							$table.append(newtrsimple);
							if(attr.is_visible == 1) {
								$table.find('tr').filter(':last').find('.ifvisibleproductpage').prop('checked',true);
							}
							$table.find('tr').filter(':last').prev('tr').find('.attrname').val(attr.name).end().find('.attrvalue').val(attr.value);
						} else {
							$table.append(newtrvar);
							if(attr.is_visible == 1) {
								$table.find('tr').filter(':last').find('.ifvisibleproductpage').prop('checked',true);
							}
							if(attr.is_variation == 1) {
								$table.find('tr').filter(':last').find('.ifusedforvars').prop('checked',true);
							}
							$table.find('tr').filter(':last').prev('tr').find('.attrname').val(attr.name).end().find('.attrvalue').val(attr.value);
						}
					}
				}
			}
			if(args.item.post_type !== "product_variation") {
				$('<a href="#" class="button insert" style="margin-left:10px;margin-top:10px;">Add Attribute</a>')
					.appendTo($wrapper);
				$wrapper.find("a.insert").bind("click",function(e) {
					if(args.item.product_type === 'simple') {
						$table.append(newtrsimple);
					} else {
						$table.append(newtrvar);
					}
					e.preventDefault();
				});
			} else {
				if(addsel !== "") {
					addsel = '<table id="defattrsholder">' + addsel + '</table>';
					$table.append(addsel);
				}
			}

			$('<DIV style="text-align:right"><BUTTON>Save</BUTTON><BUTTON>Cancel</BUTTON></DIV>')
				.appendTo($wrapper);
			$wrapper.find("button:first").bind("click", this.save);
			$wrapper.find("button:last").bind("click", this.cancel);
			jQuery(document).on( 'click', '.choosefile',this.loadFile);
			jQuery(document).on( 'click', '.deleteattr',this.deleteAttr);
			$input.bind("keydown", this.handleKeyDown);
			scope.position(args.position);
			$input.focus().select();
		};

		this.save = function () {
			var W3Ex = window.W3Ex || {};
			W3Ex.bdontgodown = true;
			args.commitChanges();
		};

		this.loadValue = function (item) {
			if(item.post_type !== 'product_variation') {
				return;
			}
			$('#defattrsholder select').each(function() {
				var $select = $(this);
				var attrslug = $select.attr('data-slug');
				for (var key in item._custom_attributes) {
					if (item._custom_attributes.hasOwnProperty(key)) {
						var attr = item._custom_attributes[key];
						if(attr.attslug === attrslug) {
							$select.val(attr.slug);
							break;
						}
					}
				}
			})
		};

		this.handleKeyDown = function (e) {
			if (e.which == $.ui.keyCode.ENTER && e.ctrlKey) {
				scope.save();
			} else if (e.which == $.ui.keyCode.ESCAPE) {
				e.preventDefault();
				scope.cancel();
			} else if (e.which == $.ui.keyCode.TAB && e.shiftKey) {
				//e.preventDefault();
				//args.grid.navigatePrev();
			} else if (e.which == $.ui.keyCode.TAB) {
				//e.preventDefault();
				//args.grid.navigateNext();
			}
		};

		this.loadFile = function(event) {
			//$(this).closest('tr').find('td .filepath').val('asd');
			//$(this).closest('tr').find('td .filename').val('qwe');
		};

		this.deleteAttr = function(event) {
			var $el = $(this);
			$el.closest('tr').next('tr').remove().end().remove();
			//var $closesttr = $el.closest('tr');
			//$closesttr.next('tr').remove();
			//$closesttr.remove();
			event.preventDefault();
		};

		this.cancel = function () {
			$input.val(defaultValue);
			args.cancelChanges();
		};

		this.hide = function () {
		  $wrapper.hide();
		};

		this.show = function () {
		  $wrapper.show();
		};

		this.position = function (position) {
			$wrapper
				.css("top", position.top - 5)
				.css("left", position.left - 255);
		};

		this.destroy = function () {
			$wrapper.find("button:first").unbind();
			$wrapper.find("button:last").unbind();
			jQuery(document).off( 'click', '.choosefile');
			jQuery(document).off( 'click', '.deleteattr');
			$wrapper.find("a.insert").unbind();
			$wrapper.remove();
			var W3Ex = window.W3Ex || {};
			W3Ex.invalidateselected = true;
		};

		this.focus = function () {
			//$input.focus();
		};

		this.serializeValue = function () {
			return $input.val();
		};

		this.applyValue = function (item, state) {
			if(args.item.post_type === 'product_variation') {
				var parent = {};
				if(W3Ex._w3ex_active_parent !== undefined) {
					parent = W3Ex._w3ex_active_parent;
				}
				W3Ex._w3ex_map_attributes = W3Ex._w3ex_map_attributes || {};
				item._custom_attributes = [];
				$('#defattrsholder select').each(function() {
					var $select = $(this);
					var attrslug = $select.attr('data-slug');
					//if($select.val() !== "")
					if(parent._custom_attributes === undefined) {
						parent._custom_attributes = [];
					}
					for (var i=0; i < parent._custom_attributes.length; i++) {
						var attr = parent._custom_attributes[i];
						var pattrslug = W3Ex._w3ex_map_attributes[attr.name];
						if(pattrslug !== attrslug) {
							continue;
						}
						var attr1 = {};
						attr1.name = attr.name;
						attr1.attslug = pattrslug;
						attr1.slug = $select.val();
						if(attr1.slug === '') {
							attr1.value = '';
						} else {
							attr1.value = $select.find('option:selected').text();
						}
						item._custom_attributes.push(attr1);
					}
				})
				return;
			}

			args.item._custom_attributes = arrcust;
			var ajaxarr = {};
			ajaxarr.action = 'wpmelon_adv_bulk_edit';
			ajaxarr.type = 'getcustomslugs';
			ajaxarr.nonce = window.W3ExABE.nonce;
			ajaxarr.data = arrcust;
			$wrapper.find('button:first').css('position','relative');
			$wrapper.find('button:first').append('<div class="showajax"></div>');
			$('.showajax').css({
				left:'10px',
				top:'3px'
			});
			jQuery.ajax({
				type : "post",
				dataType : "json",
				url : W3ExABE.ajaxurl,
				data : ajaxarr,
				success: function(response) {
					if(response.products !== null && response.products !== undefined) {
						for (var key in response.products) {
							if (response.products.hasOwnProperty(key)) {
								var attr = response.products[key];
								W3Ex._w3ex_map_attributes[key] = attr;
							}
						}
					}
				},
				complete:function (args) {
					$('.showajax').remove();
				},
				async : false
			}) ;
		};

		this.isValueChanged = function () {
			if (args.item.post_type === 'product_variation') {
				var hasattr = false;
				$('#defattrsholder select').each(function() {
					var $select = $(this);
					var attrslug = $select.attr('data-slug');
					var selval = $select.val();
					hasattr = false;
					for (var key in args.item._custom_attributes) {
						if (args.item._custom_attributes.hasOwnProperty(key)) {
							var attr = args.item._custom_attributes[key];
							if (attr.attslug === attrslug) {
								if (attr.slug !== selval) {
									hasattr = false;
									break;
								}
								hasattr = true;
								break;
							}
						}
					}
					if(!hasattr) {
						return false;
					}
				});
				if(!hasattr) {
					return true;
				}
				return false;
			}

			var counter = 0;
			var custattr = {};
			$table.find('tr').each(function() {
				var $tr = $(this);
				var $elem = $tr.find('.attrname');
				if($elem.length > 0) {
					custattr.name = $elem.val();
					custattr.value = $tr.find('.attrvalue').val();
				}
				$elem = $tr.find('.ifvisibleproductpage');
				if($elem.length > 0) {
					custattr.is_visible = 0;

					if($elem.is(':checked')) {
						custattr.is_visible = 1;
					}

					custattr.is_variation = 0;
					$elem = $tr.find('.ifusedforvars');
					if($elem.length > 0) {
						if($elem.is(':checked')) {
							custattr.is_variation = 1;
						}
					}
				}
				counter++;
				if((counter % 2) == 0) {
					if(custattr.name !== "" && custattr.value !== "") {
						arrcust.push(custattr);
					}
					custattr = {};
				}
			})

			var arrcustselitem = args.item._custom_attributes;
			if(arrcustselitem.length !== arrcust.length) {
				return true;
			}
			for(var i = 0; i < arrcust.length; i++) {
				var custitem = arrcust[i];
				var hasit = false;
				var curitem = {};
				for(var j = 0; j < arrcustselitem.length; j++) {
					if(arrcustselitem[j].name === custitem.name) {
						curitem = arrcustselitem[j];
						hasit = true;
						break;
					}
				}
				if(!hasit) {
					return true;
				}

				if(curitem.value !== custitem.value) {
					return true;
				}
				if(curitem.is_visible !== custitem.is_visible) {
					return true;
				}
				if(args.item.product_type.toUpperCase().indexOf("VARIABLE") !== -1) {
					if(curitem.is_variation !== custitem.is_variation) {
						return true;
					}
				}
			}
			return false;
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function CategoryEditor(args) {
		var $input, $wrapper;
		var defaultValue;
		var scope = this;
		var catnames = '';
		var sel_ids = [];
		var ischecked = false;
		var oldvalue = 0;
		var bonlyvisible = false;
		var bsave = false;
		this.init = function () {
			var $container = $('body');
			$wrapper = $('<DIV style="z-index:10000;position:absolute;background:white;padding:25px;padding-top:5px;padding-bottom:12px;border:3px solid gray; -moz-border-radius:10px; border-radius:10px;min-width:150px;" class="activeeditordiv" />')
				.appendTo($container);
			$input = $('<div style="max-height:320px;min-height:120px;overflow:auto;" class="editorcats"></div>')
				.appendTo($wrapper);
			scope.position(args.position);
			if($('#categoriesdialog .' + args.column.field).length === 0) {
				var W3Ex = window.W3Ex || {};
				if(W3Ex[args.column.field + 'editorhtml'] === undefined) {
					var W3ExABE = window.W3ExABE || {};
					var ajaxarr = {};
					ajaxarr.action = 'wpmelon_adv_bulk_edit';
					ajaxarr.type = 'savecustom';
					ajaxarr.nonce = W3ExABE.nonce;
					ajaxarr.foreditor = 1;
					var $elem = $('.editorcats');
					$elem.css('position','relative').append('<div class="showajax"></div>');
					$('.showajax').css({
						left:'15px'
					});
					ajaxarr.data = args.column.field;
					jQuery.ajax({
						type : 'post',
						dataType : 'json',
						url : W3ExABE.ajaxurl,
						data : ajaxarr,
						success: function(response) {
							if( response !== undefined &&
								response !== null &&
								response.editortext !== undefined &&
								response.editortext !== null
							) {
								W3Ex[args.column.field + 'editorhtml'] = response.editortext;
								$input.html(response.editortext);
							}
							$('.showajax').remove();
						},
						error:function (xhr, status, error) {
							$('.showajax').remove();
						},
						async:false
					});
				} else {
					$input.html(W3Ex[args.column.field + 'editorhtml']);
				}
			} else {
				$input.html($('#categoriesdialog .' + args.column.field).html());
			}

			var selfield = args.column.field
			if(selfield == "product_shipping_class") {
				if(args.item[selfield + '_ids'] === undefined) {
					args.item[selfield  + '_ids'] = "";
				}
				if(args.item[selfield] === undefined) {
					args.item[selfield] = "";
				}
				if(args.item[selfield] === "") {
					$('#in-product_shipping_class-none').prop('checked',true);
				}
			}
			if(selfield.indexOf("attribute_pa_") === -1) {
				var parenttext = '<select class="selectnewcategory" class="makechosen" data-placeholder="select parent(optional)" multiple></select><br/>';
				if(selfield === 'product_tag') {
					parenttext = '';
				}
				var prependHtml = '<br/><button class="butnewattribute button newcatin" type="button" style="margin-top:10px !important;"><span class="icon-plus-outline"></span>new</button><div class="divnewattribute">' +
					'<input class="inputnewattributename" type="text" placeholder="name" data-slug="'+selfield+'"></input><br/>' +
					'<input class="inputnewattributeslug" type="text" placeholder="slug (optional)"></input><br/>' +   parenttext +
					'<button class="butnewattributesave newcat" style="position:relative;">Ok</button><button class="butnewattributecancel newcat">Cancel</button></div>' +
					'<div class="divnewattributeerror" style="margin-top: 10px !important;"></div>';
				if ( selfield && jQuery.inArray(selfield, ['product_type', 'product_shipping_class', 'post_author', 'grouped_items']) < 0 ) {
					prependHtml += '<div><a id="attrsCheckAll" href="#">All</a> <a id="attrsUncheckAll" href="#">None</a></div>';
				}
				$wrapper.prepend(prependHtml);
				if(selfield !== "product_tag") {
					$wrapper.find('select').chosen({max_selected_options:1});
				}
			}
			if(selfield.indexOf("attribute_pa_") !== -1) {
				if(args.item['post_type'] == 'product_variation') {
					$('.editorcats .categorychecklist').addClass('clearothersattr');
				} else {
					if(args.item[selfield + '_ids'] === undefined) {
						if(args.item[selfield + '_visiblefp'] !== undefined) {
							delete args.item[selfield + '_visiblefp'];
						}
					}
					if(args.item[selfield + '_visiblefp'] === undefined) {
						args.item[selfield + '_visiblefp'] = 0;
					}
					oldvalue = parseInt(args.item[selfield + '_visiblefp']);
					if(args.item[selfield + '_visiblefp'] & 1) {
						ischecked = true;
						$('<hr><label style="white-space: nowrap;"><input id="ifvisibleproductpage" type="checkbox" checked=checked>Visible on the product<br/>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; page</label><hr>')
							.appendTo($wrapper);
					} else {
						ischecked = false;
						$('<hr><label style="white-space: nowrap;"><input id="ifvisibleproductpage" type="checkbox">Visible on the product<br/>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; page</label><hr>')
							.appendTo($wrapper);
					}
					if(args.item['product_type'].toUpperCase().indexOf("VARIABLE") !== -1) {
						if(args.item[selfield + '_visiblefp'] & 2) {
							ischecked = true;
							$('<label style="white-space: nowrap;"><input id="ifusedforvars" type="checkbox" checked=checked>Used for variations</label><hr>')
								.appendTo($wrapper);
						} else {
							ischecked = false;
							$('<label style="white-space: nowrap;"><input id="ifusedforvars" type="checkbox">Used for variations</label><hr>')
								.appendTo($wrapper);
						}
					}
				}
				var prependHtml = '<br/><button class="butnewattribute button" type="button" style="margin-top:10px !important;"><span class="icon-plus-outline"></span>new</button><div class="divnewattribute">' +
					'<input class="inputnewattributename" type="text" placeholder="name" data-slug="'+selfield+'"></input><br/>' +
					'<input class="inputnewattributeslug" type="text" placeholder="slug (optional)"></input><br/>' +
					'<button class="butnewattributesave" style="position:relative;">Ok</button><button class="butnewattributecancel">Cancel</button></div>' +
					'<div class="divnewattributeerror" style="margin-top: 10px !important;"></div>';
				if(selfield.indexOf("attribute_pa_") !== -1) {
					prependHtml += '<div><a id="attrsCheckAll" href="#">All</a> <a id="attrsUncheckAll" href="#">None</a></div>';
				}
				$wrapper.prepend(prependHtml);
			}

			$wrapper.prepend('<input id="searchcheckboxes" type="text" style="width:100px;" placeholder="search"></input><BUTTON id="categoryshow">checked</BUTTON>');
			$("<DIV style='text-align:right'><BUTTON id='categorysave'>Save</BUTTON><BUTTON id='categorycancel'>Cancel</BUTTON></DIV>")
				.appendTo($wrapper);
			$wrapper.find("#categorysave").bind("click", this.save);
			$wrapper.find("#categorycancel").bind("click", this.cancel);
			$wrapper.bind("keydown", this.handleKeyDown);
			if(args.item[selfield + '_ids'] !== undefined) {
				var ids = args.item[selfield + '_ids'].split(',');
				$('.editorcats input').each(function() {
					var val = $(this).attr('value');
					if(ids.indexOf(val) >= 0) {
						$(this).prop('checked','checked');
					}
				})
			}

			$('#searchcheckboxes').focus();
		};

		this.handleKeyDown = function (e) {
			if (e.which == $.ui.keyCode.ENTER && e.ctrlKey) {
				scope.save();
			} else if (e.which == $.ui.keyCode.ESCAPE) {
				e.preventDefault();
				scope.cancel();
			} else if (e.which == $.ui.keyCode.TAB && e.shiftKey) {
				//e.preventDefault();
				//args.grid.navigatePrev();
			} else if (e.which == $.ui.keyCode.TAB) {
				//e.preventDefault();
				//args.grid.navigateNext();
			}
		};

		this.save = function () {
			var W3Ex = window.W3Ex || {};
			W3Ex.bdontgodown = true;
			bsave = true;
			args.commitChanges();
		};

		this.cancel = function () {
			//$input.val(defaultValue);
			args.cancelChanges();
		};

		this.hide = function () {
			$wrapper.hide();
		};

		this.show = function () {
			$wrapper.show();
		};

		this.position = function (position) {
			$wrapper
				.css("top", position.top - 85)
				.css("left", position.left - 75)
			};

		this.destroy = function () {
			$wrapper.remove();
			var W3Ex = window.W3Ex || {};
			W3Ex.invalidateselected = true;
		};

		this.focus = function () {
			//$input.focus();
		};

		this.loadValue = function (item) {
			//$input.val(defaultValue = item[args.column.field]);
		};

		this.serializeValue = function () {
			return '';
		};

		this.applyValue = function (item, state) {
			if(!bsave) {
				return;
			}
			if(!bonlyvisible) {
				item[args.column.field] = catnames;
				item[args.column.field + '_ids'] = sel_ids.join();
			}
			if(args.column.field === 'product_shipping_class') {
				if(args.item[args.column.field + '_ids'] === '') {
					args.item[args.column.field] = '';
				}
			}

			if(
				args.item['post_type'] == 'product' &&
				args.column.field.indexOf("attribute_pa_") !== -1 &&
				$('#ifvisibleproductpage').length > 0
			) {
				if(args.item[args.column.field + '_visiblefp'] === undefined) {
					args.item[args.column.field + '_visiblefp'] = 0;
				}
				//oldvalue = parseInt(args.item[args.column.field + '_visiblefp']);
				var newvalue = 0;
				if($('#ifvisibleproductpage').is(':checked')) {
					newvalue|= 1;
				} else {
					newvalue&= ~1;
				}
				if($('#ifusedforvars').length > 0) {
					if($('#ifusedforvars').is(':checked')) {
						newvalue|= 2;
					} else {
						newvalue&= ~2;
					}
				}
				if(oldvalue !== newvalue) {
					args.item[args.column.field + '_visiblefp'] = newvalue;
				}
			}
		};

		this.isValueChanged = function () {

			$('.editorcats input:checked').each(function () {
				sel_ids.push($(this).attr('value'));
				var cattext = $(this).parent().text();
				cattext = $.trim(cattext);
				if (catnames === "") {
					catnames = cattext;
				} else {
					catnames = catnames + ', ' + cattext;
				}

			})
			if (args.item[args.column.field + '_ids'] !== undefined && args.item[args.column.field + '_ids'] !== '') {
				if (sel_ids.length == 0) return true;
				var ids = args.item[args.column.field + '_ids'].split(',');
				if (sel_ids.length !== ids.length) return true;
				for (var i = 0; i < ids.length; i++) {
					if (sel_ids.indexOf(ids[i]) === -1) {
						return true;
					}
				}
			} else {
				if (sel_ids.length > 0) return true;
			}
			if (args.item['post_type'] == 'product' && args.column.field.indexOf('attribute_pa_') !== -1 && $('#ifvisibleproductpage').length > 0) {
				var newvalue = 0;
				if ($('#ifvisibleproductpage').is(':checked')) {
					newvalue |= 1;
				} else {
					newvalue &= ~1;
				}
				if ($('#ifusedforvars').is(':checked')) {
					newvalue |= 2;
				} else {
					newvalue &= ~2;
				}
				if (newvalue !== oldvalue) {
					bonlyvisible = true;
					return true;
				}
			}
			return false;
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function DefAttrsEditor(args) {
		var $input, $wrapper;
		var defaultValue;
		var scope = this;
		var newvalue = "";
		this.init = function () {
			var $container = $('body');

			$wrapper = $("<DIV style='z-index:10000;position:absolute;background:white;padding:25px;padding-top:12px;padding-bottom:12px;border:3px solid gray; -moz-border-radius:10px; border-radius:10px;min-width:150px;max-width:300px;overflow:auto;'/>")
				.appendTo($container);
			$input = $("<div style='max-height:350px;min-height:100px;overflow:auto;' class='editorcats'></div>")
				.appendTo($wrapper);
			$('<DIV style="text-align:right"><BUTTON>Save</BUTTON><BUTTON>Cancel</BUTTON></DIV>')
				.appendTo($wrapper);
			$wrapper.find('button:first').bind('click', this.save);
			$wrapper.find('button:last').bind('click', this.cancel);
			$input.bind('keydown', this.handleKeyDown);
			scope.position(args.position);
			/////get all attributes
			var W3Ex = window.W3Ex || {};
			var W3ExABE = window.W3ExABE || {};
			var selitem = args.item;
			var attrtoget = [];
			if (selitem._product_attributes !== undefined) {
				for (var key in selitem._product_attributes) {
					if (selitem._product_attributes.hasOwnProperty(key)) {
						var attr = selitem._product_attributes[key];
						if (attr.is_taxonomy === 1 && attr.is_variation === 1) {
							if (selitem['attribute_' + key + '_ids'] === undefined && selitem['attribute_' + key] === undefined) {
								attrtoget.push('attribute_' + key);
							}
						}
					}
				}
			}
			if (attrtoget.length !== 0) {
				var ajaxarr = {};
				ajaxarr.action = 'wpmelon_adv_bulk_edit';
				ajaxarr.type = 'savecolumns';
				ajaxarr.nonce = W3ExABE.nonce;
				ajaxarr.colstoload = attrtoget;
				ajaxarr.colstoloadids = String(selitem.ID);
				$wrapper.append('<div class="showajax"></div>');
				jQuery.ajax({
					type: "post",
					dataType: "json",
					url: W3ExABE.ajaxurl,
					data: ajaxarr,
					success: function (response) {
						$('.showajax').remove();
						if (attrtoget.length > 0) {
							if (response !== undefined && response !== null && response.products !== undefined && response.products !== null) {
								var retproducts = response.products;
								for (var irow = 0; irow < retproducts.length; irow++) {
									if (retproducts[irow] === undefined) continue;
									var seliteout = retproducts[irow];
									if (seliteout.ID === undefined) continue;
									for (var key in seliteout) {
										if (seliteout.hasOwnProperty(key)) {
											if (selitem[key] === undefined || key === 'attribute_pa_ids')
												selitem[key] = seliteout[key];
										}
									}
								}
								if (selitem._product_attributes !== undefined && W3Ex.attributes_mapped !== undefined && W3Ex.attributes !== undefined) {
									if (selitem.attribute_pa_ids === undefined)
										selitem.attribute_pa_ids = '';
									var attrids = selitem.attribute_pa_ids.split(',');
									for (var key in selitem._product_attributes) {
										if (selitem._product_attributes.hasOwnProperty(key)) {
											var attr = selitem._product_attributes[key];
											if (attr.is_taxonomy === 1) {
												if (selitem['attribute_' + key + '_ids'] !== undefined && selitem['attribute_' + key] !== undefined)
													continue;
												for (var i = 0; i < attrids.length; i++) {
													var attrid = attrids[i];
													if (W3Ex.attributes_mapped[attrid] !== undefined) {
														var attritem = W3Ex.attributes[W3Ex.attributes_mapped[attrid]];
														if (attritem !== undefined) {
															var attrname = 'pa_' + attritem.attr;
															if (attrname === key)//attr.name)
															{
																if (selitem['attribute_' + attrname] === undefined) {
																	selitem['attribute_' + attrname] = '';
																	selitem['attribute_' + attrname + '_ids'] = '';
																}
																if (selitem['attribute_' + attrname] === '') {
																	selitem['attribute_' + attrname] = attritem.name;
																	selitem['attribute_' + attrname + '_ids'] = String(attritem.term_id);
																	//selitem["attribute_" + attrname +  '_visiblefp'] = 0;
																	var newvalue = 0;
																	if (attr.is_visible === 1) {
																		newvalue |= 1;
																	} else {
																		newvalue &= ~1;
																	}
																	if (attr.is_variation === 1) {
																		newvalue |= 2;
																	} else {
																		newvalue &= ~2;
																	}
																	selitem['attribute_' + attrname + '_visiblefp'] = newvalue;
																} else {
																	selitem['attribute_' + attrname] = selitem['attribute_' + attrname] + ', ' + attritem.name;
																	selitem['attribute_' + attrname + '_ids'] = selitem['attribute_' + attrname + '_ids'] + ',' + String(attritem.term_id);
																}
															}
														}
													}
												}
											}
										}
									}
								}
							}
						}

					},
					error: function (xhr, status, error) {
						$('.showajax').remove();
					},
					async: false
				});
			}
			////

			var addsel = "";
			for (var prop in selitem) {
				if (selitem.hasOwnProperty(prop)) {
					//addsel = '';
					if (
						prop.indexOf('attribute_pa_') === 0 &&
						prop.indexOf('_ids') === -1 && prop.indexOf('_visiblefp') === -1
					) {
						var attr_slug = prop.replace('attribute_pa_', '');
						addsel += '<tr><td>' + attr_slug + '</td><td><select data-slug="pa_' + attr_slug + '"><option value="" selected="selected">No default</option>';

						if (W3Ex.attributes !== undefined && W3Ex.attributes_mapped !== undefined && W3Ex.attributes_slugs_mapped !== undefined) {
							if (selitem[prop + '_ids'] !== undefined) {
								var attr_ids = selitem[prop + '_ids'];
								var ids_arr = attr_ids.split(',');
								for (var g = 0; g < ids_arr.length; g++) {
									if (W3Ex.attributes_mapped[ids_arr[g]] !== undefined) {
										var attritem = W3Ex.attributes[W3Ex.attributes_mapped[ids_arr[g]]];
										if (attritem !== undefined)
											addsel += '<option value="' + attritem.value + '">' + attritem.name + '</option>';
									}
								}
							}
						}
						addsel += "</select></td></tr>";
					}

				}
			}
			if (selitem._custom_attributes === undefined) {
				selitem._custom_attributes = [];
			}
			W3Ex._w3ex_map_attributes = W3Ex._w3ex_map_attributes || {};
			for (var i = 0; i < selitem._custom_attributes.length; i++) {
				var attr = selitem._custom_attributes[i];
				var attrslug = W3Ex._w3ex_map_attributes[attr.name];
				if (attrslug === undefined) attrslug = "";
				addsel += '<tr><td>' + attrslug + '</td><td><select data-slug="' + attrslug + '"><option value="" selected="selected">No default</option>';
				var values = attr.value.split(W3Ex._w3ex_wc_delimiter);
				for (var j = 0; j < values.length; j++) {
					var value = $.trim(values[j]);
					var valueslug = W3Ex._w3ex_map_attributes[value];
					if (valueslug === undefined) valueslug = "";
					addsel += '<option value="' + valueslug + '">' + value + '</option>';
				}
				addsel += '</select></td></tr>';
			}
			if (addsel !== "") {
				addsel = '<table id="defattrsholder">' + addsel + '</table>';
				$input.append(addsel);
			}
			if (args.item[args.column.field] !== undefined) {
				var attrs = args.item[args.column.field].split(';');
				if (attrs instanceof Array) {
					for (var i = 0; i < attrs.length; i++) {
						var arr_item = attrs[i];
						arr_item = $.trim(arr_item);
						var cur_attr = arr_item.split(',');
						if (cur_attr instanceof Array && cur_attr.length == 2) {
							$('#defattrsholder select[data-slug="' + cur_attr[0] + '"]').val(cur_attr[1]);
						}
					}
				}
				//return;
			}

			//$input.focus().select();
		};

		this.handleKeyDown = function (e) {
			if (e.which == $.ui.keyCode.ENTER && e.ctrlKey) {
				scope.save();
			} else if (e.which == $.ui.keyCode.ESCAPE) {
				e.preventDefault();
				scope.cancel();
			} else if (e.which == $.ui.keyCode.TAB && e.shiftKey) {
				e.preventDefault();
				args.grid.navigatePrev();
			} else if (e.which == $.ui.keyCode.TAB) {
				e.preventDefault();
				args.grid.navigateNext();
			}
		};

		this.save = function () {
			var W3Ex = window.W3Ex || {};
			W3Ex.bdontgodown = true;
			args.commitChanges();
		};

		this.cancel = function () {
			args.cancelChanges();
		};

		this.hide = function () {
			$wrapper.hide();
		};

		this.show = function () {
			$wrapper.show();
		};

		this.position = function (position) {
			$wrapper
				.css("top", position.top - 85)
				.css("left", position.left - 75)
		};

		this.destroy = function () {
			$wrapper.remove();
			var W3Ex = window.W3Ex || {};
			W3Ex.invalidateselected = true;
		};

		this.focus = function () {
			//$input.focus();
		};

		this.loadValue = function (item) {
			//$input.val(defaultValue = item[args.column.field]);
		};

		this.serializeValue = function () {
			return '';
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = newvalue;
		};

		this.isValueChanged = function () {
			$('#defattrsholder select').each(function () {
				var attr_slug = $(this).attr('data-slug');
				var attr_def = $(this).val();
				if (attr_def !== "") {
					if (newvalue === "") {
						newvalue = attr_slug + ',' + attr_def;
					} else {
						newvalue += ' ;' + attr_slug + ',' + attr_def;
					}
				}
			})
			if (newvalue != args.item[args.column.field]) {
				return true;
			}
			return false;
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function ImageGalleryEditor(args) {
		var $input, $wrapper, $ul, $aremove, $aset;
		var defaultValue;
		var scope = this;
		var file_frame;
		var set_to_post_id = -1;
		var attachment_id = [];
		var bapplied = false;
		var bcandestroy = false;
		var W3Ex = window.W3Ex || {};
		W3Ex.imageeditor = true;
		this.init = function () {
			$('#imagepopup').remove();

			//if(W3Ex._global_settings['openimage'] === true)
			//{
			//		this.setImage();
			//		return;
			//}
			var $container = $("body");

			$wrapper = $('<DIV id="gallerydiv" style="z-index:10000;position:absolute;background:white;padding:15px;padding-bottom:10px;border:3px solid gray; -moz-border-radius:5px; border-radius:5px;max-height: 400px;overflow: auto;"/>')
				.appendTo($container);
			$('<div id="w3exabe-note">Note! - save updates the database immediately.</div>')
				.appendTo($wrapper);
			$input = $("<div class='galleryholder'></div>")
				.appendTo($wrapper);
			$ul = $('<ul id="sortable"></ul>').appendTo($input);
			$('<div style="clear:both;"></div>').appendTo($input);
			$aset = $("<a href='#' class='setimage'> Add product gallery images</a>")
				.appendTo($input);

			//$aremove = $("<a href='#' class='removeimage'>Remove product image</a>")
			//	.appendTo($wrapper);
			var data = args.grid.getData();
			if (args.grid.getActiveCell() !== null) {
				var iRow = args.grid.getActiveCell().row;
				var selectedRows = args.grid.getSelectedRows();
				if (selectedRows.length > 0) {
					for (var irow = 0; irow < selectedRows.length; irow++) {
						var rowid = selectedRows[irow];
						if (rowid === undefined) continue;
						if (data[rowid] === undefined) continue;
						if (rowid === iRow) {//clicked on selected
							$('<br/><br/><label><input id="ifapplyaction" type="checkbox">Apply action to all<br/>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; selected products</label>').appendTo($wrapper);
							break;
						}
					}
				}
			}
			$("<DIV style='text-align:right;position:relative;'><BUTTON>Save</BUTTON><BUTTON>Cancel</BUTTON></DIV>")
				.appendTo($wrapper);

			$wrapper.find("button:first").bind("click", this.save);
			$wrapper.find("button:last").bind("click", this.cancel);
			//$aremove.bind("click", this.removeImage);
			$aset.bind("click", this.setImage);
			$input.bind("keydown", this.handleKeyDown);
			//$input.bind("click", this.handleClick);
			scope.position(args.position);
			$input.focus().select();
		};

		this.setImage = function (e) {

			var W3Ex = window.W3Ex || {};
			if (file_frame) {

				// Set the post ID to what we want
				if (set_to_post_id !== -1)
					file_frame.uploader.uploader.param('post_id', set_to_post_id);

				// Open frame
				file_frame.open();
				return;
			} else {
				// Set the wp.media post id so the uploader grabs the ID we want when initialised
				if (set_to_post_id !== -1)
					wp.media.model.settings.post.id = set_to_post_id;
			}
			// Create the media frame.
			file_frame = wp.media.frames.file_frame = wp.media({
				title: 'Add images to product gallery',
				button: {
					text: 'Add images',
				},
				states: [
					new wp.media.controller.Library({
						//library:   wp.media.query({ type: 'image' }),
						title: 'Add images to product gallery',
						filterable: 'all',
						multiple: true
					})
				],
				attachment: true,
				multiple: true // Set to true to allow multiple files to be selected
			});

			// When an image is selected, run a callback.
			file_frame.on('select', function () {
				// We set multiple to false so only get one image from the uploader
				var selection = file_frame.state().get('selection');

				if (selection) {
					// iterate through selected elements
					selection.each(function (attachment) {
						var url = attachment.attributes.url;
						var insert = '<li class="ui-state-default" data-id="' + attachment.attributes.id + '"  data-image="' + url + '" style="background:url(' + url + ') top left no-repeat;background-size: 90px auto;"><img class="delete" src="' + W3Ex.imagepath + 'images/gallerydel.png"></li>';
						$ul.append(insert);
					});
				}

				return;
				// Do something with attachment.id and/or attachment.url here
				// Restore the main post ID
				//alert('asd');
				//wp.media.model.settings.post.id = wp_media_post_id;
			});
			file_frame.on('open', function () {
				var attachment,
					selection = file_frame.state().get('selection');
				//if(attachment_id.length !== 0)
				//{
				//	var attaches = [];
				//	for(var i=0; i<attachment_id.length;i++)
				//	{
				//		attachment = wp.media.attachment(attachment_id[i]);
				//		attachment.fetch();
				//		attaches.push(attachment);
				//	}
				//	selection.add(attaches);
				//}
			});

			// Finally, open the modal
			file_frame.open();
		};

		this.removeImage = function (e) {
			var prodids = '';
			if ($('#ifapplyaction').length > 0) {
				if ($('#ifapplyaction').is(':checked')) {
					var data = args.grid.getData();
					if (args.grid.getActiveCell() !== null) {
						var iRow = args.grid.getActiveCell().row;
						var selectedRows = args.grid.getSelectedRows();
						if (selectedRows.length > 0) {
							for (var irow = 0; irow < selectedRows.length; irow++) {
								var rowid = selectedRows[irow];
								if (rowid === undefined) continue;
								if (data[rowid] === undefined) continue;
								data[rowid]._thumbnail_id = "";
								data[rowid]._thumbnail_id_val = "";
								if (prodids == "") {
									prodids = data[rowid].ID.toString() + ',';
								} else {
									prodids += ',' + data[rowid].ID.toString();
								}

							}
						}
					}
				}
			}

			var ajaxarr = {};
			ajaxarr.action = 'wpmelon_adv_bulk_edit';
			ajaxarr.type = 'removethumb';
			ajaxarr.nonce = window.W3ExABE.nonce;
			var _arrData = [];
			if (prodids !== "") {
				_arrData[0] = prodids;
				bapplied = true;
			} else
				_arrData[0] = args.item.ID;
			ajaxarr.data = _arrData;
			$wrapper.append('<div class="showajax"></div>');
			$('.showajax').css({
				left: '200px',
				top: '100px'
			});
			jQuery.ajax({
				type: "post",
				dataType: "json",
				url: W3ExABE.ajaxurl,
				data: ajaxarr,
				success: function (response) {
					$img.css('display', 'none');
					$aremove.css('display', 'none');
					$aset.css('display', 'inline');
					set_to_post_id = -1;
					attachment_id = -1;
					args.item._thumbnail_id = '';
					args.item._thumbnail_id_val = '';
					var W3Ex = window.W3Ex || {};
					W3Ex._global_settings.invalidateimages = true;
				},
				complete: function (args) {
					$('.showajax').remove();
				},
				async: false
			});
		};


		this.handleKeyDown = function (e) {
			if (e.which == $.ui.keyCode.ENTER && e.ctrlKey) {
				scope.save();
			} else if (e.which == $.ui.keyCode.ESCAPE) {
				bcandestroy = true;
				e.preventDefault();
				scope.cancel();
				//this.destroy();
			} else if (e.which == $.ui.keyCode.TAB && e.shiftKey) {
				e.preventDefault();
				args.grid.navigatePrev();
			} else if (e.which == $.ui.keyCode.TAB) {
				e.preventDefault();
				args.grid.navigateNext();
			}
		};

		this.save = function () {
			bcandestroy = true;
			var W3Ex = window.W3Ex || {};
			W3Ex.bdontgodown = true;
			var ids = '';
			var images = '';
			$ul.find('li').each(function () {
				var $item = $(this);
				if (ids == '') {
					ids = $item.attr('data-id');
				} else {
					ids += ',' + $item.attr('data-id');
				}
				if (images == "") {
					images = $item.attr('data-image');
				} else {
					images += "|" + $item.attr('data-image');
				}

			})
			if (args.item['_product_image_gallery'] === undefined) {
				args.item['_product_image_gallery'] = "";
			}
			if (ids === args.item['_product_image_gallery']) {
				args.cancelChanges();
				return;
			}
			W3Ex._w3abe_refreshlibrary = true;
			args.item['_product_image_gallery_val'] = images;
			args.item['_product_image_gallery_original'] = images;
			args.item['_product_image_gallery'] = ids;
			var ajaxarr = {};
			ajaxarr.action = 'wpmelon_adv_bulk_edit';
			ajaxarr.type = 'setgallery';
			ajaxarr.nonce = window.W3ExABE.nonce;
			var _arrData = [];
			var prodids = '';
			if ($('#ifapplyaction').length > 0) {
				if ($('#ifapplyaction').is(':checked')) {
					var data = args.grid.getData();
					if (args.grid.getActiveCell() !== null) {
						var iRow = args.grid.getActiveCell().row;
						var selectedRows = args.grid.getSelectedRows();
						if (selectedRows.length > 0) {
							for (var irow = 0; irow < selectedRows.length; irow++) {
								var rowid = selectedRows[irow];
								if (rowid === undefined) continue;
								if (data[rowid] === undefined) continue;
								//if(data[rowid].post_type == 'product_variation') continue;
								data[rowid]._product_image_gallery_val = images;
								data[rowid]._product_image_gallery_original = images;
								data[rowid]._product_image_gallery = ids;
								if (prodids == '') {
									prodids = data[rowid].ID.toString() + ',';
								} else {
									prodids += ',' + data[rowid].ID.toString();
								}
							}
						}
					}
				}
			}
			if (prodids !== '') {
				_arrData[0] = prodids;
				bapplied = true;
			} else
				_arrData[0] = args.item.ID;
			_arrData[1] = ids;
			ajaxarr.data = _arrData;
			$wrapper.find('button:first').parent().append('<div class="showajax"></div>');
			$('.showajax').css({
				left: '170px',
				top: '5px'
			});
			var upargs = args;
			jQuery.ajax({
				type: 'post',
				dataType: 'json',
				url: W3ExABE.ajaxurl,
				data: ajaxarr,
				success: function (response) {

				},
				complete: function (args) {
					$('.showajax').remove();
					upargs.cancelChanges();
					W3Ex._global_settings.invalidateimages = true;
				},
				async: false
			});

		};

		this.cancel = function () {
			bcandestroy = true;
			args.cancelChanges();
		};

		this.hide = function () {
			$wrapper.hide();
		};

		this.show = function () {
			$wrapper.show();
		};

		this.position = function (position) {
			$wrapper
				.css('top', position.top - 170)
				.css('left', position.left - 185)
		};

		this.destroy = function () {
			$wrapper.remove();
			{
				var W3Ex = window.W3Ex || {};
				W3Ex.imageeditor = false;
			}

			if (bapplied) {
				var W3Ex = window.W3Ex || {};
				W3Ex.invalidateselected = true;
			}
		};

		this.focus = function () {
			//$input.focus();
		};

		this.loadValue = function (item) {
			if (item['_product_image_gallery_val'] !== undefined && item['_product_image_gallery_val'] !== '') {
				var images = item['_product_image_gallery_val'];
				if (images.indexOf('|') === -1) {
					attachment_id.push(parseInt(item['_product_image_gallery']));
					var insert = '';
					if (images === ' ') {
						insert = '<li class="ui-state-default" data-id="' + attachment_id + '" data-image="' + images + '" style="background:url(' + W3Ex.imagepath + 'images/nophoto.png) top left no-repeat;background-size: 90px auto;"><img class="delete" src="' + W3Ex.imagepath + 'images/gallerydel.png"></li>';
					} else {
						insert = '<li class="ui-state-default" data-id="' + attachment_id + '" data-image="' + images + '" style="background:url(' + images + ') top left no-repeat;background-size: 90px auto;"><img class="delete" src="' + W3Ex.imagepath + 'images/gallerydel.png"></li>';
					}
					$ul.append(insert);

					set_to_post_id = parseInt(item['ID']);
					//$aset.css("display","none");
				} else {
					var res = images.split('|');
					if (item['_product_image_gallery'] !== undefined) {
						var ids = item['_product_image_gallery'].split(',');
						if (res instanceof Array && ids instanceof Array && res.length === ids.length) {
							for (var i = 0; i < res.length; i++) {
								attachment_id.push(ids[i]);
								var insert = '';
								if (res[i] === ' ') {
									insert = '<li class="ui-state-default" data-id="' + ids[i] + '" data-image="' + res[i] + '" style="background:url(' + W3Ex.imagepath + 'images/nophoto.png) top left no-repeat;background-size: 90px auto;"><img class="delete" src="' + W3Ex.imagepath + 'images/gallerydel.png"></li>';
								} else {
									insert = '<li class="ui-state-default" data-id="' + ids[i] + '" data-image="' + res[i] + '" style="background:url(' + res[i] + ') top left no-repeat;background-size: 90px auto;"><img class="delete" src="' + W3Ex.imagepath + 'images/gallerydel.png"></li>';
								}
								$ul.append(insert);
							}
						}
					}
				}
			} else {
				attachment_id = -1;
				set_to_post_id = -1;
			}
			set_to_post_id = parseInt(item['ID']);
			$("#sortable").sortable();
			$("#sortable").disableSelection();
		};

		this.serializeValue = function () {
			return $input.val();
		};

		this.applyValue = function (item, state) {
			//if(bapplied)
			//{
			//	var data = args.grid.getData();
			//	var selectedRows = args.grid.getSelectedRows();
			//	if(selectedRows.length > 0)
			//	{
			//		for(var irow=0; irow < selectedRows.length; irow++)
			//		{
			//			var rowid = selectedRows[irow];
			//			if(rowid === undefined) continue;
			//			if(data[rowid] === undefined) continue;
			//			data[rowid][args.column.field] = state;
			//		}
			//	}
			//} else
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return false;
			//return (!($input.val() == "" && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function TextAreaEditor(args) {
		var $input, $wrapper;
		var defaultValue;
		var scope = this;
		var W3Ex = window.W3Ex || {};
		var width = args.column.field + 'width';
		var height = args.column.field + 'height';
		this.init = function () {
			var $container = $('body');

			$wrapper = $('<DIV style="z-index:10000;position:absolute;background:white;padding:5px;border:3px solid gray; -moz-border-radius:10px; border-radius:10px;"/>')
				.appendTo($container);

			$input = $('<TEXTAREA hidefocus rows=5 style="backround:white;width:250px;height:80px;border:0;outline:0">')
				.appendTo($wrapper);

			$('<DIV style="text-align:right"><BUTTON>Save</BUTTON><BUTTON>Cancel</BUTTON></DIV>')
				.appendTo($wrapper);
			if (W3Ex[width] !== undefined && W3Ex[height] !== undefined) {
				$input.css({
					'width': W3Ex[width],
					'height': W3Ex[height]
				})
			}

			$wrapper.find('button:first').bind('click', this.save);
			$wrapper.find('button:last').bind('click', this.cancel);
			$input.bind('keydown', this.handleKeyDown);

			scope.position(args.position);
			$input.focus().select();
		};

		this.handleKeyDown = function (e) {
			if (e.which == $.ui.keyCode.ENTER && e.ctrlKey) {
				scope.save();
			} else if (e.which == $.ui.keyCode.ESCAPE) {
				e.preventDefault();
				scope.cancel();
			} else if (e.which == $.ui.keyCode.TAB && e.shiftKey) {
				e.preventDefault();
				args.grid.navigatePrev();
			} else if (e.which == $.ui.keyCode.TAB) {
				e.preventDefault();
				args.grid.navigateNext();
			}
		};

		this.save = function () {
			var W3Ex = window.W3Ex || {};
			W3Ex.bdontgodown = true;
			args.commitChanges();
		};

		this.cancel = function () {
			$input.val(defaultValue);
			args.cancelChanges();
		};

		this.hide = function () {
			$wrapper.hide();
		};

		this.show = function () {
			$wrapper.show();
		};

		this.position = function (position) {
			$wrapper
				.css('top', position.top - 5)
				.css('left', position.left - 55)
		};

		this.destroy = function () {
			W3Ex[width] = $input.css('width');
			W3Ex[height] = $input.css('height');
			$wrapper.remove();
		};

		this.focus = function () {
			$input.focus();
		};

		this.loadValue = function (item) {
			$input.val(defaultValue = item[args.column.field]);
			//$input.select();
		};

		this.serializeValue = function () {
			return $input.val();
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return (!($input.val() == "" && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function WordPressEditor(args) {
		var $input, $wrapper;
		var defaultValue;
		var scope = this;
		//var clickedsave = false;
		var W3Ex = window.W3Ex || {};
		var width = args.column.field + 'width';
		var height = args.column.field + 'height';
		this.init = function () {
			var $container = $('body');

			$wrapper = $('#editorcontainer');
			$('#editorcontainer').css('display', 'block');
			$('#editorcontainer').css("visibility", "visible");//css("display","inline");
			$("#editorcontainer").css('height', 'auto');
			$("#wp-posttext-wrap").css('height', '400px');
			//	  tinyMCE.get('editorid').setContent('asd asdasdasd');
			//	     $wrapper
			//          .css("top", position.top - 5)
			//          .css("left", position.left - 55)
			//      $input = $("<TEXTAREA hidefocus rows=5 style='backround:white;border:0;outline:0;width:0px;height:0px;display:none;'>")
			//          .appendTo($wrapper);

			//      $("<DIV style='text-align:right'><BUTTON>Save</BUTTON><BUTTON>Cancel</BUTTON></DIV>")
			//          .appendTo($wrapper);
			//	  if(W3Ex[width] !== undefined && W3Ex[height] !== undefined)
			//	  {
			//
			//	  	   $input.css({
			//		  	  'width':W3Ex[width],
			//			  'height':W3Ex[height]
			//		  })
			//	  }

			$wrapper.find('#savewordpeditor').bind('click', this.save);
			$wrapper.find('#cancelwordpeditor').bind('click', this.cancel);
			$wrapper.bind('keydown', this.handleKeyDown);
			//      var W3Ex = window.W3Ex || {};
			//	  W3Ex._w3abe_wordpres_clicked_save = false;
			scope.position(args.position);
			//      $input.focus().select();
		};

		this.handleKeyDown = function (e) {
			if (e.which == $.ui.keyCode.ENTER && e.ctrlKey) {
				scope.save();
			} else if (e.which == $.ui.keyCode.ESCAPE) {
				e.preventDefault();
				scope.cancel();
			} else if (e.which == $.ui.keyCode.TAB && e.shiftKey) {
				e.preventDefault();
				args.grid.navigatePrev();
			} else if (e.which == $.ui.keyCode.TAB) {
				e.preventDefault();
				args.grid.navigateNext();
			}
		};

		this.save = function () {
			//clickedsave = true;
			//var W3Ex = window.W3Ex || {};
			//W3Ex._w3abe_wordpres_clicked_save = true;
			W3Ex.bdontgodown = true;
			args.commitChanges();
		};

		this.cancel = function () {
			//$input.val(defaultValue);
			args.cancelChanges();
		};

		this.hide = function () {
			$wrapper.hide();
		};

		this.show = function () {
			$wrapper.show();
		};

		this.position = function (position) {
			$wrapper
				.css("top", position.top)
				.css("left", position.left - 265);
			//$('#editorcontainer').css('top',position.top - 5);
			//$('#editorcontainer').css('left',position.left - 55);
		};

		this.destroy = function () {
			//W3Ex[width] = $input.css('width');
			//W3Ex[height] = $input.css('height');

			$('#editorcontainer').css("visibility", "hidden");//css("display","inline");
			$('#editorcontainer').css("display", "none");
			$("#editorcontainer").css("height", "0px");
			$("#wp-posttext-wrap").css("visibility", "hidden");
			$("#wp-posttext-wrap").css("height", "0px");
		};

		this.focus = function () {
			//$input.focus();
		};

		this.loadValue = function (item) {
			//$input.val();
			defaultValue = item[args.column.field];
			if (defaultValue === undefined)
				defaultValue = "";
			defaultValue = defaultValue.replace(/\n$/g, "");
			if ($('#editorid').is(':visible')) {
				$('#editorid').val(defaultValue);
			} else {
				var editor = tinyMCE.get('editorid');
				editor.setContent(defaultValue);
			}
			//$input.select();
		};

		this.serializeValue = function () {
			//return tinyMCE.get('editorid').getContent();
		};

		this.applyValue = function (item, state) {

			var html = "";
			if ($('#editorid').is(':visible') || tinyMCE.get('editorid') === null) {
				html = $('#editorid').val();
			} else {
				html = tinyMCE.get('editorid').getContent();
			}

			item[args.column.field] = html;
		};

		this.isValueChanged = function () {
			//var W3Ex = window.W3Ex || {};
			//if(W3Ex._w3abe_wordpres_clicked_save !== true)
			//return false;
			var html = "";
			if ($('#editorid').is(':visible') || tinyMCE.get('editorid') === null) {
				html = $('#editorid').val();
			} else {
				html = tinyMCE.get('editorid').getContent();
			}
			return (!(html == "" && defaultValue == null)) && (html != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function customDateEditor(args) {
		var $input;
		var defaultValue;
		var scope = this;
		var calendarOpen = false;

		this.init = function () {
			$input = $('<INPUT type=text class="editor-text" />');
			$input.appendTo(args.container);
			$input.focus().select();
			$input.datepicker({
				dateFormat: 'yy-mm-dd',
				showOn: 'button',
				buttonImageOnly: true,
				buttonImage: W3Ex.imagepath + 'images/calendar.gif',
				beforeShow: function () {
					calendarOpen = true;
				},
				onClose: function () {
					calendarOpen = false;
				}
			});
			$input.width($input.width() - 18);
			if ($('.w3exabedeleditor').length === 0)
				$('.ui-datepicker').wrap('<div class="w3exabe w3exabedeleditor" />');
		};

		this.destroy = function () {
			$.datepicker.dpDiv.stop(true, true);
			$input.datepicker('hide');
			$input.datepicker('destroy');
			$input.remove();
			$('.w3exabedeleditor').contents().unwrap();
		};

		this.show = function () {
			if (calendarOpen) {
				$.datepicker.dpDiv.stop(true, true).show();
			}
		};

		this.hide = function () {
			if (calendarOpen) {
				$.datepicker.dpDiv.stop(true, true).hide();
			}
		};

		this.position = function (position) {
			if (!calendarOpen) {
				return;
			}
			$.datepicker.dpDiv
				.css('top', position.top + 30)
				.css('left', position.left);
		};

		this.focus = function () {
			$input.focus();
		};

		this.loadValue = function (item) {
			defaultValue = item[args.column.field];
			$input.val(defaultValue);
			$input[0].defaultValue = defaultValue;
			$input.select();
		};

		this.serializeValue = function () {
			return $input.val();
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return (!($input.val() == '' && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function SelectCellEditor(args) {
		var $select;
		var defaultValue;
		var scope = this;

		this.init = function () {
			var opt_values;
			if (args.column.options) {
				opt_values = args.column.options.split(',');
			} else {
				opt_values = 'yes,no'.split(',');
			}
			var option_str = '';
			if (args.column.field == '_visibility') {
				for (i in opt_values) {
					var v = opt_values[i];
					if (v == args.item._visibility)
						option_str += '<OPTION value="' + v + '" selected="selected">' + v + '</OPTION>';
					else
						option_str += '<OPTION value="' + v + '">' + v + '</OPTION>';

				}
			} else if (args.column.field == '_download_type') {
				for (i in opt_values) {
					var v = opt_values[i];
					//if(args.item.download_type != null)
					if (v == args.item._download_type)
						option_str += '<OPTION value="' + v + '" selected="selected">' + v + '</OPTION>';
					else
						option_str += '<OPTION value="' + v + '">' + v + '</OPTION>';
				}
			} else if (args.column.field == 'post_status') {
				for (i in opt_values) {
					var v = opt_values[i];
					if (v == args.item.post_status)
						option_str += '<OPTION value="' + v + '" selected="selected">' + v + '</OPTION>';
					else
						option_str += '<OPTION value="' + v + '">' + v + '</OPTION>';

				}
			} else {
				for (i in opt_values) {
					var v = opt_values[i];
					if (v == args.item[args.column.field])
						option_str += '<OPTION value="' + v + '" selected="selected">' + v + '</OPTION>';
					else
						option_str += '<OPTION value="' + v + '">' + v + '</OPTION>';
				}
			}

			$select = $('<SELECT tabIndex="0" class="editor-select">' + option_str + '</SELECT>');
			$select.appendTo(args.container);
			$select.focus();
		};

		this.destroy = function () {
			$select.remove();
		};

		this.focus = function () {
			$select.focus();
		};

		this.loadValue = function (item) {
			defaultValue = item[args.column.field];
			$select.val(defaultValue);
		};

		this.serializeValue = function () {
			if (args.column.options) {
				return $select.val();
			} else {
				return ($select.val() == 'yes');
			}
		};

		this.applyValue = function (item, state) {
			if (state != null)
				item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return ($select.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function TextEditor(args) {
		var $input;
		var defaultValue;
		var scope = this;

		this.init = function () {
			$input = $('<INPUT type="text" class="editor-text" />')
				.appendTo(args.container)
				.bind('keydown.nav', function (e) {
					if (e.keyCode === $.ui.keyCode.LEFT || e.keyCode === $.ui.keyCode.RIGHT) {
						e.stopImmediatePropagation();
					}
				})
				.focus()
				.select();
		};

		this.destroy = function () {
			$input.remove();
		};

		this.focus = function () {
			$input.focus();
		};

		this.getValue = function () {
			return $input.val();
		};

		this.setValue = function (val) {
			$input.val(val);
		};

		this.loadValue = function (item) {
			if (item[args.column.field] === undefined || item[args.column.field] === null)
				item[args.column.field] = "";
			item[args.column.field] = String(item[args.column.field]);
			defaultValue = item[args.column.field] || "";
			$input.val(defaultValue);
			$input[0].defaultValue = defaultValue;
			$input.select();
		};

		this.serializeValue = function () {
			return $input.val();
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return (!($input.val() == "" && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			if (args.column.validator) {
				var validationResults = args.column.validator($input.val());
				if (!validationResults.valid) {
					return validationResults;
				}
			}

			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function IntegerEditor(args) {
		var $input;
		var defaultValue;
		var scope = this;

		this.init = function () {
			$input = $('<INPUT type="text" class="editor-text" />');

			$input.bind('keydown.nav', function (e) {
				if (e.keyCode === $.ui.keyCode.LEFT || e.keyCode === $.ui.keyCode.RIGHT) {
					e.stopImmediatePropagation();
				}
			});

			$input.appendTo(args.container);
			$input.focus().select();
		};

		this.destroy = function () {
			$input.remove();
		};

		this.focus = function () {
			$input.focus();
		};

		this.loadValue = function (item) {
			defaultValue = item[args.column.field];
			$input.val(defaultValue);
			$input[0].defaultValue = defaultValue;
			$input.select();
		};

		this.serializeValue = function () {
			return parseInt($input.val(), 10) || 0;
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return (!($input.val() == '' && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			if (isNaN($input.val())) {
				return {
					valid: false,
					msg: 'Please enter a valid integer'
				};
			}

			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function DateEditor(args) {
		var $input;
		var defaultValue;
		var scope = this;
		var calendarOpen = false;

		this.init = function () {
			$input = $('<INPUT type="text" class="editor-text" />');
			$input.appendTo(args.container);
			$input.focus().select();
			$input.datepicker({
				showOn: 'button',
				buttonImageOnly: true,
				buttonImage: '../images/calendar.gif',
				beforeShow: function () {
					calendarOpen = true
				},
				onClose: function () {
					calendarOpen = false
				}
			});
			$input.width($input.width() - 18);
		};

		this.destroy = function () {
			$.datepicker.dpDiv.stop(true, true);
			$input.datepicker('hide');
			$input.datepicker('destroy');
			$input.remove();
		};

		this.show = function () {
			if (calendarOpen) {
				$.datepicker.dpDiv.stop(true, true).show();
			}
		};

		this.hide = function () {
			if (calendarOpen) {
				$.datepicker.dpDiv.stop(true, true).hide();
			}
		};

		this.position = function (position) {
			if (!calendarOpen) {
				return;
			}
			$.datepicker.dpDiv
				.css('top', position.top + 30)
				.css('left', position.left);
		};

		this.focus = function () {
			$input.focus();
		};

		this.loadValue = function (item) {
			defaultValue = item[args.column.field];
			$input.val(defaultValue);
			$input[0].defaultValue = defaultValue;
			$input.select();
		};

		this.serializeValue = function () {
			return $input.val();
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return (!($input.val() == "" && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function YesNoSelectEditor(args) {
		var $select;
		var defaultValue;
		var scope = this;

		this.init = function () {
			$select = $('<SELECT tabIndex="0" class="editor-yesno"><OPTION value="yes">Yes</OPTION><OPTION value="no">No</OPTION></SELECT>');
			$select.appendTo(args.container);
			$select.focus();
		};

		this.destroy = function () {
			$select.remove();
		};

		this.focus = function () {
			$select.focus();
		};

		this.loadValue = function (item) {
			$select.val((defaultValue = item[args.column.field]) ? 'yes' : 'no');
			$select.select();
		};

		this.serializeValue = function () {
			return ($select.val() == 'yes');
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return ($select.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function CheckboxEditor(args) {
		var $select;
		var defaultValue;
		var scope = this;
		var bempty = false;
		this.init = function () {
			$select = $('<INPUT type="checkbox" value="true" class="editor-checkbox" hideFocus>');
			$select.appendTo(args.container);
			$select.focus();
		};

		this.destroy = function () {
			$select.remove();
		};

		this.focus = function () {
			$select.focus();
		};

		this.loadValue = function (item) {
			defaultValue = item[args.column.field];
			if (args.column.field == '_stock_status') {
				var W3Ex = window.W3Ex || {};
				var skip = false;
				if (W3Ex._wooversion !== undefined && W3Ex._wooversion >= 3.3) {
					skip = true;
				}
				if (!skip) {
					if (defaultValue == 'instock') {
						$select.prop('checked', true);
					} else {
						$select.prop('checked', false);
					}
				}

			} else {
				if (defaultValue == 'yes') {
					$select.prop('checked', true);
				} else {
					$select.prop('checked', false);
				}
			}
		};

		this.serializeValue = function () {
			return $select.prop('checked');
		};

		this.applyValue = function (item, state) {
			if (args.column.field == '_stock_status') {
				var W3Ex = window.W3Ex || {};
				var skip = false;
				if (W3Ex._wooversion !== undefined && W3Ex._wooversion >= 3.3) {
					skip = true;
				}
				if (!skip) {
					if (state == false)
						item[args.column.field] = 'outofstock';  //<-- remove that line, add lines below
					else
						item[args.column.field] = 'instock';
				}
			} else {
				if (state == false) {
					item[args.column.field] = 'no';  //<-- remove that line, add lines below
				} else
					item[args.column.field] = 'yes';
			}

			//if (state == 'checked') {
			//		item[args.column.field] = true;
			//} else {
			//		item[args.column.field] = false;
			//}
		};

		this.isValueChanged = function () {
			return (this.serializeValue() !== defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function PercentCompleteEditor(args) {
		var $input, $picker;
		var defaultValue;
		var scope = this;

		this.init = function () {
			$input = $('<INPUT type="text" class="editor-percentcomplete" />');
			$input.width($(args.container).innerWidth() - 25);
			$input.appendTo(args.container);

			$picker = $('<div class="editor-percentcomplete-picker" />').appendTo(args.container);
			$picker.append('<div class="editor-percentcomplete-helper"><div class="editor-percentcomplete-wrapper"><div class="editor-percentcomplete-slider" /><div class="editor-percentcomplete-buttons" /></div></div>');

			$picker.find('.editor-percentcomplete-buttons').append('<button val=0>Not started</button><br/><button val=50>In Progress</button><br/><button val=100>Complete</button>');

			$input.focus().select();

			$picker.find('.editor-percentcomplete-slider').slider({
				orientation: 'vertical',
				range: 'min',
				value: defaultValue,
				slide: function (event, ui) {
					$input.val(ui.value)
				}
			});

			$picker.find('.editor-percentcomplete-buttons button').bind('click', function (e) {
				$input.val($(this).attr('val'));
				$picker.find('.editor-percentcomplete-slider').slider('value', $(this).attr('val'));
			})
		};

		this.destroy = function () {
			$input.remove();
			$picker.remove();
		};

		this.focus = function () {
			$input.focus();
		};

		this.loadValue = function (item) {
			$input.val(defaultValue = item[args.column.field]);
			$input.select();
		};

		this.serializeValue = function () {
			return parseInt($input.val(), 10) || 0;
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return (!($input.val() == "" && defaultValue == null)) && ((parseInt($input.val(), 10) || 0) != defaultValue);
		};

		this.validate = function () {
			if (isNaN(parseInt($input.val(), 10))) {
				return {
					valid: false,
					msg: 'Please enter a valid positive number'
				};
			}

			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}


	/*
	* An example of a "detached" editor.
	* The UI is added onto document BODY and .position(), .show() and .hide() are implemented.
	* KeyDown events are also handled to provide handling for Tab, Shift-Tab, Esc and Ctrl-Enter.
	*/
	////////////
	//Files
	////

	function LongTextEditor(args) {
		var $input, $input_val, $wrapper, $table;
		var input_val;
		var defaultValue;
		var scope = this;
		var downloadable_file_frame;
		var file_path_field;
		var newtr;

		this.init = function () {
			var $container = $('body');

			$wrapper = $('<DIV style="z-index:10000;position:absolute;background:white;padding:5px;border:3px solid gray; -moz-border-radius:10px; border-radius:10px;width:400px;"/>')
				.appendTo($container);

			$input = $('<TEXTAREA hidefocus rows=1 style="backround:white;width:0px;height:0px;border:0;outline:0;display:none;">')
				.appendTo($wrapper);
			//$input_val = $("<TEXTAREA hidefocus rows=1 style='backround:white;width:0px;height:0px;border:0;outline:0;display:none;'>")
			//	.appendTo($wrapper);
			newtr = '<tr> \
						<td> \
						<input style="width:80px;" class="input_text filename" placeholder="File Name" name="_wc_file_names[]" value="" type="text">\
						</td>\
						<td> \
						<input style="width:160px;" class="input_text filepath" placeholder="File Path" name="_wc_file_path[]" value="" type="text">\
						</td>\
						<td style="width:20%;"> \
						<a href="#" class="button upload_file_button choosefile" data-choose="Choose file" data-update="Insert file URL">Choose&nbsp;file</a>\
						</td>\
						<td style="width:10%;"> \
						<a href="#" class="deletefile">Del</a>\
						</td>\
					</tr>';
			$table = $('<table style="max-width:390px;"></table>')
				.appendTo($wrapper);
			$('<a href="#" class="button insert" style="margin-left:10px;">Add File</a>')
				.appendTo($wrapper);
			$('<DIV style="text-align:right"><BUTTON>Save</BUTTON><BUTTON>Cancel</BUTTON></DIV>')
				.appendTo($wrapper);

			$wrapper.find('button:first').bind('click', this.save);
			$wrapper.find('button:last').bind('click', this.cancel);
			jQuery(document).on('click', '.choosefile', this.loadFile);
			jQuery(document).on('click', '.deletefile', this.deleteFile);
			$wrapper.find('a.insert').bind('click', function (e) {
				$table.append(newtr);
				e.preventDefault();
			});
			$input.bind('keydown', this.handleKeyDown);

			scope.position(args.position);
			$input.focus().select();

		};

		this.save = function () {
			var fieldtext = "";
			var fieldtext_val = "";
			$table.find('tr').each(function () {
					var path = $(this).find('.filepath').val();
					if (!(!path || !/\S/.test(path))) {

						var name = $(this).find('.filename').val();
						//			 if(!(!name || !/\S/.test(name)))
						{
							fieldtext = fieldtext + ' Name:' + name + ' URL:' + path;
							if (fieldtext_val == '')
								fieldtext_val = name + '#####' + path;
							else
								fieldtext_val = fieldtext_val + '*****' + name + '#####' + path;
						}
					}
				}
			)
			$input.val(fieldtext);
			input_val = fieldtext_val;
			var W3Ex = window.W3Ex || {};
			W3Ex.bdontgodown = true;
			args.commitChanges();
		};

		this.loadValue = function (item) {
			$input.val(defaultValue = item[args.column.field]);
			$input.select();
			var value_val = item[args.column.field + '_val'];
			if (value_val !== undefined && value_val !== '') {
				var cols = value_val.split('*****');
				for (var i = 0; i < cols.length; i++) {
					var itemsarr = cols[i];
					if (itemsarr === undefined || itemsarr === '') continue;
					var items = itemsarr.split('#####');
					var bappend = false;
					for (var j = 0; j < items.length; j++) {
						var item = items[j];
						if (item === undefined || item === '') continue;
						if (!bappend) {
							$table.append(newtr);
							bappend = true;
						}
						if (j == 0) {//name
							$table.find('.filename:last').val(item);
						} else {//url
							$table.find('.filepath:last').val(item);
						}
					}
				}
			}
		};

		this.handleKeyDown = function (e) {
			if (e.which == $.ui.keyCode.ENTER && e.ctrlKey) {
				scope.save();
			} else if (e.which == $.ui.keyCode.ESCAPE) {
				e.preventDefault();
				scope.cancel();
			} else if (e.which == $.ui.keyCode.TAB && e.shiftKey) {
				e.preventDefault();
				args.grid.navigatePrev();
			} else if (e.which == $.ui.keyCode.TAB) {
				e.preventDefault();
				args.grid.navigateNext();
			}
		};
		this.loadFile = function (event) {
			//$(this).closest('tr').find('td .filepath').val('asd');
			//$(this).closest('tr').find('td .filename').val('qwe');

			var $el = $(this);

			file_path_field = $el.closest('tr').find('td .filepath');
			event.preventDefault();
			// If the media frame already exists, reopen it.
			if (downloadable_file_frame) {
				downloadable_file_frame.open();
				return;
			}

			var downloadable_file_states = [
				// Main states.
				new wp.media.controller.Library({
					library: wp.media.query(),
					multiple: true,
					title: $el.data('choose'),
					priority: 20,
					filterable: 'uploaded',
				})
			];

			// Create the media frame.
			downloadable_file_frame = wp.media.frames.downloadable_file = wp.media({
				// Set the title of the modal.
				title: $el.data('choose'),
				library: {
					type: ''
				},
				button: {
					text: $el.data('update'),
				},
				multiple: true,
				states: downloadable_file_states,
			});

			// When an image is selected, run a callback.
			downloadable_file_frame.on('select', function () {

				var file_path = '';
				var selection = downloadable_file_frame.state().get('selection');

				selection.map(function (attachment) {

					attachment = attachment.toJSON();

					if (attachment.url)
						file_path = attachment.url

				});

				file_path_field.val(file_path);
			});

			// Set post to 0 and set our custom type
			downloadable_file_frame.on('ready', function () {
				downloadable_file_frame.uploader.options.uploader.params = {
					type: 'downloadable_product'
				};
			});

			// Finally, open the modal.
			downloadable_file_frame.open();
		};

		this.deleteFile = function (event) {
			var $el = $(this);
			$el.closest('tr').remove();
			event.preventDefault();
		};


		this.cancel = function () {
			$input.val(defaultValue);
			args.cancelChanges();
		};

		this.hide = function () {
			$wrapper.hide();
		};

		this.show = function () {
			$wrapper.show();
		};

		this.position = function (position) {
			$wrapper
				.css('top', position.top - 5)
				.css('left', position.left - 255);
		};

		this.destroy = function () {
			$wrapper.find('button:first').unbind();
			$wrapper.find('button:last').unbind();
			jQuery(document).off('click', '.choosefile');
			jQuery(document).off('click', '.deletefile');
			$wrapper.find('a.insert').unbind();
			$wrapper.remove();
		};

		this.focus = function () {
			$input.focus();
		};


		this.serializeValue = function () {
			return $input.val();
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
			item[args.column.field + '_val'] = input_val;
		};

		this.isValueChanged = function () {
			return (!($input.val() == "" && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function timeOptionsBuilder(optionsCount) {
		var optionsHtml = '';
		var optionText = '';
		for (var i=0; i<optionsCount; i++) {
			optionText = (i<10) ? '0'+i : i;
			optionsHtml += '<option value="'+i+'">'+optionText+'</option>';
		}

		return optionsHtml;
	}

	function DateTimeEditor(args) {
		var $inputDate;
		var $inputHours;
		var $inputMinutes;
		var $inputSeconds;
		var defaultDateValue;
		var defaultHoursValue;
		var defaultMinutesValue;
		var defaultSecondsValue;
		var scope = this;
		var calendarOpen = false;
		var inputHoursHtml = '<select class="datetime-hours-select">' + timeOptionsBuilder(24) + '</select>';
		var inputMinutesHtml = '<select class="datetime-minutes-select">' + timeOptionsBuilder(60) + '</select>';
		var inputSecondsHtml = '<select class="datetime-seconds-select">' + timeOptionsBuilder(60) + '</select>';

		this.init = function () {
			$inputDate = $('<INPUT type="text" class="editor-text datetime-date-input" />');
			$inputDate.appendTo(args.container);

			$(args.container).append(' ');

			$inputHours = $(inputHoursHtml);
			$inputHours.appendTo(args.container);

			$(args.container).append(' : ');

			$inputMinutes = $(inputMinutesHtml);
			$inputMinutes.appendTo(args.container);

			$(args.container).append(' : ');

			$inputSeconds = $(inputSecondsHtml);
			$inputSeconds.appendTo(args.container);

			$inputDate.focus().select();
			$inputDate.datepicker({
				dateFormat: 'yy-mm-dd',
				showOn: 'button',
				buttonImageOnly: true,
				buttonImage: W3Ex.imagepath + 'images/calendar.gif',
				beforeShow: function () {
					calendarOpen = true;
				},
				onClose: function () {
					calendarOpen = false;
				}
			});
			$inputDate.width($inputDate.width() - 18);
		};

		this.destroy = function () {
			$.datepicker.dpDiv.stop(true, true);
			$inputDate.datepicker('hide');
			$inputDate.datepicker('destroy');
			$inputDate.remove();

			$inputHours.remove();
			$inputMinutes.remove();
			$inputSeconds.remove();
		};

		this.show = function () {
			if (calendarOpen) {
				$.datepicker.dpDiv.stop(true, true).show();
			}
		};

		this.hide = function () {
			if (calendarOpen) {
				$.datepicker.dpDiv.stop(true, true).hide();
			}
		};

		this.position = function (position) {
			if (!calendarOpen) {
				return;
			}
			$.datepicker.dpDiv
				.css('top', position.top + 30)
				.css('left', position.left);
		};

		this.focus = function () {
			$inputDate.focus();
		};

		this.loadValue = function (item) {

			if (item[args.column.field] == undefined || item[args.column.field] == "") {
				$inputDate.val('');
				$inputDate[0].defaultValue = '';
				$inputHours.val('');
				$inputHours[0].defaultValue = '';
				$inputMinutes.val('');
				$inputMinutes[0].defaultValue = '';
				$inputSeconds.val('');
				$inputSeconds[0].defaultValue = '';

				return;
			}

			const dateTimeVal = item[args.column.field];
			const dateTimeArr = dateTimeVal.split(' ');
			let date = dateTimeArr[0];
			if ('created_date' in item) {
				date = item.created_date;
			}
			const time = dateTimeArr[1].split(':');
			let timeH = time[0];
			if ('created_time_h' in item) {
				timeH = item.created_time_h;
			}
			let timeM = time[1];
			if ('created_time_m' in item) {
				timeM = item.created_time_m;
			}
			let timeS = time[2];
			if ('created_time_s' in item) {
				timeS = item.created_time_s;
			}

			defaultDateValue = date;
			$inputDate.val(defaultDateValue);
			$inputDate[0].defaultValue = defaultDateValue;

			defaultHoursValue = timeH;
			$inputHours.val(parseInt(defaultHoursValue));
			$inputHours[0].defaultValue = defaultHoursValue;

			defaultMinutesValue = timeM;
			$inputMinutes.val(parseInt(defaultMinutesValue));
			$inputMinutes[0].defaultValue = defaultMinutesValue;

			defaultSecondsValue = timeS;
			$inputSeconds.val(parseInt(defaultSecondsValue));
			$inputSeconds[0].defaultValue = defaultSecondsValue;

			$inputDate.select();
		};

		this.serializeValue = function () {
			if ($inputDate.val() == null || $inputDate.val() == '') {
				return '';
			}

			let timeH  = '00';
			if ($inputHours.val() != null) {
				timeH = $inputHours.val() < 10 ? '0'+$inputHours.val() : $inputHours.val();
			}
			let timeM  = '00';
			if ($inputMinutes.val() != null) {
				timeM = $inputMinutes.val() < 10 ? '0'+$inputMinutes.val() : $inputMinutes.val();
			}
			let timeS  = '00';
			if ($inputSeconds.val() != null) {
				timeS = $inputSeconds.val() < 10 ? '0'+$inputSeconds.val() : $inputSeconds.val();
			}
			return $inputDate.val() + ' ' + timeH + ':' + timeM + ':' + timeS;
		};

		this.applyValue = function (item, state) {
			if (
				(args.column.field in item && item[args.column.field] != '') ||
				state
			   ) {

				item[args.column.field] = state;
			} else {
				item[args.column.field] = "";
			}
		};

		this.isValueChanged = function () {
			return (!($inputDate.val() == '' && defaultDateValue == null)) &&
			       (!($inputHours.val() == '' && defaultHoursValue == null)) &&
			       (!($inputMinutes.val() == '' && defaultMinutesValue == null)) &&
			       (!($inputSeconds.val() == '' && defaultSecondsValue == null)) &&
				   (
				   	($inputDate.val() != defaultDateValue) ||
				   	($inputHours.val() != defaultHoursValue) ||
				   	($inputMinutes.val() != defaultMinutesValue) ||
				   	($inputSeconds.val() != defaultSecondsValue)
				   );
		};

		this.validate = function () {
			if (args.column.validator) {
				var validationResults = args.column.validator($inputDate.val());
				if (!validationResults.valid) {
					return validationResults;
				}
			}

			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function SyncWithEditor(args) {
		let $input;
		let defaultValue;
		let scope = this;
		let SyncWithValues = {};

		//this.keyCaptureList = [ Slick.keyCode.UP, Slick.keyCode.DOWN, Slick.keyCode.ENTER ];
		this.keyCaptureList = [ 38, 40, 13 ];

		this.init = function () {
			function split( val ) {
				return val.split( /,\s*/ );
			}
			function extractLast( term ) {
				return split( term ).pop();
			}
			$input = $("<INPUT id='tags' class='editor-text' />");
			$input.appendTo(args.container);
			$input.focus().select();

			$input.autocomplete({
				minLength: 2,
				source: function (request, response) {
					//let term = request.term;
					let term = extractLast(request.term);

					let ajaxarr = {};
					ajaxarr.action = 'wpmelon_wcabe_custom';
					ajaxarr.type = 'autocomplete_search_customsyncwith';
					ajaxarr.nonce = W3ExABE.nonce;
					ajaxarr.term = term;
					jQuery.ajax({
						type : 'post',
						dataType : 'json',
						url : W3ExABE.ajaxurl,
						data : ajaxarr,
						success: function(reply) {
							response( reply.keys );
							//SyncWithValues = reply.values;
							SyncWithValues = {
								...SyncWithValues,
								...reply.values
							};
							response(reply.keys);
						},
						error: function(xhr, status, error) {
							console.log('Error refreshing the Sync With autocomplete field: '+xhr.responseText+' Check if integration is activated.');
						}
					});

				},
				search: function() {
					// custom minLength
					var term = extractLast( this.value );
					if ( term.length < 2 ) {
						return false;
					}
				},
				select: function( event, ui ) {
					var terms = split( this.value );
					// remove the current input
					terms.pop();
					// add the selected item
					terms.push( ui.item.value );
					// add placeholder to get the comma-and-space at the end
					terms.push( '' );
					this.value = terms.join( ', ' );
					return false;
				}
			});
		};

		this.destroy = function () {
			$input.autocomplete('destroy');
			$input.remove();
		};

		this.focus = function () {
			$input.focus();
		};

		this.loadValue = function (item) {
			defaultValue = item[args.column.field];
			$input.val(defaultValue);
			$input[0].defaultValue = defaultValue;
			$input.select();
		};

		this.serializeValue = function () {
			if ($input.val() && $input.val() != '') {
				let values = $input.val().split(', ');
				let valuesArr = [];
				values.forEach(function(val) {
					if (SyncWithValues[val] != undefined) {
						valuesArr.push(SyncWithValues[val]);
					} else {
						valuesArr.push(val);
					}
				});
				return valuesArr.join(', ');
			} else {
				return $input.val();
			}
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return (!($input.val() == "" && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function AutocompleteProductSelectorEditor(args) {
		let $input;
		let defaultValue;
		let scope = this;
		let AutocompleteProductSelectorValues = {};

		//this.keyCaptureList = [ Slick.keyCode.UP, Slick.keyCode.DOWN, Slick.keyCode.ENTER ];
		this.keyCaptureList = [ 38, 40, 13 ];

		this.init = function () {
			function split( val ) {
				return val.split( /,\s*/ );
			}
			function extractLast( term ) {
				return split( term ).pop();
			}
			$input = $("<INPUT id='tags' class='editor-text' />");
			$input.appendTo(args.container);
			$input.focus().select();

			$input.autocomplete({
				minLength: 2,
				source: function (request, response) {
					//let term = request.term;
					let term = extractLast(request.term);

					let ajaxarr = {};
					ajaxarr.action = 'wpmelon_wcabe';
					ajaxarr.type = 'autocomplete_product_selector';
					ajaxarr.nonce = W3ExABE.nonce;
					ajaxarr.term = term;
					jQuery.ajax({
						type : 'post',
						dataType : 'json',
						url : W3ExABE.ajaxurl,
						data : ajaxarr,
						success: function(reply) {
							response( reply.keys );
							AutocompleteProductSelectorValues = {
								...AutocompleteProductSelectorValues,
								...reply.values
							};
							response(reply.keys);
						},
						error: function(xhr, status, error) {
							console.log('Error refreshing the Autocomplete Product Selector field: '+xhr.responseText);
						}
					});

				},
				search: function() {
					let term = extractLast( this.value );
					if ( term.length < 2 ) {
						return false;
					}
				},
				select: function( event, ui ) {
					let terms = split( this.value );
					// remove the current input
					terms.pop();
					// add the selected item
					terms.push( ui.item.value );
					// add placeholder to get the comma-and-space at the end
					terms.push( '' );
					this.value = terms.join( ', ' );
					return false;
				}
			});
		};

		this.destroy = function () {
			$input.autocomplete('destroy');
			$input.remove();
		};

		this.focus = function () {
			$input.focus();
		};

		this.loadValue = function (item) {
			defaultValue = item[args.column.field];
			$input.val(defaultValue);
			$input[0].defaultValue = defaultValue;
			$input.select();
		};

		this.serializeValue = function () {
			if ($input.val() && $input.val() != '') {
				let values = $input.val().split(', ');
				let valuesArr = [];
				values.forEach(function(val) {
					if (AutocompleteProductSelectorValues[val] != undefined) {
						valuesArr.push(AutocompleteProductSelectorValues[val]);
					} else {
						valuesArr.push(val);
					}
				});
				return valuesArr.join(', ');
			} else {
				return $input.val();
			}
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return (!($input.val() == '' && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

	function SerializedToCSVEditor(args) {
		var $input;
		var defaultValue;
		var scope = this;

		this.init = function () {
			$input = $('<INPUT type="text" class="editor-text" />')
				.appendTo(args.container)
				.bind('keydown.nav', function (e) {
					if (e.keyCode === $.ui.keyCode.LEFT || e.keyCode === $.ui.keyCode.RIGHT) {
						e.stopImmediatePropagation();
					}
				})
				.focus()
				.select();
		};

		this.destroy = function () {
			$input.remove();
		};

		this.focus = function () {
			$input.focus();
		};

		this.getValue = function () {
			return $input.val();
		};

		this.setValue = function (val) {
			$input.val(val);
		};

		this.loadValue = function (item) {
			if(item[args.column.field] === undefined || item[args.column.field] === null)
				item[args.column.field] = '';
			item[args.column.field] = wcabehelper.unserializeCSV(String(item[args.column.field]));
			item[args.column.field] = String(item[args.column.field]);
			defaultValue = item[args.column.field] || '';
			$input.val(defaultValue);
			$input[0].defaultValue = defaultValue;
			$input.select();
		};

		this.serializeValue = function () {
			return $input.val();
		};

		this.applyValue = function (item, state) {
			item[args.column.field] = state;
		};

		this.isValueChanged = function () {
			return (!($input.val() == "" && defaultValue == null)) && ($input.val() != defaultValue);
		};

		this.validate = function () {
			if (args.column.validator) {
				var validationResults = args.column.validator($input.val());
				if (!validationResults.valid) {
					return validationResults;
				}
			}

			return {
				valid: true,
				msg: null
			};
		};

		this.init();
	}

})(jQuery);