freecol/src/net/sf/freecol/client/gui/panel/UnitLabel.java

411 lines
14 KiB
Java

/**
* Copyright (C) 2002-2007 The FreeCol Team
*
* This file is part of FreeCol.
*
* FreeCol 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 2 of the License, or
* (at your option) any later version.
*
* FreeCol 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 FreeCol. If not, see <http://www.gnu.org/licenses/>.
*/
package net.sf.freecol.client.gui.panel;
import java.awt.Color;
import java.awt.Component;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Image;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.image.BufferedImage;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.logging.Logger;
import javax.swing.BorderFactory;
import javax.swing.ImageIcon;
import javax.swing.JLabel;
import javax.swing.UIManager;
import net.sf.freecol.client.control.InGameController;
import net.sf.freecol.client.gui.Canvas;
import net.sf.freecol.client.gui.ImageLibrary;
import net.sf.freecol.client.gui.i18n.Messages;
import net.sf.freecol.common.model.Building;
import net.sf.freecol.common.model.BuildingType;
import net.sf.freecol.common.model.ColonyTile;
import net.sf.freecol.common.model.GoodsType;
import net.sf.freecol.common.model.StringTemplate;
import net.sf.freecol.common.model.TileType;
import net.sf.freecol.common.model.Unit;
import net.sf.freecol.common.model.Unit.UnitState;
/**
* This label holds Unit data in addition to the JLabel data, which makes it
* ideal to use for drag and drop purposes.
*/
public final class UnitLabel extends JLabel implements ActionListener, PropertyChangeListener {
@SuppressWarnings("unused")
private static Logger logger = Logger.getLogger(UnitLabel.class.getName());
public static enum UnitAction { ASSIGN,
CLEAR_SPECIALITY, ACTIVATE_UNIT, FORTIFY, SENTRY,
COLOPEDIA, LEAVE_TOWN, WORK_TILE, WORK_BUILDING, CLEAR_ORDERS }
private final Unit unit;
private final Canvas parent;
private boolean selected;
private boolean isSmall = false;
private boolean ignoreLocation;
private InGameController inGameController;
/**
* Initializes this JLabel with the given unit data.
*
* @param unit The Unit that this JLabel will visually represent.
* @param parent The parent that knows more than we do.
*/
public UnitLabel(Unit unit, Canvas parent) {
ImageLibrary lib = parent.getImageLibrary();
setIcon(lib.getUnitImageIcon(unit));
setDisabledIcon(lib.getUnitImageIcon(unit, true));
this.unit = unit;
unit.addPropertyChangeListener(Unit.EQUIPMENT_CHANGE, this);
setDescriptionLabel(Messages.message(Messages.getLabel(unit)));
StringTemplate label = unit.getEquipmentLabel();
if (label != null) {
setDescriptionLabel(getDescriptionLabel() + " ("
+ Messages.message(label) + ")");
}
this.parent = parent;
selected = false;
setSmall(false);
setIgnoreLocation(false);
this.inGameController = parent.getClient().getInGameController();
}
/**
* Initializes this JLabel with the given unit data.
*
* @param unit The Unit that this JLabel will visually represent.
* @param parent The parent that knows more than we do.
* @param isSmall The image will be smaller if set to <code>true</code>.
*/
public UnitLabel(Unit unit, Canvas parent, boolean isSmall) {
this(unit, parent);
setSmall(isSmall);
setIgnoreLocation(false);
}
/**
* Initializes this JLabel with the given unit data.
*
* @param unit The Unit that this JLabel will visually represent.
* @param parent The parent that knows more than we do.
* @param isSmall The image will be smaller if set to <code>true</code>.
* @param ignoreLocation The image will not include production or state
* information if set to <code>true</code>.
*/
public UnitLabel(Unit unit, Canvas parent, boolean isSmall, boolean ignoreLocation) {
this(unit, parent);
setSmall(isSmall);
setIgnoreLocation(ignoreLocation);
}
/**
* Returns the parent Canvas object.
*
* @return This UnitLabel's Canvas.
*/
public Canvas getCanvas() {
return parent;
}
/**
* Returns this UnitLabel's unit data.
*
* @return This UnitLabel's unit data.
*/
public Unit getUnit() {
return unit;
}
/**
* Sets whether or not this unit should be selected.
*
* @param b Whether or not this unit should be selected.
*/
public void setSelected(boolean b) {
selected = b;
}
/**
* Sets whether or not this unit label should include production and state
* information.
*
* @param b Whether or not this unit label should include production and
* state information.
*/
public void setIgnoreLocation(boolean b) {
ignoreLocation = b;
}
/**
* Makes a smaller version.
*
* @param isSmall The image will be smaller if set to <code>true</code>.
*/
public void setSmall(boolean isSmall) {
ImageIcon imageIcon = parent.getImageLibrary().getUnitImageIcon(unit);
ImageIcon disabledImageIcon = parent.getImageLibrary().getUnitImageIcon(unit, true);
if (isSmall) {
setPreferredSize(null);
// setIcon(new
// ImageIcon(imageIcon.getImage().getScaledInstance(imageIcon.getIconWidth()
// / 2, imageIcon.getIconHeight() / 2, Image.SCALE_DEFAULT)));
setIcon(new ImageIcon(imageIcon.getImage().getScaledInstance((imageIcon.getIconWidth() / 3) * 2,
(imageIcon.getIconHeight() / 3) * 2, Image.SCALE_SMOOTH)));
setDisabledIcon(new ImageIcon(disabledImageIcon.getImage().getScaledInstance(
(imageIcon.getIconWidth() / 3) * 2, (imageIcon.getIconHeight() / 3) * 2, Image.SCALE_SMOOTH)));
setBorder(BorderFactory.createEmptyBorder(0, 2, 0, 0));
this.isSmall = true;
} else {
if (unit.getLocation() instanceof ColonyTile) {
TileType tileType = ((ColonyTile) unit.getLocation()).getTile().getType();
setSize(new Dimension(parent.getImageLibrary().getTerrainImageWidth(tileType) / 2,
imageIcon.getIconHeight()));
} else {
setPreferredSize(null);
}
setIcon(imageIcon);
setDisabledIcon(disabledImageIcon);
if (unit.getLocation() instanceof ColonyTile) {
setBorder(BorderFactory.createEmptyBorder(0, 15, 0, 15));
} else {
setBorder(BorderFactory.createEmptyBorder(0, 5, 0, 5));
}
this.isSmall = false;
}
}
/**
* Gets the description label.
*
* The description label is a tooltip with the unit name and description of
* the terrain its on if applicable *
*
* @return This UnitLabel's description label.
*/
public String getDescriptionLabel() {
return getToolTipText();
}
/**
* Sets the description label.
*
* The description label is a tooltip with the unit name and description of
* the terrain its on if applicable
*
* @param label The string to set the label to.
*/
public void setDescriptionLabel(String label) {
setToolTipText(label);
}
/**
* Paints this UnitLabel.
*
* @param g The graphics context in which to do the painting.
*/
public void paintComponent(Graphics g) {
if (ignoreLocation || selected || (!unit.isCarrier() && unit.getState() != UnitState.SENTRY)) {
setEnabled(true);
} else if (unit.getOwner() != parent.getClient().getMyPlayer() && unit.getColony() == null) {
setEnabled(true);
} else {
setEnabled(false);
}
super.paintComponent(g);
if (ignoreLocation)
return;
if (unit.getLocation() instanceof ColonyTile) {
GoodsType workType = unit.getWorkType();
int production = ((ColonyTile) unit.getLocation()).getProductionOf(workType);
ProductionLabel pl = new ProductionLabel(workType, production, getCanvas());
g.translate(0, 10);
pl.paintComponent(g);
g.translate(0, -10);
} else if (getParent() instanceof ColonyPanel.OutsideColonyPanel ||
getParent() instanceof ColonyPanel.InPortPanel ||
getParent() instanceof EuropePanel.InPortPanel ||
getParent() instanceof EuropePanel.DocksPanel ||
getParent().getParent() instanceof ReportPanel) {
g.drawImage(parent.getGUI().getOccupationIndicatorImage(g, unit), 0, 0, null);
if (unit.isUnderRepair()) {
String underRepair = Messages.message("underRepair",
"%turns%", Integer.toString(unit.getTurnsForRepair()));
String underRepair1 = underRepair.substring(0, underRepair.indexOf('(')).trim();
String underRepair2 = underRepair.substring(underRepair.indexOf('(')).trim();
Font font = ((Font) UIManager.get("NormalFont")).deriveFont(14f);
Image repairImage1 = parent.getGUI()
.createStringImage((Graphics2D)g, underRepair1, Color.RED, font);
Image repairImage2 = parent.getGUI()
.createStringImage((Graphics2D)g, underRepair2, Color.RED, font);
int textHeight = repairImage1.getHeight(null) + repairImage2.getHeight(null);
int leftIndent = Math.min(5, Math.min(getWidth() - repairImage1.getWidth(null),
getWidth() - repairImage2.getWidth(null)));
g.drawImage(repairImage1,
leftIndent, // indent from left side of label (icon is placed at the left side)
((getHeight() - textHeight) / 2),
null);
g.drawImage(repairImage2,
leftIndent,
((getHeight() - textHeight) / 2) + repairImage1.getHeight(null),
null);
}
}
}
/**
* Analyzes an event and calls the right external methods to take care of
* the user's request.
*
* @param event The incoming action event
*/
public void actionPerformed(ActionEvent event) {
String commandString = event.getActionCommand();
String arg = null;
int index = commandString.indexOf(':');
if (index > 0) {
arg = commandString.substring(index + 1);
commandString = commandString.substring(0, index);
}
UnitAction command = Enum.valueOf(UnitAction.class, commandString.toUpperCase());
switch(command) {
case ASSIGN:
Unit teacher = (Unit) unit.getGame().getFreeColGameObject(arg);
inGameController.assignTeacher(unit, teacher);
Component uc = getParent();
while (uc != null) {
/*
if (uc instanceof ColonyPanel) {
((ColonyPanel) uc).reinitialize();
break;
}
*/
uc = uc.getParent();
}
break;
case WORK_TILE:
GoodsType goodsType = parent.getSpecification().getGoodsType(arg);
// Change workType first for the benefit of change listeners
inGameController.changeWorkType(unit, goodsType);
// Move unit to best producing ColonyTile
ColonyTile bestTile = unit.getColony().getVacantColonyTileFor(unit, false, goodsType);
if (bestTile != unit.getLocation()) {
inGameController.work(unit, bestTile);
}
break;
case WORK_BUILDING:
BuildingType buildingType = parent.getSpecification().getBuildingType(arg);
Building building = unit.getColony().getBuilding(buildingType);
inGameController.work(unit, building);
break;
case ACTIVATE_UNIT:
inGameController.changeState(unit, UnitState.ACTIVE);
parent.getGUI().setActiveUnit(unit);
break;
case FORTIFY:
inGameController.changeState(unit, UnitState.FORTIFYING);
break;
case SENTRY:
inGameController.changeState(unit, UnitState.SENTRY);
break;
case COLOPEDIA:
getCanvas().showPanel(new ColopediaPanel(getCanvas(), ColopediaPanel.PanelType.UNITS, unit.getType()));
break;
case LEAVE_TOWN:
inGameController.putOutsideColony(unit);
break;
case CLEAR_SPECIALITY:
inGameController.clearSpeciality(unit);
break;
case CLEAR_ORDERS:
inGameController.clearOrders(unit);
}
updateIcon();
}
public void updateIcon() {
setSmall(isSmall);
Component uc = getParent();
while (uc != null) {
if (uc instanceof ColonyPanel) {
if (unit.getColony() == null) {
parent.remove(uc);
parent.getClient().getActionManager().update();
}
break;
}
if (uc instanceof EuropePanel) {
break;
}
uc = uc.getParent();
}
}
public boolean canUnitBeEquipedWith(JLabel data){
if(!getUnit().hasAbility("model.ability.canBeEquipped")){
return false;
}
if(data instanceof GoodsLabel && ((GoodsLabel)data).isToEquip()){
return true;
}
if(data instanceof MarketLabel && ((MarketLabel)data).isToEquip()){
return true;
}
return false;
}
public void propertyChange(PropertyChangeEvent evt) {
if(evt.getPropertyName() == Unit.EQUIPMENT_CHANGE){
updateIcon();
}
}
}