blob: d97864b23cf4ac9f27714d7b52aac89304486f8f [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.common.dialogs;
import java.io.File;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import org.apache.directory.api.util.FileUtils;
import org.apache.directory.studio.connection.ui.ConnectionUIPlugin;
import org.apache.directory.studio.ldapbrowser.common.BrowserCommonActivator;
import org.apache.directory.studio.ldapbrowser.common.BrowserCommonConstants;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.jface.dialogs.Dialog;
import org.eclipse.jface.dialogs.IDialogConstants;
import org.eclipse.jface.resource.JFaceResources;
import org.eclipse.swt.SWT;
import org.eclipse.swt.events.ModifyEvent;
import org.eclipse.swt.events.ModifyListener;
import org.eclipse.swt.layout.GridData;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Control;
import org.eclipse.swt.widgets.FileDialog;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.swt.widgets.Text;
/**
* Dialog to display binary data in hex format. It could be
* used to load and save binary data from and to disk.
*
* @author <a href="mailto:dev@directory.apache.org">Apache Directory Project</a>
*/
public class HexDialog extends Dialog
{
public static final String LOAD_FILE_NAME_TOOLTIP = "LoadFileName";
/** The default title. */
private static final String DIALOG_TITLE = Messages.getString( "HexDialog.HexEditor" ); //$NON-NLS-1$
/** The button ID for the edit as text button. */
private static final int EDIT_AS_TEXT_BUTTON_ID = 9997;
/** The button ID for the load button. */
private static final int LOAD_BUTTON_ID = 9998;
/** The button ID for the save button. */
private static final int SAVE_BUTTON_ID = 9999;
/** Hidden text to set the filename, used for UI tests. */
private Text loadFilenameText;
/** The current data. */
private byte[] currentData;
/** The return data. */
private byte[] returnData;
/** The text field with the binary data. */
private Text hexText;
/**
* Creates a new instance of HexDialog.
*
* @param parentShell the parent shell
* @param initialData the initial data
*/
public HexDialog( Shell parentShell, byte[] initialData )
{
super( parentShell );
super.setShellStyle( super.getShellStyle() | SWT.RESIZE );
this.currentData = initialData;
}
/**
* @see org.eclipse.jface.dialogs.Dialog#buttonPressed(int)
*/
protected void buttonPressed( int buttonId )
{
if ( buttonId == IDialogConstants.OK_ID )
{
returnData = currentData;
}
else if ( buttonId == EDIT_AS_TEXT_BUTTON_ID )
{
TextDialog dialog = new TextDialog( getShell(), new String( currentData, StandardCharsets.UTF_8 ) );
if ( dialog.open() == TextDialog.OK )
{
String text = dialog.getText();
currentData = text.getBytes( StandardCharsets.UTF_8 );
hexText.setText( toFormattedHex( currentData ) );
}
}
else if ( buttonId == SAVE_BUTTON_ID )
{
FileDialog fileDialog = new FileDialog( getShell(), SWT.SAVE );
fileDialog.setText( Messages.getString( "HexDialog.SaveData" ) ); //$NON-NLS-1$
String returnedFileName = fileDialog.open();
if ( returnedFileName != null )
{
try
{
File file = new File( returnedFileName );
FileUtils.writeByteArrayToFile( file, currentData );
}
catch ( IOException e )
{
ConnectionUIPlugin.getDefault().getExceptionHandler().handleException(
new Status( IStatus.ERROR, BrowserCommonConstants.PLUGIN_ID, IStatus.ERROR, Messages
.getString( "HexDialog.CantWriteToFile" ), e ) ); //$NON-NLS-1$
}
}
}
else if ( buttonId == LOAD_BUTTON_ID )
{
FileDialog fileDialog = new FileDialog( getShell(), SWT.OPEN );
fileDialog.setText( Messages.getString( "HexDialog.LoadData" ) ); //$NON-NLS-1$
String returnedFileName = fileDialog.open();
if ( returnedFileName != null )
{
loadFile( returnedFileName );
}
}
else
{
returnData = null;
}
super.buttonPressed( buttonId );
}
private void loadFile( String fileName )
{
try
{
File file = new File( fileName );
currentData = FileUtils.readFileToByteArray( file );
hexText.setText( toFormattedHex( currentData ) );
}
catch ( IOException e )
{
ConnectionUIPlugin.getDefault().getExceptionHandler().handleException(
new Status( IStatus.ERROR, BrowserCommonConstants.PLUGIN_ID, IStatus.ERROR, Messages
.getString( "HexDialog.CantReadFile" ), e ) ); //$NON-NLS-1$
}
}
/**
* Small helper.
*/
private boolean isEditable( byte[] b )
{
if ( b == null )
{
return false;
}
return !( new String( b, StandardCharsets.UTF_8 ).contains( "\uFFFD" ) );
}
/**
* @see org.eclipse.jface.window.Window#configureShell(org.eclipse.swt.widgets.Shell)
*/
protected void configureShell( Shell shell )
{
super.configureShell( shell );
shell.setText( DIALOG_TITLE );
shell.setImage( BrowserCommonActivator.getDefault().getImage( BrowserCommonConstants.IMG_HEXEDITOR ) );
}
/**
* @see org.eclipse.jface.dialogs.Dialog#createButtonsForButtonBar(org.eclipse.swt.widgets.Composite)
*/
protected void createButtonsForButtonBar( Composite parent )
{
((GridLayout) parent.getLayout()).numColumns++;
loadFilenameText = new Text( parent, SWT.NONE );
loadFilenameText.setToolTipText( LOAD_FILE_NAME_TOOLTIP );
loadFilenameText.setBackground( parent.getBackground() );
loadFilenameText.addModifyListener( new ModifyListener()
{
public void modifyText( ModifyEvent e )
{
loadFile( loadFilenameText.getText() );
}
} );
if ( isEditable( currentData ) )
{
createButton( parent, EDIT_AS_TEXT_BUTTON_ID, Messages.getString( "HexDialog.EditAsText" ), false ); //$NON-NLS-1$
}
createButton( parent, LOAD_BUTTON_ID, Messages.getString( "HexDialog.LoadDataButton" ), false ); //$NON-NLS-1$
createButton( parent, SAVE_BUTTON_ID, Messages.getString( "HexDialog.SaveDataButton" ), false ); //$NON-NLS-1$
createButton( parent, IDialogConstants.OK_ID, IDialogConstants.OK_LABEL, false );
createButton( parent, IDialogConstants.CANCEL_ID, IDialogConstants.CANCEL_LABEL, false );
}
/**
* @see org.eclipse.jface.dialogs.Dialog#createDialogArea(org.eclipse.swt.widgets.Composite)
*/
protected Control createDialogArea( Composite parent )
{
// create composite
Composite composite = ( Composite ) super.createDialogArea( parent );
hexText = new Text( composite, SWT.MULTI | SWT.BORDER | SWT.H_SCROLL | SWT.V_SCROLL | SWT.READ_ONLY );
hexText.setFont( JFaceResources.getFont( JFaceResources.TEXT_FONT ) );
hexText.setText( toFormattedHex( currentData ) );
GridData gd = new GridData( GridData.FILL_BOTH );
gd.widthHint = convertHorizontalDLUsToPixels( ( int ) ( IDialogConstants.MINIMUM_MESSAGE_AREA_WIDTH * 1.4 ) );
gd.heightHint = convertHorizontalDLUsToPixels( IDialogConstants.MINIMUM_MESSAGE_AREA_WIDTH / 2 );
hexText.setLayoutData( gd );
applyDialogFont( composite );
return composite;
}
/**
* Formats the binary data in two columns. One containing the hex
* presentation and one containing the ASCII presentation of each byte.
*
* 91 a1 08 23 42 b1 c1 15 52 d1 f0 24 33 62 72 82 ...#B... R..$3br.
* 09 0a 16 17 18 19 1a 25 26 27 28 29 2a 34 35 36 .......% &'()*456
*
* @param data the data
*
* @return the formatted string
*/
private String toFormattedHex( byte[] data )
{
StringBuffer sb = new StringBuffer();
for ( int i = 0; i < data.length; i++ )
{
// get byte
int b = ( int ) data[i];
if ( b < 0 )
{
b = 256 + b;
}
// format to hex, optionally prepend a 0
String s = Integer.toHexString( b );
if ( s.length() == 1 )
{
s = "0" + s; //$NON-NLS-1$
}
// space between hex numbers
sb.append( s ).append( " " ); //$NON-NLS-1$
// extra space after 8 hex numbers
if ( ( i + 1 ) % 8 == 0 && ( i + 1 ) % 16 != 0 )
{
sb.append( " " ); //$NON-NLS-1$
}
// if end of data is reached then fill with spaces
if ( i == data.length - 1 )
{
while ( ( i + 1 ) % 16 != 0 )
{
sb.append( " " ); //$NON-NLS-1$
if ( ( i + 1 ) % 8 == 0 )
{
sb.append( " " ); //$NON-NLS-1$
}
i++;
}
}
// print ASCII characters after 16 hex numbers
if ( ( i + 1 ) % 16 == 0 )
{
sb.append( " " ); //$NON-NLS-1$
for ( int x = i - 16 + 1; x <= i && x < data.length; x++ )
{
// print ASCII character if printable
// otherwise print a dot
if ( data[x] > 32 && data[x] < 127 )
{
sb.append( ( char ) data[x] );
}
else
{
sb.append( '.' );
}
// space after 8 characters
if ( ( x + 1 ) % 8 == 0 )
{
sb.append( " " ); //$NON-NLS-1$
}
}
}
// start new line after 16 hex numbers
if ( ( i + 1 ) % 16 == 0 )
{
sb.append( "\r\n" ); //$NON-NLS-1$
}
}
return sb.toString();
}
/**
* Gets the data.
*
* @return the data
*/
public byte[] getData()
{
return returnData;
}
}