Compare commits

..

6 Commits

Author SHA1 Message Date
4988cdc31e feat: GUI单例模式 2024-09-26 22:18:55 +08:00
4e1af001c2 fix: 修复更新后版本号显示为配置文件内旧版本号的错误 2024-09-26 21:14:48 +08:00
octopus yan
ef7f4461c2
Release v1.0.1
fix Release v1.0.1
2024-09-26 20:47:52 +08:00
6670dc7210 ci: Modify the name of the file to be uploaded 2024-09-26 20:43:56 +08:00
octopus yan
98c4ec2cd8
Release v1.0.1
Release v1.0.1
2024-09-26 20:36:15 +08:00
7e0ec8b598 docs: version 1.0.1 2024-09-26 20:31:06 +08:00
5 changed files with 99 additions and 8 deletions

View File

@ -118,7 +118,7 @@ jobs:
- name: Build with Maven
run: |
mvn clean package -f pom.xml
mkdir zipball && cp target/*-windows.zip zipball
mkdir zipball && cp target/*.zip zipball
- name: Upload AListGUI to Github
uses: actions/upload-artifact@v4

View File

@ -6,7 +6,7 @@
<groupId>cn.octopusyan</groupId>
<artifactId>alist-gui</artifactId>
<version>1.0.0</version>
<version>1.0.1</version>
<name>alist-gui</name>
<organization>
@ -262,7 +262,7 @@
</configuration>
<executions>
<execution>
<id>bundling-for-windows</id>
<id>windows</id>
<phase>package</phase>
<goals>
<goal>package</goal>
@ -278,7 +278,7 @@
</configuration>
</execution>
<execution>
<id>bundling-for-windows-nojre</id>
<id>windows-nojre</id>
<phase>package</phase>
<goals>
<goal>package</goal>

View File

@ -18,11 +18,12 @@ import lombok.Getter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.ProxySelector;
import java.io.*;
import java.net.*;
import java.net.http.HttpClient;
import java.util.Objects;
import java.util.UUID;
import java.util.concurrent.CountDownLatch;
public class Application extends javafx.application.Application {
private static final Logger logger = LoggerFactory.getLogger(Application.class);
@ -32,6 +33,10 @@ public class Application extends javafx.application.Application {
@Override
public void init() {
logger.info("application init ...");
// 单例模式检查
makeSingle();
// 初始化客户端配置
ConfigManager.load();
@ -83,6 +88,7 @@ public class Application extends javafx.application.Application {
primaryStage.setTitle(String.format("%s v%s", Constants.APP_TITLE, Constants.APP_VERSION));
Scene scene = Context.initScene();
primaryStage.setScene(scene);
primaryStage.show();
// 静默启动
@ -113,4 +119,84 @@ public class Application extends javafx.application.Application {
Platform.exit();
System.exit(0);
}
private static final int SINGLE_INSTANCE_LISTENER_PORT = 9009;
private static final String SINGLE_INSTANCE_FOCUS_MESSAGE = "focus";
private static final String instanceId = UUID.randomUUID().toString();
/**
* 我们在聚焦现有实例之前定义一个暂停
* 因为有时启动实例的命令行或窗口
* 可能会在第二个实例执行完成后重新获得焦点
* 所以我们在聚焦原始窗口之前引入了一个短暂的延迟
* 以便原始窗口可以保留焦点
*/
private static final int FOCUS_REQUEST_PAUSE_MILLIS = 500;
/**
* 单实例检测
*
* @see <a href='https://www.cnblogs.com/shihaiming/p/13553278.html'>JavaFX单实例运行应用程序</url>
*/
public static void makeSingle() {
CountDownLatch instanceCheckLatch = new CountDownLatch(1);
Thread instanceListener = new Thread(() -> {
try (ServerSocket serverSocket = new ServerSocket(SINGLE_INSTANCE_LISTENER_PORT, 10)) {
instanceCheckLatch.countDown();
while (true) {
try (
Socket clientSocket = serverSocket.accept();
BufferedReader in = new BufferedReader(
new InputStreamReader(clientSocket.getInputStream()))
) {
String input = in.readLine();
logger.info(STR."Received single instance listener message: \{input}");
if (input.startsWith(SINGLE_INSTANCE_FOCUS_MESSAGE) && primaryStage != null) {
//noinspection BusyWait
Thread.sleep(FOCUS_REQUEST_PAUSE_MILLIS);
Platform.runLater(() -> {
logger.info(STR."To front \{instanceId}");
primaryStage.setIconified(false);
primaryStage.show();
primaryStage.toFront();
});
}
} catch (IOException e) {
logger.error("Single instance listener unable to process focus message from client");
}
}
} catch (java.net.BindException b) {
logger.error("SingleInstanceApp already running");
try (
Socket clientSocket = new Socket(InetAddress.getLocalHost(), SINGLE_INSTANCE_LISTENER_PORT);
PrintWriter out = new PrintWriter(new OutputStreamWriter(clientSocket.getOutputStream()))
) {
logger.info("Requesting existing app to focus");
out.println(STR."\{SINGLE_INSTANCE_FOCUS_MESSAGE} requested by \{instanceId}");
} catch (IOException e) {
logger.error("", e);
}
logger.info(STR."Aborting execution for instance \{instanceId}");
Platform.exit();
} catch (Exception e) {
logger.error("", e);
} finally {
instanceCheckLatch.countDown();
}
}, "instance-listener");
instanceListener.setDaemon(true);
instanceListener.start();
try {
instanceCheckLatch.await();
} catch (InterruptedException e) {
//noinspection ResultOfMethodCallIgnored
Thread.interrupted();
}
}
}

View File

@ -114,12 +114,13 @@ public class RootController extends BaseController<RootViewModel> {
@Override
public void initViewAction() {
closeIcon.addEventHandler(MouseEvent.MOUSE_CLICKED, _ -> {
Platform.setImplicitExit(!ConfigManager.closeToTray());
if (ConfigManager.closeToTray()) {
SystemTrayManager.show();
} else {
SystemTrayManager.hide();
Platform.exit();
}
Platform.setImplicitExit(!ConfigManager.closeToTray());
getWindow().close();
});
minimizeIcon.addEventHandler(MouseEvent.MOUSE_CLICKED, _ -> getWindow().setIconified(true));

View File

@ -278,6 +278,10 @@ public class ConfigManager {
}
public static String guiVersion() {
// 覆盖配置文件读取的版本号
if (!Constants.APP_VERSION.equals(gui().getVersion())) {
guiVersion(Constants.APP_VERSION);
}
return gui().getVersion();
}