| /******************************************************************************* |
| * Copyright (c) 2005, 2016 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 |
| * |
| |
| *******************************************************************************/ |
| package org.eclipse.dltk.internal.core; |
| |
| import java.io.ByteArrayInputStream; |
| import java.io.IOException; |
| import java.io.InputStream; |
| |
| import org.eclipse.core.resources.IContainer; |
| import org.eclipse.core.resources.IFile; |
| import org.eclipse.core.resources.IResource; |
| import org.eclipse.core.resources.IWorkspace; |
| import org.eclipse.core.runtime.CoreException; |
| import org.eclipse.core.runtime.NullProgressMonitor; |
| import org.eclipse.core.runtime.Path; |
| import org.eclipse.core.runtime.jobs.ISchedulingRule; |
| import org.eclipse.dltk.core.IBuffer; |
| import org.eclipse.dltk.core.IModelElement; |
| import org.eclipse.dltk.core.IModelElementDelta; |
| import org.eclipse.dltk.core.IModelStatus; |
| import org.eclipse.dltk.core.IModelStatusConstants; |
| import org.eclipse.dltk.core.IScriptFolder; |
| import org.eclipse.dltk.core.ISourceModule; |
| import org.eclipse.dltk.core.ModelException; |
| import org.eclipse.dltk.internal.core.util.Messages; |
| import org.eclipse.dltk.internal.core.util.Util; |
| |
| /** |
| * <p> |
| * This operation creates a compilation unit (CU). If the CU doesn't exist yet, |
| * a new compilation unit will be created with the content provided. Otherwise |
| * the operation will override the contents of an existing CU with the new |
| * content. |
| * |
| * <p> |
| * Note: It is possible to create a CU automatically when creating a class or |
| * interface. Thus, the preferred method of creating a CU is to perform a create |
| * type operation rather than first creating a CU and secondly creating a type |
| * inside the CU. |
| * |
| * <p> |
| * Required Attributes: |
| * <ul> |
| * <li>The package fragment in which to create the compilation unit. |
| * <li>The name of the compilation unit. Do not include the <code>".java"</code> |
| * suffix (ex. <code>"Object"</code> - the <code>".java"</code> will be added |
| * for the name of the compilation unit.) |
| * <li> |
| * </ul> |
| */ |
| public class CreateSourceModuleOperation extends ModelOperation { |
| |
| /** |
| * The name of the compilation unit being created. |
| */ |
| protected String fName; |
| /** |
| * The source code to use when creating the element. |
| */ |
| protected String fSource = null; |
| |
| /** |
| * When executed, this operation will create a compilation unit with the |
| * given name. The name should have the ".java" suffix. |
| */ |
| public CreateSourceModuleOperation(IScriptFolder parentElement, |
| String name, String source, boolean force) { |
| super(null, new IModelElement[] { parentElement }, force); |
| fName = name; |
| fSource = source; |
| } |
| |
| /** |
| * Creates a compilation unit. |
| * |
| * @exception ModelException |
| * if unable to create the compilation unit. |
| */ |
| @Override |
| protected void executeOperation() throws ModelException { |
| try { |
| // beginTask(Messages.operation_createUnitProgress, 2); |
| ModelElementDelta delta = newModelElementDelta(); |
| ISourceModule unit = getSourceModule(); |
| IScriptFolder pkg = (IScriptFolder) getParentElement(); |
| IContainer folder = (IContainer) pkg.getResource(); |
| worked(1); |
| IFile compilationUnitFile = folder.getFile(new Path(fName)); |
| if (compilationUnitFile.exists()) { |
| // update the contents of the existing unit if fForce is true |
| if (force) { |
| IBuffer buffer = unit.getBuffer(); |
| if (buffer == null) |
| return; |
| buffer.setContents(fSource); |
| unit.save(new NullProgressMonitor(), false); |
| resultElements = new IModelElement[] { unit }; |
| if (!Util.isExcluded(unit) && unit.getParent().exists()) { |
| for (int i = 0; i < resultElements.length; i++) { |
| delta.changed(resultElements[i], |
| IModelElementDelta.F_CONTENT); |
| } |
| addDelta(delta); |
| } |
| } else { |
| throw new ModelException(new ModelStatus( |
| IModelStatusConstants.NAME_COLLISION, Messages |
| .bind(Messages.status_nameCollision, |
| compilationUnitFile.getFullPath() |
| .toString()))); |
| } |
| } else { |
| try { |
| String encoding = null; |
| try { |
| encoding = folder.getDefaultCharset(); // get folder |
| // encoding as |
| // file is not |
| // accessible |
| } catch (CoreException ce) { |
| // use no encoding |
| } |
| InputStream stream = new ByteArrayInputStream( |
| encoding == null ? fSource.getBytes() : fSource |
| .getBytes(encoding)); |
| createFile(folder, unit.getElementName(), stream, force); |
| resultElements = new IModelElement[] { unit }; |
| if (!Util.isExcluded(unit) && unit.getParent().exists()) { |
| for (int i = 0; i < resultElements.length; i++) { |
| delta.added(resultElements[i]); |
| } |
| addDelta(delta); |
| } |
| } catch (IOException e) { |
| throw new ModelException(e, |
| IModelStatusConstants.IO_EXCEPTION); |
| } |
| } |
| worked(1); |
| } finally { |
| done(); |
| } |
| } |
| |
| /** |
| * @see CreateElementInCUOperation#getSourceModule() |
| */ |
| protected ISourceModule getSourceModule() { |
| return ((IScriptFolder) getParentElement()).getSourceModule(fName); |
| } |
| |
| @Override |
| protected ISchedulingRule getSchedulingRule() { |
| IResource resource = getSourceModule().getResource(); |
| IWorkspace workspace = resource.getWorkspace(); |
| if (resource.exists()) { |
| return workspace.getRuleFactory().modifyRule(resource); |
| } else { |
| return workspace.getRuleFactory().createRule(resource); |
| } |
| } |
| |
| /** |
| * Possible failures: |
| * <ul> |
| * <li>NO_ELEMENTS_TO_PROCESS - the package fragment supplied to the |
| * operation is <code>null</code>. |
| * <li>INVALID_NAME - the compilation unit name provided to the operation is |
| * <code>null</code> or has an invalid syntax |
| * <li>INVALID_CONTENTS - the source specified for the compiliation unit is |
| * null |
| * </ul> |
| */ |
| @Override |
| public IModelStatus verify() { |
| if (getParentElement() == null) { |
| return new ModelStatus(IModelStatusConstants.NO_ELEMENTS_TO_PROCESS); |
| } |
| if (fSource == null) { |
| return new ModelStatus(IModelStatusConstants.INVALID_CONTENTS); |
| } |
| return IModelStatus.VERIFIED_OK; |
| } |
| } |