GUI: added gui scale support for skip and phase button on game panel (part of #969, #6887):

- reworked skip and phase buttons to use layouts instead IDE designer;
- added GUI scale support skip and phase buttons (depend on dialogs font size from preferences settings);
This commit is contained in:
Oleg Agafonov 2024-07-28 18:01:15 +04:00
parent 8186b35dfb
commit 1d701df0e8
8 changed files with 274 additions and 144 deletions

View file

@ -78,7 +78,7 @@ public class HoverButton extends JPanel implements MouseListener {
private boolean doGainFade = true;
public HoverButton(String text, Image image, Rectangle size) {
this(text, image, image, null, image, size);
this(text, image, image, image, image, size);
if (image == null) {
throw new IllegalArgumentException("Image can't be null");
}
@ -370,10 +370,6 @@ public class HoverButton extends JPanel implements MouseListener {
this.drawSet = true;
}
public void update(String text, Image image) {
update(text, image, this.hoverImage, this.selectedImage, this.disabledImage, this.imageSize);
}
public void update(String text, Image image, Image hover, Image selected, Image disabled, Rectangle size) {
this.text = text;
this.image = image;
@ -381,7 +377,7 @@ public class HoverButton extends JPanel implements MouseListener {
this.selectedImage = selected;
this.disabledImage = disabled;
this.imageSize = size;
repaint();
this.invalidate();
}
public void execute() {

View file

@ -1,6 +1,7 @@
package mage.client.components;
import mage.client.dialog.PreferencesDialog;
import mage.client.util.GUISizeHelper;
import javax.swing.*;
import java.awt.*;
@ -12,19 +13,25 @@ import java.awt.*;
*/
public class KeyboundButton extends JButton {
private float guiScale = 1.0f;
private final String key;
private boolean showKey;
private String drawingText;
private static final Font keyFont = new Font(Font.SANS_SERIF, Font.BOLD, 13);
private Font keyFont;
private boolean tinting = false;
public KeyboundButton(String key, boolean showKey) {
this.key = key;
this.showKey = showKey;
updateFont();
updateDrawingText();
}
private void updateFont() {
this.keyFont = new Font(Font.SANS_SERIF, Font.BOLD, sizeMod(13));
}
private void updateDrawingText() {
if (this.showKey) {
this.drawingText = PreferencesDialog.getCachedKeyText(key);
@ -33,6 +40,12 @@ public class KeyboundButton extends JButton {
}
}
public void updateGuiScale(float guiScale) {
this.guiScale = guiScale;
updateFont();
invalidate();
}
public void setShowKey(boolean showKey) {
this.showKey = showKey;
updateDrawingText();
@ -48,7 +61,7 @@ public class KeyboundButton extends JButton {
if (tinting) {
sg.setColor(new Color(0, 0, 0, 32));
sg.fillRoundRect(2, 2, getWidth() - 4, getHeight() - 4, 6, 6);
sg.fillRoundRect(sizeMod(2), sizeMod(2), getWidth() - sizeMod(4), getHeight() - sizeMod(4), sizeMod(6), sizeMod(6));
}
sg.setColor(tinting ? Color.lightGray : Color.white);
@ -58,7 +71,7 @@ public class KeyboundButton extends JButton {
int textWidth = sg.getFontMetrics(keyFont).stringWidth(this.drawingText);
int centerX = (getWidth() - textWidth) / 2;
sg.drawString(this.drawingText, centerX, 28);
sg.drawString(this.drawingText, centerX, sizeMod(28));
}
} finally {
sg.dispose();
@ -70,4 +83,12 @@ public class KeyboundButton extends JButton {
this.tinting = tinting;
repaint();
}
private int sizeMod(int value) {
return GUISizeHelper.guiSizeScale(value, this.guiScale);
}
private float sizeMod(float value) {
return GUISizeHelper.guiSizeScale(value, this.guiScale);
}
}

View file

@ -29,6 +29,7 @@ import mage.constants.*;
import mage.game.events.PlayerQueryEvent;
import mage.players.PlayableObjectStats;
import mage.players.PlayableObjectsList;
import mage.util.DebugUtil;
import mage.util.MultiAmountMessage;
import mage.view.*;
import org.apache.log4j.Logger;
@ -64,7 +65,12 @@ public final class GamePanel extends javax.swing.JPanel {
private static final Logger logger = Logger.getLogger(GamePanel.class);
private static final String YOUR_HAND = "Your hand";
private static final int X_PHASE_WIDTH = 55;
private static final int SKIP_BUTTONS_SPACE_H = 3;
private static final int SKIP_BUTTONS_SPACE_V = 3;
private static final int PHASE_BUTTONS_SPACE_H = 3;
private static final int PHASE_BUTTONS_SPACE_V = 3;
private static final String CMD_AUTO_ORDER_FIRST = "cmdAutoOrderFirst";
private static final String CMD_AUTO_ORDER_LAST = "cmdAutoOrderLast";
@ -188,6 +194,27 @@ public final class GamePanel extends javax.swing.JPanel {
initComponents = true;
initComponents();
// prepare commands buttons panel with flow layout (instead custom from IDE)
// size changes in helper method at the end
// TODO: remove IDE form file (it useless anyway due many custom code in init)
if (DebugUtil.GUI_GAME_DRAW_COMMAND_BUTTONS_PANEL_BORDER) {
pnlShortCuts.setBorder(BorderFactory.createLineBorder(Color.red));
}
pnlShortCuts.removeAll();
pnlShortCuts.setLayout(null); // real layout on size settings
pnlShortCuts.add(btnSkipToNextTurn);
pnlShortCuts.add(btnSkipToEndTurn);
pnlShortCuts.add(btnSkipToNextMain);
pnlShortCuts.add(btnSkipToYourTurn);
pnlShortCuts.add(btnSkipStack);
pnlShortCuts.add(btnSkipToEndStepBeforeYourTurn);
pnlShortCuts.add(txtHoldPriority);
//pnlShortCuts.add(btnToggleMacro);
pnlShortCuts.add(btnSwitchHands);
pnlShortCuts.add(btnCancelSkip);
pnlShortCuts.add(btnConcede);
pnlShortCuts.add(btnStopWatching);
pickNumber = new PickNumberDialog();
MageFrame.getDesktop().add(pickNumber, JLayeredPane.MODAL_LAYER);
@ -426,7 +453,7 @@ public final class GamePanel extends javax.swing.JPanel {
// hand + stack panels
// the stack takes up a portion of the possible space (GUISizeHelper.stackWidth)
// TODO: research and delete rare used settings
int newStackWidth = pnlHelperHandButtonsStackArea.getWidth() * GUISizeHelper.stackWidth / 100;
newStackWidth = Math.max(410, newStackWidth);
Dimension newDimension = new Dimension(
@ -451,11 +478,48 @@ public final class GamePanel extends javax.swing.JPanel {
userChatPanel.changeGUISize(GUISizeHelper.chatFont);
gameChatPanel.changeGUISize(GUISizeHelper.chatFont);
// skip buttons
newDimension = new Dimension(newStackWidth, (int) pnlShortCuts.getPreferredSize().getHeight());
// skip buttons - sizes
// must be able to put controls in 2 rows
float guiScale = GUISizeHelper.dialogGuiScale;
int hGap = GUISizeHelper.guiSizeScale(SKIP_BUTTONS_SPACE_H, guiScale);
int vGap = GUISizeHelper.guiSizeScale(SKIP_BUTTONS_SPACE_V, guiScale);
newDimension = new Dimension(newStackWidth, (4 * vGap) + (2 * GUISizeHelper.gameCommandButtonHeight));
pnlShortCuts.setLayout(new FlowLayout(FlowLayout.RIGHT, hGap, vGap));
pnlShortCuts.setPreferredSize(newDimension);
pnlShortCuts.setMinimumSize(newDimension);
pnlShortCuts.setMaximumSize(newDimension);
// skip buttons - sizes
Dimension strictSize = new Dimension(2 * GUISizeHelper.gameCommandButtonHeight, GUISizeHelper.gameCommandButtonHeight);
setSkipButtonSize(btnCancelSkip, guiScale, strictSize);
setSkipButtonSize(btnSkipToNextTurn, guiScale, strictSize);
setSkipButtonSize(btnSkipToEndTurn, guiScale, strictSize);
setSkipButtonSize(btnSkipToEndStepBeforeYourTurn, guiScale, strictSize);
setSkipButtonSize(btnSkipToYourTurn, guiScale, strictSize);
setSkipButtonSize(btnSkipToNextMain, guiScale, strictSize);
setSkipButtonSize(btnSkipStack, guiScale, strictSize);
setSkipButtonSize(btnConcede, guiScale, strictSize);
setSkipButtonSize(btnToggleMacro, guiScale, strictSize);
setSkipButtonSize(btnSwitchHands, guiScale, strictSize);
setSkipButtonSize(btnStopWatching, guiScale, strictSize);
pnlShortCuts.invalidate();
// phase buttons - sizes
int buttonSize = GUISizeHelper.gamePhaseButtonSize;
guiScale = GUISizeHelper.dialogGuiScale;
hGap = GUISizeHelper.guiSizeScale(PHASE_BUTTONS_SPACE_H, guiScale);
vGap = GUISizeHelper.guiSizeScale(PHASE_BUTTONS_SPACE_V, guiScale);
BoxLayout layout = new BoxLayout(jPhases, BoxLayout.Y_AXIS);
jPhases.setLayout(layout);
int fullPhaseWidth = Math.round(1.5f * GUISizeHelper.gamePhaseButtonSize);
jPhases.setPreferredSize(new Dimension(fullPhaseWidth, (vGap * phaseButtons.size()) + (buttonSize * phaseButtons.size())));
jPhases.setMaximumSize(new Dimension(fullPhaseWidth, Short.MAX_VALUE));
phaseButtons.forEach((phaseName, phaseButton) -> {
phaseButton.setPreferredSize(new Dimension(buttonSize, buttonSize));
});
// phase buttons - active size
if (lastGameData.game != null) {
updateActivePhase(lastGameData.game.getStep());
}
if (themeReload) {
reloadThemeRelatedGraphic();
@ -463,18 +527,19 @@ public final class GamePanel extends javax.swing.JPanel {
}
private void reloadThemeRelatedGraphic() {
// skip buttons
btnCancelSkip.setIcon(new ImageIcon(ImageManagerImpl.instance.getCancelSkipButtonImage()));
btnSkipToNextTurn.setIcon(new ImageIcon(ImageManagerImpl.instance.getSkipNextTurnButtonImage()));
btnSkipToEndTurn.setIcon(new ImageIcon(ImageManagerImpl.instance.getSkipEndTurnButtonImage()));
btnSkipToEndStepBeforeYourTurn.setIcon(new ImageIcon(ImageManagerImpl.instance.getSkipEndStepBeforeYourTurnButtonImage()));
btnSkipToYourTurn.setIcon(new ImageIcon(ImageManagerImpl.instance.getSkipYourNextTurnButtonImage()));
btnSkipToNextMain.setIcon(new ImageIcon(ImageManagerImpl.instance.getSkipMainButtonImage()));
btnSkipStack.setIcon(new ImageIcon(ImageManagerImpl.instance.getSkipStackButtonImage()));
btnConcede.setIcon(new ImageIcon(ImageManagerImpl.instance.getConcedeButtonImage()));
btnToggleMacro.setIcon(new ImageIcon(ImageManagerImpl.instance.getToggleRecordMacroButtonImage()));
btnSwitchHands.setIcon(new ImageIcon(ImageManagerImpl.instance.getSwitchHandsButtonImage()));
btnStopWatching.setIcon(new ImageIcon(ImageManagerImpl.instance.getStopWatchButtonImage()));
// skip buttons - images
int buttonHeight = GUISizeHelper.gameCommandButtonHeight;
setSkipButtonImage(btnCancelSkip, ImageManagerImpl.instance.getCancelSkipButtonImage(buttonHeight));
setSkipButtonImage(btnSkipToNextTurn, ImageManagerImpl.instance.getSkipNextTurnButtonImage(buttonHeight));
setSkipButtonImage(btnSkipToEndTurn, ImageManagerImpl.instance.getSkipEndTurnButtonImage(buttonHeight));
setSkipButtonImage(btnSkipToEndStepBeforeYourTurn, ImageManagerImpl.instance.getSkipEndStepBeforeYourTurnButtonImage(buttonHeight));
setSkipButtonImage(btnSkipToYourTurn, ImageManagerImpl.instance.getSkipYourNextTurnButtonImage(buttonHeight));
setSkipButtonImage(btnSkipToNextMain, ImageManagerImpl.instance.getSkipMainButtonImage(buttonHeight));
setSkipButtonImage(btnSkipStack, ImageManagerImpl.instance.getSkipStackButtonImage(buttonHeight));
setSkipButtonImage(btnConcede, ImageManagerImpl.instance.getConcedeButtonImage(buttonHeight));
setSkipButtonImage(btnToggleMacro, ImageManagerImpl.instance.getToggleRecordMacroButtonImage(buttonHeight));
setSkipButtonImage(btnSwitchHands, ImageManagerImpl.instance.getSwitchHandsButtonImage(buttonHeight));
setSkipButtonImage(btnStopWatching, ImageManagerImpl.instance.getStopWatchButtonImage(buttonHeight));
// hotkeys for skip buttons
boolean displayButtonText = PreferencesDialog.getCurrentTheme().isShortcutsVisibleForSkipButtons();
@ -489,7 +554,9 @@ public final class GamePanel extends javax.swing.JPanel {
// phase buttons
phaseButtons.forEach((phaseName, phaseButton) -> {
phaseButton.update(phaseButton.getText(), ImageManagerImpl.instance.getPhaseImage(phaseName));
Image buttonImage = ImageManagerImpl.instance.getPhaseImage(phaseName, GUISizeHelper.gamePhaseButtonSize);
Rectangle buttonRect = new Rectangle(buttonImage.getWidth(null), buttonImage.getHeight(null));
phaseButton.update(phaseButton.getText(), buttonImage, buttonImage, buttonImage, buttonImage, buttonRect);
});
// player panels
@ -513,6 +580,22 @@ public final class GamePanel extends javax.swing.JPanel {
}
}
private void setSkipButtonImage(JButton button, Image image) {
button.setIcon(new ImageIcon(image));
}
private void setSkipButtonSize(JComponent button, float guiScale, Dimension size) {
if (button instanceof KeyboundButton) {
((KeyboundButton) button).updateGuiScale(guiScale);
}
// no needs in size - it controlled by button's icon
if (true) return;
button.setMinimumSize(size);
button.setPreferredSize(size);
button.setMaximumSize(size);
}
private void saveDividerLocations() {
// save panel sizes and divider locations.
Rectangle rec = MageFrame.getDesktop().getBounds();
@ -564,7 +647,6 @@ public final class GamePanel extends javax.swing.JPanel {
bigCard.setMinimumSize(bbDimension);
bigCard.setPreferredSize(bbDimension);
pnlShortCuts.revalidate();
pnlShortCuts.repaint();
for (PlayAreaPanel p : players.values()) {
p.getPlayerPanel().sizePlayerPanel(smallMode);
}
@ -576,7 +658,6 @@ public final class GamePanel extends javax.swing.JPanel {
bigCard.setMinimumSize(bbDimension);
bigCard.setPreferredSize(bbDimension);
pnlShortCuts.revalidate();
pnlShortCuts.repaint();
for (PlayAreaPanel p : players.values()) {
p.getPlayerPanel().sizePlayerPanel(smallMode);
}
@ -893,7 +974,7 @@ public final class GamePanel extends javax.swing.JPanel {
}
if (lastGameData.game.getStep() != null) {
updatePhases(lastGameData.game.getStep());
updateActivePhase(lastGameData.game.getStep());
this.txtStep.setText(lastGameData.game.getStep().toString());
} else {
logger.debug("Step is empty");
@ -1261,63 +1342,63 @@ public final class GamePanel extends javax.swing.JPanel {
this.stackObjects.loadCards(game.getStack(), bigCard, gameId, true);
}
/**
* Update phase buttons\labels.
*/
private void updatePhases(PhaseStep step) {
if (step == null) {
logger.warn("step is null");
private void updateActivePhase(PhaseStep currentStep) {
if (currentStep == null) {
return;
}
if (currentStep != null) {
currentStep.setLocation(prevPoint);
}
switch (step) {
switch (currentStep) {
case UNTAP:
updateButton("Untap");
updatePhaseButtons("Untap");
break;
case UPKEEP:
updateButton("Upkeep");
updatePhaseButtons("Upkeep");
break;
case DRAW:
updateButton("Draw");
updatePhaseButtons("Draw");
break;
case PRECOMBAT_MAIN:
updateButton("Main1");
updatePhaseButtons("Main1");
break;
case BEGIN_COMBAT:
updateButton("Combat_Start");
updatePhaseButtons("Combat_Start");
break;
case DECLARE_ATTACKERS:
updateButton("Combat_Attack");
updatePhaseButtons("Combat_Attack");
break;
case DECLARE_BLOCKERS:
updateButton("Combat_Block");
updatePhaseButtons("Combat_Block");
break;
case FIRST_COMBAT_DAMAGE:
case COMBAT_DAMAGE:
updateButton("Combat_Damage");
updatePhaseButtons("Combat_Damage");
break;
case END_COMBAT:
updateButton("Combat_End");
updatePhaseButtons("Combat_End");
break;
case POSTCOMBAT_MAIN:
updateButton("Main2");
updatePhaseButtons("Main2");
break;
case END_TURN:
updateButton("Cleanup");
case CLEANUP:
updatePhaseButtons("Cleanup");
break;
default:
break;
}
}
private void updateButton(String name) {
if (phaseButtons.containsKey(name)) {
currentStep = phaseButtons.get(name);
prevPoint = currentStep.getLocation();
currentStep.setLocation(prevPoint.x - 15, prevPoint.y);
}
private void updatePhaseButtons(String currentPhaseName) {
phaseButtons.forEach((phaseName, phaseButton) -> {
if (phaseName.equals(currentPhaseName)) {
//phaseButton.setBorder(this.phaseButtonBorderActive);
phaseButton.setAlignmentX(Component.CENTER_ALIGNMENT);
} else {
//phaseButton.setBorder(this.phaseButtonBorderInactive);
phaseButton.setAlignmentX(Component.LEFT_ALIGNMENT);
}
});
jPhases.invalidate();
}
// Called if the game frame is deactivated because the tabled the deck editor or other frames go to foreground
@ -2530,8 +2611,7 @@ public final class GamePanel extends javax.swing.JPanel {
jPhases = new JPanel();
jPhases.setBackground(new Color(0, 0, 0, 0));
jPhases.setLayout(null);
jPhases.setPreferredSize(new Dimension(X_PHASE_WIDTH, 435));
// layout on gui size
MouseAdapter phasesMouseAdapter = new MouseAdapter() {
@Override
@ -2539,6 +2619,11 @@ public final class GamePanel extends javax.swing.JPanel {
mouseClickPhaseBar(evt);
}
};
/// phase buttons
if (DebugUtil.GUI_GAME_DRAW_PHASE_BUTTONS_PANEL_BORDER) {
jPhases.setBorder(BorderFactory.createLineBorder(Color.red));
}
String[] phases = {"Untap", "Upkeep", "Draw", "Main1",
"Combat_Start", "Combat_Attack", "Combat_Block", "Combat_Damage", "Combat_End",
"Main2", "Cleanup", "Next_Turn"};
@ -2546,15 +2631,6 @@ public final class GamePanel extends javax.swing.JPanel {
createPhaseButton(name, phasesMouseAdapter);
}
int i = 0;
for (String name : phaseButtons.keySet()) {
HoverButton hoverButton = phaseButtons.get(name);
hoverButton.setAlignmentX(LEFT_ALIGNMENT);
hoverButton.setBounds(X_PHASE_WIDTH - 36, i * 36, 36, 36);
jPhases.add(hoverButton);
i++;
}
pnlReplay.setOpaque(false);
helper = new HelperPanel();
@ -2938,12 +3014,14 @@ public final class GamePanel extends javax.swing.JPanel {
}
private void createPhaseButton(String name, MouseAdapter mouseAdapter) {
Rectangle rect = new Rectangle(36, 36);
HoverButton button = new HoverButton("", ImageManagerImpl.instance.getPhaseImage(name), rect);
int buttonSize = GUISizeHelper.gamePhaseButtonSize;
Rectangle rect = new Rectangle(buttonSize, buttonSize);
HoverButton button = new HoverButton("", ImageManagerImpl.instance.getPhaseImage(name, buttonSize), rect);
button.setToolTipText(name.replaceAll("_", " "));
button.setPreferredSize(new Dimension(36, 36));
button.setPreferredSize(new Dimension(buttonSize, buttonSize));
button.addMouseListener(mouseAdapter);
phaseButtons.put(name, button);
jPhases.add(button);
}
// Event listener for the ShowCardsDialog
@ -3156,9 +3234,6 @@ public final class GamePanel extends javax.swing.JPanel {
private JPanel phasesContainer;
private javax.swing.JLabel txtHoldPriority;
private HoverButton currentStep;
private Point prevPoint;
private boolean imagePanelState;
}

View file

@ -370,9 +370,9 @@ public class PlayerPanelExt extends javax.swing.JPanel {
path = "/avatars/special/" + avatarId + ".gif";
}
Image image = ImageHelper.getImageFromResources(path);
Rectangle r = new Rectangle(sizeMod(80), sizeMod(80));
BufferedImage resized = ImageHelper.getResizedImage(BufferedImageBuilder.bufferImage(image, BufferedImage.TYPE_INT_ARGB), r);
this.avatar.update(this.player.getName(), resized, resized, resized, resized, r);
Rectangle buttonRect = new Rectangle(sizeMod(80), sizeMod(80));
BufferedImage buttonImage = ImageHelper.getResizedImage(BufferedImageBuilder.bufferImage(image, BufferedImage.TYPE_INT_ARGB), buttonRect);
this.avatar.update(this.player.getName(), buttonImage, buttonImage, buttonImage, buttonImage, buttonRect);
}
}
if (this.timer != null) {

View file

@ -50,10 +50,10 @@ public final class GUISizeHelper {
public static int gameFeedbackPanelExtraMessageFontSize = 11;
public static int gameFeedbackPanelMaxHeight = 0;
public static int gameDialogAreaButtonHigh = 16;
public static int gamePhaseButtonSize = 36;
public static int gameCommandButtonHeight = 32;
public static Font gameFeedbackPanelFont = new java.awt.Font("Arial", 0, 12);
public static float gameDialogAreaDefaultFontSize = gameFeedbackPanelFont.getSize2D();
public static int gameFeedbackPanelButtonHeight;
public static int gameFeedbackPanelButtonWidth;
@ -132,6 +132,10 @@ public final class GUISizeHelper {
gameFeedbackPanelButtonHeight = dialogFontSize + 6;
gameFeedbackPanelButtonWidth = dialogFontSize * 2 + 40;
// game - phase and command buttons
gamePhaseButtonSize = dialogGuiScaleSize(36);
gameCommandButtonHeight = dialogGuiScaleSize(32);
// app - chats and game logs
int chatFontSize = PreferencesDialog.getCachedValue(PreferencesDialog.KEY_GUI_CHAT_FONT_SIZE, 14);
chatFont = new java.awt.Font("Arial", 0, chatFontSize);

View file

@ -5,6 +5,11 @@ import mage.abilities.icon.CardIconColor;
import java.awt.*;
import java.awt.image.BufferedImage;
/**
* GUI: images source for GUI components like buttons. With theme supports.
*
* @author JayDi85
*/
public interface ImageManager {
Image getAppImage();
@ -57,29 +62,29 @@ public interface ImageManager {
Image getDlgActiveNextButtonImage();
Image getSwitchHandsButtonImage();
Image getSwitchHandsButtonImage(int height);
Image getStopWatchButtonImage();
Image getStopWatchButtonImage(int height);
Image getConcedeButtonImage();
Image getConcedeButtonImage(int height);
Image getCancelSkipButtonImage();
Image getCancelSkipButtonImage(int height);
Image getSkipNextTurnButtonImage();
Image getSkipNextTurnButtonImage(int height);
Image getSkipEndTurnButtonImage();
Image getSkipEndTurnButtonImage(int height);
Image getSkipMainButtonImage();
Image getSkipMainButtonImage(int height);
Image getSkipStackButtonImage();
Image getSkipStackButtonImage(int height);
Image getSkipEndStepBeforeYourTurnButtonImage();
Image getSkipEndStepBeforeYourTurnButtonImage(int height);
Image getSkipYourNextTurnButtonImage();
Image getSkipYourNextTurnButtonImage(int height);
Image getToggleRecordMacroButtonImage();
Image getToggleRecordMacroButtonImage(int height);
BufferedImage getCardIcon(String resourceName, int size, CardIconColor cardIconColor);
Image getPhaseImage(String phase);
Image getPhaseImage(String phase, int size);
}

View file

@ -21,7 +21,7 @@ import java.util.Map;
import java.util.Objects;
/**
* GUI plugin: images and icons with theme support
* GUI: images source for GUI components like buttons. With theme supports.
*
* @author JayDi85
*/
@ -61,6 +61,7 @@ public enum ImageManagerImpl implements ImageManager {
private static final SoftValuesLoadingCache<Key, BufferedImage> THEME_BUTTON_IMAGES_CACHE;
private static final SoftValuesLoadingCache<Key, Image> PHASE_THEME_BUTTON_IMAGES_CACHE;
static {
THEME_BUTTON_IMAGES_CACHE = ImageCaches.register(SoftValuesLoadingCache.from(ImageManagerImpl::createThemeButtonImage));
PHASE_THEME_BUTTON_IMAGES_CACHE = ImageCaches.register(SoftValuesLoadingCache.from(ImageManagerImpl::createPhaseThemeButtonImage));
@ -69,15 +70,21 @@ public enum ImageManagerImpl implements ImageManager {
private static final class Key {
final String resourceName;
final int width;
final int height;
public Key(String resourceName) {
public Key(String resourceName, int width, int height) {
this.resourceName = resourceName;
this.width = width;
this.height = height;
}
@Override
public int hashCode() {
int hash = 3;
hash = 53 * hash + Objects.hashCode(this.resourceName);
hash = 53 * hash + this.width;
hash = 53 * hash + this.height;
return hash;
}
@ -96,31 +103,40 @@ public enum ImageManagerImpl implements ImageManager {
if (!this.resourceName.equals(other.resourceName)) {
return false;
}
if (this.width != other.width) {
return false;
}
if (this.height != other.height) {
return false;
}
return true;
}
}
private static BufferedImage createThemeButtonImage(ImageManagerImpl.Key key) {
return getBufferedImageFromResource(PreferencesDialog.getCurrentTheme().getButtonPath(key.resourceName));
String resName = PreferencesDialog.getCurrentTheme().getButtonPath(key.resourceName);
return getBufferedImageFromResource(resName, key.width, key.height);
}
private static Image createPhaseThemeButtonImage(ImageManagerImpl.Key key) {
return getImageFromResource(
PreferencesDialog.getCurrentTheme().getPhasePath("phase_" + key.resourceName.toLowerCase(Locale.ENGLISH) + ".png"),
new Rectangle(36, 36)
);
String resName = PreferencesDialog.getCurrentTheme().getPhasePath("phase_" + key.resourceName.toLowerCase(Locale.ENGLISH) + ".png");
return getImageFromResource(resName, key.width, key.height);
}
ImageManagerImpl() {
}
public BufferedImage getThemeButton(String resourceName) {
return THEME_BUTTON_IMAGES_CACHE.getOrThrow(new ImageManagerImpl.Key(resourceName));
private BufferedImage getThemeButton(String resourceName, int height) {
// all command buttons are 64 x 32
// TODO: add theme support with any proportion buttons
return THEME_BUTTON_IMAGES_CACHE.getOrThrow(new ImageManagerImpl.Key(resourceName, 2 * height, height));
}
@Override
public Image getPhaseImage(String phaseName) {
return PHASE_THEME_BUTTON_IMAGES_CACHE.getOrThrow(new ImageManagerImpl.Key(phaseName));
public Image getPhaseImage(String phaseName, int size) {
// all phase buttons are 36 x 36
// TODO: add theme support with any proportion buttons
return PHASE_THEME_BUTTON_IMAGES_CACHE.getOrThrow(new ImageManagerImpl.Key(phaseName, size, size));
}
@Override
@ -344,58 +360,58 @@ public enum ImageManagerImpl implements ImageManager {
}
@Override
public Image getConcedeButtonImage() {
return getThemeButton("concede.png");
public Image getConcedeButtonImage(int height) {
return getThemeButton("concede.png", height);
}
@Override
public Image getSwitchHandsButtonImage() {
return getThemeButton("switch_hands.png");
public Image getSwitchHandsButtonImage(int height) {
return getThemeButton("switch_hands.png", height);
}
@Override
public Image getStopWatchButtonImage() {
return getThemeButton("stop_watching.png");
public Image getStopWatchButtonImage(int height) {
return getThemeButton("stop_watching.png", height);
}
@Override
public Image getCancelSkipButtonImage() {
return getThemeButton("cancel_skip.png");
public Image getCancelSkipButtonImage(int height) {
return getThemeButton("cancel_skip.png", height);
}
@Override
public Image getSkipNextTurnButtonImage() {
return getThemeButton("skip_turn.png");
public Image getSkipNextTurnButtonImage(int height) {
return getThemeButton("skip_turn.png", height);
}
@Override
public Image getSkipEndTurnButtonImage() {
return getThemeButton("skip_to_end.png");
public Image getSkipEndTurnButtonImage(int height) {
return getThemeButton("skip_to_end.png", height);
}
@Override
public Image getSkipMainButtonImage() {
return getThemeButton("skip_to_main.png");
public Image getSkipMainButtonImage(int height) {
return getThemeButton("skip_to_main.png", height);
}
@Override
public Image getSkipStackButtonImage() {
return getThemeButton("skip_stack.png");
public Image getSkipStackButtonImage(int height) {
return getThemeButton("skip_stack.png", height);
}
@Override
public Image getSkipEndStepBeforeYourTurnButtonImage() {
return getThemeButton("skip_to_previous_end.png");
public Image getSkipEndStepBeforeYourTurnButtonImage(int height) {
return getThemeButton("skip_to_previous_end.png", height);
}
@Override
public Image getSkipYourNextTurnButtonImage() {
return getThemeButton("skip_all.png");
public Image getSkipYourNextTurnButtonImage(int height) {
return getThemeButton("skip_all.png", height);
}
@Override
public Image getToggleRecordMacroButtonImage() {
return getThemeButton("toggle_macro.png");
public Image getToggleRecordMacroButtonImage(int height) {
return getThemeButton("toggle_macro.png", height);
}
@Override
@ -414,51 +430,62 @@ public enum ImageManagerImpl implements ImageManager {
}
}
protected static Image getImageFromResourceTransparent(String path, Color mask, Rectangle rec) {
BufferedImage image;
Image imageCardTransparent;
Image resized = null;
private static Image getImageFromResourceTransparent(String path, Color mask, Rectangle rec) {
Image res = null;
URL imageURL = ImageManager.class.getResource(path);
try {
image = ImageIO.read(imageURL);
imageCardTransparent = Transparency.makeColorTransparent(image, mask);
resized = imageCardTransparent.getScaledInstance(rec.width, rec.height, java.awt.Image.SCALE_SMOOTH);
BufferedImage original = ImageIO.read(imageURL);
Image transparent = Transparency.makeColorTransparent(original, mask);
res = transparent.getScaledInstance(rec.width, rec.height, java.awt.Image.SCALE_SMOOTH);
} catch (Exception e) {
e.printStackTrace();
}
return resized;
return res;
}
protected static Image getImageFromResource(String path, Rectangle rec) {
Image resized = null;
private static Image getImageFromResource(String path) {
return getImageFromResource(path, 0, 0);
}
private static Image getImageFromResource(String path, int width, int height) {
Image res = null;
URL imageURL = ImageManager.class.getResource(path);
try {
BufferedImage image = ImageIO.read(imageURL);
resized = image.getScaledInstance(rec.width, rec.height, java.awt.Image.SCALE_SMOOTH);
if (width > 0 && height > 0) {
// use new size
res = image.getScaledInstance(width, height, java.awt.Image.SCALE_SMOOTH);
} else {
// keep file size
res = image;
}
} catch (Exception e) {
e.printStackTrace();
}
return resized;
return res;
}
protected static BufferedImage getBufferedImageFromResource(String path) {
URL imageURL = ImageManager.class.getResource(path);
BufferedImage image = null;
private static BufferedImage getBufferedImageFromResource(String path) {
return getBufferedImageFromResource(path, 0, 0);
}
private static BufferedImage getBufferedImageFromResource(String path, int width, int height) {
Image original = getImageFromResource(path, width, height);
BufferedImage output = new BufferedImage(original.getWidth(null), original.getHeight(null), BufferedImage.TYPE_INT_ARGB);
Graphics2D g2 = output.createGraphics();
try {
image = ImageIO.read(imageURL);
} catch (Exception e) {
e.printStackTrace();
g2.drawImage(original, 0, 0, null);
} finally {
g2.dispose();
}
return image;
return output;
}
public static BufferedImage deepCopy(BufferedImage bi) {

View file

@ -35,6 +35,8 @@ public class DebugUtil {
public static boolean GUI_GAME_DRAW_BATTLEFIELD_BORDER = false;
public static boolean GUI_GAME_DRAW_HAND_AND_STACK_BORDER = false;
public static boolean GUI_GAME_DRAW_PLAYER_PANEL_BORDER = false;
public static boolean GUI_GAME_DRAW_COMMAND_BUTTONS_PANEL_BORDER = false;
public static boolean GUI_GAME_DRAW_PHASE_BUTTONS_PANEL_BORDER = false;
// game dialogs
public static boolean GUI_GAME_DIALOGS_DRAW_CARDS_AREA_BORDER = false;