/*
* 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.wizards;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import org.apache.directory.studio.ldapbrowser.common.widgets.BaseWidgetUtils;
import org.apache.directory.studio.ldapbrowser.core.events.EventRegistry;
import org.apache.directory.studio.ldapbrowser.core.model.IAttribute;
import org.apache.directory.studio.ldapbrowser.core.model.IValue;
import org.apache.directory.studio.ldapbrowser.core.model.impl.Attribute;
import org.apache.directory.studio.ldapbrowser.core.model.impl.DummyEntry;
import org.apache.directory.studio.ldapbrowser.core.model.impl.Value;
import org.apache.directory.studio.ldapbrowser.core.model.schema.ObjectClassDescription;
import org.apache.directory.studio.ldapbrowser.ui.BrowserUIConstants;
import org.apache.directory.studio.ldapbrowser.ui.BrowserUIPlugin;
import org.eclipse.jface.fieldassist.DecoratedField;
import org.eclipse.jface.fieldassist.FieldDecoration;
import org.eclipse.jface.fieldassist.FieldDecorationRegistry;
import org.eclipse.jface.fieldassist.IControlCreator;
import org.eclipse.jface.viewers.ArrayContentProvider;
import org.eclipse.jface.viewers.DoubleClickEvent;
import org.eclipse.jface.viewers.IDoubleClickListener;
import org.eclipse.jface.viewers.ISelection;
import org.eclipse.jface.viewers.IStructuredSelection;
import org.eclipse.jface.viewers.LabelProvider;
import org.eclipse.jface.viewers.StructuredSelection;
import org.eclipse.jface.viewers.TableViewer;
import org.eclipse.jface.viewers.Viewer;
import org.eclipse.jface.viewers.ViewerFilter;
import org.eclipse.jface.viewers.ViewerSorter;
import org.eclipse.jface.wizard.WizardPage;
import org.eclipse.swt.SWT;
import org.eclipse.swt.events.KeyAdapter;
import org.eclipse.swt.events.KeyEvent;
import org.eclipse.swt.events.ModifyEvent;
import org.eclipse.swt.events.ModifyListener;
import org.eclipse.swt.events.SelectionAdapter;
import org.eclipse.swt.events.SelectionEvent;
import org.eclipse.swt.graphics.Image;
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.Control;
import org.eclipse.swt.widgets.Label;
import org.eclipse.swt.widgets.Text;
/**
* The NewEntryTypeWizardPage is used to select the entry's object classes.
*
* @author <a href="mailto:dev@directory.apache.org">Apache Directory Project</a>
* @version $Rev$, $Date$
*/
public class NewEntryObjectclassWizardPage extends WizardPage
{
/** The Constant SIZING_SELECTION_WIDGET_HEIGHT. */
private final static int SIZING_SELECTION_WIDGET_HEIGHT = 250;
/** The Constant SIZING_SELECTION_WIDGET_WIDTH. */
private final static int SIZING_SELECTION_WIDGET_WIDTH = 400;
/** The wizard. */
private NewEntryWizard wizard;
/** The available object classes. */
private List<ObjectClassDescription> availableObjectClasses;
/** The available object classes instant search. */
private Text availableObjectClassesInstantSearch;
/** The available object classes viewer. */
private TableViewer availableObjectClassesViewer;
/** The selected object classes. */
private List<ObjectClassDescription> selectedObjectClasses;
/** The selected object classes viewer. */
private TableViewer selectedObjectClassesViewer;
/** The add button. */
private Button addButton;
/** The remove button. */
private Button removeButton;
private LabelProvider labelProvider = new LabelProvider()
{
/* (non-Javadoc)
* @see org.eclipse.jface.viewers.LabelProvider#getImage(java.lang.Object)
*/
public Image getImage( Object element )
{
if ( element instanceof ObjectClassDescription )
{
ObjectClassDescription ocd = ( ObjectClassDescription ) element;
if ( ocd.isAbstract() )
{
return BrowserUIPlugin.getDefault().getImage( BrowserUIConstants.IMG_OCD_ABSTRACT );
}
else if ( ocd.isAuxiliary() )
{
return BrowserUIPlugin.getDefault().getImage( BrowserUIConstants.IMG_OCD_AUXILIARY );
}
else if ( ocd.isStructural() )
{
return BrowserUIPlugin.getDefault().getImage( BrowserUIConstants.IMG_OCD_STRUCTURAL );
}
else
{
return BrowserUIPlugin.getDefault().getImage( BrowserUIConstants.IMG_OCD );
}
}
// Default
return super.getImage( element );
}
};
/**
* Creates a new instance of NewEntryObjectclassWizardPage.
*
* @param pageName the page name
* @param wizard the wizard
*/
public NewEntryObjectclassWizardPage( String pageName, NewEntryWizard wizard )
{
super( pageName );
setTitle( "Object Classes" );
setDescription( "Please select object classes of the new entry. Select at least one structural object class." );
setImageDescriptor( BrowserUIPlugin.getDefault().getImageDescriptor( BrowserUIConstants.IMG_ENTRY_WIZARD ) );
setPageComplete( false );
this.wizard = wizard;
this.availableObjectClasses = new ArrayList<ObjectClassDescription>();
this.selectedObjectClasses = new ArrayList<ObjectClassDescription>();
}
/**
* Validates the input fields.
*/
private void validate()
{
if ( !selectedObjectClasses.isEmpty() )
{
boolean hasOneStructuralOC = false;
for ( ObjectClassDescription ocd : selectedObjectClasses )
{
if ( ocd.isStructural() )
{
hasOneStructuralOC = true;
break;
}
}
if ( !hasOneStructuralOC )
{
setMessage( "Please select at least one structural object class.", WizardPage.WARNING );
}
else
{
setMessage( null );
}
setPageComplete( true );
saveState();
}
else
{
setPageComplete( false );
setMessage( null );
}
}
/**
* Loads the state of selected and available object classes from
* the prototype entry. Called when this page becomes visible.
*/
private void loadState()
{
availableObjectClasses.clear();
selectedObjectClasses.clear();
if ( wizard.getSelectedConnection() != null )
{
// if ( !wizard.getSelectedConnection().isOpened() )
// {
// OpenConnectionsJob job = new OpenConnectionsJob( wizard.getSelectedConnection().getConnection() );
// RunnableContextJobAdapter.execute( job, getContainer() );
// }
availableObjectClasses.addAll( Arrays.asList( wizard.getSelectedConnection().getSchema()
.getObjectClassDescriptions() ) );
DummyEntry newEntry = wizard.getPrototypeEntry();
IAttribute ocAttribute = newEntry.getAttribute( IAttribute.OBJECTCLASS_ATTRIBUTE );
if ( ocAttribute != null )
{
String[] ocValues = ocAttribute.getStringValues();
for ( int i = 0; i < ocValues.length; i++ )
{
String ocValue = ocValues[i];
ObjectClassDescription ocd = wizard.getSelectedConnection().getSchema().getObjectClassDescription(
ocValue );
availableObjectClasses.remove( ocd );
selectedObjectClasses.add( ocd );
}
}
}
availableObjectClassesViewer.refresh();
selectedObjectClassesViewer.refresh();
}
/**
* Saves the state of selected object classes to the entry.
*/
private void saveState()
{
DummyEntry newEntry = wizard.getPrototypeEntry();
try
{
EventRegistry.suspendEventFireingInCurrentThread();
// set new objectClass values
IAttribute ocAttribute = newEntry.getAttribute( IAttribute.OBJECTCLASS_ATTRIBUTE );
if ( ocAttribute == null )
{
ocAttribute = new Attribute( newEntry, IAttribute.OBJECTCLASS_ATTRIBUTE );
newEntry.addAttribute( ocAttribute );
}
IValue[] values = ocAttribute.getValues();
for ( int i = 0; i < values.length; i++ )
{
ocAttribute.deleteValue( values[i] );
}
for ( Iterator<ObjectClassDescription> it = selectedObjectClasses.iterator(); it.hasNext(); )
{
ObjectClassDescription ocd = it.next();
ocAttribute.addValue( new Value( ocAttribute, ocd.getNames()[0] ) );
}
}
finally
{
EventRegistry.resumeEventFireingInCurrentThread();
}
}
/**
* {@inheritDoc}
*
* This implementation initializes the list of available and selected
* object classes when this page becomes visible.
*/
public void setVisible( boolean visible )
{
super.setVisible( visible );
if ( visible )
{
loadState();
validate();
availableObjectClassesInstantSearch.setFocus();
}
}
/**
* {@inheritDoc}
*/
public void createControl( Composite parent )
{
Composite composite = new Composite( parent, SWT.NONE );
GridLayout gl = new GridLayout( 3, false );
composite.setLayout( gl );
composite.setLayoutData( new GridData( GridData.FILL_BOTH ) );
Label availableLabel = new Label( composite, SWT.NONE );
availableLabel.setText( "Available object classes" );
Label buttonLabel = new Label( composite, SWT.NONE );
buttonLabel.setText( "" );
Label selectedLabel = new Label( composite, SWT.NONE );
selectedLabel.setText( "Selected object classes" );
Composite availableObjectClassesComposite = BaseWidgetUtils.createColumnContainer( composite, 1, 1 );
if ( FieldDecorationRegistry.getDefault().getFieldDecoration( getClass().getName() ) == null )
{
FieldDecoration dummy = FieldDecorationRegistry.getDefault().getFieldDecoration(
FieldDecorationRegistry.DEC_CONTENT_PROPOSAL );
FieldDecorationRegistry.getDefault().registerFieldDecoration( getClass().getName(),
"You may enter a filter to restrict the list below", dummy.getImage() );
}
final FieldDecoration fieldDecoration = FieldDecorationRegistry.getDefault().getFieldDecoration(
getClass().getName() );
final DecoratedField availabeObjectClassesInstantSearchField = new DecoratedField(
availableObjectClassesComposite, SWT.BORDER, new IControlCreator()
{
public Control createControl( Composite parent, int style )
{
return BaseWidgetUtils.createText( parent, "", 1 );
}
} );
availabeObjectClassesInstantSearchField.addFieldDecoration( fieldDecoration, SWT.TOP | SWT.LEFT, true );
availabeObjectClassesInstantSearchField.getLayoutControl().setLayoutData(
new GridData( SWT.FILL, SWT.CENTER, true, false ) );
availableObjectClassesInstantSearch = ( Text ) availabeObjectClassesInstantSearchField.getControl();
availableObjectClassesInstantSearch.addModifyListener( new ModifyListener()
{
public void modifyText( ModifyEvent e )
{
availableObjectClassesViewer.refresh();
if ( availableObjectClassesViewer.getTable().getItemCount() >= 1 )
{
Object item = availableObjectClassesViewer.getElementAt( 0 );
availableObjectClassesViewer.setSelection( new StructuredSelection( item ) );
}
}
} );
availableObjectClassesInstantSearch.addKeyListener( new KeyAdapter()
{
public void keyPressed( KeyEvent e )
{
if ( e.keyCode == SWT.ARROW_DOWN )
{
availableObjectClassesViewer.getTable().setFocus();
}
}
} );
availableObjectClassesViewer = new TableViewer( availableObjectClassesComposite );
GridData data = new GridData( GridData.FILL_BOTH );
data.heightHint = SIZING_SELECTION_WIDGET_HEIGHT;
data.widthHint = ( int ) ( SIZING_SELECTION_WIDGET_WIDTH * 0.4 );
availableObjectClassesViewer.getTable().setLayoutData( data );
availableObjectClassesViewer.setContentProvider( new ArrayContentProvider() );
availableObjectClassesViewer.setLabelProvider( labelProvider );
availableObjectClassesViewer.setSorter( new ViewerSorter() );
availableObjectClassesViewer.addFilter( new InstantSearchFilter( availableObjectClassesInstantSearch ) );
availableObjectClassesViewer.setInput( availableObjectClasses );
availableObjectClassesViewer.addDoubleClickListener( new IDoubleClickListener()
{
public void doubleClick( DoubleClickEvent event )
{
add( event.getSelection() );
}
} );
availableObjectClassesViewer.getTable().addKeyListener( new KeyAdapter()
{
public void keyPressed( KeyEvent e )
{
if ( e.keyCode == SWT.ARROW_UP )
{
if ( availableObjectClassesViewer.getTable().getSelectionIndex() <= 0 )
{
availableObjectClassesInstantSearch.setFocus();
}
}
}
} );
Composite buttonComposite = new Composite( composite, SWT.NONE );
gl = new GridLayout( 1, true );
buttonComposite.setLayout( gl );
data = new GridData( GridData.FILL_BOTH );
data.heightHint = SIZING_SELECTION_WIDGET_HEIGHT;
// data.widthHint = (int)(SIZING_SELECTION_WIDGET_WIDTH * 0.2);
data.horizontalAlignment = SWT.CENTER;
buttonComposite.setLayoutData( data );
Label label0 = new Label( buttonComposite, SWT.NONE );
data = new GridData();
data.grabExcessHorizontalSpace = true;
data.grabExcessVerticalSpace = true;
label0.setLayoutData( data );
addButton = BaseWidgetUtils.createButton( buttonComposite, "&Add", 1 );
removeButton = BaseWidgetUtils.createButton( buttonComposite, "&Remove", 1 );
Label label3 = new Label( buttonComposite, SWT.NONE );
data = new GridData();
data.grabExcessHorizontalSpace = true;
data.grabExcessVerticalSpace = true;
label3.setLayoutData( data );
addButton.addSelectionListener( new SelectionAdapter()
{
public void widgetSelected( SelectionEvent e )
{
add( availableObjectClassesViewer.getSelection() );
}
} );
removeButton.addSelectionListener( new SelectionAdapter()
{
public void widgetSelected( SelectionEvent e )
{
remove( selectedObjectClassesViewer.getSelection() );
}
} );
selectedObjectClassesViewer = new TableViewer( composite );
data = new GridData( GridData.FILL_BOTH );
data.heightHint = SIZING_SELECTION_WIDGET_HEIGHT;
data.widthHint = ( int ) ( SIZING_SELECTION_WIDGET_WIDTH * 0.4 );
selectedObjectClassesViewer.getTable().setLayoutData( data );
selectedObjectClassesViewer.setContentProvider( new ArrayContentProvider() );
selectedObjectClassesViewer.setLabelProvider( labelProvider );
selectedObjectClassesViewer.setSorter( new ViewerSorter() );
selectedObjectClassesViewer.setInput( selectedObjectClasses );
selectedObjectClassesViewer.addDoubleClickListener( new IDoubleClickListener()
{
public void doubleClick( DoubleClickEvent event )
{
remove( event.getSelection() );
}
} );
setControl( composite );
}
/**
* Adds the selected object classes to the list of selected
* object classes.
*
* @param iselection the selection
*/
private void add( ISelection iselection )
{
IStructuredSelection selection = ( IStructuredSelection ) iselection;
Iterator it = selection.iterator();
while ( it.hasNext() )
{
ObjectClassDescription ocd = ( ObjectClassDescription ) it.next();
if ( availableObjectClasses.contains( ocd ) && !selectedObjectClasses.contains( ocd ) )
{
availableObjectClasses.remove( ocd );
selectedObjectClasses.add( ocd );
// recursively add superior object classes
ObjectClassDescription[] superiorObjectClassDescriptions = ocd.getSuperiorObjectClassDescriptions();
if ( superiorObjectClassDescriptions.length > 0 )
{
add( new StructuredSelection( superiorObjectClassDescriptions ) );
}
}
}
availableObjectClassesViewer.refresh();
selectedObjectClassesViewer.refresh();
validate();
if ( !"".equals( availableObjectClassesInstantSearch.getText() ) )
{
availableObjectClassesInstantSearch.setText( "" );
availableObjectClassesInstantSearch.setFocus();
}
}
/**
* Removes the selected object classes from the list of selected
* object classes.
*
* @param iselection the iselection
*/
private void remove( ISelection iselection )
{
IStructuredSelection selection = ( IStructuredSelection ) iselection;
Iterator it = selection.iterator();
while ( it.hasNext() )
{
ObjectClassDescription ocd = ( ObjectClassDescription ) it.next();
if ( !availableObjectClasses.contains( ocd ) && selectedObjectClasses.contains( ocd ) )
{
selectedObjectClasses.remove( ocd );
availableObjectClasses.add( ocd );
// recursively remove sub object classes
ObjectClassDescription[] subObjectClassDescriptions = ocd.getSubObjectClassDescriptions();
if ( subObjectClassDescriptions.length > 0 )
{
remove( new StructuredSelection( subObjectClassDescriptions ) );
}
}
}
availableObjectClassesViewer.refresh();
selectedObjectClassesViewer.refresh();
validate();
}
/**
* Saves dialog settings.
*/
public void saveDialogSettings()
{
}
/**
* The Class InstantSearchFilter.
*/
private class InstantSearchFilter extends ViewerFilter
{
/** The filter text. */
private Text filterText;
/**
* Creates a new instance of InstantSearchFilter.
*
* @param filterText the filter text
*/
private InstantSearchFilter( Text filterText )
{
this.filterText = filterText;
}
/**
* {@inheritDoc}
*/
public boolean select( Viewer viewer, Object parentElement, Object element )
{
if ( element instanceof ObjectClassDescription )
{
ObjectClassDescription ocd = ( ObjectClassDescription ) element;
Set<String> lowerCaseIdentifierSet = ocd.getLowerCaseIdentifierSet();
for ( Iterator<String> it = lowerCaseIdentifierSet.iterator(); it.hasNext(); )
{
String s = it.next();
if ( s.toLowerCase().startsWith( filterText.getText().toLowerCase() ) )
{
return true;
}
}
}
return false;
}
}
}