AUI().add(
	'liferay-dynamic-select',
	function(A) {

		/**
		 * OPTIONS
		 *
		 * Required
		 * array {array}: An array of options.
		 * array[i].select {string}: An id of a select box.
		 * array[i].selectId {string}: A JSON object field name for an option value.
		 * array[i].selectDesc {string}: A JSON object field name for an option description.
		 * array[i].selectVal {string}: The value that is displayed in an option field.
		 *
		 * Callbacks
		 * array[i].selectData {function}: Returns a JSON array to populate the next select box.
		 */

		var DynamicSelect = function(array) {
			var instance = this;

			instance.array = array;

			A.each(
				array,
				function(item, index, collection) {
					var id = item.select;
					var select = A.one('#' + id);
					var selectData = item.selectData;

					if (select) {
						select.attr('data-componentType', 'dynamic_select');

						var prevSelectVal = null;

						if (index > 0) {
							prevSelectVal = array[index - 1].selectVal;
						}

						selectData(
							function(list) {
								instance._updateSelect(index, list);
							},
							prevSelectVal
						);

						select.attr('name', id);

						select.on(
							'change',
							function() {
								instance._callSelectData(index);
							}
						);
					}
				}
			);
		};

		DynamicSelect.prototype = {
			_callSelectData: function(i) {
				var instance = this;

				var array = instance.array;

				if ((i + 1) < array.length) {
					var curSelect = A.one('#' + array[i].select);
					var nextSelectData = array[i + 1].selectData;

					nextSelectData(
						function(list) {
							instance._updateSelect(i + 1, list);
						},
						curSelect && curSelect.val()
					);
				}
			},

			_updateSelect: function(i, list) {
				var instance = this;

				var options = instance.array[i];

				var select = A.one('#' + options.select);
				var selectId = options.selectId;
				var selectDesc = options.selectDesc;
				var selectVal = options.selectVal;
				var selectNullable = options.selectNullable || true;

				var selectOptions = [];

				if (selectNullable) {
					selectOptions.push('<option value="0"></option>');
				}

				A.each(
					list,
					function(item, index, collection) {
						var key = item[selectId];
						var value = item[selectDesc];

						selectOptions.push('<option value="' + key + '">' + value + '</option>');
					}
				);

				selectOptions = selectOptions.join('');

				if (select) {
					select.empty().append(selectOptions).val(selectVal);

					if (Liferay.Browser.isIe()) {
						select.setStyle('width', 'auto');
					}
				}
			}
		};

		Liferay.DynamicSelect = DynamicSelect;
	},
	'',
	{
		requires: ['aui-base']
	}
);
