com.alee.extended.style.CodeLinkGenerator Maven / Gradle / Ivy
/*
* This file is part of WebLookAndFeel library.
*
* WebLookAndFeel library is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* WebLookAndFeel library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with WebLookAndFeel library. If not, see .
*/
package com.alee.extended.style;
import com.alee.api.jdk.Objects;
import com.alee.extended.window.PopOverDirection;
import com.alee.extended.window.WebPopOver;
import com.alee.laf.colorchooser.WebColorChooserPanel;
import com.alee.laf.list.WebList;
import com.alee.laf.scroll.WebScrollPane;
import com.alee.laf.slider.WebSlider;
import com.alee.managers.hotkey.Hotkey;
import com.alee.managers.style.ComponentDescriptor;
import com.alee.managers.style.StyleId;
import com.alee.managers.style.StyleManager;
import com.alee.managers.style.data.ComponentStyleConverter;
import com.alee.utils.MathUtils;
import com.alee.utils.collection.ImmutableList;
import com.alee.utils.xml.ColorConverter;
import com.alee.utils.xml.InsetsConverter;
import com.thoughtworks.xstream.converters.basic.FloatConverter;
import net.htmlparser.jericho.*;
import org.fife.ui.rsyntaxtextarea.LinkGenerator;
import org.fife.ui.rsyntaxtextarea.LinkGeneratorResult;
import org.fife.ui.rsyntaxtextarea.RSyntaxTextArea;
import org.slf4j.LoggerFactory;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import javax.swing.event.HyperlinkEvent;
import javax.swing.text.BadLocationException;
import java.awt.*;
import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.util.List;
import java.util.Locale;
/**
* @author Mikle Garin
*/
public class CodeLinkGenerator implements LinkGenerator
{
/**
* Code constants.
*/
private static final ImmutableList propertyNodes = new ImmutableList (
ComponentStyleConverter.COMPONENT_NODE,
ComponentStyleConverter.UI_NODE,
ComponentStyleConverter.PAINTER_NODE
);
private static final String trueString = "true";
private static final String falseString = "false";
private static final ImmutableList booleanContent = new ImmutableList ( trueString, falseString );
private static final ImmutableList colorContent = new ImmutableList ( "foreground", "fg", "background", "bg", "color" );
private static final ImmutableList insetsContent = new ImmutableList ( "insets", "margin" );
private static final ImmutableList opacityContent = new ImmutableList ( "opacity", "transparency" );
/**
* Data converters.
*/
private static final ColorConverter colorConverter = new ColorConverter ();
private static final InsetsConverter insetsConverter = new InsetsConverter ();
private static final FloatConverter floatConverter = new FloatConverter ();
/**
* Parent component.
*/
protected final Component parentComponent;
/**
* Runtime variables.
*/
protected Source src = null;
protected String text = null;
/**
* Constructs new code link generator.
*
* @param parentComponent parent component
*/
public CodeLinkGenerator ( final Component parentComponent )
{
super ();
this.parentComponent = parentComponent;
}
@Override
public LinkGeneratorResult isLinkAtOffset ( final RSyntaxTextArea source, final int pos )
{
final String code = source.getText ();
if ( src == null || Objects.notEquals ( src, code ) )
{
text = code;
src = new Source ( code );
src.setLogger ( null );
src.fullSequentialParse ();
}
final Element element = src.getEnclosingElement ( pos );
if ( element == null )
{
return null;
}
final Element parent = element.getParentElement ();
if ( parent == null )
{
return null;
}
final String name = element.getName ();
final String parentName = parent.getName ();
if ( ComponentStyleConverter.STYLE_NODE.equals ( name ) )
{
final Attributes attributes = element.getAttributes ();
for ( final Attribute attribute : attributes )
{
if ( attribute.getBegin () < pos && pos < attribute.getEnd () )
{
final String attributeName = attribute.getName ();
if ( attributeName.equals ( ComponentStyleConverter.COMPONENT_TYPE_ATTRIBUTE ) )
{
final Segment content = attribute.getValueSegment ();
final String type = element.getAttributeValue ( ComponentStyleConverter.COMPONENT_TYPE_ATTRIBUTE );
final ComponentDescriptor descriptor = StyleManager.getDescriptor ( type );
return new LinkGeneratorResult ()
{
@Override
public HyperlinkEvent execute ()
{
try
{
final WebPopOver typeChooser = new WebPopOver ( parentComponent );
typeChooser.setCloseOnFocusLoss ( true );
typeChooser.setPadding ( 5, 0, 5, 0 );
final List types = StyleManager.getDescriptors ();
final WebList typesList = new WebList ( types );
typesList.setOpaque ( false );
typesList.setVisibleRowCount ( Math.min ( 10, types.size () ) );
typesList.setSelectOnHover ( true );
typesList.setSelectedValue ( descriptor );
final Runnable commitChanges = new Runnable ()
{
@Override
public void run ()
{
final String typeString = typesList.getSelectedValue ().toString ();
source.replaceRange ( typeString, content.getBegin (), content.getEnd () );
typeChooser.dispose ();
}
};
typesList.addMouseListener ( new MouseAdapter ()
{
@Override
public void mouseReleased ( final MouseEvent e )
{
commitChanges.run ();
}
} );
typesList.addKeyListener ( new KeyAdapter ()
{
@Override
public void keyReleased ( final KeyEvent e )
{
if ( Hotkey.ENTER.isKeyTriggered ( e ) )
{
commitChanges.run ();
}
}
} );
final WebScrollPane scrollPane = new WebScrollPane ( StyleId.scrollpanePopup, typesList );
typeChooser.add ( scrollPane );
final int position = ( content.getBegin () + content.getEnd () ) / 2;
final Rectangle wb = source.getUI ().modelToView ( source, position );
typeChooser.show ( source, wb.x, wb.y, wb.width, wb.height, PopOverDirection.down );
return new HyperlinkEvent ( this, HyperlinkEvent.EventType.EXITED, null );
}
catch ( final BadLocationException e )
{
LoggerFactory.getLogger ( CodeLinkGenerator.class ).error ( e.toString (), e );
return null;
}
}
@Override
public int getSourceOffset ()
{
return content.getBegin ();
}
};
}
}
}
}
else if ( propertyNodes.contains ( parentName ) )
{
final Segment content = element.getContent ();
final String contentString = content.toString ();
if ( booleanContent.contains ( contentString ) )
{
return new LinkGeneratorResult ()
{
@Override
public HyperlinkEvent execute ()
{
final String str = contentString.equals ( trueString ) ? falseString : trueString;
source.replaceRange ( str, content.getBegin (), content.getEnd () );
return new HyperlinkEvent ( this, HyperlinkEvent.EventType.EXITED, null );
}
@Override
public int getSourceOffset ()
{
return content.getBegin ();
}
};
}
else
{
if ( contains ( name.toLowerCase ( Locale.ROOT ), colorContent ) )
{
final Color color = ( Color ) colorConverter.fromString ( contentString );
if ( color != null || contentString.equals ( ColorConverter.NULL_COLOR ) )
{
return new LinkGeneratorResult ()
{
@Override
public HyperlinkEvent execute ()
{
try
{
final WebPopOver colorChooser = new WebPopOver ( parentComponent );
colorChooser.setCloseOnFocusLoss ( true );
final WebColorChooserPanel colorChooserPanel = new WebColorChooserPanel ( false );
colorChooserPanel.setColor ( color != null ? color : Color.WHITE );
colorChooserPanel.addChangeListener ( new ChangeListener ()
{
private int length = content.getEnd () - content.getBegin ();
@Override
public void stateChanged ( final ChangeEvent e )
{
final Color newColor = colorChooserPanel.getColor ();
if ( color == null || newColor != null && !newColor.equals ( color ) )
{
final String colorString = colorConverter.toString ( newColor );
source.replaceRange ( colorString, content.getBegin (), content.getBegin () + length );
length = colorString.length ();
}
}
} );
colorChooser.add ( colorChooserPanel );
final int position = ( content.getBegin () + content.getEnd () ) / 2;
final Rectangle wb = source.getUI ().modelToView ( source, position );
colorChooser.show ( source, wb.x, wb.y, wb.width, wb.height, PopOverDirection.down );
return new HyperlinkEvent ( this, HyperlinkEvent.EventType.EXITED, null );
}
catch ( final BadLocationException e )
{
LoggerFactory.getLogger ( CodeLinkGenerator.class ).error ( e.toString (), e );
return null;
}
}
@Override
public int getSourceOffset ()
{
return content.getBegin ();
}
};
}
}
else if ( contains ( name.toLowerCase ( Locale.ROOT ), opacityContent ) )
{
final Float f = ( Float ) floatConverter.fromString ( contentString );
if ( f != null )
{
return new LinkGeneratorResult ()
{
@Override
public HyperlinkEvent execute ()
{
try
{
final WebPopOver opacityChooser = new WebPopOver ( parentComponent );
opacityChooser.setCloseOnFocusLoss ( true );
final int value = MathUtils.limit ( 0, Math.round ( 1000 * f ), 1000 );
final WebSlider slider = new WebSlider ( WebSlider.HORIZONTAL, 0, 1000, value );
slider.setPadding ( 10 );
slider.setPaintTicks ( true );
slider.setSnapToTicks ( true );
slider.setPaintLabels ( false );
slider.setMajorTickSpacing ( 50 );
slider.setMinorTickSpacing ( 10 );
slider.setPreferredWidth ( 500 );
slider.addChangeListener ( new ChangeListener ()
{
private int length = content.getEnd () - content.getBegin ();
@Override
public void stateChanged ( final ChangeEvent e )
{
final String floatString = floatConverter.toString ( ( float ) slider.getValue () / 1000 );
source.replaceRange ( floatString, content.getBegin (), content.getBegin () + length );
length = floatString.length ();
}
} );
opacityChooser.add ( slider );
final int position = ( content.getBegin () + content.getEnd () ) / 2;
final Rectangle wb = source.getUI ().modelToView ( source, position );
opacityChooser.show ( source, wb.x, wb.y, wb.width, wb.height, PopOverDirection.down );
return new HyperlinkEvent ( this, HyperlinkEvent.EventType.EXITED, null );
}
catch ( final BadLocationException e )
{
LoggerFactory.getLogger ( CodeLinkGenerator.class ).error ( e.toString (), e );
return null;
}
}
@Override
public int getSourceOffset ()
{
return content.getBegin ();
}
};
}
}
else if ( contains ( name.toLowerCase ( Locale.ROOT ), insetsContent ) )
{
final Insets insets = ( Insets ) insetsConverter.fromString ( contentString );
if ( insets != null )
{
// todo Margin editor
return null;
}
}
}
}
// todo Parse property types properly later (so far types are "guessed" from property names)
// todo This sounds like a small IDE already, but it is better than nothing
// if ( parentName.equals ( ComponentStyleConverter.COMPONENT_NODE ) )
// {
//
// }
// else if ( parentName.equals ( ComponentStyleConverter.UI_NODE ) )
// {
//
// }
// else if ( parentName.equals ( ComponentStyleConverter.PAINTER_NODE ) )
// {
//
// }
return null;
}
/**
* Returns whether text contains any of the tokens from the specified list or not.
*
* @param text text to look for tokens
* @param tokens tokens list
* @return true if text contains any of the tokens from the specified list, false otherwise
*/
protected boolean contains ( final String text, final List tokens )
{
for ( final String token : tokens )
{
if ( text.contains ( token ) )
{
return true;
}
}
return false;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy