質問

Word 2007のズームスライダーを模倣する(または既存のコンポーネントを見つける)ことを試みています:

 2状態ズームバー

このコンポーネントと標準のJava JSliderには2つの主な違いがあります:

  1. 100%以外のティックにスナップせず、マウスを離したときではなく、バーをスライドさせているときにスナップします
  2. スライダーは全体にわたって線形ではありません。スライダーの左半分は10%から100%になります。右側は100%から500%になります。

これまでの所持品は次のとおりです。 Javaクローン

出典:

    import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JSlider;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;

/**
 *
 * @author NDUNN
 * @date Nov 25, 2009
 */
public class ZoomBar extends JPanel implements ActionListener, ChangeListener {

    private JLabel zoomAmount;
    private JButton minus;
    private JButton plus;
    private JSlider slider;

    private static final int MIN_ZOOM = 10;
    private static final int MAX_ZOOM = 200;
    private static final int DEFAULT_ZOOM = 100;

    private static final int MAJOR_ZOOM_SPACING = 50;
    private static final int MINOR_ZOOM_SPACING = 10;

    public ZoomBar() {
        super();

        minus = new JButton("-");
        plus = new JButton("+");
        slider = new JSlider(MIN_ZOOM, MAX_ZOOM, DEFAULT_ZOOM);

        slider.setMinorTickSpacing(MINOR_ZOOM_SPACING);
        slider.setMajorTickSpacing(MAJOR_ZOOM_SPACING);
        slider.setPaintTicks(true);
        slider.setSnapToTicks(true);

        zoomAmount = new JLabel(slider.getValue() + "%");

        add(zoomAmount);
        add(minus);
        add(slider);
        add(plus);

        plus.addActionListener(this);
        minus.addActionListener(this);

        slider.addChangeListener(this);
    }

    public static void main(String[] args) {
        JFrame frame = new JFrame("Zoom bar clone");
        frame.setContentPane(new ZoomBar());
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.pack();
        frame.setVisible(true);
    }

    public void actionPerformed(ActionEvent e) {
        if (e.getSource() == plus) {
            slider.setValue(slider.getValue() + MINOR_ZOOM_SPACING);
        }
        else if (e.getSource() == minus) {
            slider.setValue(slider.getValue() - MINOR_ZOOM_SPACING);
        }
    }

    public void stateChanged(ChangeEvent e) {
        if (slider.getValueIsAdjusting()) {
            return;
        }
        zoomAmount.setText(slider.getValue() + "%");
    }

}

基本的には外観を模倣しますが、上記の2つの機能はありません。 JSlider API に何も表示されませんこれにより、この動作を取得できます。この動作を得るためにゼロから始めなければなりませんか?もしそうなら、それは私の時間の価値はありませんが、誰かがこの振る舞いを得る良い方法を知っているなら、私が取り組んでいるプロジェクトに参加するのは良いと思います。

ありがとう、

ニック


編集:

0..50と50..100を異なる値にマッピングするだけのアイデアについてPSpeedに感謝します。そのためのコードは以下のとおりです。

残念ながら、setValueをsnapに変更するという考えはうまくいきませんでした。

public void stateChanged(ChangeEvent e) {
    // While slider is moving, snap it to midpoint
    int value = slider.getValue();
    if (slider.getValueIsAdjusting()) {
        return;
    }

    zoomValue = fromSlider(value);
    zoomLabel.setText(zoomValue + "%");
}

public int fromSlider(int sliderValue) {
    int mappedValue = 0;
    if (sliderValue <= 50) {
        // Map from [0, 50] to [MIN ... DEFAULT]
        mappedValue = (int) map(sliderValue, 0, 50, MIN_ZOOM, DEFAULT_ZOOM);
    }
    else {
        // Convert from  (50, 100] to (DEFAULT ... MAX]
        mappedValue = (int) map(sliderValue, 50, 100, DEFAULT_ZOOM, MAX_ZOOM);
    }
    return mappedValue;
}

public int toSlider(int modelValue) {
    int mappedValue = 0;
    if (modelValue <= DEFAULT_ZOOM) {
        // Map from [MIN_ZOOM, DEFAULT_ZOOM] to [0 ... 50]
        mappedValue = (int) map(modelValue, MIN_ZOOM, DEFAULT_ZOOM, 0, 50);
    }
    else {
        // Convert from  (DEFAULT ... MAX] to (50, 100]
        mappedValue = (int) map(modelValue, DEFAULT_ZOOM, MAX_ZOOM, 50, 100);
    }
    return mappedValue;
}


/**
 * @param value The incoming value to be converted
 * @param low1  Lower bound of the value's current range
 * @param high1 Upper bound of the value's current range
 * @param low2  Lower bound of the value's target range
 * @param high2 Upper bound of the value's target range
 * @return
 */
public static final double map(double value, double low1, double high1, double low2, double high2) {

    double diff = value - low1;
    double proportion = diff / (high1 - low1);

    return lerp(low2, high2, proportion);
}

public static final double lerp(double value1, double value2, double amt) {
    return ((value2 - value1) * amt) + value1;
}

編集2: 私が気づくもう1つの違いは、JButtonではボタンを何度も押してボタンを放すことはできませんが、オフィスの+/-ボタンはそうすることです。これを模倣する方法はありますか?

役に立ちましたか?

解決

カスタムのBoundedRangeModelを使用すると、このすべての動作を取得できると思います。重要なのは、モデルが通常のタイプの値の範囲をレポートするようにすることですが、ズーム率が必要な場合は異なる方法で処理することです。

たとえば、範囲を0〜100の範囲で実行する場合、0〜50を一方向に、50〜100を別の方法で処理します(それぞれ10〜100%と100〜500%)。

スナップ動作を取得するには、setValue()をオーバーライドして必要な範囲をスナップするだけでよいと確信しています。したがって、値の範囲として0〜100を使用し、47〜53でsetValue()が呼び出された場合は、値を50にスナップします。

ライセンス: CC-BY-SA帰属
所属していません StackOverflow
scroll top