Java中的Windows 7任务栏进度条

32

我想知道是否有可能在任务栏上显示进度条,就像Windows资源管理器在进行文件操作时所做的那样?

我看到了许多示例,但它们都涉及到C#。

SWT不可行。

在任务栏上显示进度条


1
你能否为那些没有Windows设备的人发布一张截图? - OscarRyz
你想让它像资源管理器进度条一样吗?还是你想在任务栏上显示它?(提示:其中一个比另一个难得多)。 - John Knoeller
你是怎么解决的?有进展吗? - Paul Lammertsma
6个回答

18

我发现这个功能包含在Java 9中。它是AWT的一部分,使用起来非常简单。

以下是一个简短的示例:

import java.awt.Taskbar;
import java.awt.Taskbar.State;

import javax.swing.JFrame;
import javax.swing.WindowConstants;

/**
 * @author fxl
 */
public class TaskbarSample {

    public static void main(String[] args) {

        // JavaDoc:
        // https://docs.oracle.com/javase/9/docs/api/java/awt/Taskbar.html

        // MSDNDoc:
        // https://msdn.microsoft.com/en-us/library/dd391692(VS.85).aspx
        if (Taskbar.isTaskbarSupported() == false) {
            return;
        }

        JFrame dialog = new JFrame("Test - 50%");
        dialog.setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);
        dialog.setVisible(true);    


        Taskbar taskbar = Taskbar.getTaskbar();
        taskbar.setWindowProgressState(dialog, State.ERROR);
        taskbar.setWindowProgressValue(dialog, 50);     
    }

}

11

7
这个示例可以胜任:

此网页

任务栏:
enter image description here

代码:

import org.bridj.Platform;
import org.bridj.Pointer;
import org.bridj.cpp.com.COMRuntime;
import org.bridj.cpp.com.shell.ITaskbarList3;
import org.bridj.jawt.JAWTUtils;

import javax.swing.*;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

public class TaskBarListDemo extends JFrame implements ActionListener, ChangeListener
{
    private ITaskbarList3 list;
    private JSlider slider;
    private Pointer<?> hwnd;

    private TaskBarListDemo() throws ClassNotFoundException
    {
        super("TaskbarList Demo (" + (Platform.is64Bits() ? "64 bits" : "32 bits") + ")");

        list = COMRuntime.newInstance(ITaskbarList3.class);

        getContentPane().add("Center", new JLabel("Hello Native Windows 7 World !"));
        Box box = Box.createVerticalBox();

        int min = 0;
        int max = 300;
        int val = (min + max / 2);
        slider = new JSlider(min, max, val);
        slider.addChangeListener(this);
        box.add(slider);

        ButtonGroup group = new ButtonGroup();
        for (ITaskbarList3.TbpFlag state : ITaskbarList3.TbpFlag.values())
        {
            JRadioButton cb = new JRadioButton(state.name());
            group.add(cb);
            cb.putClientProperty(ITaskbarList3.TbpFlag.class, state);
            cb.setSelected(state == ITaskbarList3.TbpFlag.TBPF_NORMAL);
            cb.addActionListener(this);
            box.add(cb);
        }
        getContentPane().add("South", box);

    }

    @Override
    protected void finalize() throws Throwable
    {
        super.finalize();
        list.Release();
    }

    public void setVisible(boolean visible)
    {
        super.setVisible(visible);

        long hwndVal = JAWTUtils.getNativePeerHandle(this);
        hwnd = Pointer.pointerToAddress(hwndVal);
        list.SetProgressValue((Pointer) hwnd, slider.getValue(), slider.getMaximum());
    }

    @Override
    public void stateChanged(ChangeEvent actionEvent)
    {
        list.SetProgressValue((Pointer) hwnd, slider.getValue(), slider.getMaximum());
    }

    @Override
    public void actionPerformed(ActionEvent actionEvent)
    {
        JRadioButton button = ((JRadioButton) actionEvent.getSource());
        if (button.isSelected())
        {
            ITaskbarList3.TbpFlag flag = (ITaskbarList3.TbpFlag) button.getClientProperty(ITaskbarList3.TbpFlag.class);
            list.SetProgressValue((Pointer) hwnd, slider.getValue(), slider.getMaximum());
            list.SetProgressState((Pointer) hwnd, flag);
        }
    }

    public static void main(String[] arguments) throws Exception
    {
        TaskBarListDemo f = new TaskBarListDemo();
        f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
        f.pack();
        f.setVisible(true);
    }
}

Maven 依赖:

<dependencies>
    <dependency>
        <groupId>com.nativelibs4java</groupId>
        <artifactId>bridj</artifactId>
        <version>LATEST</version>
    </dependency>
    <dependency>
        <groupId>net.java.dev.jna</groupId>
        <artifactId>jna</artifactId>
        <version>LATEST</version>
    </dependency>
    <dependency>
        <groupId>net.java.dev.jna</groupId>
        <artifactId>jna-platform</artifactId>
        <version>LATEST</version>
    </dependency>
</dependencies>

只要您的顶层窗口是JFrame,这就可以工作。但是,JDialog却不行。 - spy

5
在Java中,目前没有标准设施可以实现这一点。
因此,你需要直接与Windows进行通信来完成此操作。因此,你需要找到正确的Windows例程,并使用JNA(可能是最简单的方法)来调用该例程。我不知道是否有供应商或项目已经完成了这项工作。
编辑:看起来http://code.google.com/p/nativelibs4java/项目可能会做你想要的。

不,JNA比JNI更容易使用。 - Thorbjørn Ravn Andersen
BridJ(NativeLibs4Java的子项目:http://code.google.com/p/bridj/)确实提供了ITaskbarList3的即用包装器和一个工作示例: http://code.google.com/p/nativelibs4java/source/browse/trunk/libraries/Runtime/BridJ/src/main/java/org/bridj/demos/TaskbarListDemo.java - zOlive
看起来演示链接已经移动到http://code.google.com/p/nativelibs4java/source/browse/trunk/libraries/Runtime/BridJ/src/main/demos/TaskbarListDemo.java。 - rogerdpack
1
顺便说一下,演示已经移动到了更远的地方(GitHub):https://github.com/ochafik/nativelibs4java/blob/master/libraries/Runtime/BridJ/src/main/demos/TaskbarListDemo.java - zOlive
再次LOL https://github.com/ochafik/nativelibs4java/blob/master/libraries/BridJ/src/main/demos/TaskbarListDemo.java - rogerdpack
1
还有...又出现了?>_< https://github.com/nativelibs4java/BridJ/blob/master/src/main/demos/TaskbarListDemo.java - Xr.

1
作为Java9的java.awt.Taskbar仅适用于旧的swing框架(他们不知道如何为javafx.stage.Stage实现此功能),而com.nativelibs4java bridj不再工作(请参见https://github.com/nativelibs4java/BridJ/issues/94),因此我使用JNA 4.1.0实现了一种解决方案。

请注意:

  • 依赖于调用内部javafx api(com.sun.javafx.stage.WindowHelper) - 因此它可能会在下一个Java更新中出现问题
  • 它仅设置“不确定”的进度状态 - 但是通过一些调整应该也可以实现正常的进度状态

希望这有所帮助。

ITaskbarList3.java

package example;

import com.sun.jna.platform.win32.Guid.IID;
import com.sun.jna.platform.win32.WinDef.HWND;
import com.sun.jna.platform.win32.WinNT.HRESULT;

public interface ITaskbarList3 {
    IID IID_ITASKBARLIST3 = new IID("ea1afb91-9e28-4b86-90e9-9e9f8a5eefaf"); // from ShObjIdl.h

    int TBPF_NOPROGRESS = 0;
    int TBPF_INDETERMINATE = 0x1;
    int TBPF_NORMAL = 0x2;
    int TBPF_ERROR = 0x4;
    int TBPF_PAUSED = 0x8;

    HRESULT SetProgressState(HWND hwnd, int tbpFlags);
}

TaskbarList3.java

package example;

import com.sun.jna.Pointer;
import com.sun.jna.platform.win32.WinDef.HWND;
import com.sun.jna.platform.win32.WinNT.HRESULT;
import com.sun.jna.platform.win32.COM.COMInvoker;

public final class TaskbarList3 extends COMInvoker implements ITaskbarList3 {

    public TaskbarList3(Pointer pointer) {
        setPointer(pointer);
    }

    @Override
    public HRESULT SetProgressState(HWND hwnd, int tbpFlags) {
        return (HRESULT) this._invokeNativeObject(
                10, // magic number (gathered by trial and error)
                new Object[] { this.getPointer(), hwnd, tbpFlags },
                HRESULT.class);
    }
}

TaskbarPeer.java

package example;

import com.sun.javafx.stage.WindowHelper;
import com.sun.jna.Pointer;
import com.sun.jna.platform.win32.Guid.CLSID;
import com.sun.jna.platform.win32.Ole32;
import com.sun.jna.platform.win32.W32Errors;
import com.sun.jna.platform.win32.WTypes;
import com.sun.jna.platform.win32.WinDef.HWND;
import com.sun.jna.ptr.PointerByReference;

import javafx.stage.Stage;

public final class TaskbarPeer {

    public static void setIndeterminateProgress(Stage stage, boolean indeterminate) {
        final var peer = WindowHelper.getPeer(stage);
        final long windowHandle = peer.getRawHandle();

        final var clsid = new CLSID("56FDF344-FD6D-11d0-958A-006097C9A090"); // from ShObjIdl.h
        final var taskbarListPointerRef = new PointerByReference();

        var hr = Ole32.INSTANCE.CoCreateInstance(clsid, null, WTypes.CLSCTX_SERVER,
                ITaskbarList3.IID_ITASKBARLIST3, taskbarListPointerRef);

        if (W32Errors.FAILED(hr)) {
            throw new RuntimeException("failed with code: " + hr.intValue());
        }

        final TaskbarList3 taskbarList = new TaskbarList3(taskbarListPointerRef.getValue());
        final var hwnd = new HWND(new Pointer(windowHandle));
        final int progressState = indeterminate ? ITaskbarList3.TBPF_INDETERMINATE : ITaskbarList3.TBPF_NOPROGRESS;
        hr = taskbarList.SetProgressState(hwnd, progressState);

        if (W32Errors.FAILED(hr)) {
            throw new RuntimeException("failed with code: " + hr.intValue());
        }
    }
}

Sample.java

package example;

import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.stage.Stage;

public final class Sample extends Application {

    private boolean indeterminateProgressState = false;

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

    @Override
    public void start(Stage primaryStage) throws Exception {
        final Button btn = new Button("Click me!");

        primaryStage.setScene(new Scene(btn));
        primaryStage.sizeToScene();
        primaryStage.show();

        btn.setOnAction(evt -> {
            indeterminateProgressState = !indeterminateProgressState;
            TaskbarPeer.setIndeterminateProgress(primaryStage, indeterminateProgressState);
        });
    }
}

我结合了这个答案使它工作,并获取了窗口句柄。 - SDIDSA

0

Windows 通过 COM 公开了这个功能。我相信对于您来说,“平面 DLL” 调用可能更容易,但如果您可以使用 COM,则可以完成此操作。COM 接口是 ITaskbarList3(还有一个从它继承的 ITaskbarList4)。http://msdn.microsoft.com/en-us/library/dd391692(VS.85).aspx 对其进行了文档化。SetProgressState 和 SetProgressValue 是您想要调用的方法。状态包括正常(绿色)、暂停(黄色)、错误(红色)、不确定(扫描绿色)和无。在 MSDN 页面上,一些社区成员添加了从 VB 和 C# 调用此 COM 组件的详细信息 - 这可能会帮助您弄清所需的 Java 设置和拆卸。


好的,我已经安装了javax.comm,但是文档很差。你有什么可以提供的吗? - user400935
抱歉,我的Java日子已经过去了,如果我曾经知道如何从Java调用COM方法,现在也已经忘记了。如果您需要关于传递给这些方法的内容的文档,我有调用它们的C#代码,因此我知道参数是什么。随时问吧。 - Kate Gregory
nativelibs4java似乎使用COM。 - rogerdpack

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接