com.alee.extended.list.WebFileListCellRenderer 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.list;
import com.alee.extended.layout.AbstractLayoutManager;
import com.alee.global.GlobalConstants;
import com.alee.laf.WebLookAndFeel;
import com.alee.laf.label.WebLabel;
import com.alee.laf.list.WebListCellRenderer;
import com.alee.utils.FileUtils;
import com.alee.utils.ImageUtils;
import com.alee.utils.file.FileDescription;
import javax.swing.*;
import java.awt.*;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.io.File;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/**
* Custom list cell renderer for WebFileList component.
* This renderer is also able to generate image thumbnails for image file elements.
*
* @author Mikle Garin
*/
public class WebFileListCellRenderer extends WebListCellRenderer
{
/**
* Constant cell sizes.
*/
protected static final Dimension tileCellSize = new Dimension ( 220, 65 );
protected static final Dimension iconCellSize = new Dimension ( 90, 90 );
/**
* Constant cell margins.
*/
protected static final Insets tileCellMargin = new Insets ( 6, 6, 5, 8 );
protected static final Insets iconCellMargin = new Insets ( 5, 5, 8, 5 );
/**
* Image thumbnails size.
*/
protected static final int thumbSize = 50;
/**
* Image side length.
*/
protected static final int imageSide = 54;
/**
* Gap between renderer elements.
*/
protected static final int gap = 4;
/**
* File list in which this list cell renderer is used.
*/
protected WebFileList fileList;
/**
* Thumbnail icon label.
*/
protected WebLabel iconLabel;
/**
* File name label.
*/
protected WebLabel nameLabel;
/**
* File size label.
*/
protected WebLabel sizeLabel;
/**
* File description label.
*/
protected WebLabel descriptionLabel;
/**
* Thumbnails queue lock object.
*/
protected final Object thumbnailsLock = new Object ();
/**
* Executor service for thumbnails generation.
*/
protected ExecutorService executorService = Executors.newSingleThreadExecutor ();
/**
* Constructs cell renderer for the specified file list.
*
* @param fileList file list in which this cell renderer is used
*/
public WebFileListCellRenderer ( final WebFileList fileList )
{
super ();
this.fileList = fileList;
iconLabel = new WebLabel ();
iconLabel.setHorizontalAlignment ( JLabel.CENTER );
iconLabel.setPreferredSize ( new Dimension ( imageSide, imageSide ) );
nameLabel = new WebLabel ();
nameLabel.setFont ( nameLabel.getFont ().deriveFont ( Font.PLAIN ) );
nameLabel.setForeground ( Color.BLACK );
nameLabel.setVerticalAlignment ( JLabel.CENTER );
descriptionLabel = new WebLabel ( WebLabel.LEADING );
descriptionLabel.setFont ( descriptionLabel.getFont ().deriveFont ( Font.PLAIN ) );
descriptionLabel.setForeground ( Color.GRAY );
sizeLabel = new WebLabel ( WebLabel.LEADING );
sizeLabel.setFont ( sizeLabel.getFont ().deriveFont ( Font.PLAIN ) );
sizeLabel.setForeground ( new Color ( 49, 77, 179 ) );
setLayout ( new FileCellLayout () );
add ( iconLabel );
add ( nameLabel );
add ( descriptionLabel );
add ( sizeLabel );
fileList.addPropertyChangeListener ( WebLookAndFeel.ENABLED_PROPERTY, new PropertyChangeListener ()
{
@Override
public void propertyChange ( final PropertyChangeEvent evt )
{
final boolean enabled = fileList.isEnabled ();
iconLabel.setEnabled ( enabled );
nameLabel.setEnabled ( enabled );
descriptionLabel.setEnabled ( enabled );
sizeLabel.setEnabled ( enabled );
}
} );
fileList.addPropertyChangeListener ( WebLookAndFeel.ORIENTATION_PROPERTY, new PropertyChangeListener ()
{
@Override
public void propertyChange ( final PropertyChangeEvent evt )
{
final ComponentOrientation orientation = fileList.getComponentOrientation ();
nameLabel.setComponentOrientation ( orientation );
descriptionLabel.setComponentOrientation ( orientation );
sizeLabel.setComponentOrientation ( orientation );
}
} );
updateFilesView ();
}
/**
* Returns description bounds for list cell.
*
* @return description bounds for list cell
*/
public Rectangle getDescriptionBounds ()
{
return ( ( FileCellLayout ) getLayout () ).getDescriptionBounds ();
}
/**
* Updates renderer elements view.
*/
public void updateFilesView ()
{
if ( isTilesView () )
{
nameLabel.setHorizontalAlignment ( JLabel.LEADING );
fileList.setFixedCellWidth ( tileCellSize.width );
fileList.setFixedCellHeight ( tileCellSize.height );
}
else
{
nameLabel.setHorizontalAlignment ( JLabel.CENTER );
fileList.setFixedCellWidth ( iconCellSize.width );
fileList.setFixedCellHeight ( iconCellSize.height );
}
}
/**
* Returns thumbnail icon label.
*
* @return thumbnail icon label
*/
public JLabel getIconLabel ()
{
return iconLabel;
}
/**
* Returns file name label.
*
* @return file name label
*/
public JLabel getNameLabel ()
{
return nameLabel;
}
/**
* Returns file size label.
*
* @return file size label
*/
public JLabel getSizeLabel ()
{
return sizeLabel;
}
/**
* Returns file description label.
*
* @return file description label
*/
public JLabel getDescriptionLabel ()
{
return descriptionLabel;
}
/**
* Returns list cell renderer component.
*
* @param list tree
* @param value cell value
* @param index cell index
* @param isSelected whether cell is selected or not
* @param cellHasFocus whether cell has focus or not
* @return cell renderer component
*/
@Override
public Component getListCellRendererComponent ( final JList list, final Object value, final int index, final boolean isSelected,
final boolean cellHasFocus )
{
super.getListCellRendererComponent ( list, "", index, isSelected, cellHasFocus );
final FileElement element = ( FileElement ) value;
final File file = element.getFile ();
// Proper margin
setMargin ( isTilesView () ? tileCellMargin : iconCellMargin );
// Renderer icon
String imageSize = null;
if ( iconLabel.isEnabled () )
{
// Thumbnail loading
synchronized ( thumbnailsLock )
{
if ( !element.isThumbnailQueued () && !element.isDisabledThumbnailQueued () )
{
queueThumbnailLoad ( element, false );
}
}
// Image thumbnail
final ImageIcon thumbnail = element.getEnabledThumbnail ();
iconLabel.setIcon ( thumbnail );
// Image description
if ( thumbnail != null )
{
imageSize = thumbnail.getDescription ();
}
}
else
{
// Disabled thumbnail loading
synchronized ( thumbnailsLock )
{
if ( !element.isDisabledThumbnailQueued () )
{
queueThumbnailLoad ( element, true );
}
}
// Image disabled thumbnail
iconLabel.setDisabledIcon ( element.getDisabledThumbnail () );
}
// Updating file description elements
if ( fileList.getEditedCell () != index )
{
// Settings description
final FileDescription fileDescription = FileUtils.getFileDescription ( file, imageSize );
nameLabel.setText ( fileDescription.getName () );
// Updating tile view additional description
if ( isTilesView () )
{
descriptionLabel.setText ( fileDescription.getDescription () );
// Updating size label
if ( fileDescription.getSize () != null )
{
sizeLabel.setText ( fileDescription.getSize () );
}
else
{
sizeLabel.setText ( null );
}
}
else
{
descriptionLabel.setText ( null );
sizeLabel.setText ( null );
}
}
else
{
nameLabel.setText ( null );
descriptionLabel.setText ( null );
sizeLabel.setText ( null );
}
return this;
}
/**
* Returns whether list is currently displaying tiles or not.
*
* @return true if list is currently displaying tiles, false otherwise
*/
protected boolean isTilesView ()
{
return fileList.getFileListViewType ().equals ( FileListViewType.tiles );
}
/**
* Adds specified element into thumbnails queue.
*
* @param element element to add
*/
protected void queueThumbnailLoad ( final FileElement element, final boolean disabled )
{
element.setThumbnailQueued ( true );
element.setDisabledThumbnailQueued ( disabled );
executorService.submit ( new Runnable ()
{
@Override
public void run ()
{
final String absolutePath = element.getFile ().getAbsolutePath ();
final String ext = FileUtils.getFileExtPart ( element.getFile ().getName (), false ).toLowerCase ();
if ( fileList.isGenerateThumbnails () && GlobalConstants.IMAGE_FORMATS.contains ( ext ) )
{
final ImageIcon thumb = element.getEnabledThumbnail () != null ? element.getEnabledThumbnail () :
ImageUtils.createThumbnailIcon ( absolutePath, thumbSize );
if ( thumb != null )
{
element.setEnabledThumbnail ( thumb );
if ( disabled )
{
element.setDisabledThumbnail ( ImageUtils.createDisabledCopy ( thumb ) );
}
}
else
{
element.setEnabledThumbnail ( FileUtils.getStandartFileIcon ( element.getFile (), true, true ) );
if ( disabled )
{
element.setDisabledThumbnail ( FileUtils.getStandartFileIcon ( element.getFile (), true, false ) );
}
}
}
else
{
element.setEnabledThumbnail ( FileUtils.getStandartFileIcon ( element.getFile (), true, true ) );
if ( disabled )
{
element.setDisabledThumbnail ( FileUtils.getStandartFileIcon ( element.getFile (), true, false ) );
}
}
if ( disabled != fileList.isEnabled () )
{
fileList.repaint ( element );
}
}
} );
}
/**
* Custom layout for file list cell element.
*/
protected class FileCellLayout extends AbstractLayoutManager
{
/**
* {@inheritDoc}
*/
@Override
public Dimension preferredLayoutSize ( final Container parent )
{
return isTilesView () ? tileCellSize : iconCellSize;
}
/**
* {@inheritDoc}
*/
@Override
public void layoutContainer ( final Container parent )
{
// Constants for futher layout calculations
final boolean ltr = fileList.getComponentOrientation ().isLeftToRight ();
final Insets i = getInsets ();
final boolean tilesView = isTilesView ();
final boolean hasName = nameLabel.getText () != null;
final boolean hasDescription = tilesView && descriptionLabel.getText () != null;
final boolean hasFileSize = tilesView && sizeLabel.getText () != null;
// Updating elements visibility
nameLabel.setVisible ( hasName );
descriptionLabel.setVisible ( hasDescription );
sizeLabel.setVisible ( hasFileSize );
// Updating elements bounds
if ( tilesView )
{
// Left-middle icon
iconLabel.setBounds ( ltr ? i.left : tileCellSize.width - i.right - imageSide, i.top, imageSide, imageSide );
// Description elements
if ( hasName )
{
// Constants for futher description positioning calculations
final Dimension nps = nameLabel.getPreferredSize ();
final Dimension dps = hasDescription ? descriptionLabel.getPreferredSize () : new Dimension ( 0, 0 );
final Dimension sps = hasFileSize ? sizeLabel.getPreferredSize () : new Dimension ( 0, 0 );
final int dh = nps.height + ( hasDescription ? gap + dps.height : 0 ) + ( hasFileSize ? gap + sps.height : 0 );
final int dx = ltr ? i.left + imageSide + gap : i.left;
final int dw = tileCellSize.width - i.left - i.right - imageSide - gap;
// Name element
int dy = i.top + ( tileCellSize.height - i.top - i.bottom ) / 2 - dh / 2;
nameLabel.setBounds ( dx, dy, dw, nps.height );
dy += nps.height + gap;
// Description element
if ( hasDescription )
{
descriptionLabel.setBounds ( dx, dy, dw, dps.height );
dy += dps.height + gap;
}
// File size element
if ( hasFileSize )
{
sizeLabel.setBounds ( dx, dy, dw, sps.height );
}
}
}
else
{
// Top-middle icon
final int cw = iconCellSize.width - i.left - i.right;
iconLabel.setBounds ( i.left + cw / 2 - 27, i.top, imageSide, imageSide );
// Name element
if ( hasName )
{
final int ny = i.top + imageSide + gap;
nameLabel.setBounds ( i.left, ny, cw, iconCellSize.height - ny - i.bottom );
}
}
}
/**
* Returns description bounds for list cell.
*
* @return description bounds for list cell
*/
public Rectangle getDescriptionBounds ()
{
// Constants for futher size calculations
final boolean ltr = fileList.getComponentOrientation ().isLeftToRight ();
final Insets i = getInsets ();
final boolean tilesView = isTilesView ();
// Determining
if ( tilesView )
{
// Tile view
if ( ltr )
{
// Icon at the left side, description at the right side
final int x = i.left + imageSide + gap;
return new Rectangle ( x, i.top, tileCellSize.width - x - i.right, tileCellSize.height - i.top - i.bottom );
}
else
{
// Icon at the right side, description at the left side
return new Rectangle ( i.left, i.top, tileCellSize.width - gap - imageSide - i.right,
tileCellSize.height - i.top - i.bottom );
}
}
else
{
// Icon view
final int ny = i.top + imageSide + gap;
return new Rectangle ( i.left, ny, iconCellSize.width - i.left - i.right, iconCellSize.height - ny - i.bottom );
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy