今天共享一个局域网唤醒Windows,Flutter局域网关闭Windows的方案。

这个方案是我根据我的需求而定制,并不一定适用所有人,仅供参考。 由于我有一个局域网NAS,装置的是Windows Server 2019,用作家庭的电影文件存储,跨设备传输,挂载了2个机械盘,耗电是25-30W,平常没有运用的时候,习气将它关机,但是由于放在柜子内部,而且没有外接显卡,之前都是手动开机,运用其他Windows连入进行关机,非常的不方便,所以自己做了一套开关机方案。


【开机方案】

关于开机方案,方案有很多种。 1.网卡唤醒 2.主板通电唤醒 3.主板开关跳线

第一种方案如同是支撑外网的,由于我没有外网开机需求,所以这次没做,它的原理是通过网络给指定网卡发送数据包,收到数据包的网卡调用电脑开机。 第三种方案需要购买设备,如果会硬件开发那是最优方案,这次我没选。

Flutter局域网关闭Windows

我挑选的第二种方案,我购买了一个 小米智能插座,大概是几十块钱,然后在电脑主板Bios里边设置 通电开机,在米家里边进行通电操作,这样就完结了 长途开机的功能。

【关机方案】

关于关机方案,也有几种 1.小米插座直接断电(直接断电,伤电脑) 2.开柜子按关机键(太麻烦了) 3.长途桌面进入体系,点关机(…)

最后我挑选了运用软件关机的方案,趁便也 宣扬一波flutter。 设计的方案是,在方针客户端上面发动一个进程,监听某个端口,然后用手机连入端口发送指令,这样就可以完结关机和其他操作。

【技能方案】

一开始预备Windows上面运用WPF开发,这个我比较了解,手机端运用Flutter。 后边决定用flutter 完全开发Windows+Android。终究也是满足了我的需求。

【网络协议】挑选

网络协议方面,首先是抛弃TCP,然后我运用的是Http监听,觉得结构有点重了(Http协议优势是,flutter端也不必写了,直接浏览器访问一个网址就可以,但是后边我发现由于NAS主机没有固定IP,IP地址会变化,只有主机名,在手机上Linux核心的体系如同原生就不支撑解析hostname,考虑到今后这个功能或许会给多个电脑运用,终究抛弃http协议),最后挑选了UDP通讯。

【Windows端】

Flutter局域网关闭Windows

发动后UDP监听19999端口,当收到固定音讯包的音讯时,将自己的信息封装后发给对方主机,当收到关机指令,运用MethodChannel 调用Windows cpp里边的函数进行关机

RawDatagramSocket.bind(InternetAddress.anyIPv4, 19999).then(
      (RawDatagramSocket udpSocket) {
        udpSocket.forEach((RawSocketEvent event) async {
          if (event == RawSocketEvent.write) {
            state.value = "服务已发动";
          }
          if (event == RawSocketEvent.read) {
            Datagram? dg = udpSocket.receive();
            if (dg != null) {
              //dg.data.forEach((x) => print(x));
              if (dg.data.first == 0) {
                //播送音讯,回发自己的计算机信息
                List<int> data = const Utf8Encoder()
                    .convert(windowsDeviceInfo?.computerName ?? "");
                udpSocket.send(data, dg.address, dg.port);
              } else if (dg.data.first == 1) {
                //关机
                state.value = "已收到开机指令";
              } else if (dg.data.first == 2) {
                //关机
                state.value = "已收到关机指令";
                platform.invokeMethod("CloseWindows");
              } else {
                state.value = "已收到指令:${const Utf8Decoder().convert(dg.data)}";
              }
            }
          }
        });
      },
    );
 #include "flutter/method_channel.h"
  #include "flutter/standard_method_codec.h"
  void configMethodChannel(flutter::FlutterEngine *engine)
{
        const std::string test_channel("DMSkin.Channel");
        const flutter::StandardMethodCodec &codec = flutter::StandardMethodCodec::GetInstance();
        flutter::MethodChannel method_channel_(engine->messenger(), test_channel, &codec);
        method_channel_.SetMethodCallHandler([](const auto &call, auto result)
                                       {
            std::cout << "Inside method call" << std::endl;
            if (call.method_name().compare("CloseWindows") == 0) {
            std::cout << "Close window message recieved!" << std::endl;
            system("shutdown -s -t 3");
            std::cout << "Close window Success!" << std::endl;
         result->Success();
    }
         else if (call.method_name().compare("goToNativeScanPage") == 0) {
            std::cout << "goToNativeScanPage!" << std::endl;
        result->Success();
    } });
}

【手机端】

Flutter局域网关闭Windows

发动后运用bind绑定本机端口,由于是any + 0,所有体系会分配一个随机的端口号,这个用来发送数据,不需要知道详细的端口号,所以无所谓。数据包比较简单,只用了一个 int 长度。

void init() async {
    port = 19999;
    socket = await RawDatagramSocket.bind(InternetAddress.anyIPv4, 0);
    if (socket != null) {
      socket!.broadcastEnabled = true;
      socket!.listen((RawSocketEvent event) {
        if (event == RawSocketEvent.write) {
          sendBroadcast();
        }
        else if (event == RawSocketEvent.read) {
          Datagram? datagram = socket!.receive();
          if (datagram != null) {
            String name = const Utf8Decoder().convert(datagram.data);
            if (!data.any(
                (element) => element.ip!.address == datagram.address.address)) {
              Windows windows = Windows()
                ..name = name
                ..ip = datagram.address;
              data.add(windows);
              change(null, status: RxStatus.success());
            }
          }
        }
      });
      change(null, status: RxStatus.success());
    }
  }
  /* 发送播送 - 在线 */
  void sendBroadcast() {
    if (socket != null && port != 0) {
      socket!.send([0], InternetAddress("255.255.255.255"), port);
    }
  }
  /* 关闭电脑 */
  void sendShutDown(InternetAddress ip) async {
    socket.send([2], InternetAddress(ip.address), port);
  }

【开机自启】遇到的问题

shell:startup 不登录桌面,程序不翻开, 终究运用的是: 1.右键点击此电脑图标,在弹出菜单中挑选“办理”菜单项。 2.然后在翻开的计算机办理窗口中,找到“任务方案程序”菜单项。 3.右键,点击创建基本任务,挑选开机发动一个进程。 4.修正这个方案,在设置中修正为不登录桌面就发动,输入电脑的开机密码。

【广域网扩展】

这套方案中,只需要将端口映射到公网中,将Flutter端的播送改为固定IP,就可以完成外网关机,当然你的手机必须要在公网网段上,否则NAT地址无法回发到真实的手机19999端口上。

Github源码