blob: f00e9416c06db0252eefdd335de8a5d7b77a9a6c [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*/
package org.apache.directory.studio.ldapbrowser.ui.views.browser;
import java.util.HashMap;
import java.util.Map;
import org.apache.directory.studio.connection.core.Connection;
import org.apache.directory.studio.connection.core.event.ConnectionEventRegistry;
import org.apache.directory.studio.connection.ui.ConnectionUIPlugin;
import org.apache.directory.studio.entryeditors.EntryEditorExtension;
import org.apache.directory.studio.entryeditors.EntryEditorInput;
import org.apache.directory.studio.entryeditors.EntryEditorManager;
import org.apache.directory.studio.entryeditors.IEntryEditor;
import org.apache.directory.studio.ldapbrowser.common.BrowserCommonActivator;
import org.apache.directory.studio.ldapbrowser.common.BrowserCommonConstants;
import org.apache.directory.studio.ldapbrowser.common.actions.BrowserSelectionUtils;
import org.apache.directory.studio.ldapbrowser.common.widgets.browser.BrowserContentProvider;
import org.apache.directory.studio.ldapbrowser.common.widgets.browser.BrowserUniversalListener;
import org.apache.directory.studio.ldapbrowser.core.BrowserCorePlugin;
import org.apache.directory.studio.ldapbrowser.core.events.AttributesInitializedEvent;
import org.apache.directory.studio.ldapbrowser.core.events.BookmarkUpdateEvent;
import org.apache.directory.studio.ldapbrowser.core.events.BookmarkUpdateListener;
import org.apache.directory.studio.ldapbrowser.core.events.BulkModificationEvent;
import org.apache.directory.studio.ldapbrowser.core.events.ChildrenInitializedEvent;
import org.apache.directory.studio.ldapbrowser.core.events.EntryAddedEvent;
import org.apache.directory.studio.ldapbrowser.core.events.EntryDeletedEvent;
import org.apache.directory.studio.ldapbrowser.core.events.EntryModificationEvent;
import org.apache.directory.studio.ldapbrowser.core.events.EntryMovedEvent;
import org.apache.directory.studio.ldapbrowser.core.events.EntryRenamedEvent;
import org.apache.directory.studio.ldapbrowser.core.events.EventRegistry;
import org.apache.directory.studio.ldapbrowser.core.events.SearchUpdateEvent;
import org.apache.directory.studio.ldapbrowser.core.events.SearchUpdateEvent.EventDetail;
import org.apache.directory.studio.ldapbrowser.core.events.SearchUpdateListener;
import org.apache.directory.studio.ldapbrowser.core.model.IBookmark;
import org.apache.directory.studio.ldapbrowser.core.model.IBrowserConnection;
import org.apache.directory.studio.ldapbrowser.core.model.IContinuation;
import org.apache.directory.studio.ldapbrowser.core.model.IEntry;
import org.apache.directory.studio.ldapbrowser.core.model.IQuickSearch;
import org.apache.directory.studio.ldapbrowser.core.model.IRootDSE;
import org.apache.directory.studio.ldapbrowser.core.model.ISearch;
import org.apache.directory.studio.ldapbrowser.core.model.ISearchResult;
import org.apache.directory.studio.ldapbrowser.core.model.impl.DummyEntry;
import org.apache.directory.studio.ldapbrowser.ui.BrowserUIConstants;
import org.apache.directory.studio.ldapbrowser.ui.BrowserUIPlugin;
import org.apache.directory.studio.ldapbrowser.ui.editors.searchresult.SearchResultEditor;
import org.apache.directory.studio.ldapbrowser.ui.editors.searchresult.SearchResultEditorInput;
import org.apache.directory.studio.ldapbrowser.ui.views.connection.ConnectionView;
import org.eclipse.jface.util.IPropertyChangeListener;
import org.eclipse.jface.viewers.IOpenListener;
import org.eclipse.jface.viewers.ISelection;
import org.eclipse.jface.viewers.ISelectionChangedListener;
import org.eclipse.jface.viewers.ITreeContentProvider;
import org.eclipse.jface.viewers.OpenEvent;
import org.eclipse.jface.viewers.SelectionChangedEvent;
import org.eclipse.jface.viewers.StructuredSelection;
import org.eclipse.jface.viewers.TreeViewer;
import org.eclipse.ui.IEditorReference;
import org.eclipse.ui.INullSelectionListener;
import org.eclipse.ui.IPartListener2;
import org.eclipse.ui.IWorkbenchPage;
import org.eclipse.ui.IWorkbenchPart;
import org.eclipse.ui.IWorkbenchPartReference;
import org.eclipse.ui.PartInitException;
import org.eclipse.ui.PlatformUI;
import org.eclipse.ui.contexts.IContextActivation;
import org.eclipse.ui.contexts.IContextService;
/**
* The BrowserViewUniversalListener manages all events for the browser view.
*
* @author <a href="mailto:dev@directory.apache.org">Apache Directory Project</a>
*/
public class BrowserViewUniversalListener extends BrowserUniversalListener implements SearchUpdateListener,
BookmarkUpdateListener
{
/** This map contains all expanded elements for a particular connection */
private Map<IBrowserConnection, Object[]> connectionToExpandedElementsMap;
/** This map contains all selected elements for a particular connection */
private Map<IBrowserConnection, ISelection> connectionToSelectedElementMap;
/** The browser view */
private BrowserView view;
/** Token used to activate and deactivate shortcuts in the view */
private IContextActivation contextActivation;
/** Listener that listens for selections of connections */
private INullSelectionListener connectionSelectionListener = new INullSelectionListener()
{
/**
* {@inheritDoc}
*
* This implementation sets the input when another connection was selected.
*/
public void selectionChanged( IWorkbenchPart part, ISelection selection )
{
if ( view != null && part != null )
{
if ( view.getSite().getWorkbenchWindow() == part.getSite().getWorkbenchWindow() )
{
Connection[] connections = BrowserSelectionUtils.getConnections( selection );
if ( connections.length == 1 )
{
IBrowserConnection connection = BrowserCorePlugin.getDefault().getConnectionManager()
.getBrowserConnection( connections[0] );
setInput( connection );
}
else
{
setInput( null );
}
}
}
}
};
/** The part listener used to activate and deactivate the shortcuts */
private IPartListener2 partListener = new IPartListener2()
{
/**
* {@inheritDoc}
*
* This implementation deactivates the shortcuts when the part is deactivated.
*/
public void partDeactivated( IWorkbenchPartReference partRef )
{
if ( partRef.getPart( false ) == view && contextActivation != null )
{
view.getActionGroup().deactivateGlobalActionHandlers();
IContextService contextService = ( IContextService ) PlatformUI.getWorkbench().getAdapter(
IContextService.class );
contextService.deactivateContext( contextActivation );
contextActivation = null;
}
}
/**
* {@inheritDoc}
*
* This implementation activates the shortcuts when the part is activated.
*/
public void partActivated( IWorkbenchPartReference partRef )
{
if ( partRef.getPart( false ) == view )
{
IContextService contextService = ( IContextService ) PlatformUI.getWorkbench().getAdapter(
IContextService.class );
contextActivation = contextService.activateContext( BrowserCommonConstants.CONTEXT_WINDOWS );
// org.eclipse.ui.contexts.dialogAndWindow
// org.eclipse.ui.contexts.window
// org.eclipse.ui.text_editor_context
view.getActionGroup().activateGlobalActionHandlers();
}
}
/**
* {@inheritDoc}
*/
public void partBroughtToTop( IWorkbenchPartReference partRef )
{
}
/**
* {@inheritDoc}
*/
public void partClosed( IWorkbenchPartReference partRef )
{
}
/**
* {@inheritDoc}
*/
public void partOpened( IWorkbenchPartReference partRef )
{
}
/**
* {@inheritDoc}
*/
public void partHidden( IWorkbenchPartReference partRef )
{
}
/**
* {@inheritDoc}
*/
public void partVisible( IWorkbenchPartReference partRef )
{
}
/**
* {@inheritDoc}
*/
public void partInputChanged( IWorkbenchPartReference partRef )
{
}
};
/** This listener is used to ensure that the entry editor and search result editor are opened
when an object in the browser view is selected */
private ISelectionChangedListener viewerSelectionListener = new ISelectionChangedListener()
{
/**
* {@inheritDoc}
*/
public void selectionChanged( SelectionChangedEvent event )
{
openEditor( event.getSelection() );
}
};
/** This listerner is used to listen on the preference settings modifications, especially
* the open mode preference value change. */
private IPropertyChangeListener preferencePropertyChangeListener = new IPropertyChangeListener()
{
/**
* {@inheritDoc}
*/
public void propertyChange( org.eclipse.jface.util.PropertyChangeEvent event )
{
if ( BrowserUIConstants.PREFERENCE_ENTRYEDITORS_OPEN_MODE.equals( event.getProperty() ) )
{
setupOpenModeListeners();
}
};
};
/** The open mode listener */
private IOpenListener openListener = new IOpenListener()
{
public void open( OpenEvent event )
{
openEditor( event.getSelection() );
}
};
/**
* Creates a new instance of BrowserViewUniversalListener.
*
* @param view the browser view
*/
public BrowserViewUniversalListener( BrowserView view )
{
super( view.getMainWidget() );
this.view = view;
// create maps
connectionToExpandedElementsMap = new HashMap<IBrowserConnection, Object[]>();
connectionToSelectedElementMap = new HashMap<IBrowserConnection, ISelection>();
// register listeners
EventRegistry.addSearchUpdateListener( this, BrowserCommonActivator.getDefault().getEventRunner() );
EventRegistry.addBookmarkUpdateListener( this, BrowserCommonActivator.getDefault().getEventRunner() );
EventRegistry.addEntryUpdateListener( this, BrowserCommonActivator.getDefault().getEventRunner() );
ConnectionEventRegistry.addConnectionUpdateListener( this, ConnectionUIPlugin.getDefault().getEventRunner() );
// listener for shortcuts activation/deactivation
view.getSite().getPage().addPartListener( partListener );
// listener for connections
view.getSite().getWorkbenchWindow().getSelectionService().addPostSelectionListener( ConnectionView.getId(),
connectionSelectionListener );
// listener for open mode
BrowserUIPlugin.getDefault().getPreferenceStore().addPropertyChangeListener( preferencePropertyChangeListener );
setupOpenModeListeners();
}
/**
* Sets up the open mode listeners according the preferences.
*/
private void setupOpenModeListeners()
{
int openMode = BrowserUIPlugin.getDefault().getPluginPreferences().getInt(
BrowserUIConstants.PREFERENCE_ENTRYEDITORS_OPEN_MODE );
if ( openMode == BrowserUIConstants.PREFERENCE_ENTRYEDITORS_OPEN_MODE_HISTORICAL_BEHAVIOR )
{
// Historical Behavior
viewer.removeOpenListener( openListener );
viewer.addSelectionChangedListener( viewerSelectionListener );
}
else if ( openMode == BrowserUIConstants.PREFERENCE_ENTRYEDITORS_OPEN_MODE_APPLICATION_WIDE )
{
// Application Wide Setting
viewer.removeSelectionChangedListener( viewerSelectionListener );
viewer.addOpenListener( openListener );
}
}
/**
* Opens an editor to show the given selection.
*
* @param selection the browser's selection.
*/
private void openEditor( ISelection selection )
{
if ( view != null )
{
IEntry[] entries = BrowserSelectionUtils.getEntries( selection );
ISearchResult[] searchResults = BrowserSelectionUtils.getSearchResults( selection );
IBookmark[] bookmarks = BrowserSelectionUtils.getBookmarks( selection );
ISearch[] searches = BrowserSelectionUtils.getSearches( selection );
EntryEditorManager entryEditorManager = BrowserUIPlugin.getDefault().getEntryEditorManager();
if ( entries.length + searchResults.length + bookmarks.length == 1 )
{
entryEditorManager.openEntryEditor( entries, searchResults, bookmarks );
// atm it is not necessary to blank the search result editor, it blanks itself
}
else
{
// Checking if there's at least one entry editor open.
// We need to blank them.
// This is done before the search result editor is opened,
// otherwise the entry editor would be activated.
// We can blank them directly here, without using the OpenEntryEditorRunnable.
blankSingleTabEntryEditors();
if ( searches.length == 1 )
{
try
{
SearchResultEditorInput input = new SearchResultEditorInput( searches[0] );
view.getSite().getPage().openEditor( input, SearchResultEditor.getId(), false );
}
catch ( PartInitException e )
{
}
}
}
}
}
/**
* Blanks all single-tab entry editors.
*/
private void blankSingleTabEntryEditors()
{
IWorkbenchPage activePage = PlatformUI.getWorkbench().getActiveWorkbenchWindow().getActivePage();
for ( IEditorReference ref : activePage.getEditorReferences() )
{
IWorkbenchPart part = ref.getPart( false );
if ( part instanceof IEntryEditor )
{
IEntryEditor editor = ( IEntryEditor ) part;
if ( ( editor != null ) && ( editor.getEntryEditorInput() != null )
&& ( editor.getEntryEditorInput().getResolvedEntry() != null )
&& ( editor.getEntryEditorInput().getExtension() != null )
&& ( !editor.getEntryEditorInput().getExtension().isMultiWindow() ) )
{
EntryEditorExtension extension = editor.getEntryEditorInput().getExtension();
String editorId = extension.getEditorId();
EntryEditorInput input = new EntryEditorInput( ( IEntry ) null, extension );
try
{
view.getSite().getPage().openEditor( input, editorId, false );
}
catch ( PartInitException e )
{
}
}
}
}
}
/**
* {@inheritDoc}
*/
public void dispose()
{
if ( view != null )
{
EventRegistry.removeSearchUpdateListener( this );
EventRegistry.removeBookmarkUpdateListener( this );
EventRegistry.removeEntryUpdateListener( this );
ConnectionEventRegistry.removeConnectionUpdateListener( this );
view.getSite().getPage().removePartListener( partListener );
view.getSite().getWorkbenchWindow().getSelectionService().removePostSelectionListener(
ConnectionView.getId(), connectionSelectionListener );
viewer.removeOpenListener( openListener );
viewer.removeSelectionChangedListener( viewerSelectionListener );
view = null;
connectionToExpandedElementsMap.clear();
connectionToExpandedElementsMap = null;
connectionToSelectedElementMap.clear();
connectionToSelectedElementMap = null;
}
super.dispose();
}
/**
* Sets the input to the viewer and saves/restores the expanded and selected elements.
*
* @param connection the connection input
*/
void setInput( IBrowserConnection connection )
{
// only if another connection is selected
if ( connection != viewer.getInput() )
{
IBrowserConnection currentConnection = viewer.getInput() instanceof IBrowserConnection ? ( IBrowserConnection ) viewer
.getInput()
: null;
// save expanded elements and selection
if ( currentConnection != null )
{
connectionToExpandedElementsMap.put( currentConnection, viewer.getExpandedElements() );
if ( !viewer.getSelection().isEmpty() )
{
connectionToSelectedElementMap.put( currentConnection, viewer.getSelection() );
}
}
// change input
viewer.setInput( connection );
view.getActionGroup().setInput( connection );
view.getMainWidget().getQuickSearchWidget().setInput( connection );
// restore expanded elements and selection
if ( view != null && connection != null )
{
if ( connectionToExpandedElementsMap.containsKey( connection ) )
{
viewer.setExpandedElements( ( Object[] ) connectionToExpandedElementsMap.get( connection ) );
}
if ( connectionToSelectedElementMap.containsKey( connection )
&& this.view.getSite().getPage().isPartVisible( view ) )
{
viewer.setSelection( ( ISelection ) connectionToSelectedElementMap.get( connection ), true );
}
}
}
}
/**
* @see org.apache.directory.studio.connection.core.event.ConnectionUpdateListener#connectionOpened(org.apache.directory.studio.connection.core.Connection)
*/
public void connectionOpened( Connection connection )
{
IBrowserConnection browserConnection = BrowserCorePlugin.getDefault().getConnectionManager()
.getBrowserConnection( connection );
if ( browserConnection == null )
{
//
// If browser connection is null then it has been temporarily created
// by a dialog or other transient entity. Only display in the view those
// browser connections stored in the connection manager
//
return;
}
// expand viewer
viewer.refresh( browserConnection );
viewer.expandToLevel( 2 );
// expand root DSE to show base entries
IRootDSE rootDSE = browserConnection.getRootDSE();
viewer.expandToLevel( rootDSE, 1 );
// expand base entries, if requested
if ( view.getConfiguration().getPreferences().isExpandBaseEntries() )
{
viewer.expandToLevel( rootDSE, 2 );
}
}
/**
* @see org.apache.directory.studio.connection.core.event.ConnectionUpdateListener#connectionClosed(org.apache.directory.studio.connection.core.Connection)
*/
public void connectionClosed( Connection connection )
{
IBrowserConnection browserConnection = BrowserCorePlugin.getDefault().getConnectionManager()
.getBrowserConnection( connection );
if (browserConnection == null)
{
//
// If browser connection is null then it has been temporarily created
// by a dialog or other transient entity. Only display in the view those
// browser connections stored in the connection manager
//
return;
}
viewer.collapseAll();
connectionToExpandedElementsMap.remove( browserConnection );
connectionToSelectedElementMap.remove( browserConnection );
viewer.refresh( browserConnection );
}
/**
* {@inheritDoc}
*
* This viewer selects the updated search.
*/
public void searchUpdated( SearchUpdateEvent searchUpdateEvent )
{
ISearch search = searchUpdateEvent.getSearch();
if ( ( search instanceof IQuickSearch ) && ( searchUpdateEvent.getDetail() == EventDetail.SEARCH_REMOVED ) )
{
if ( search.getBrowserConnection().getQuickSearch() == search )
{
search.getBrowserConnection().setQuickSearch( null );
}
}
viewer.refresh();
if ( search instanceof IContinuation )
{
viewer.setSelection( new StructuredSelection( search ), true );
}
else if ( search.getBrowserConnection().getSearchManager().getSearches().contains( search ) )
{
viewer.setSelection( new StructuredSelection( search ), true );
}
else if ( ( search instanceof IQuickSearch ) && ( searchUpdateEvent.getDetail() != EventDetail.SEARCH_REMOVED ) )
{
if ( search.getBrowserConnection().getQuickSearch() == search )
{
viewer.setSelection( new StructuredSelection( search ), true );
}
}
else
{
Object searchCategory = ( ( ITreeContentProvider ) viewer.getContentProvider() ).getParent( search );
if ( searchCategory != null )
{
viewer.setSelection( new StructuredSelection( searchCategory ), true );
}
}
}
/**
* {@inheritDoc}
*/
public void bookmarkUpdated( BookmarkUpdateEvent bookmarkUpdateEvent )
{
viewer.refresh();
}
/**
* {@inheritDoc}
*
* This implementation refreshes the tree and
* selects an entry depending on the event type.
*/
public void entryUpdated( EntryModificationEvent event )
{
// Don't handle attribute initalization, could cause double
// retrieval of children.
//
// When double-clicking an entry two Jobs/Threads are started:
// - InitializeAttributesJob and
// - InitializeChildrenJob
// If the InitializeAttributesJob is finished first the
// AttributesInitializedEvent is fired. If this causes
// a refresh of the tree before the children are initialized
// another InitializeChildrenJob is executed.
if ( event instanceof AttributesInitializedEvent && !( event.getModifiedEntry() instanceof IRootDSE ) )
{
return;
}
if ( event instanceof EntryAddedEvent )
{
viewer.refresh( event.getModifiedEntry(), true );
viewer.refresh( event.getModifiedEntry().getParententry(), true );
viewer.setSelection( new StructuredSelection( event.getModifiedEntry() ), true );
}
else if ( event instanceof EntryRenamedEvent )
{
EntryRenamedEvent ere = ( EntryRenamedEvent ) event;
viewer.refresh( ere.getNewEntry().getParententry(), true );
viewer.setSelection( new StructuredSelection( ere.getNewEntry() ), true );
}
else if ( event instanceof EntryMovedEvent )
{
EntryMovedEvent eme = ( EntryMovedEvent ) event;
viewer.refresh( eme.getOldEntry().getParententry(), true );
viewer.refresh( eme.getNewEntry().getParententry(), true );
viewer.setSelection( new StructuredSelection( eme.getNewEntry() ), true );
}
else if ( event instanceof EntryDeletedEvent )
{
EntryDeletedEvent ede = ( EntryDeletedEvent ) event;
if ( ede.getModifiedEntry().getParententry() != null )
{
viewer.refresh( ede.getModifiedEntry().getParententry(), true );
viewer.setSelection( new StructuredSelection( ede.getModifiedEntry().getParententry() ), true );
}
else
{
viewer.refresh();
}
}
else if ( event instanceof BulkModificationEvent )
{
viewer.refresh();
}
else if ( event instanceof ChildrenInitializedEvent )
{
// Getting the children of the entry to collapse their nodes
// See DIRSTUDIO-481 (refreshing of attributes and children)
Object[] children = ( ( BrowserContentProvider ) viewer.getContentProvider() ).getChildren( event
.getModifiedEntry() );
for ( Object child : children )
{
// We're only collapsing the node if it is expanded
if ( viewer.getExpandedState( child ) )
{
viewer.collapseToLevel( child, TreeViewer.ALL_LEVELS );
// There seem to be a bug (maybe it's a feature?!?) with nodes that were expanded
// but does not have any child.
// The call to 'viewer.collapseToLevel(...)' has no effect, the node stays expanded...
if ( viewer.getExpandedState( child ) )
{
// In that particular case, we need to remove the child from the tree viewer.
// As it's a costly�operation we're only using this in that particular case,
// and not as default option.
viewer.remove( child );
}
}
}
viewer.refresh( event.getModifiedEntry(), true );
}
else if ( !( event.getModifiedEntry() instanceof DummyEntry ) )
{
viewer.refresh( event.getModifiedEntry(), true );
}
}
}