/******************************************************************************* * Copyright (c) 2011 Wind River Systems, Inc. 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: * William Chen (Wind River) - [345552] Edit the remote files with a proper editor *******************************************************************************/ package org.eclipse.tm.te.tcf.filesystem.internal.autosave; import java.io.File; import java.net.URI; import java.util.ArrayList; import java.util.List; import org.eclipse.compare.CompareUI; import org.eclipse.core.commands.ExecutionEvent; import org.eclipse.core.commands.ExecutionException; import org.eclipse.core.commands.IExecutionListener; import org.eclipse.core.commands.NotHandledException; import org.eclipse.core.filesystem.EFS; import org.eclipse.core.filesystem.IFileStore; import org.eclipse.core.runtime.CoreException; import org.eclipse.core.runtime.NullProgressMonitor; import org.eclipse.jface.dialogs.MessageDialog; import org.eclipse.osgi.util.NLS; import org.eclipse.swt.widgets.Shell; import org.eclipse.tm.te.tcf.filesystem.internal.compare.LocalTypedElement; import org.eclipse.tm.te.tcf.filesystem.internal.compare.MergeEditorInput; import org.eclipse.tm.te.tcf.filesystem.internal.compare.RemoteTypedElement; import org.eclipse.tm.te.tcf.filesystem.internal.exceptions.TCFException; import org.eclipse.tm.te.tcf.filesystem.internal.handlers.CacheManager; import org.eclipse.tm.te.tcf.filesystem.internal.handlers.PersistenceManager; import org.eclipse.tm.te.tcf.filesystem.internal.handlers.StateManager; import org.eclipse.tm.te.tcf.filesystem.internal.nls.Messages; import org.eclipse.tm.te.tcf.filesystem.model.CacheState; import org.eclipse.tm.te.tcf.filesystem.model.FSModel; import org.eclipse.tm.te.tcf.filesystem.model.FSTreeNode; import org.eclipse.ui.IEditorInput; import org.eclipse.ui.IEditorPart; import org.eclipse.ui.IURIEditorInput; import org.eclipse.ui.IWorkbenchPage; import org.eclipse.ui.IWorkbenchWindow; import org.eclipse.ui.PlatformUI; import org.eclipse.ui.handlers.HandlerUtil; /** * The execution listener of command "SAVE ALL", which synchronizes the local * file with the one on the target server after it is saved. */ public class SaveAllListener implements IExecutionListener { // Dirty nodes that should be saved and synchronized. private List<FSTreeNode> fDirtyNodes; // The file system fModel storing the existing FSTreeNodes. private FSModel fModel; /** * Create the listener listening to command "SAVE ALL". */ public SaveAllListener() { this.fModel = FSModel.getInstance(); this.fDirtyNodes = new ArrayList<FSTreeNode>(); } /* (non-Javadoc) * @see org.eclipse.core.commands.IExecutionListener#postExecuteSuccess(java.lang.String, java.lang.Object) */ @Override public void postExecuteSuccess(String commandId, Object returnValue) { if (!fDirtyNodes.isEmpty()) { try { List<FSTreeNode> modified = new ArrayList<FSTreeNode>(); List<FSTreeNode> conflicts = new ArrayList<FSTreeNode>(); for (FSTreeNode node : fDirtyNodes) { // Refresh the dirty nodes and get their latest states. StateManager.getInstance().refreshState(node); CacheState state = StateManager.getInstance().getCacheState(node); switch (state) { case consistent: break; case outdated: break; case modified: // Reclassifying modified.add(node); break; case conflict: // Reclassifying conflicts.add(node); break; } } if (PersistenceManager.getInstance().isAutoSaving()) { // If auto saving is on. if (!modified.isEmpty()) { // Upload the modified nodes. CacheManager.getInstance().upload(modified.toArray(new FSTreeNode[modified.size()])); } if (!conflicts.isEmpty()) { // Merge the conflicting ones. mergeConflicts(conflicts); } } } catch (TCFException tcfe) { Shell parent = PlatformUI.getWorkbench().getActiveWorkbenchWindow().getShell(); MessageDialog.openError(parent, Messages.StateManager_RefreshFailureTitle, tcfe.getLocalizedMessage()); } } } /** * Merge those conflicting nodes. * * @param conflicts The conflicting nodes. */ private void mergeConflicts(List<FSTreeNode> conflicts) { for (FSTreeNode node : conflicts) { String title = Messages.SaveAllListener_StateChangedDialogTitle; String message = NLS.bind(Messages.SaveAllListener_SingularMessage, node.name); IWorkbenchWindow window = PlatformUI.getWorkbench().getActiveWorkbenchWindow(); IWorkbenchPage page = window.getActivePage(); Shell parent = window.getShell(); MessageDialog msgDialog = new MessageDialog(parent, title, null, message, MessageDialog.QUESTION, new String[] { Messages.SaveAllListener_Merge, Messages.SaveAllListener_SaveAnyway, Messages.SaveAllListener_Cancel }, 0); int index = msgDialog.open(); if (index == 0) { // Merge LocalTypedElement local = new LocalTypedElement(node); RemoteTypedElement remote = new RemoteTypedElement(node); MergeEditorInput mergeInput = new MergeEditorInput(local, remote, page); CompareUI.openCompareDialog(mergeInput); } else if (index == 1) { // Save anyway CacheManager.getInstance().upload(conflicts.toArray(new FSTreeNode[conflicts.size()])); } } } /* (non-Javadoc) * @see org.eclipse.core.commands.IExecutionListener#preExecute(java.lang.String, org.eclipse.core.commands.ExecutionEvent) */ @Override public void preExecute(String commandId, ExecutionEvent event) { fDirtyNodes.clear(); IWorkbenchPage page = HandlerUtil.getActiveWorkbenchWindow(event).getActivePage(); IEditorPart[] editors = page.getDirtyEditors(); for (IEditorPart editor : editors) { IEditorInput input = editor.getEditorInput(); FSTreeNode node = getEditedNode(input); if (node != null) { // If it is a modified node, add it to the dirty node list. fDirtyNodes.add(node); } } } /** * Get the corresponding FSTreeNode from the input. * If the input has no corresponding FSTreeNode, return null; * @param input The editor input. * @return The corresponding FSTreeNode or null if it has not. */ private FSTreeNode getEditedNode(IEditorInput input){ if (input instanceof IURIEditorInput) { //Get the file that is being edited. IURIEditorInput fileInput = (IURIEditorInput) input; URI uri = fileInput.getURI(); try { IFileStore store = EFS.getStore(uri); File localFile = store.toLocalFile(0, new NullProgressMonitor()); if (localFile != null) { // Get the file's mapped FSTreeNode. FSTreeNode node = fModel.getTreeNode(localFile.toString()); return node; } }catch(CoreException e){} } return null; } /* (non-Javadoc) * @see org.eclipse.core.commands.IExecutionListener#notHandled(java.lang.String, org.eclipse.core.commands.NotHandledException) */ @Override public void notHandled(String commandId, NotHandledException exception) { } /* (non-Javadoc) * @see org.eclipse.core.commands.IExecutionListener#postExecuteFailure(java.lang.String, org.eclipse.core.commands.ExecutionException) */ @Override public void postExecuteFailure(String commandId, ExecutionException exception) { } }