//------------------------------------------------------------------------------
// Copyright (c) 2005, 2006 IBM Corporation and others.
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// which accompanies this distribution, and is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// Contributors:
// IBM Corporation - initial implementation
//------------------------------------------------------------------------------
package org.eclipse.epf.authoring.ui.forms;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

import org.eclipse.emf.edit.ui.provider.AdapterFactoryContentProvider;
import org.eclipse.emf.edit.ui.provider.AdapterFactoryLabelProvider;
import org.eclipse.epf.authoring.ui.AuthoringUIPlugin;
import org.eclipse.epf.authoring.ui.AuthoringUIResources;
import org.eclipse.epf.authoring.ui.AuthoringUIText;
import org.eclipse.epf.authoring.ui.dialogs.ItemsFilterDialog;
import org.eclipse.epf.authoring.ui.editors.MethodElementEditor;
import org.eclipse.epf.authoring.ui.filters.ContentFilter;
import org.eclipse.epf.authoring.ui.filters.WorkProductFilter;
import org.eclipse.epf.authoring.ui.richtext.IMethodRichText;
import org.eclipse.epf.authoring.ui.util.UIHelper;
import org.eclipse.epf.library.edit.IFilter;
import org.eclipse.epf.library.edit.TngAdapterFactory;
import org.eclipse.epf.library.edit.command.IActionManager;
import org.eclipse.epf.library.edit.itemsfilter.FilterConstants;
import org.eclipse.epf.library.edit.itemsfilter.VariabilityBaseElementFilter;
import org.eclipse.epf.library.edit.ui.UserInteractionHelper;
import org.eclipse.epf.library.edit.util.TngUtil;
import org.eclipse.epf.uma.Artifact;
import org.eclipse.epf.uma.ArtifactDescription;
import org.eclipse.epf.uma.ContentElement;
import org.eclipse.epf.uma.Deliverable;
import org.eclipse.epf.uma.DeliverableDescription;
import org.eclipse.epf.uma.FulfillableElement;
import org.eclipse.epf.uma.Outcome;
import org.eclipse.epf.uma.UmaPackage;
import org.eclipse.epf.uma.VariabilityElement;
import org.eclipse.epf.uma.WorkProduct;
import org.eclipse.epf.uma.util.AssociationHelper;
import org.eclipse.jface.viewers.ILabelProvider;
import org.eclipse.jface.viewers.ISelectionChangedListener;
import org.eclipse.jface.viewers.IStructuredContentProvider;
import org.eclipse.jface.viewers.IStructuredSelection;
import org.eclipse.jface.viewers.SelectionChangedEvent;
import org.eclipse.jface.viewers.TableViewer;
import org.eclipse.swt.SWT;
import org.eclipse.swt.events.FocusAdapter;
import org.eclipse.swt.events.FocusEvent;
import org.eclipse.swt.events.SelectionAdapter;
import org.eclipse.swt.events.SelectionEvent;
import org.eclipse.swt.layout.GridData;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Button;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.Listener;
import org.eclipse.swt.widgets.Table;
import org.eclipse.ui.IEditorInput;
import org.eclipse.ui.IEditorSite;
import org.eclipse.ui.PlatformUI;
import org.eclipse.ui.forms.editor.FormEditor;
import org.eclipse.ui.forms.widgets.FormToolkit;
import org.eclipse.ui.forms.widgets.Section;

/**
 * The Description page in a Work Product (Artifact, Deliverable and Outcome)
 * editor.
 * 
 * @author Shilpa Toraskar
 * @author Kelvin Low
 * @since 1.0
 */
public class WorkProductDescriptionPage extends DescriptionFormPage {

	private static final String FORM_PAGE_ID = "workProductDescriptionPage"; //$NON-NLS-1$	

	private IMethodRichText ctrl_impact;

	private IMethodRichText ctrl_reason;

	// For Artifact
	private IMethodRichText ctrl_brief_outline;

	private IMethodRichText ctrl_representation_options;

	// For Deliverable
	private IMethodRichText ctrl_external_desc, ctrl_packaging_guidance;

	private IMethodRichText ctrl_representation;

	private IMethodRichText ctrl_notation;

	private WorkProduct workProduct;

	// For slots
	private Button ctrl_slot_button;
	
	private Section slotSection;
	
	private Composite slotComposite;
	
	private String slotSectionDescription;
	
	private Button addSlotButton, removeSlotButton;
	
	private Table ctrl_slot;
	
	private TableViewer slotViewer;

	// slots content provider
	private IStructuredContentProvider slotContentProvider = new AdapterFactoryContentProvider(
			TngAdapterFactory.INSTANCE
					.getNavigatorView_ComposedAdapterFactory()) {
		public Object[] getElements(Object object) {
			List<FulfillableElement> list = new ArrayList<FulfillableElement>();
			list.addAll(workProduct.getFulfills());
			return list.toArray();
		}
	};

	private ILabelProvider slotLabelProvider = new AdapterFactoryLabelProvider(
			TngAdapterFactory.INSTANCE
					.getNavigatorView_ComposedAdapterFactory());
	/**
	 * Creates a new instance.
	 */
	public WorkProductDescriptionPage(FormEditor editor) {
		super(editor, FORM_PAGE_ID, AuthoringUIText.DESCRIPTION_PAGE_TITLE);
	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#init(org.eclipse.ui.IEditorSite,
	 *      org.eclipse.ui.IEditorInput)
	 */
	public void init(IEditorSite site, IEditorInput input) {
		super.init(site, input);
		workProduct = (WorkProduct) contentElement;
		purposeOn = true;
		externalIdOn = true;
		if (!(contentElement instanceof Outcome)) {
			notationSectionOn = true;
		}
		tailoringSectionOn = true;
		iconSectionOn = true;
		slotSectionOn = true;
	}

	protected void createGeneralSectionContent() {
		super.createGeneralSectionContent();

		ctrl_slot_button = toolkit.createButton(generalComposite,
				AuthoringUIResources.workproductDescriptionPage_slot_text,
				SWT.CHECK);
		GridData data = new GridData();
		data.horizontalSpan = 3;
		ctrl_slot_button.setLayoutData(data);
	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#createNotationSectionContent()
	 */
	protected void createNotationSectionContent() {
		super.createNotationSectionContent();

		if (workProduct instanceof Artifact) {
			ctrl_brief_outline = createRichTextEditWithLinkForSection(toolkit,
					notationComposite, AuthoringUIText.BRIEF_OUTLINE_TEXT, 40,
					400, NOTATION_SECTION_ID);
			ctrl_representation = createRichTextEditWithLinkForSection(toolkit,
					notationComposite, AuthoringUIText.REPRESENTATION_TEXT, 40,
					400, NOTATION_SECTION_ID);
			ctrl_notation = createRichTextEditWithLinkForSection(toolkit,
					notationComposite, AuthoringUIText.NOTATION_TEXT, 40, 400,
					NOTATION_SECTION_ID);
		} else if (workProduct instanceof Deliverable) {
			ctrl_external_desc = createRichTextEditWithLinkForSection(toolkit,
					notationComposite,
					AuthoringUIText.EXTERNAL_DESCRIPTION_TEXT, 40, 400,
					NOTATION_SECTION_ID);
			ctrl_packaging_guidance = createRichTextEditWithLinkForSection(
					toolkit, notationComposite,
					AuthoringUIText.PACKAGING_GUIDANCE_TEXT, 40, 400,
					NOTATION_SECTION_ID);
		}
	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#createTailoringSectionContent()
	 */
	protected void createTailoringSectionContent() {
		super.createTailoringSectionContent();
		ctrl_impact = createRichTextEditWithLinkForSection(toolkit,
				tailoringComposite, AuthoringUIText.IMPACT_OF_NOT_HAVING_TEXT,
				40, 400, TAILORING_SECTION_ID);
		ctrl_reason = createRichTextEditWithLinkForSection(toolkit,
				tailoringComposite,
				AuthoringUIText.REASON_FOR_NOT_NEEDING_TEXT, 40, 400,
				TAILORING_SECTION_ID);
		if (workProduct instanceof Artifact) {
			ctrl_representation_options = createRichTextEditWithLinkForSection(
					toolkit, tailoringComposite,
					AuthoringUIText.REPRESENTATION_OPTIONS_TEXT, 40, 400,
					NOTATION_SECTION_ID);
		}
	}

	protected void createEditorContent(FormToolkit toolkit) {
		super.createEditorContent(toolkit);
	}

	/**
	 * Create slot section
	 * 
	 * @param toolkit
	 */
	protected void createSlotSection(FormToolkit toolkit) {

		slotSection = createSection(toolkit, sectionComposite,
				AuthoringUIText.SLOT_SECTION_NAME, this.slotSectionDescription);
		slotComposite = createComposite(toolkit, slotSection);
		slotComposite.setLayoutData(new GridData(GridData.FILL_HORIZONTAL));
		slotComposite.setLayout(new GridLayout(5, false));
		createSlotSectionContent();
		toolkit.paintBordersFor(slotComposite);
	}

	/**
	 * 
	 */
	private void createSlotSectionContent() {
		createLabel(toolkit, slotComposite,
				AuthoringUIResources.slotsLabel_text, 2);
		ctrl_slot = createTable(toolkit, slotComposite, SWT.MULTI
				| SWT.READ_ONLY, GridData.FILL_HORIZONTAL | GridData.BEGINNING,
				5, 300, 1, 2);
		{
			GridData gridData = new GridData(GridData.FILL_HORIZONTAL
					| GridData.BEGINNING);
			gridData.horizontalSpan = 2;
			gridData.heightHint = 30;
			gridData.widthHint = 300;
			ctrl_slot.setLayoutData(gridData);
		}

		slotViewer = new TableViewer(ctrl_slot);
		slotViewer.setContentProvider(slotContentProvider);
		slotViewer.setLabelProvider(slotLabelProvider);
		slotViewer.setInput(methodElement);

		Composite buttonPane = createComposite(toolkit, slotComposite,
				GridData.HORIZONTAL_ALIGN_END, 1, 1, 1);
		{
			GridData gridData = new GridData(GridData.HORIZONTAL_ALIGN_END);
			buttonPane.setLayoutData(gridData);
		}

		addSlotButton = toolkit.createButton(buttonPane,
				AuthoringUIText.ADD_BUTTON_TEXT, SWT.SIMPLE);
		{
			GridData gridData = new GridData(GridData.FILL_HORIZONTAL);
			gridData.widthHint = BUTTON_WIDTH;
			addSlotButton.setLayoutData(gridData);
		}
		
		removeSlotButton = toolkit.createButton(buttonPane,
				AuthoringUIText.REMOVE_BUTTON_TEXT, SWT.SIMPLE);
		{
			GridData gridData = new GridData(GridData.FILL_HORIZONTAL);
			gridData.widthHint = BUTTON_WIDTH;
			removeSlotButton.setLayoutData(gridData);
		}
		removeSlotButton.setEnabled(false);
	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#addListeners()
	 */
	protected void addListeners() {
		final IActionManager actionMgr = ((MethodElementEditor) getEditor())
				.getActionManager();

		super.addListeners();

		// add all slot related listeners
		
		if (slotSectionOn) {
			addSlotListeners();
		}
		
		
		if (purposeOn) {
			ctrl_purpose.setModalObject(contentElement.getPresentation());
			ctrl_purpose.setModalObjectFeature(UmaPackage.eINSTANCE
					.getWorkProductDescription_Purpose());
			ctrl_purpose.addModifyListener(contentModifyListener);
			ctrl_purpose.addListener(SWT.Deactivate, new Listener() {
				public void handleEvent(Event e) {
					IMethodRichText control = descExpandFlag ? ctrl_expanded
							: ctrl_purpose;
					if (!control.getModified()) {
						return;
					}
					String oldContent = ((org.eclipse.epf.uma.WorkProductDescription) workProduct
							.getPresentation()).getPurpose();
					if (((MethodElementEditor) getEditor()).mustRestoreValue(
							control, oldContent)) {
						return;
					}
					String newContent = control.getText();
					if (!newContent.equals(oldContent)) {
						boolean success = actionMgr.doAction(
								IActionManager.SET, contentElement
										.getPresentation(),
								UmaPackage.eINSTANCE
										.getWorkProductDescription_Purpose(),
								newContent, -1);
						if (success && isVersionSectionOn()) {
							updateChangeDate();
						}
					}
				}
			});
		}

		if (externalIdOn) {
			ctrl_external_id.addModifyListener(contentModifyListener);
			ctrl_external_id.addFocusListener(new FocusAdapter() {
				public void focusGained(FocusEvent e) {
					((MethodElementEditor) getEditor())
							.setCurrentFeatureEditor(e.widget,
									UmaPackage.eINSTANCE
											.getContentDescription_ExternalId());
				}

				public void focusLost(FocusEvent e) {
					String oldContent = ((org.eclipse.epf.uma.WorkProductDescription) workProduct
							.getPresentation()).getExternalId();
					if (((MethodElementEditor) getEditor()).mustRestoreValue(
							ctrl_external_id, oldContent)) {
						return;
					}
					String newContent = ctrl_external_id.getText();
					if (!newContent.equals(oldContent)) {
						boolean success = actionMgr.doAction(
								IActionManager.SET, contentElement
										.getPresentation(),
								UmaPackage.eINSTANCE
										.getContentDescription_ExternalId(),
								newContent, -1);
						if (success && isVersionSectionOn()) {
							updateChangeDate();
						}
					}
				}
			});
		}

		if (tailoringSectionOn) {
			ctrl_impact.setModalObject(contentElement.getPresentation());
			ctrl_impact.setModalObjectFeature(UmaPackage.eINSTANCE
					.getWorkProductDescription_ImpactOfNotHaving());
			ctrl_impact.addModifyListener(contentModifyListener);
			ctrl_impact.addListener(SWT.Deactivate, new Listener() {
				public void handleEvent(Event e) {
					IMethodRichText control = descExpandFlag ? ctrl_expanded
							: ctrl_impact;
					if (!control.getModified()) {
						return;
					}
					String oldContent = ((org.eclipse.epf.uma.WorkProductDescription) workProduct
							.getPresentation()).getImpactOfNotHaving();
					if (((MethodElementEditor) getEditor()).mustRestoreValue(
							control, oldContent)) {
						return;
					}
					String newContent = control.getText();
					if (!newContent.equals(oldContent)) {
						boolean success = actionMgr
								.doAction(
										IActionManager.SET,
										contentElement.getPresentation(),
										UmaPackage.eINSTANCE
												.getWorkProductDescription_ImpactOfNotHaving(),
										newContent, -1);
						if (success && isVersionSectionOn()) {
							updateChangeDate();
						}
					}
				}
			});

			ctrl_reason.setModalObject(contentElement.getPresentation());
			ctrl_reason.setModalObjectFeature(UmaPackage.eINSTANCE
					.getWorkProductDescription_ReasonsForNotNeeding());
			ctrl_reason.addModifyListener(contentModifyListener);
			ctrl_reason.addListener(SWT.Deactivate, new Listener() {
				public void handleEvent(Event e) {
					IMethodRichText control = descExpandFlag ? ctrl_expanded
							: ctrl_reason;
					if (!control.getModified()) {
						return;
					}
					String oldContent = ((org.eclipse.epf.uma.WorkProductDescription) workProduct
							.getPresentation()).getReasonsForNotNeeding();
					if (((MethodElementEditor) getEditor()).mustRestoreValue(
							control, oldContent)) {
						return;
					}
					String newContent = control.getText();
					if (!newContent.equals(oldContent)) {
						boolean success = actionMgr
								.doAction(
										IActionManager.SET,
										contentElement.getPresentation(),
										UmaPackage.eINSTANCE
												.getWorkProductDescription_ReasonsForNotNeeding(),
										newContent, -1);
						if (success && isVersionSectionOn()) {
							updateChangeDate();
						}
					}
				}
			});

			if (workProduct instanceof Artifact) {
				ctrl_representation_options.setModalObject(contentElement
						.getPresentation());
				ctrl_representation_options
						.setModalObjectFeature(UmaPackage.eINSTANCE
								.getArtifactDescription_RepresentationOptions());
				ctrl_representation_options
						.addModifyListener(contentModifyListener);
				ctrl_representation_options.addListener(SWT.Deactivate,
						new Listener() {
							public void handleEvent(Event e) {
								IMethodRichText control = descExpandFlag ? ctrl_expanded
										: ctrl_representation_options;
								if (!control.getModified()) {
									return;
								}
								String oldContent = ((org.eclipse.epf.uma.ArtifactDescription) workProduct
										.getPresentation())
										.getRepresentationOptions();
								if (((MethodElementEditor) getEditor())
										.mustRestoreValue(control, oldContent)) {
									return;
								}
								String newContent = control.getText();
								if (!newContent.equals(oldContent)) {
									boolean success = actionMgr
											.doAction(
													IActionManager.SET,
													contentElement
															.getPresentation(),
													UmaPackage.eINSTANCE
															.getArtifactDescription_RepresentationOptions(),
													newContent, -1);
									if (success && isVersionSectionOn()) {
										updateChangeDate();
									}
								}
							}
						});
			}
		}

		if (notationSectionOn) {
			if (workProduct instanceof Artifact) {
				ctrl_brief_outline.setModalObject(contentElement
						.getPresentation());
				ctrl_brief_outline.setModalObjectFeature(UmaPackage.eINSTANCE
						.getArtifactDescription_BriefOutline());
				ctrl_brief_outline.addModifyListener(contentModifyListener);
				ctrl_brief_outline.addListener(SWT.Deactivate, new Listener() {
					public void handleEvent(Event e) {
						IMethodRichText control = descExpandFlag ? ctrl_expanded
								: ctrl_brief_outline;
						if (!control.getModified()) {
							return;
						}
						String oldContent = ((org.eclipse.epf.uma.ArtifactDescription) workProduct
								.getPresentation()).getBriefOutline();
						if (((MethodElementEditor) getEditor())
								.mustRestoreValue(control, oldContent)) {
							return;
						}
						String newContent = descExpandFlag ? ctrl_expanded
								.getText() : ctrl_brief_outline.getText();
						if (!newContent.equals(oldContent)) {
							boolean success = actionMgr
									.doAction(
											IActionManager.SET,
											contentElement.getPresentation(),
											UmaPackage.eINSTANCE
													.getArtifactDescription_BriefOutline(),
											newContent, -1);
							if (success && isVersionSectionOn()) {
								updateChangeDate();
							}
						}
					}
				});

				ctrl_representation.setModalObject(contentElement
						.getPresentation());
				ctrl_representation.setModalObjectFeature(UmaPackage.eINSTANCE
						.getArtifactDescription_Representation());
				ctrl_representation.addModifyListener(contentModifyListener);
				ctrl_representation.addListener(SWT.Deactivate, new Listener() {
					public void handleEvent(Event e) {
						IMethodRichText control = descExpandFlag ? ctrl_expanded
								: ctrl_representation;
						if (!control.getModified()) {
							return;
						}
						String oldContent = ((org.eclipse.epf.uma.ArtifactDescription) workProduct
								.getPresentation()).getRepresentation();
						if (((MethodElementEditor) getEditor())
								.mustRestoreValue(control, oldContent)) {
							return;
						}
						String newContent = descExpandFlag ? ctrl_expanded
								.getText() : ctrl_representation.getText();
						if (!newContent.equals(oldContent)) {
							boolean success = actionMgr
									.doAction(
											IActionManager.SET,
											contentElement.getPresentation(),
											UmaPackage.eINSTANCE
													.getArtifactDescription_Representation(),
											newContent, -1);
							if (success && isVersionSectionOn()) {
								updateChangeDate();
							}
						}
					}
				});

				ctrl_notation.setModalObject(contentElement.getPresentation());
				ctrl_notation.setModalObjectFeature(UmaPackage.eINSTANCE
						.getArtifactDescription_Notation());
				ctrl_notation.addModifyListener(contentModifyListener);
				ctrl_notation.addListener(SWT.Deactivate, new Listener() {
					public void handleEvent(Event e) {
						IMethodRichText control = descExpandFlag ? ctrl_expanded
								: ctrl_notation;
						if (!control.getModified()) {
							return;
						}
						String oldContent = ((org.eclipse.epf.uma.ArtifactDescription) workProduct
								.getPresentation()).getNotation();
						if (((MethodElementEditor) getEditor())
								.mustRestoreValue(control, oldContent)) {
							return;
						}
						String newContent = descExpandFlag ? ctrl_expanded
								.getText() : ctrl_notation.getText();
						if (!newContent.equals(oldContent)) {
							boolean success = actionMgr.doAction(
									IActionManager.SET, contentElement
											.getPresentation(),
									UmaPackage.eINSTANCE
											.getArtifactDescription_Notation(),
									newContent, -1);
							if (success && isVersionSectionOn()) {
								updateChangeDate();
							}
						}
					}
				});
			}

			if (workProduct instanceof Deliverable) {
				ctrl_external_desc.setModalObject(contentElement
						.getPresentation());
				ctrl_external_desc.setModalObjectFeature(UmaPackage.eINSTANCE
						.getDeliverableDescription_ExternalDescription());
				ctrl_external_desc.addModifyListener(contentModifyListener);
				ctrl_external_desc.addListener(SWT.Deactivate, new Listener() {
					public void handleEvent(Event e) {
						IMethodRichText control = descExpandFlag ? ctrl_expanded
								: ctrl_external_desc;
						if (!control.getModified()) {
							return;
						}
						String oldContent = ((org.eclipse.epf.uma.DeliverableDescription) workProduct
								.getPresentation()).getExternalDescription();
						if (((MethodElementEditor) getEditor())
								.mustRestoreValue(control, oldContent)) {
							return;
						}
						String newContent = control.getText();
						if (!newContent.equals(oldContent)) {
							boolean success = actionMgr
									.doAction(
											IActionManager.SET,
											contentElement.getPresentation(),
											UmaPackage.eINSTANCE
													.getDeliverableDescription_ExternalDescription(),
											newContent, -1);
							if (success && isVersionSectionOn()) {
								updateChangeDate();
							}
						}
					}
				});

				ctrl_packaging_guidance.setModalObject(contentElement
						.getPresentation());
				ctrl_packaging_guidance
						.setModalObjectFeature(UmaPackage.eINSTANCE
								.getDeliverableDescription_PackagingGuidance());
				ctrl_packaging_guidance
						.addModifyListener(contentModifyListener);
				ctrl_packaging_guidance.addListener(SWT.Deactivate,
						new Listener() {
							public void handleEvent(Event e) {
								IMethodRichText control = descExpandFlag ? ctrl_expanded
										: ctrl_packaging_guidance;
								if (!control.getModified()) {
									return;
								}
								String oldContent = ((org.eclipse.epf.uma.DeliverableDescription) workProduct
										.getPresentation())
										.getPackagingGuidance();
								if (((MethodElementEditor) getEditor())
										.mustRestoreValue(control, oldContent)) {
									return;
								}
								String newContent = control.getText();
								if (!newContent.equals(oldContent)) {
									boolean success = actionMgr
											.doAction(
													IActionManager.SET,
													contentElement
															.getPresentation(),
													UmaPackage.eINSTANCE
															.getDeliverableDescription_PackagingGuidance(),
													newContent, -1);
									if (success && isVersionSectionOn()) {
										updateChangeDate();
									}
								}
							}
						});
			}
		}
	}
	
	
	/**
	 * Slot Listeners
	 */
	private void addSlotListeners()	{
		// work product slot check button
		ctrl_slot_button.addSelectionListener(new SelectionAdapter() {
			public void widgetSelected(SelectionEvent e) {
				Boolean isSlot = new Boolean(ctrl_slot_button.getSelection());
				if (isSlot.booleanValue()) {
					// if this becoming a slot itself
					List<FulfillableElement> fulFills = workProduct
							.getFulfills();
					if (fulFills != null && fulFills.size() != 0) {
						boolean val = AuthoringUIPlugin
								.getDefault()
								.getMsgDialog()
								.displayConfirmation(
										AuthoringUIResources.slotConfirmDialog_title,
										AuthoringUIResources.wpFulFillsConfirmDialog_message);
						if (!val) {
							ctrl_slot_button.setSelection(workProduct
									.getIsAbstract().booleanValue());
							return;
						} else {
							actionMgr.doAction(IActionManager.REMOVE_MANY,
									workProduct, UmaPackage.eINSTANCE
											.getFulfillableElement_Fulfills(),
									fulFills, -1);
						}
					}
				} else {					
					List fulFills = new ArrayList();
					fulFills.addAll(AssociationHelper.getFullFills(workProduct));
					if (fulFills != null && fulFills.size() != 0) {
						boolean val = AuthoringUIPlugin
								.getDefault()
								.getMsgDialog()
								.displayConfirmation(
										AuthoringUIResources.slotConfirmDialog_title,
										AuthoringUIResources.wpSlotFulFillsConfirmDialog_message);
						if (!val) {
							ctrl_slot_button.setSelection(workProduct
									.getIsAbstract().booleanValue());
							return;
						} else {
							for (Iterator itor = fulFills.iterator(); itor
									.hasNext();) {
								WorkProduct wp = (WorkProduct) itor.next();
								actionMgr
										.doAction(
												IActionManager.REMOVE,
												wp,
												UmaPackage.eINSTANCE
														.getFulfillableElement_Fulfills(),
												workProduct, -1);
							}
						}
					}				
				}
				actionMgr.doAction(IActionManager.SET, methodElement,
						UmaPackage.eINSTANCE.getClassifier_IsAbstract(), isSlot,
						-1);
				
				if (isSlot) {		
					addSlotButton.setEnabled(false);
				}
				else {
					addSlotButton.setEnabled(true);
				}
			}
		});

		addSlotButton.addSelectionListener(new SelectionAdapter() {
			public void widgetSelected(SelectionEvent e) {
				IFilter filter = new WorkProductFilter() {
					protected boolean childAccept(Object obj) {
						if (obj instanceof WorkProduct) {
							System.out.println(obj + " " + ((WorkProduct) obj).getIsAbstract());
							return ((WorkProduct) obj).getIsAbstract().booleanValue();
						}
						return false;
					}
				};
				List<FulfillableElement> alreadyExisting = new ArrayList<FulfillableElement>();
				alreadyExisting.addAll(workProduct.getFulfills());
				ItemsFilterDialog fd = new ItemsFilterDialog(PlatformUI
						.getWorkbench().getActiveWorkbenchWindow().getShell(),
						filter, methodElement,
						FilterConstants.WORKPRODUCTS, alreadyExisting);
				fd.setViewerSelectionSingle(false);
				fd.setBlockOnOpen(true);
				fd.setTitle(FilterConstants.WORKPRODUCTS);
				fd.open();
				
				if (fd.getSelectedItems().size() > 0) {
					boolean status = actionMgr.doAction(
							IActionManager.ADD_MANY, workProduct,
							UmaPackage.eINSTANCE
									.getFulfillableElement_Fulfills(), fd
									.getSelectedItems(), -1);
					if (!status) {
						return;
					}
					// refresh slot viewer
					slotViewer.refresh();	
				}		
			}
		});
		
		removeSlotButton.addSelectionListener(new SelectionAdapter() {
			public void widgetSelected(SelectionEvent e){
				IStructuredSelection selection = (IStructuredSelection) slotViewer
						.getSelection();
				if (selection.size() > 0) {
					boolean status = actionMgr.doAction(IActionManager.REMOVE_MANY, workProduct,
							UmaPackage.eINSTANCE
									.getFulfillableElement_Fulfills(),
							selection.toList(), -1);
					
					if (!status) {
						return;
					}
					// refresh slot viewer
					slotViewer.refresh();
					removeSlotButton.setEnabled(false);
				}
			}	
		});
		
		slotViewer.addSelectionChangedListener(new ISelectionChangedListener() {
			public void selectionChanged(SelectionChangedEvent e) {
				IStructuredSelection selection = (IStructuredSelection) slotViewer
						.getSelection();
				if (selection.size() > 0 && !TngUtil.isLocked(workProduct)) {
					removeSlotButton.setEnabled(true);
				}
			}
		});

	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#refresh(boolean)
	 */
	protected void refresh(boolean editable) {
		super.refresh(editable);
		
		if (slotSectionOn)  {
			ctrl_slot_button.setEnabled(editable);
			
			Boolean isAbstract = workProduct.getIsAbstract();
			if (isAbstract != null && editable) {
				if (isAbstract.booleanValue())  {
					addSlotButton.setEnabled(false);
				} else {
					addSlotButton.setEnabled(true);
				}
			}
			else
				addSlotButton.setEnabled(editable);
			
			IStructuredSelection selection = (IStructuredSelection) slotViewer.getSelection();
			if (selection.size() > 0 && editable) {
				removeSlotButton.setEnabled(true);
			} else {
				removeSlotButton.setEnabled(false);
			}
		}
		
		if (tailoringSectionOn) {
			ctrl_impact.setEditable(editable);
			ctrl_reason.setEditable(editable);
			if (workProduct instanceof Artifact)
				ctrl_representation_options.setEditable(editable);
		}
		if (notationSectionOn) {
			if (workProduct instanceof Artifact) {
				ctrl_brief_outline.setEditable(editable);
				ctrl_representation.setEditable(editable);
				ctrl_notation.setEditable(editable);
			}
			if (workProduct instanceof Deliverable) {
				ctrl_external_desc.setEditable(editable);
				ctrl_packaging_guidance.setEditable(editable);
			}
		}
	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#loadData()
	 */
	protected void loadData() {
		super.loadData();

		// IsAbstract attribute
		if (slotSectionOn) {
			Boolean isAbstract = workProduct.getIsAbstract();
			if (isAbstract != null) {
				ctrl_slot_button.setSelection(isAbstract.booleanValue());
			} else
				ctrl_slot_button.setSelection(false);
		}

		
		org.eclipse.epf.uma.WorkProductDescription content = ((org.eclipse.epf.uma.WorkProductDescription) workProduct
				.getPresentation());

		if (purposeOn) {
			String purpose = content.getPurpose();
			ctrl_purpose.setText(purpose == null ? "" : purpose); //$NON-NLS-1$
		}
		if (tailoringSectionOn) {
			String impact = content.getImpactOfNotHaving();
			String reason = content.getReasonsForNotNeeding();
			ctrl_impact.setText(impact == null ? "" : impact); //$NON-NLS-1$	
			ctrl_reason.setText(reason == null ? "" : reason); //$NON-NLS-1$
			if (workProduct instanceof Artifact) {
				ctrl_representation_options
						.setText(((ArtifactDescription) content)
								.getRepresentationOptions() == null ? "" : ((ArtifactDescription) content).getRepresentationOptions()); //$NON-NLS-1$
			}
		}

		if (externalIdOn) {
			String external_id = content.getExternalId();
			ctrl_external_id.setText(external_id == null ? "" : external_id); //$NON-NLS-1$
		}

		if (notationSectionOn) {
			if (workProduct instanceof Artifact) {
				ctrl_brief_outline
						.setText(((ArtifactDescription) content)
								.getBriefOutline() == null ? "" : ((ArtifactDescription) content).getBriefOutline()); //$NON-NLS-1$			
				ctrl_representation
						.setText(((ArtifactDescription) content)
								.getRepresentation() == null ? "" : ((ArtifactDescription) content).getRepresentation()); //$NON-NLS-1$
				ctrl_notation
						.setText(((ArtifactDescription) content).getNotation() == null ? "" : ((ArtifactDescription) content).getNotation()); //$NON-NLS-1$
			} else if (workProduct instanceof Deliverable) {
				ctrl_external_desc
						.setText(((DeliverableDescription) content)
								.getExternalDescription() == null ? "" : ((DeliverableDescription) content).getExternalDescription()); //$NON-NLS-1$
				ctrl_packaging_guidance
						.setText(((DeliverableDescription) content)
								.getPackagingGuidance() == null ? "" : ((DeliverableDescription) content).getPackagingGuidance()); //$NON-NLS-1$
			}
		}

	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#getContentElement()
	 */
	protected Object getContentElement() {
		return workProduct;
	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#getTabString()
	 */
	protected String getTabString() {
		return FilterConstants.WORKPRODUCTS;
	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#getFilter()
	 */
	protected IFilter getFilter() {
		filter = new WorkProductFilter() {
			List badlist = new ArrayList();

			protected boolean childAccept(Object obj) {
				if (super.childAccept(obj)) {
					if (obj instanceof ContentElement) {
						if (obj instanceof WorkProduct
								&& ((ContentElement) getHelper()
										.getContentElement()).getType().equals(
										((ContentElement) obj).getType())) {
							// prevent circular dependency from deliverable
							// parts and variability
							if (obj instanceof Deliverable
									&& workProduct instanceof Deliverable) {
								if (!UIHelper.checkCircularDeliverables(
										(Deliverable) obj,
										(Deliverable) workProduct)) {
									badlist.add(obj);
									return false;
								}
								if (((Deliverable) obj)
										.getVariabilityBasedOnElement() != null) {
									if (!UIHelper
											.checkCircularDeliverables(
													(Deliverable) ((Deliverable) obj)
															.getVariabilityBasedOnElement(),
													(Deliverable) workProduct)) {
										badlist.add(obj);
										return false;
									}
								}
								Iterator it1 = TngUtil
										.getGeneralizers((Deliverable) obj);
								while (it1.hasNext()) {
									VariabilityElement ve = (VariabilityElement) it1
											.next();
									if (ve != null) {
										if (ve.equals(obj))
											return false;
										if (!UIHelper
												.checkCircularDeliverables(
														(Deliverable) ve,
														(Deliverable) workProduct)) {
											badlist.add(obj);
											return false;
										}
									}

								}
								// browse through badlist and its deliverable
								// parts.
								List templist = new ArrayList();
								for (Iterator iterator = badlist.iterator(); iterator
										.hasNext();) {
									Object bad = iterator.next();
									UIHelper.deliverablePartsChain(
											(Deliverable) bad, templist);
								}
								if (templist.contains(obj))
									return false;
							}

							return true;
						}
					}
				}
				return false;
			}
		};

		// set additional filter for variability base element checking
		((ContentFilter) filter)
				.setAdditionalFilters(new IFilter[] { new VariabilityBaseElementFilter(
						workProduct) });
		return filter;
	}

	/**
	 * @see org.eclipse.epf.authoring.ui.forms.DescriptionFormPage#loadSectionDescription()
	 */
	public void loadSectionDescription() {
		if (contentElement instanceof Artifact) {
			this.generalSectionDescription = AuthoringUIResources.artifact_generalInfoSection_desc;
			this.detailSectionDescription = AuthoringUIResources.artifact_detailSection_desc;
			this.variabilitySectionDescription = AuthoringUIResources.artifact_variabilitySection_desc;
			this.versionSectionDescription = AuthoringUIResources.artifact_versionInfoSection_desc;
			this.notationSectionDescription = AuthoringUIResources.artifact_notationSection_desc;
			this.tailoringSectionDescription = AuthoringUIResources.artifact_tailoringSection_desc;
			this.iconSectionDescription = AuthoringUIResources.artifact_iconSection_desc;
			this.slotSectionDescription = AuthoringUIResources.artifact_slotSection_desc;
		}
		if (contentElement instanceof Outcome) {
			this.generalSectionDescription = AuthoringUIResources.outcome_generalInfoSection_desc;
			this.detailSectionDescription = AuthoringUIResources.outcome_detailSection_desc;
			this.variabilitySectionDescription = AuthoringUIResources.outcome_variabilitySection_desc;
			this.versionSectionDescription = AuthoringUIResources.outcome_versionInfoSection_desc;
			this.notationSectionDescription = AuthoringUIResources.outcome_notationSection_desc;
			this.tailoringSectionDescription = AuthoringUIResources.outcome_tailoringSection_desc;
			this.iconSectionDescription = AuthoringUIResources.outcome_iconSection_desc;
			this.slotSectionDescription = AuthoringUIResources.outcome_slotSection_desc;
		}
		if (contentElement instanceof Deliverable) {
			this.generalSectionDescription = AuthoringUIResources.deliverable_generalInfoSection_desc;
			this.detailSectionDescription = AuthoringUIResources.deliverable_detailSection_desc;
			this.variabilitySectionDescription = AuthoringUIResources.deliverable_variabilitySection_desc;
			this.versionSectionDescription = AuthoringUIResources.deliverable_versionInfoSection_desc;
			this.notationSectionDescription = AuthoringUIResources.deliverable_notationSection_desc;
			this.tailoringSectionDescription = AuthoringUIResources.deliverable_tailoringSection_desc;
			this.iconSectionDescription = AuthoringUIResources.deliverable_iconSection_desc;
			this.slotSectionDescription = AuthoringUIResources.deliverable_slotSection_desc;
		}
	}

}
