【问题标题】:custom round skin gui自定义圆形皮肤gui
【发布时间】:2014-10-13 04:47:54
【问题描述】:

在寻找如何创建圆形界面的起点。我曾尝试查看文档并尝试查看 Eclipse 的其他插件。我刚开始用 java 构建 gui,我发现的一切要么要求我成为公司的一员以使用他们的产品,要么想要几百美元。

我只是一个谦逊的程序员,尝试着做 gui,而我正在从事的一个个人项目想要一个圆形的 gui 皮肤。

【问题讨论】:

  • “圆形接口”....?这是什么意思?
  • 我所说的圆形是我不想要方形/矩形窗口。我想要一个圆形的界面。
  • “我刚刚开始用 java 构建 gui。” 暂时不要使用非矩形 GUI。它们实际上是一个“专家”主题,用途非常有限,学习如何布局(甚至)矩形 GUI 本身对某些人来说是一条陡峭的学习曲线。

标签: java swing user-interface window


【解决方案1】:

至少有两种方法可以实现这一目标...

你可以...

使用JFrame#setShape 改变主窗口的形状,例如...

JFrame frame = new JFrame("Testing");
frame.getContentPane().setBackground(Color.RED);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setLayout(new GridBagLayout());
frame.add(new JLabel("Boo!"));
frame.setSize(200, 200);
frame.setLocationRelativeTo(null);
frame.setUndecorated(true);
frame.setShape(new Ellipse2D.Double(0, 0, 200, 200));
frame.setVisible(true);

这很简单,但坦率地说,看起来很糟糕。使用这种技术无法实现软剪辑以平滑边缘...

你可以...

创建一个透明窗口并“伪造”形状...

import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.GridBagLayout;
import java.awt.RenderingHints;
import java.awt.geom.Ellipse2D;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;

public class CircleUI {

    public static void main(String[] args) {
        new CircleUI();
    }

    public CircleUI() {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
                }

                JFrame frame = new JFrame("Testing");
                frame.setUndecorated(true);
                frame.setBackground(new Color(0, 0, 0, 0));
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.add(new CirclePane());
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }

    public class CirclePane extends JPanel {

        public CirclePane() {
            setOpaque(false);
            setLayout(new GridBagLayout());
            add(new JLabel("Boo!"));
            setBackground(Color.RED);
        }

        @Override
        public Dimension getPreferredSize() {
            return new Dimension(200, 200);
        }

        @Override
        protected void paintComponent(Graphics g) {
            super.paintComponent(g);
            Graphics2D g2d = (Graphics2D) g.create();
            g2d.setRenderingHint(RenderingHints.KEY_ALPHA_INTERPOLATION, RenderingHints.VALUE_ALPHA_INTERPOLATION_QUALITY);
            g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
            g2d.setRenderingHint(RenderingHints.KEY_COLOR_RENDERING, RenderingHints.VALUE_COLOR_RENDER_QUALITY);
            g2d.setRenderingHint(RenderingHints.KEY_DITHERING, RenderingHints.VALUE_DITHER_ENABLE);
            g2d.setRenderingHint(RenderingHints.KEY_FRACTIONALMETRICS, RenderingHints.VALUE_FRACTIONALMETRICS_ON);
            g2d.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BILINEAR);
            g2d.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
            g2d.setRenderingHint(RenderingHints.KEY_STROKE_CONTROL, RenderingHints.VALUE_STROKE_PURE);
            g2d.setColor(Color.RED);
            g2d.fill(new Ellipse2D.Double(0, 0, getWidth() - 1, getHeight() - 1));
            g2d.dispose();
        }

    }

}

可以说,这会产生更好看的结果,但会允许您显示超出形状的组件(允许它们溢出),因此您需要能够管理内容以确保不会发生这种情况。 .

你可以...

两者都做...

JFrame frame = new JFrame("Testing");
frame.setUndecorated(true);
frame.setBackground(new Color(0, 0, 0, 0));
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

// This is the secret here...
JPanel content = new JPanel(new BorderLayout());
content.setOpaque(false);
content.setBorder(new EmptyBorder(1, 1, 1, 1));
content.add(new CirclePane());

frame.setContentPane(content);
frame.pack();
int width = frame.getWidth();
int height = frame.getHeight();
frame.setShape(new Ellipse2D.Double(0, 0, width, height));
frame.setLocationRelativeTo(null);
frame.setVisible(true);

这样做是使用第一个示例中的setShape 方法,以及第二个示例中的软剪辑并将它们组合起来。这是通过使用另一个JPanel 作为我们“假”形状面板的容器并提供一个非常小的(1 像素)空边框来完成的。这会将框架的大小从我们的软剪裁面板的边缘推出,但意味着任何溢出它的组件都将被剪裁......

【讨论】:

  • “创建一个透明窗口并“伪造”形状......可以说,这会产生更好看的结果,但允许您显示超出形状的组件(允许它们溢出).." 您可以将其与第一个示例中稍大的剪辑相结合,然后在内容窗格中添加 EmptyBorder 以确保内容位于第二个示例的纯色区域内。
  • @AndrewThompson Shhhh....如果你一直告诉他们所有的秘密,我怎么能用我明显的“才华”来“惊叹”人们...
【解决方案2】:

在您的 JFrame 上调用 setOpacity 方法。更多细节在这里: http://java-demos.blogspot.com/2012/09/how-to-create-shaped-jframes-in-java.html

Java 6 和 Java 7 的其他选项:click here for examples in another solution

JFrame frame = new JFrame();
frame.setUndecorated(true);
AWTUtilities.setWindowShape(frame, new Ellipse2D.Double(0, 0, 100, 100));

不要粗鲁,但如果您花 5 分钟在 Google 上搜索或在 Stack Overflow 上搜索,您真的可以找到这个...

如果你真的想学习 Java GUI,你应该花一些时间学习 AWT、Swing 或 JavaFX,这些是用于执行 GUI 工作的核心 Java 库,以及 JFrame 和 JWindow 等基本对象,以及创建菜单、按钮、ActionListener 和布局。

【讨论】:

    猜你喜欢
    • 2011-08-17
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-10-25
    • 1970-01-01
    • 2011-05-02
    相关资源
    最近更新 更多