/*******************************************************************************
 * Copyright (c) 2001, 2004 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 API and implementation
 *     Jens Lukowski/Innoopract - initial renaming/restructuring
 *     
 *******************************************************************************/
package org.eclipse.wst.sse.ui.internal.reconcile.validator;

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

import org.eclipse.core.resources.IFile;
import org.eclipse.core.resources.IProject;
import org.eclipse.core.resources.ResourcesPlugin;
import org.eclipse.core.runtime.IPath;
import org.eclipse.core.runtime.Path;
import org.eclipse.core.runtime.Platform;
import org.eclipse.jface.text.IDocument;
import org.eclipse.jface.text.IRegion;
import org.eclipse.jface.text.Position;
import org.eclipse.jface.text.reconciler.DirtyRegion;
import org.eclipse.jface.text.reconciler.IReconcilableModel;
import org.eclipse.jface.text.reconciler.IReconcileResult;
import org.eclipse.jface.text.reconciler.IReconcileStep;
import org.eclipse.wst.sse.core.internal.provisional.IStructuredModel;
import org.eclipse.wst.sse.core.internal.provisional.StructuredModelManager;
import org.eclipse.wst.sse.ui.internal.Logger;
import org.eclipse.wst.sse.ui.internal.reconcile.DocumentAdapter;
import org.eclipse.wst.sse.ui.internal.reconcile.ReconcileAnnotationKey;
import org.eclipse.wst.sse.ui.internal.reconcile.StructuredReconcileStep;
import org.eclipse.wst.sse.ui.internal.reconcile.TemporaryAnnotation;
import org.eclipse.wst.validation.internal.provisional.core.IMessage;
import org.eclipse.wst.validation.internal.provisional.core.IValidator;


/**
 * A reconcile step for an IValidator for reconcile. Used the reconcile
 * framework to create TemporaryAnnotations from the validator messages.
 * 
 * @author pavery
 */
public class ReconcileStepForValidator extends StructuredReconcileStep {

	/** debug flag */
	protected static final boolean DEBUG;
	static {
		String value = Platform.getDebugOption("org.eclipse.wst.sse.ui/debug/reconcilerjob"); //$NON-NLS-1$
		DEBUG = value != null && value.equalsIgnoreCase("true"); //$NON-NLS-1$
	}

	private final IReconcileResult[] EMPTY_RECONCILE_RESULT_SET = new IReconcileResult[0];
	private IncrementalHelper fHelper = null;
	private IncrementalReporter fReporter = null;
	private int fScope = -1;
	private IValidator fValidator = null;


	public ReconcileStepForValidator(IValidator v, int scope) {
		super();

		if (v == null)
			throw new IllegalArgumentException("validator cannot be null"); //$NON-NLS-1$

		fValidator = v;
		fScope = scope;
	}

	public ReconcileStepForValidator(IValidator v, IReconcileStep step, int scope) {
//		super(step);
		fValidator = v;
		fScope = scope;
	}

	/**
	 * Converts a map of IValidatorForReconcile to List to annotations based
	 * on those messages
	 * 
	 * @param messages
	 * @return
	 */
//	protected IReconcileResult[] createAnnotations(List messageList) {
//		List annotations = new ArrayList();
//		for (int i = 0; i < messageList.size(); i++) {
//			IMessage validationMessage = (IMessage) messageList.get(i);
//
//			int offset = validationMessage.getOffset();
//
//			if (offset < 0)
//				continue;
//
//			String messageText = null;
//			try {
//				messageText = validationMessage.getText(validationMessage.getClass().getClassLoader());
//			}
//			catch (Exception t) {
//				Logger.logException("exception reporting message from validator", t); //$NON-NLS-1$
//				continue;
//			}
//			
//			String type = getSeverity(validationMessage);
//			// this position seems like it would be possibly be the wrong
//			// length
//			int length = validationMessage.getLength();
//			if (length >= 0) {
//				Position p = new Position(offset, length);
//				ReconcileAnnotationKey key = createKey(getPartitionType(getDocument(), offset), getScope());
//				annotations.add(new TemporaryAnnotation(p, type, messageText, key));
//			}
//		}
//
//		return (IReconcileResult[]) annotations.toArray(new IReconcileResult[annotations.size()]);
//	}

	/**
	 * Converts a map of IValidatorForReconcile to List to annotations based
	 * on those messages
	 * 
	 * @param messages
	 * @return
	 */
	protected IReconcileResult[] createAnnotations(AnnotationInfo[] infos) {
		
	
		List annotations = new ArrayList();
		for (int i = 0; i < infos.length; i++) {
			
			AnnotationInfo info = infos[i];
			
			IMessage validationMessage = info.getMessage();
			int offset = validationMessage.getOffset();
			if (offset < 0)
				continue;

			String messageText = null;
			try {
				messageText = validationMessage.getText(validationMessage.getClass().getClassLoader());
			}
			catch (Exception t) {
				Logger.logException("exception reporting message from validator", t); //$NON-NLS-1$
				continue;
			}
			String type = getSeverity(validationMessage);
			// this position seems like it would be possibly be the wrong
			// length
			int length = validationMessage.getLength();
			if (length >= 0) {

				Position p = new Position(offset, length);
				ReconcileAnnotationKey key = createKey(getPartitionType(getDocument(), offset), getScope());
				
				if(info.getProblemId() == AnnotationInfo.NO_PROBLEM_ID) {
					// create an annotation w/ no fix info
					annotations.add(new TemporaryAnnotation(p, type, messageText, key));
				}
				else {
					// create an annotation w/ problem ID and fix info
					TemporaryAnnotation annotation = new TemporaryAnnotation(p, type, messageText, key, info.getProblemId());
					annotation.setAdditionalFixInfo(info.getAdditionalFixInfo());
					annotations.add(annotation);
				}
			}
		}
		return (IReconcileResult[]) annotations.toArray(new IReconcileResult[annotations.size()]);
	}

	private String getSeverity(IMessage validationMessage) {
		String type = TemporaryAnnotation.ANNOT_INFO;
		switch (validationMessage.getSeverity()) {
			case IMessage.HIGH_SEVERITY :
				type = TemporaryAnnotation.ANNOT_ERROR;
				break;
			case IMessage.NORMAL_SEVERITY :
				type = TemporaryAnnotation.ANNOT_WARNING;
				break;
			case IMessage.LOW_SEVERITY :
				type = TemporaryAnnotation.ANNOT_WARNING;
				break;
			case IMessage.ERROR_AND_WARNING :
				type = TemporaryAnnotation.ANNOT_WARNING;
				break;
		}
		return type;
	}
	
	private IFile getFile() {
		IStructuredModel model = null;
		IFile file = null;
		try {
			model = StructuredModelManager.getModelManager().getExistingModelForRead(getDocument());
			if (model != null) {
				String baseLocation = model.getBaseLocation();
				// The baseLocation may be a path on disk or relative to the
				// workspace root. Don't translate on-disk paths to
				// in-workspace resources.
				IPath basePath = new Path(baseLocation);
				if (basePath.segmentCount() > 1 && !basePath.toFile().exists()) {
					file = ResourcesPlugin.getWorkspace().getRoot().getFile(basePath);
				}
			}
		}
		finally {
			if (model != null) {
				model.releaseFromRead();
			}
		}
		return file;
	}

	private IncrementalHelper getHelper(IProject project) {
		if (fHelper == null)
			fHelper = new IncrementalHelper(getStructuredDocument(), project);
		return fHelper;
	}

	private IncrementalReporter getReporter() {
		if (fReporter == null)
			fReporter = new IncrementalReporter(getProgressMonitor());
		return fReporter;
	}

	/**
	 * If this validator is partial or total
	 * @return
	 */
	public int getScope() {
		return fScope;
	}

	public void initialReconcile() {
		// do nothing
	}

	protected IReconcileResult[] reconcileModel(DirtyRegion dirtyRegion, IRegion subRegion) {
		if (DEBUG)
			System.out.println("[trace reconciler] > reconciling model in VALIDATOR step w/ dirty region: [" + dirtyRegion.getText() + "]"); //$NON-NLS-1$ //$NON-NLS-2$

		IReconcileResult[] results = EMPTY_RECONCILE_RESULT_SET;
		if (dirtyRegion != null) {
			try {
				if(fValidator instanceof ISourceValidator) {
					results = validate(dirtyRegion, subRegion);
				}
				else {
					results = validate();
				}
			}
			catch (Exception ex) {
				Logger.logException("EXEPTION IN RECONCILE STEP FOR VALIDATOR", ex); //$NON-NLS-1$
			}
		}

		if (DEBUG)
			System.out.println("[trace reconciler] > VALIDATOR step done"); //$NON-NLS-1$

		return results;
	}

	public String toString() {
		StringBuffer debugString = new StringBuffer("ValidatorStep: "); //$NON-NLS-1$
		if (fValidator != null)
			debugString.append(fValidator.getClass().toString());
		return debugString.toString();
	}
	
	protected IReconcileResult[] validate() {
		IReconcileResult[] results = EMPTY_RECONCILE_RESULT_SET;

		IFile file = getFile();

		if (file != null) {
			try {
				IProject project = file.getProject();
				IncrementalHelper helper = getHelper(project);
				IncrementalReporter reporter = getReporter();

				helper.setURI(file.getFullPath().toString());

				fValidator.validate(helper, reporter);

				//results = createAnnotations(reporter.getMessages());
				results = createAnnotations(reporter.getAnnotationInfo());
				reporter.removeAllMessages(fValidator);

			}
			catch (Exception e) {
				Logger.logException(e);
			}
		}
		return results;
	}
	
	public void setInputModel(IReconcilableModel inputModel) {
		if(inputModel instanceof DocumentAdapter) {
			IDocument document = ((DocumentAdapter)inputModel).getDocument();
			if(document != null) {
				if(fValidator instanceof ISourceValidator) {
					// notify that document connecting
					((ISourceValidator)fValidator).connect(document);
				}
			}
		}
		super.setInputModel(inputModel);
	}
	
	public void release() {
		if(fValidator instanceof ISourceValidator) {
			IDocument document = getDocument();
			if(document != null) {
				// notify that document disconnecting
				((ISourceValidator)fValidator).disconnect(document);
			}
		}
		super.release();
	}
	
	protected IReconcileResult[] validate(DirtyRegion dirtyRegion, IRegion subRegion) {
 		IReconcileResult[] results = EMPTY_RECONCILE_RESULT_SET;
 
 		IFile file = getFile();
		try {
			IProject project = null;
			if (file != null)
				project = file.getProject();
 
			IncrementalHelper helper = getHelper(project);

			if (file != null)
				helper.setURI(file.getFullPath().toString());
 
			if(fValidator instanceof ISourceValidator) {
				IncrementalReporter reporter = getReporter();
				((ISourceValidator)fValidator).validate(dirtyRegion, helper, reporter);
				//results = createAnnotations(reporter.getMessages());
				results = createAnnotations(reporter.getAnnotationInfo());
				reporter.removeAllMessages(fValidator);
			}

		} catch (Exception e) {
			Logger.logException(e);
 		}
 		return results;
 	}
	
	
	
}