AUI().add(
	'liferay-auto-fields',
	function(A) {
		var Lang = A.Lang;

		var CSS_AUTOROW_CONTROLS = 'lfr-autorow-controls';

		var CSS_ICON_LOADING = 'loading-animation';

		var TPL_INPUT_HIDDEN = '<input name="{name}" type="hidden" />';

		var TPL_AUTOROW_CONTROLS = '<span class="row-controls ' + CSS_AUTOROW_CONTROLS + '"><a href="javascript:;" class="add-row">{0}</a><a href="javascript:;" class="delete-row modify-link">{1}</a></span>';

		var TPL_ADD_BUTTON = '<button type="button" class="add-row aui-buttonitem-content aui-buttonitem aui-state-default aui-buttonitem-icon-only aui-toolbar-first aui-toolbar-item" title=""><span class="aui-buttonitem-icon aui-icon aui-icon-plus"></span></button>';

		var TPL_DELETE_BUTTON = '<button type="button" class="delete-row aui-buttonitem-content aui-buttonitem aui-state-default aui-buttonitem-icon-only aui-toolbar-last aui-toolbar-item" title=""><span class="aui-buttonitem-icon aui-icon aui-icon-minus"></span></button>';

		var TPL_AUTOROW_CONTROLS =
			'<span class="lfr-autorow-controls aui-toolbar aui-toolbar-horizontal"><span class="aui-toolbar-content">' +
				TPL_ADD_BUTTON +
				TPL_DELETE_BUTTON +
			'</span></span>';

		var TPL_LOADING = '<div class="' + CSS_ICON_LOADING + '"></div>';

		/**
		 * OPTIONS
		 *
		 * Required
		 * container {string|object}: A selector that contains the rows you wish to duplicate.
		 * baseRows {string|object}: A selector that defines which fields are duplicated.
		 *
		 * Optional
		 * fieldIndexes {string}: The name of the POST parameter that will contain a list of the order for the fields.
		 * sortable{boolean}: Whether or not the rows should be sortable
		 * sortableHandle{string}: A selector that defines a handle for the sortables
		 *
		 */

		var AutoFields = A.Component.create(
			{
				EXTENDS: A.Base,
				prototype: {

					initializer: function(config) {
						var instance = this;

						instance.config = config;
					},

					render: function() {
						var instance = this;

						var config = instance.config;

						var contentBox = A.one(config.contentBox);

						var baseRows = contentBox.all(config.baseRows || '.lfr-form-row');
						var baseContainer = A.Node.create('<div class="lfr-form-row"><div class="row-fields"></div></div>');

						instance._contentBox = contentBox;
						instance._guid = baseRows.size();

						instance.url = config.url;

						instance._undoManager = new Liferay.UndoManager().render(contentBox);

						if (config.fieldIndexes) {
							instance._fieldIndexes = A.all('[name=' + config.fieldIndexes + ']');

							if (!instance._fieldIndexes.size()) {
								instance._fieldIndexes = A.Node.create('<input name="' + config.fieldIndexes + '" type="hidden" />');

								contentBox.append(instance._fieldIndexes);
							}
						}
						else {
							instance._fieldIndexes = A.all([]);
						}

						contentBox.delegate(
							'click',
							function(event) {
								var link = event.currentTarget;
								var currentRow = link.ancestor('.lfr-form-row');

								if (link.hasClass('add-row')) {
									instance.addRow(currentRow);
								}
								else if (link.hasClass('delete-row')) {
									link.fire('change');

									instance.deleteRow(currentRow);
								}
							},
							'.lfr-autorow-controls .aui-buttonitem'
						);

						baseRows.each(
							function(item, index, collection) {
								var formRow;
								var firstChild;

								if (item.hasClass('lfr-form-row')) {
									formRow = item;
								}
								else {
									formRow = baseContainer.clone();
									firstChild = formRow.one('> div');
									firstChild.append(item);
								}

								formRow.append(TPL_AUTOROW_CONTROLS);

								if (!contentBox.contains(formRow)) {
									contentBox.append(formRow);
								}

								if (index == 0) {
									instance._rowTemplate = formRow.clone();
									instance._clearForm(instance._rowTemplate);
								}
							}
						);

						if (config.sortable){
							instance._makeSortable(config.sortableHandle);
						}

						Liferay.on(
							'submitForm',
							function(event) {
								var form = event.form;

								form.all('.lfr-form-row').each(instance._clearHiddenRows);

								var fieldOrder = instance.serialize();

								instance._fieldIndexes.val(fieldOrder);
							}
						);

						instance._undoManager.on(
							'clearList',
							function(event) {
								contentBox.all('.lfr-form-row').each(instance._clearHiddenRows);
							}
						);
					},

					addRow: function(node) {
						var instance = this;

						var clone = instance._createClone(node);

						clone.resetId();

						node.placeAfter(clone);

						Liferay.Util.focusFormField(clone.one('input[type=text], input[type=password], textarea'));

						instance.fire(
							'clone',
							{
								guid: instance._guid,
								originalRow: node,
								row: clone
							}
						);

						if (instance._sortable) {
							instance._addHandleClass(clone);
						}
					},

					deleteRow: function(node) {
						var instance = this;

						var visible = 0;

						instance._contentBox.all('.lfr-form-row').each(
							function(item, index, collection) {
								if (!item.hasClass('aui-helper-hidden') && !item.hasClass('aui-helper-hidden-accessible')) {
									visible++;
								}
							}
						);

						var deleteRow = (visible > 1);

						if (visible == 1) {
							instance.addRow(node);

							deleteRow = true;
						}

						if (deleteRow) {
							node.hide();

							instance._undoManager.add(
								function(stateData) {
									node.show();
								}
							);

							instance.fire(
								'delete',
								{
									deletedRow: node,
									guid: instance._guid
								}
							);
						}
					},

					serialize: function(filter) {
						var instance = this;

						var visible = [];

						instance._contentBox.all('.lfr-form-row').each(
							function(item, index, collection) {
								if (!item.hasClass('aui-helper-hidden') && !item.hasClass('aui-helper-hidden-accessible')) {
									visible.push(item);
								}
							}
						);

						var visibleRows = A.all(visible);
						var serializedData = [];

						if (filter) {
							serializedData = filter.call(instance, visibleRows) || [];
						}
						else {
							visibleRows.each(
								function(item, index, collection) {
									var formField = item.one('input, textarea, select');
									var fieldId = formField.attr('id');

									if (!fieldId) {
										fieldId = formField.attr('name');
									}

									fieldId = (fieldId || '').match(/([0-9]+)$/);

									if (fieldId && fieldId[0]) {
										serializedData.push(fieldId[0]);
									}
								}
							);
						}

						return serializedData.join();
					},

					_addHandleClass: function(node) {
						var instance = this;

						var sortableHandle = instance.config.sortableHandle;

						if (sortableHandle) {
							node.all(sortableHandle).addClass('handle-sort-vertical');
						}
					},

					_clearForm: function(node) {
						node.all('input, select, textarea').each(
							function(item, index, collection) {
								var type = item.getAttribute('type');
								var tag = item.get('nodeName').toLowerCase();

								if (type == 'text' || type == 'password' || tag == 'textarea') {
									item.val('');
								}
								else if (type == 'checkbox' || type == 'radio') {
									item.set('checked', false);
								}
								else if (tag == 'select') {
									item.set('selectedIndex', -1);
								}
							}
						);
					},

					_clearHiddenRows: function(item, index, collection) {
						if (item.hasClass('aui-helper-hidden') || item.hasClass('aui-helper-hidden-accessible')) {
							item.remove(true);
						}
					},

					_createClone: function(node) {
						var instance = this;

						var currentRow = node;
						var clone = currentRow.clone();
						var guid = (++instance._guid);

						var clonedRow;

						if (instance.url) {
							clonedRow = instance._createCloneFromURL(clone, guid);
						}
						else {
							clonedRow = instance._createCloneFromMarkup(clone, guid);
						}

						return clonedRow;
					},

					_createCloneFromMarkup: function(node, guid) {
						var instance = this;

						node.all('input, select, textarea, span').each(
							function(item, index, collection) {
								var oldName = item.attr('name') || item.attr('id');
								var originalName = oldName.replace(/([0-9]+)$/, '');
								var newName = originalName + guid;
								var inputType = item.attr('type');
								var inputNodeName = item.attr('nodeName');

								if (inputType == 'radio') {
									oldName = item.attr('id');

									item.attr('checked', '');
									item.attr('value', guid);
									item.attr('id', newName);
								}
								else if (inputNodeName == 'button' || inputNodeName == 'span') {
									if (oldName) {
										item.attr('id', newName);
									}
								}
								else {
									item.attr('name', newName);
									item.attr('id', newName);
								}

								node.all('label[for=' + oldName + ']').attr('for', newName);
							}
						);

						instance._clearForm(node);

						node.all('input[type=hidden]').val('');

						return node;
					},

					_createCloneFromURL: function(node, guid) {
						var instance = this;

						var contentBox = node.one('> div');

						contentBox.html(TPL_LOADING);

						contentBox.plug(A.Plugin.ParseContent);

						A.io.request(
							instance.url,
							{
								data: {
									index: guid
								},
								on: {
									success: function(event, id, obj) {
										var responseData = this.get('responseData');

										contentBox.setContent(responseData);
									}
								}
							}
						);

						return node;
					},

					_makeSortable: function(sortableHandle) {
						var instance = this;

						var rows = instance._contentBox.all('.lfr-form-row');

						instance._addHandleClass(rows);

						instance._sortable = new A.Sortable(
							{
								constrain: {
									stickY: true
								},
								dd: {
									handles: [sortableHandle]
								},
								nodes: rows
							}
						);

						instance._undoManager.on(
							'clearList',
							function(event) {
								rows.each(
									function(item, index, collection) {
										if (item.hasClass('aui-helper-hidden') || item.hasClass('aui-helper-hidden-accessible')) {
											A.DD.DDM.getDrag(item).destroy();
										}
									}
								);
							}
						);
					},

					_guid: 0
				}
			}
		);

		Liferay.AutoFields = AutoFields;
	},
	'',
	{
		requires: ['aui-base', 'aui-data-set', 'aui-io-request', 'aui-parse-content', 'aui-sortable', 'base', 'liferay-undo-manager']
	}
);
