微信、企业微信来消息打开窗口
本帖最后由 ArGuard 于 2023-1-29 12:35 编辑由来:
老是收到微信不弹窗,导致我错过很多消息。所以决定自己写个。
原理:
1、自动获取微信、企业微信,工作目录、存档目录、加密数据库文件(不知道这个词对不对)路径。
2、根据加密数据库文件的最后修改时间来判断有没有新消息。有就弹窗。作用:
来消息自动打开微信、企业微信窗口。
环境:
我用的是JDK13.0.2,应该没用到新特性。JDK\JRE 8及以上应该都能用。
使用:
只有在微信、企业微信打开时才能够正常使用、微信、企业微信退出、没有运行自动关闭。
PS:
因为热爱,差不多刚学完javase 代码比较捞,希望各位大佬提出宝贵建议。
还有就是我想问下,打开窗口这个功能有没有办法置顶呢?当时百度半天没找到这么实现
代码:
package cn.xuexiceshi.wechat.tools;
import java.io.IOException;
import java.util.Timer;
/**
* 微信来消息弹窗小工具
*@作者: 清风不踏雪
*@版本: v1.0
*@作用; 实现了微信、企业微信来消息弹窗,会直接打开程序窗口
*@实现原理:
* 通过CMD命令获取注册表中微信、企业微信的安装和存档信息。
* 再通过存档信息循环遍历,找出加密的数据库文件,
* 最后判断文件的最后修改时间,然后根据安装信息,弹窗。
*@感谢:
* 判断思路来源:(找不到链接了,是个批处理来着)作者:phenix
* 读取注册表方法来源:csdn,天真吴邪xie
* (学会了怎么使用 Runtime.getRuntime().exec() 执行CMD命令,并获取返回值)
*
*/
public class Main {
public static void main(String[] args) {
start();
}
/**
* 开始运行程序方法
*/
public static void start(){
//创建工具类
try {
//创建工具类
Tools weChat = new Tools();
Tools wXWork = new Tools(true);
//判断程序是否在运行
if((weChat.getWXStatus() || wXWork.getWXStatus())
&& (!weChat.getFile().isDirectory() || !wXWork.getFile().isDirectory())){
Timer timer = new Timer("WXTanChuan");//创建定时器
//weChat是微信、wXWork是企业微信;0表示等待0毫秒后开始执行;1000*3表示每隔3秒重复执行一次
timer.schedule(weChat, 0,1000 * 3);//开始执行任务
timer.schedule(wXWork, 0,1000 * 3);//开始执行任务
while (weChat.getWXStatus() || wXWork.getWXStatus()){
if(!wXWork.getWXStatus()){
wXWork.cancel();
}else if(!weChat.getWXStatus()){
weChat.cancel();
}
System.gc();
}
timer.cancel();
}
} catch (IOException e) {
throw new RuntimeException(e);
}
}
}
package cn.xuexiceshi.wechat.tools;
import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
import java.text.SimpleDateFormat;
import java.util.TimerTask;
public class Tools extends TimerTask {
//加密数据库文件名
final private String weChatTargetFile = "MSG0.db-wal";
final private String wXWorktargetFile = "message.db-wal";
//cmd查询注册表返回结果前缀
final private String workPathPrefix;
final private String filesSavePathPrefix;
final private String exeName;
//微信、企业微信的工作、存档的绝对路径
//默认路径是为了防止未安装程序,出现的异常
private String workPath;
private String filesSavePath;
//获取工作路径的CMD命令
private String getWorkPathCmd;
//获取存档路径的CMD命令
private String getFilesSavePathCmd;
//是否企业微信
private boolean isEnterprise;
//加密数据库文件的绝对路径
private File file;
//数据库最后修改的时间
private String date = "1970-01-01 08:00:00";
@Override
public void run() {
try {
String newDate = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(file.lastModified());
if(!date.equals(newDate) && getWXStatus()){
date = newDate;
Process wx = Runtime.getRuntime().exec(workPath + "\\" +exeName);
//System.out.println(exeName + "消息最新更新于:" + date );//测试、预留、备用
Thread.sleep(1000);
if(wx.isAlive()){
wx.destroy();
}
}
} catch (IOException e) {
throw new RuntimeException(e);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
/**
* 这个方法用来获取微信、企业为的工作和存档路径
* 实现原理为读取注册表的内容
*/
private void getPath(){
String[] workPath = getRunCmdResult(getWorkPathCmd);
String[] filesSavePath = getRunCmdResult(getFilesSavePathCmd);
//判断然后再给变量赋值,得到运行目录,和存档目录。
if(workPath != null && workPath.length >= 3 ){
if(workPath.equals(workPathPrefix)){
if(workPath.endsWith("exe")){
workPath = workPath.substring(0,workPath.lastIndexOf("\\"));
}
this.workPath = workPath;
}
}
if(filesSavePath != null && workPath.length >= 3){
if(filesSavePath.equals(filesSavePathPrefix)){
if(filesSavePath.endsWith("exe")){
filesSavePath = filesSavePath.substring(0,filesSavePath.lastIndexOf("\\"));
}
this.filesSavePath = filesSavePath;
}
}
}
/**
* 因为有些是改过默认存档位置的,所以直接从注册表获取方便。
* 这个方法用来获取当前系统用户的文档位置
* @Return 文档的位置
*/
private String getUsersDocumentsPath(){
String[] strs = getRunCmdResult("reg query \"HKEY_CURRENT_USER\\SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Explorer\\User Shell Folders\" | findstr \"Personal\"");
if(strs != null && strs.length >= 3){
if(strs.startsWith("%USERPROFILE%")){
return System.getProperty("user.home") + "\\Documents";
}
return strs;
}
return System.getProperty("user.home") + "\\Documents";
}
/**
* 这个方法用来执行CMD命令
* @Param cmd cmd命令,最好配合findstr命令使用,因为此方法只返回一行数据
* @return 执行结果,此方法只返回一行数据,命令不对、结果为空,返回null
*/
private String[] getRunCmdResult(String cmd){
try {
Process pos = Runtime.getRuntime().exec("cmd /c " + cmd);
InputStreamReader inp = new InputStreamReader(pos.getInputStream());
BufferedReader buf = new BufferedReader(inp);
String str;
if((str = buf.readLine()) != null){
str = str.trim();
if(pos.isAlive()){
pos.destroy();
}
return str.split(" ");
}
if(pos.isAlive()){
pos.destroy();
}
return null;
} catch (IOException e) {
throw new RuntimeException(e);
}
}
/**
* 获取微信、企业微信运行状态
* @return true表示正在运行,false表示未在运行
*/
public boolean getWXStatus(){
return getExeStatus(exeName);
}
/**
* 获取程序的运行状态
* @param name 程序名字
* @return true表示正在运行,false表示未在运行
*/
private boolean getExeStatus(String name){
String[] strings = getRunCmdResult("tasklist | findstr \" "+ exeName +"\"");
if(strings != null){
return true;
}
return false;
}
/**
* 获取微信加密数据库文件的绝对路径
* @param filePath 微信的安装目录
*/
privatevoid getWeChatTargetFile(String filePath){
File workDir = new File(filePath);
if(workDir.isDirectory()){
File[] files = workDir.listFiles();
for (File file1 : files){
if(file1.isDirectory()){
getWeChatTargetFile(file1.getAbsolutePath());
}else if(String.valueOf(file1.getName()).startsWith("wxid_")){
getWeChatTargetFile(file1.getAbsolutePath());
}else if("Msg".equals(String.valueOf(file1.getName()))){
getWeChatTargetFile(file1.getAbsolutePath());
}else if("Multi".equals(String.valueOf(file1.getName()))){
getWeChatTargetFile(file1.getAbsolutePath());
}else if(weChatTargetFile.equals(String.valueOf(file1.getName()))){
file = file1;
return;
}
}
}
}
/**
* 用来获取企业微信的加密数据库文件的绝对路径
* @param filePath 企业微信的安装目录
*/
privatevoid getWXWorkTargetFile(String filePath){
File workDir = new File(filePath);
if(workDir.isDirectory()){
File[] files = workDir.listFiles();
for (File file1 : files){
boolean b = false;
//判断文件夹名字是不是数字
try{
Long.valueOf(String.valueOf(file1.getName()));
b = true;
}catch (NumberFormatException e){
}
if(file1.isDirectory()){
getWXWorkTargetFile(file1.getAbsolutePath());
}else if(b){
getWXWorkTargetFile(file1.getAbsolutePath());
}else if("Data".equals(String.valueOf(file1.getName()))){
getWXWorkTargetFile(file1.getAbsolutePath());
}else if(wXWorktargetFile.equals(String.valueOf(file1.getName()))){
file = file1;
return;
}
}
}
}
/**
* 无参构造方法
*/
public Tools() throws IOException{
this(false);
}
/**
* 有参构造
* @param isEnterprise true表示是企业微信,false表示是微信
*/
public Tools(boolean isEnterprise) throws IOException {
this.isEnterprise = isEnterprise;
exeName = isEnterprise?"WXWork.exe":"WeChat.exe";
String s = "reg query HKEY_CURRENT_USER\\SOFTWARE\\Tencent\\" + (isEnterprise?"WXWork":"WeChat");
getWorkPathCmd = s + " | findstr \"" + (isEnterprise?"Executable":"InstallPath") +"\"";
getFilesSavePathCmd = s + " | findstr \"" + (isEnterprise?"DataLocationPath":"FileSavePath") +"\"";
filesSavePath = getUsersDocumentsPath() + "\\" + (isEnterprise?"WXWork":"WeChat Files");
if(isEnterprise){
workPathPrefix = "Executable";
filesSavePathPrefix = "DataLocationPath";
workPath = "C:\\Program Files\\WXWork";//默认安装位置
getPath();
file = new File(filesSavePath);
getWXWorkTargetFile(filesSavePath);
}else{
workPathPrefix = "InstallPath" ;
filesSavePathPrefix = "FileSavePath";
workPath = "C:\\Program Files (x86)\\Tencent\\WeChat";//默认安装位置
getPath();
file = new File(filesSavePath);
getWeChatTargetFile(filesSavePath);
}
}
public String getWorkPath() {
return workPath;
}
public String getFilesSavePath() {
return filesSavePath;
}
public File getFile() {
return file;
}
}
ArGuard 发表于 2023-2-21 14:19
把贴子更新了下在审核
java.lang.NullPointerException: Cannot read the array length because "workPath" is null
at cn.xuexiceshi.wechat.tools.Tools.getPath(Tools.java:104)
at cn.xuexiceshi.wechat.tools.Tools.<init>(Tools.java:273)
at cn.xuexiceshi.wechat.tools.Tools.<init>(Tools.java:246)
at cn.xuexiceshi.wechat.tools.Main.start(Main.java:35)
at cn.xuexiceshi.wechat.tools.Main.main(Main.java:25)
at java.base/jdk.internal.reflect.DirectMethodHandleAccessor.invoke(DirectMethodHandleAccessor.java:104)
at java.base/java.lang.reflect.Method.invoke(Method.java:578)
at com.exe4j.runtime.LauncherEngine.launch(LauncherEngine.java:84)
at com.exe4j.runtime.WinLauncher.main(WinLauncher.java:94)
这个报错是怎么回事,我看你回复了,但是帖子点进去是删了 本帖最后由 zim954520huahua 于 2023-7-19 16:12 编辑
win11上单独微信不支持(企业微信没测),我调试了(执行截图,以及新消息时间截图),可能因为微信在win11上不是来消息后更新MSG0.db-wal文件(文件修改时间没变)。后面再次看来一下文件修改时间,有改变,但是时间变更的频率很低。由于是在工作时间抽空看了一下,暂时到此,有时间再看看,有结果再来回复!
----
7/19日
我用其他的思路来解决了win11上无法使用的问题(本机),通过检测wechat的托盘图标,有新消息时发送点击事件。因为java没有此类直接操作托盘图标的方法,我通过读取屏幕坐标颜色变化来进行。连续检测10次,每次间隔200ms,如果10次里有至少一次变化说明有新的消息。 {:1_921:} 基础不错 现在用java来写这些很少了 这个是什么原理?我看着像是检测微信数据库的修改时间,如果被修改了就弹出窗口?那消息内容呢? /│\云。 发表于 2023-1-29 15:10
基础不错 现在用java来写这些很少了
感谢大佬夸奖,新人不太懂,那Java现在用来写啥? ccwuax 发表于 2023-1-29 15:49
这个是什么原理?我看着像是检测微信数据库的修改时间,如果被修改了就弹出窗口?那消息内容呢?
就是检测修改时间然后打开窗口啊,消息内容要解密,我没有这本事解密啊。主要代码都写在了自动获取微信、企业微信的安装目录、存档目录、数据库文件路径上了。检测打开微信、企业微信代码不多。 ArGuard 发表于 2023-1-29 15:50
感谢大佬夸奖,新人不太懂,那Java现在用来写啥?
现在很少调用本地化了 做窗体都没了 虽然都能做,现在是服务后端 数据分析 类似电商这种大型服务后端 各种分布式等等 /│\云。 发表于 2023-1-29 15:57
现在很少调用本地化了 做窗体都没了 虽然都能做,现在是服务后端 数据分析 类似电商这种大型服务后端 各 ...
{:301_1009:}原来是搞后端去了,那现在这种本地的小工具都用啥来写? ArGuard 发表于 2023-1-29 16:03
原来是搞后端去了,那现在这种本地的小工具都用啥来写?
c# ,python 如果基础有的话可以看看 类似aardio这种混合语言编写小工具 里面能调用这种C++、C#、Java、Python 语言 当然还有其他很多就是 看个人喜好 /│\云。 发表于 2023-1-29 16:15
c# ,python 如果基础有的话可以看看 类似aardio这种混合语言编写小工具 里面能调用这种C++、C#、Java、Py ...
好的。感谢大佬 dabenchong 发表于 2023-1-29 16:39
這個 比較沒有隱私把
也不是非得用呀,我是有需求才要用。微信企业微信来消息不弹窗的呀,就是有个声,然后状态栏图标闪烁。你在干别的事情时,很有可能没注意到别人给你发了消息,为了及时知道有人给我发了消息。这个才是这个小工具主要的目的。况且都登录微信在电脑了。来消息也省去了自己打开微信窗口这一步了,而且离开电脑一般不都是自己直接锁屏或者退出微信吗?应该没那么“没有隐私”吧