概要

Swing関係の自分用のメモとかネタ帳みたいなものです。

google八分メモ

  • 状況
    • 2014/08中頃からハブられている模様
      • 2014/09中頃、何日か復活してたけど、また駄目になる
    • ウェブマスターツールには特にメッセージ無し
  • 対策(効果なし?)
    • 新着RSS用に更新してた「はてな」を非公開
    • 一覧やタグページを<meta name="robots" content="NOINDEX,FOLLOW" />
    • sitemapを分割
    • コメントをDISQUSに移行
      • コメントユーザーページへの404ページを作成しない
    • schema.orgでマークアップ
    • xrea.jp自体がペナルティを受けていると考えたほうが良さそう
      • 他のサイト(*.xrea.jp)でも、同じ症状(タイトルで検索しても上位に表示されない)が確認できる?

さくらレンタルサーバ(+独自ドメイン)に移行(2014/11/01)

  • PHP設定の編集
    date.timezone = Asia/Tokyo
    zlib.output_compression=On
  • .htaccess で301リダイレクト
    RewriteEngine on
    RewriteBase /
    RewriteCond %{http_host} ^terai\.xrea\.jp
    RewriteRule ^(.*)$ http://ateraimemo.com/$1 [R=301,L]
  • 2014/11末現在、アクセス数などは以前の水準に戻っている

移動: ネタのリクエスト、Swingのサンプルに関するメモ

  • 以前ここで頂いたコメントは、Comments/Swingに移動しました。

Swing調整中のネタ一覧

調整中

以下テスト中なので、Exceptionが発生したり、まともに動作しなくても(゚ε゚)キニシナイ!!

検討中で未実装

こっそり消えても(゚ε゚)キニシナイ!!

メモ

  • JTree Table
  • JPopupMenu BlankIcon
  • UIDefaults ProxyLazyValue
  • JTable setSurrendersFocusOnKeystroke
  • File createTempFile
  • JTable Vertical Table Header
  • File getTotalSpace, getFreeSpace, getUsableSpace
  • JTabbedPane tab height animation
  • Formatter and password char
  • Cells with rouded corners in JTable
  • Tearaway tabs
  • Multi selection JComboBox
  • JSplitPane hide unhide tab
  • JScrollBar BlockIncrement
  • JScrollPane MapView subview
  • JDesktopPane JInternalFrame Tab Loop
  • JTabbedPane ScrollButton Layout
  • JTabbedPane Transparency
  • JTextField(search) in JButton(Tab, ButtonBar)
  • switch JInternalFrame to JTabbedPane
  • JTree JList(1.8.0) Nimbus Selection Rect
  • Highlighter#removeAllHighlights()
  • http://docs.oracle.com/javase/jp/7/api/javax/swing/text/html/HTMLDocument.html#getIterator(javax.swing.text.html.HTML.Tag)
  • java - Can you make a JToolBar undetachable? - Stack Overflow
    //package example;
    //-*- mode:java; encoding:utf-8 -*-
    // vim:set fileencoding=utf-8:
    //@homepage@
    import java.awt.*;
    import javax.swing.*;
    import javax.swing.plaf.basic.*;
    
    public final class MainPanel extends JPanel {
        private final JComboBox<String> combo = new JComboBox<>(makeModel());
        private final JLabel label     = new JLabel();
        private final JToolBar toolbar = new JToolBar("toolbar");
        private final JButton button   = new JButton("button");
        public MainPanel() {
            super(new BorderLayout());
    
            toolbar.setUI(new BasicToolBarUI() {
                @Override public void setFloating(boolean b, Point p) {
                    if (b) {
                        Toolkit.getDefaultToolkit().beep();
                    }
                    super.setFloating(false, p);
                }
    //             @Override public boolean canDock(Component c, Point p) {
    //                 return super.canDock(c, p) ? isHorizontalDockingConstraint(c, p) : false;
    //             }
    //             private boolean isHorizontalDockingConstraint(Component c, Point p) {
    //                 if (!c.contains(p)) { return false; }
    //                 int iv = (toolBar.getOrientation() == JToolBar.HORIZONTAL) ? toolBar.getSize().height : toolBar.getSize().width;
    //                 return p.x < c.getWidth() - iv && p.x >= iv;
    //             }
            });
            add(Box.createRigidArea(new Dimension()), BorderLayout.WEST);
            add(Box.createRigidArea(new Dimension()), BorderLayout.EAST);
    
            button.setFocusable(false);
            toolbar.add(new JLabel("label"));
            toolbar.add(Box.createRigidArea(new Dimension(5, 5)));
            toolbar.add(button);
            toolbar.add(Box.createRigidArea(new Dimension(5, 5)));
            toolbar.add(combo);
            toolbar.add(Box.createGlue());
    
            label.setText("<html>dockable: NORTH, SOUTH<br>undockable: EAST, WEST");
            label.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
            add(toolbar, BorderLayout.NORTH);
            add(label);
            setPreferredSize(new Dimension(320, 240));
        }
        private static ComboBoxModel<String> makeModel() {
            DefaultComboBoxModel<String> model = new DefaultComboBoxModel<>();
            model.addElement("1111111");
            model.addElement("22222");
            model.addElement("3333333333333333");
            model.addElement("44444444444");
            return model;
        }
        public static void main(String... args) {
            EventQueue.invokeLater(new Runnable() {
                @Override public void run() {
                    createAndShowGUI();
                }
            });
        }
        public static void createAndShowGUI() {
            try {
                UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
            } catch (ClassNotFoundException | InstantiationException
                   | IllegalAccessException | UnsupportedLookAndFeelException ex) {
                ex.printStackTrace();
            }
            JFrame frame = new JFrame("@title@");
            frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
            frame.getContentPane().add(new MainPanel());
            frame.pack();
            frame.setLocationRelativeTo(null);
            frame.setVisible(true);
        }
    }
    

Swing関係のバグやテスト中のコードなどのメモ

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;

public class Main2 {
  private static Robot robot;
  public static void main(String... args) throws Exception {
    robot = new Robot();
    robot.setAutoDelay(100);
    EventQueue.invokeLater(new Runnable() {
      @Override public void run() {
        createAndShowGUI();
      }
    });
    EventQueue.invokeAndWait(new Runnable() {
      @Override public void run() {
        Point pt = new Point(50, 10);
        robot.mouseMove(pt.x, pt.y);
        robot.mousePress(InputEvent.BUTTON1_MASK);
        pt.translate(50, 50);
        robot.mouseMove(pt.x, pt.y);
        //robot.mouseRelease(InputEvent.BUTTON1_MASK);
      }
    });
  }
  public static void createAndShowGUI() {
    final JFrame f = new JFrame();
    f.add(new JLabel("Try JFrame draging 2sec"));
    f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
    f.setSize(320, 240);
    f.setVisible(true);
    final JWindow w = new JWindow(f);
    w.add(new JButton("button"));
    w.pack();
    Timer t = new Timer(2000, new ActionListener() {
      @Override public void actionPerformed(ActionEvent e) {
        w.setVisible(true);
      }
    });
    t.setRepeats(false);
    t.start();
  }
}

java - swing gui flickering white error - Stack Overflowのサンプルを試していたら面白い挙動に遭遇。

  • JDK1.8.0_40で修正された
  • JDK1.7.0_21
    • IllegalArgumentException: Comparison method violates its general contract!
      Exception in thread "AWT-EventQueue-0" java.lang.IllegalArgumentException: Comparison method violates its general contract!
      	at java.util.TimSort.mergeLo(TimSort.java:747)
      	at java.util.TimSort.mergeAt(TimSort.java:483)
      	at java.util.TimSort.mergeCollapse(TimSort.java:410)
      	at java.util.TimSort.sort(TimSort.java:214)
      	at java.util.TimSort.sort(TimSort.java:173)
      	at java.util.Arrays.sort(Arrays.java:659)
      	at java.util.Collections.sort(Collections.java:217)
    • フレームのリサイズで上下左右に余白?
  • JDK1.6.0_41
    • フレームのリサイズで右と下に余白?
import java.awt.*;
import javax.swing.*;

public class Gui {
  public JComponent makeUI() {
    JPanel p = new JPanel(new GridLayout(22,12,10,10));
    p.setBackground(Color.WHITE);
    p.setBorder(BorderFactory.createLineBorder(Color.BLUE, 10));
    for (int i = 0; i < 22; i++) {
      for (int j = 0; j < 12; j++) {
        JLabel label = new JLabel();
        label.setBorder(BorderFactory.createLineBorder(Color.RED));
        label.setBackground(Color.GRAY);
        label.setOpaque(true);
        //label.setPreferredSize(new Dimension(50, 50));
        p.add(label);
      }
    }
    return p;
  }
  public static void main(String[] args) {
    EventQueue.invokeLater(new Runnable() {
      @Override public void run() {
        createAndShowGUI();
      }
    });
  }
  public static void createAndShowGUI() {
    JFrame f = new JFrame();
    f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
    f.getContentPane().add(new Gui().makeUI());
    f.setSize(320, 240);
    //f.pack();
    f.setLocationRelativeTo(null);
    f.setVisible(true);
  }
}

JMenuBarを検索してJRadioButtonMenuItemに、ActionListenerを追加

Arrays.stream(menuBar.getSubElements())
  .flatMap(new Function<MenuElement, Stream<MenuElement>> () {
      @Override public Stream<MenuElement> apply(MenuElement me) {
          MenuElement[] sub = me.getSubElements();
          if (sub.length != 0) {
              return Arrays.stream(sub).flatMap(sme -> apply(sme));
          } else {
              return Stream.of(me);
          }
      }
  })
  .filter(mi -> mi instanceof JRadioButtonMenuItem)
  .forEach(mi -> ((JRadioButtonMenuItem) mi).addActionListener(al));

Swingのサンプルに関するメモ(引用)

目標にしようと思っているけど、なかなか難しそう…。

シンプルで、小さなサンプル(Simple Client Applications)

「チュートリアル」「小さなサンプル」「レビュー」

  • 社内ライブラリのチュートリアル
    社内にはそのライブラリで作られたアプリケーションが色々あり, ライブラリを使う時はその中から欲しいコード片を掘り起こすのが常だった.苦労の多い作業だった. オブジェクトの初期化, 設定, 利用のシーケンスはコード内に分散しており, それを漏れなく抜き出すのは難しい. コードの質も玉石混合で, バグを持ったままコピペされることもあった. 私達の目的は, 極端に言えば便利で正しいコピペ素材を提供することだった.

Examples should be exemplary (例は典型的であるべき)

  • Interested in Writing New Tutorial Demos for Sun?についてたコメントを引用
    • 開発者はサンプルコードに頼って(コピペして)いる
    • Examples should be exemplary
      He references a research study that found that a bug in one of the original JDK applet examples was replicated in 75% of applets attempting similar goal in their sampling of applets found on the internet. This underscores the fact that developers rely on example code, even, unfortunately, to the point that we copy bugs too. This is sometimes described as "Monkey see, monkey do", or "Rape and Paste".

コピー指向プログラミング

  • http://ameblo.jp/argv/entry-10144604985.html
    • あらかじめ完成度の高い「コピー元」を用意する
    • コピー元を品質の高いコードに集中させることで、「バグのコピー」や「似て非なるバージョンの散在」を防ぐ。
    • 「コピーされたもの」が分かるようにする
      • 例えば、コピー元のコードに特殊なコメントを埋め込む(もちろん、コピー先でも消さないでおく)などして、後からコピー先を検索しやすくする。これは、不具合修正や仕様変更などの際に、影響範囲(全てのコピー先)を特定しやすくするためである。

Best Practice for Swing coding

  • Alexander Potochkin's Blog: Why I don't use Swing hacks (in production code)についてたコメントを引用
    • For deities sake, coordinate your documentation and example in one single place. And make sure they demonstrate best practice.
    • 米Microsoft、コードサンプル集「All-In-One Code Framework」 - SourceForge.JP Magazine
      http://weblogs.java.net/blog/alexfromsun/archive/2007/09/why_i_dont_like.html:title=Best Practice for Swing coding>
      The problem with the "Swing Hacks" book is that it contains solutions in search for problems. And that is just the wrong motivation to add a hack to production code. 
      
      There are other, good motivations to add hacks to production code. Let's start with a better definition of the word hack: 
      A creative circumvention of a limitation 
      And unfortunately Swing has many limitations which hurt in the real world. If your customer insists that you make something happen "or else ..." you have a very good motivation for a hack. I do not agree that a programmer has to understand all implications of a hack. Swing is such a complex system, itself full of ugly hacks, bugs, and badly documented, that this is just not feasible. 
      
      When it comes to Swing then part of the problem is not only that Swing is so limited, but also that the limitations don't get fixed. This is an attitude problem at Sun. Sun lacks the understanding and imagination of what is needed on the desktop. Many Sun people still seem to live in SunView (ups, I am showing my age here) lala-land. Sun thinks it's always right. 
      
      Sun has totally failed to establish a best-practice for Swing coding. The examples that ship with the JDK are some of the worst Swing code. What is even worse, while Sun employees were busy writing shiny conference demos, no one at Sun could be arsed to clean up the JDK example mess during the last ten years. This is not an isolated case. The Swing tutorial examples are equally bad. They can't even be bothered to use JavaDoc for documentation and typically demonstrate badly structured code for badly structured GUIs. 
      
      Sun also has fragmented and hidden GUI information in unbelievable ways. For example, try to find a single page listing all Swing system properties. Good luck. In general, some Swing information comes with the JDK. Some can be found in the TSC (which is on life support). Others on ominous pages like JDCTechTips, Almanac (blech!), Code samples, Online Training article, Java.net articles, multi-media samples, technical articles, 2D examples in four places (did you know Sun has a half-finished Postscript Interpreter in Java http://java.sun.com/products/java-media/2D/samples/postscript/PostscriptDemo.zip?). Then there are dev.java.net examples, JDC, SDN, whatnot. I have probably forgot another ten places where Sun hides Swing, 2D, media, GUI or desktop information. 
      
      No one at Sun seems to coordinate the stuff. Instead everyone at Sun seems to run the his own show, trying to establish an own kingdom. For deities sake, coordinate your documentation and example in one single place. And make sure they demonstrate best practice. 
      
      Posted by: ewin on September 28, 2007 at 02:58 PM
      

Swing 修正予定・検討メモ

実際にやるかどうかは未定ですが、今後Swing Tipsを改良したいと思っていることをメモしています(順不同)。

  • Swing Tipsのみ分離して、Markdown、静的サイト生成に移行できないか検討中
    • http://ateraimemo.com/index.html で、Jekyll BootstrapDISQUSのテスト中
    • Jekyll、タグ付け
    • inline<code>タグ
      • JDK 1.6.0_10などのアンダーライン
      • new JPanel()などの翻訳拒否 <code>で翻訳されないはず
  • https://sites.google.com/site/javaswingtips/ は、500ページ近く手動でアップロードすることになりそうなので無理かも
  • ソースコード
    • Wiki中のサンプルコードでクラスなどをjavadocに自動リンク
    • 日本語、文字コード
  • gistプラグインでの行番号がズレてしまう 修正済み
  • 簡単にコンパイル、実行
    • Ant(build.xml)、バッチファイル、シェルスクリプトの改良
      • build.xml の整理
    • NetBeans (Eclipse) .classpath、.projectの追加
    • src.zipのファイル名を、それぞれ適当なものに変更?
  • 各Tipsの難易度表示
    • こちらで適当に評価するか、ユーザに評価してもらうか…
    • レーティングバー
  • Web Start
    • 「プログラムの追加と削除」に登録されない方法を探す
      • キャッシュを無効にする
    • UI操作無しで全部削除できる方法を探す
      • javaws -uninstall
    • 起動時にスプラッシュスクリーンを表示しない方法を探す
      • javaws -Xnosplash <jnlp>

変更済み

  • スパム対策
    • 英数のみのコメント禁止 別の対策を導入 再度英数のみのコメント禁止
  • ソースコード、FindBugs, PMD
  • ソースコード、Ubuntuなどでも実行できるように
    • e.isPopupTrigger()を使ってJPopupMenuを表示している箇所(JDK 5 以前のソース)を、すべてsetComponentPopupMenuに変更する。
  • ソースコード、1.5以上対応に変更
  • Color クラスにおける定数の大文字化
  • ソースコード、codeプラグインで色分け
  • ソースコード、build.xmlなどをすべてUTF-8化
  • バッチファイルの代わりにantのタスクを使うように変更
    • antのbuild.xmlにrunタスクを追加、ただしバッチファイルも残すことに
  • sampleをexampleに統一
  • Swing/JarFileAntの下に移動する
  • 分類: JTree JButton
  • TableSorter.javaをJDK6のTableRwoSorterに置き換える
    • ソートする意味のないサンプルで使用しているTableSorter.javaは削除
  • スクリーンショット
    • すべて撮り直す(prefixのコロン:)
    • screenshot一覧ページの分割、画像サイズ削減
    • キャッシュ一覧で探しやすいように、タイトルの頭に「サ:」(サンプル)を付けていたのを、「JST 」(Java Swing Tips)に変更
    • 日本標準時ではない…
  • 分類
    • JTableのTipsを分割
    • ソースを一つにまとめる(ワンクリックで全ソースコードを表示したい)ためにTestModel.javaを検索してpackageプライベートなクラスに変更中
      >find . -type f -name TestModel.java -ls | gawk "{print $7, $11}" | sort
    • Look&Feel関係でpublic クラスでないとダメなのは…、放置。

リンクメモ(自分用)

Bugリンクメモ(自分用)

メモ: 痒いところに手が届かないJavaのコード

初期化時にデッドロックの可能性があるからということで、それはいいとして。
代わりに使うのが
Logger.getLogger(Logger.GLOBAL_LOGGER_NAME)
だなんて、どうにかしてほしい。
Logger.getGlobal()
を用意してくれればいいのに。なに考えてんだろう?EoDじゃなかったの?
Javaのコード書きにくさって、静的動的よりも、こういう痒いところに手が届かないことの積み重ねが大きいと思う。

Swing関係

  • 表の全行を削除するメソッド
    • 表の全行を削除する場合、DefaultTableModel#setRowCount(0);ではなく、もっと分かりやすいメソッドが欲しい。
    • 例えばDefaultComboBoxModel#removeAllElements()DefaultListModel#clear()のような(これはこれでどれかひとつに統一して欲しい…)。
  • フレームを画面中央に表示するメソッド
    • フレームを画面中央に表示する場合、Window#setLocationRelativeTo(null);ではなく、もっと分かりやすいメソッドが欲しい。
    • JavaFX Scriptでは、centerOnScreen: trueが用意されていたが…。