Squashed commit of the following: commit 02c7556cdadf8de1ef3c54d2831920927cadbe30 Merge: 6bfa42e0 991003e6 Author: zkk <1007518571@qq.com> Date: Tue Dec 31 15:07:54 2024 +0800 Merge commit '991003e6cbea335eca73d3783aa1837059614724' into release commit 991003e6cbea335eca73d3783aa1837059614724 Author: zkk <1007518571@qq.com> Date: Tue Dec 31 15:03:34 2024 +0800 优化完整 简体中文和繁体中文的翻译 commit 1a177e90d09b9b9949bff2a1e3c6b12173420620 Author: zkk <1007518571@qq.com> Date: Tue Dec 31 14:10:03 2024 +0800 优化排除对象的英文语法错误 commit e8d509cb6c2883b1fadb5ab9f9ca658e61849055 Author: zkk <1007518571@qq.com> Date: Fri Dec 27 16:40:52 2024 +0800 优化耗材检测显示内容 commit 1b7670485a918cb334119175777525f768e670be Author: zkk <1007518571@qq.com> Date: Fri Dec 27 14:56:25 2024 +0800 完善断料自动切头功能描述 commit ecc0c3dd16442497e56f5f8ab3a0c6be00bd180e Author: zkk <1007518571@qq.com> Date: Fri Dec 27 13:40:01 2024 +0800 实现弹窗翻译功能 commit bc6d60fa183af50b7b16fea685139ca9d5df90ea Author: zkk <1007518571@qq.com> Date: Fri Dec 27 09:53:16 2024 +0800 优化不合理名称和不合理大小写 commit 2e650926bdba1c65baa506be1b1dd09621e8d8ea Author: zkk <1007518571@qq.com> Date: Thu Dec 26 15:06:11 2024 +0800 修复10寸屏幕偏移值微调页面出界问题 commit c481b551f777a3a6cadf7be3bfcab38448b51694 Author: zkk <1007518571@qq.com> Date: Wed Dec 25 15:28:26 2024 +0800 优化移轴设置页面没有返回按钮的问题 commit 2b3c9936bd693c8fc265d6352799f382680f4920 Author: zkk <1007518571@qq.com> Date: Wed Dec 25 10:05:04 2024 +0800 删除移动设置页面中轴反转的选项 commit 5e37d59497c54086d3f1e215cbabc03254857a58 Author: zkk <1007518571@qq.com> Date: Wed Dec 25 09:05:03 2024 +0800 优化选择语言标题 commit 5214c3697baac78bd0a465e33d6f4f3c1e30862e Author: zkk <1007518571@qq.com> Date: Tue Dec 24 10:57:41 2024 +0800 增加设置hostname功能 commit 3709a7465cc2e0d677ecc9979fb18144f7e0ec33 Author: zkk <1007518571@qq.com> Date: Tue Dec 24 10:43:46 2024 +0800 增加工厂设置中打包模式 commit 0c2a2618beb68d50601e433beef06122fb0ffd40 Author: zkk <1007518571@qq.com> Date: Mon Dec 23 16:25:37 2024 +0800 实现恢复出厂设置功能 commit e3a38f85613d53a4a063e835c34c701196010c6e Author: zkk <1007518571@qq.com> Date: Mon Dec 23 15:48:17 2024 +0800 一点格式化 commit 4e9e452e8bed4b53eaaae8f0a69142de8aa50611 Author: zkk <1007518571@qq.com> Date: Sat Dec 21 14:20:22 2024 +0800 优化探针偏移值校准时如果当前激活为第一个喷头时不会重复激活喷头 commit 3b85e8e8d59b0d4eae75f80d9e637fe9dc272cbe Merge: 882850db 10ec2029 Author: zkk <1007518571@qq.com> Date: Wed Dec 18 17:26:08 2024 +0800 Merge branch 'develop' of https://server.creatbot.com/Gitea/CreatBot/CreatBotKlipperScreen into develop commit 882850dbde648598e5f91281101d0ea01fd56d56 Author: zkk <1007518571@qq.com> Date: Wed Dec 18 17:21:16 2024 +0800 补充双喷头偏移校准页面二维码图片 commit 10ec2029eb73e66874d5a0ec492c016480641110 Author: ruipeng <1041589370@qq.com> Date: Wed Dec 18 17:17:18 2024 +0800 增加自动切换喷头开关选项 commit 3d6eed9d9526b77472ba7df29014b5768c594026 Author: zkk <1007518571@qq.com> Date: Mon Dec 16 16:33:34 2024 +0800 喷头偏移值校准功能的实现 commit 40ecbb3ea4827c9bac1aa271cccc958c94c1114a Author: zkk <1007518571@qq.com> Date: Fri Dec 13 09:20:00 2024 +0800 z探针校准时指定默认激活工具头为第一个头 commit bce3caa409618ef5bc2067865940d14a1441c0f8 Author: zkk <1007518571@qq.com> Date: Fri Dec 13 09:19:19 2024 +0800 优化z探针校准时候移动过慢问题 commit 6bfa42e036a521c8ff7db1bf8ccb65500eabb6ea Merge: 1a87ced3 4f3aa9aa Author: zkk <1007518571@qq.com> Date: Thu Dec 12 10:33:58 2024 +0800 Merge commit '4f3aa9aa4c581ae9e7a740bd37f9e80ba064c27f' into release commit 4f3aa9aa4c581ae9e7a740bd37f9e80ba064c27f Merge: e3fd413d 1a69b518 Author: zkk <1007518571@qq.com> Date: Thu Dec 12 10:30:00 2024 +0800 Merge branch 'develop' of https://server.creatbot.com/Gitea/CreatBot/CreatBotKlipperScreen into develop commit 1a69b5180de733f15550a1294a7221d6070306a0 Author: ruipeng <1041589370@qq.com> Date: Wed Dec 11 11:44:36 2024 +0800 新增D600pro2、D1000的V0版机型 commit e3fd413d6256414441e9fe653c1132b1799a5cb2 Author: zkk <1007518571@qq.com> Date: Tue Dec 10 14:34:13 2024 +0800 修复打印时没有修改z偏移值 仍提示保存z偏移的按钮的bug commit 594fb668fe94fe907c028bf65ecced43ea8660cb Author: zkk <1007518571@qq.com> Date: Tue Dec 10 14:25:04 2024 +0800 设置自适应调平选项默认为关闭状态 commit 1a87ced3f5725569a6b9a7ee5f5250044d01d852 Merge: 629416d1 6064a6e1 Author: zkk <1007518571@qq.com> Date: Fri Nov 29 10:18:32 2024 +0800 Merge branch 'develop' into release # Conflicts: # panels/advanced.py # panels/nozzle_offset.py # panels/offset_fine_tune.py # panels/zcalibrate.py
398 lines
12 KiB
Python
398 lines
12 KiB
Python
#!/usr/bin/python
|
|
|
|
import threading
|
|
import json
|
|
import logging
|
|
|
|
import gi
|
|
import websocket
|
|
|
|
gi.require_version("Gtk", "3.0")
|
|
from gi.repository import GLib
|
|
from ks_includes.KlippyGcodes import KlippyGcodes
|
|
|
|
|
|
class KlippyWebsocket(threading.Thread):
|
|
_req_id = 0
|
|
connected = False
|
|
connecting = True
|
|
callback_table = {}
|
|
reconnect_count = 0
|
|
max_retries = 4
|
|
|
|
def __init__(self, callback, host, port, api_key):
|
|
threading.Thread.__init__(self)
|
|
self._wst = None
|
|
self.ws_url = None
|
|
self._callback = callback
|
|
self.klippy = MoonrakerApi(self)
|
|
self.ws = None
|
|
self.closing = False
|
|
self.host = host
|
|
self.port = port
|
|
self.header = {"x-api-key": api_key} if api_key else {}
|
|
self.api_key = api_key
|
|
|
|
@property
|
|
def _url(self):
|
|
return f"{self.host}:{self.port}"
|
|
|
|
@property
|
|
def ws_proto(self):
|
|
return "wss" if int(self.port) in {443, 7130} else "ws"
|
|
|
|
def initial_connect(self):
|
|
if self.connect() is not False:
|
|
GLib.timeout_add_seconds(10, self.reconnect)
|
|
|
|
def reconnect(self):
|
|
if self.reconnect_count > self.max_retries:
|
|
logging.debug("Stopping reconnections")
|
|
self.connecting = False
|
|
GLib.idle_add(self._callback['on_cancel'])
|
|
return False
|
|
return self.connect()
|
|
|
|
def connect(self):
|
|
if self.connected:
|
|
logging.debug("Already connected")
|
|
return False
|
|
logging.debug("Attempting to connect")
|
|
self.reconnect_count += 1
|
|
|
|
self.ws_url = f"{self.ws_proto}://{self._url}/websocket?token={self.api_key}"
|
|
self.ws = websocket.WebSocketApp(
|
|
self.ws_url,
|
|
on_close=self.on_close,
|
|
on_error=self.on_error,
|
|
on_message=self.on_message,
|
|
on_open=self.on_open,
|
|
header=self.header
|
|
)
|
|
self._wst = threading.Thread(target=self.ws.run_forever, daemon=True)
|
|
try:
|
|
logging.debug("Starting websocket thread")
|
|
self._wst.start()
|
|
except Exception as e:
|
|
logging.debug(f"Error starting web socket {e}")
|
|
return True
|
|
return False
|
|
|
|
def close(self):
|
|
logging.debug("Closing websocket")
|
|
self.closing = True
|
|
self.connecting = False
|
|
if self.ws is not None:
|
|
self.ws.keep_running = False
|
|
self.ws.close()
|
|
|
|
def on_message(self, *args):
|
|
message = args[1] if len(args) == 2 else args[0]
|
|
response = json.loads(message)
|
|
if "id" in response and response['id'] in self.callback_table:
|
|
args = (response,
|
|
self.callback_table[response['id']][1],
|
|
self.callback_table[response['id']][2],
|
|
*self.callback_table[response['id']][3])
|
|
GLib.idle_add(self.callback_table[response['id']][0], *args, priority=GLib.PRIORITY_HIGH_IDLE)
|
|
self.callback_table.pop(response['id'])
|
|
return
|
|
|
|
if "method" in response and "on_message" in self._callback:
|
|
args = (response['method'], response['params'][0] if "params" in response else {})
|
|
GLib.idle_add(self._callback['on_message'], *args, priority=GLib.PRIORITY_HIGH_IDLE)
|
|
if self.closing:
|
|
timer = threading.Timer(2, self.ws.close)
|
|
timer.start()
|
|
return
|
|
|
|
def send_method(self, method, params=None, callback=None, *args):
|
|
if not self.connected or self.closing:
|
|
return False
|
|
if params is None:
|
|
params = {}
|
|
|
|
self._req_id += 1
|
|
if callback is not None:
|
|
self.callback_table[self._req_id] = [callback, method, params, [*args]]
|
|
|
|
data = {
|
|
"jsonrpc": "2.0",
|
|
"method": method,
|
|
"params": params,
|
|
"id": self._req_id
|
|
}
|
|
self.ws.send(json.dumps(data))
|
|
return True
|
|
|
|
def on_open(self, *args):
|
|
logging.info("Moonraker Websocket Open")
|
|
self.connected = True
|
|
self.connecting = False
|
|
self.reconnect_count = 0
|
|
if "on_connect" in self._callback:
|
|
GLib.idle_add(self._callback['on_connect'], priority=GLib.PRIORITY_HIGH_IDLE)
|
|
|
|
def on_close(self, *args):
|
|
# args: ws, status, message
|
|
# sometimes ws is not passed due to bugs
|
|
if len(args) == 3:
|
|
status = args[1]
|
|
message = args[2]
|
|
else:
|
|
status = args[0]
|
|
message = args[1]
|
|
if message is not None:
|
|
logging.info(f"{status} {message}")
|
|
if not self.connected:
|
|
logging.debug("Connection already closed")
|
|
return
|
|
if "on_close" in self._callback:
|
|
GLib.idle_add(self._callback['on_close'], priority=GLib.PRIORITY_HIGH_IDLE)
|
|
logging.info("Moonraker Websocket Closed")
|
|
self.connected = False
|
|
|
|
@staticmethod
|
|
def on_error(*args):
|
|
error = args[1] if len(args) == 2 else args[0]
|
|
logging.debug(f"Websocket error: {error}")
|
|
|
|
|
|
class MoonrakerApi:
|
|
def __init__(self, ws):
|
|
self._ws = ws
|
|
|
|
def emergency_stop(self):
|
|
logging.info("Sending printer.emergency_stop")
|
|
return self._ws.send_method(
|
|
"printer.emergency_stop"
|
|
)
|
|
|
|
def gcode_script(self, script, callback=None, *args):
|
|
logging.debug(f"Sending printer.gcode.script: {script}")
|
|
return self._ws.send_method(
|
|
"printer.gcode.script",
|
|
{"script": script},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def get_file_dir(self, path='gcodes', callback=None, *args):
|
|
logging.debug(f"Sending server.files.directory {path}")
|
|
return self._ws.send_method(
|
|
"server.files.list",
|
|
{"path": path},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def get_file_list(self, callback=None, *args):
|
|
logging.debug("Sending server.files.list")
|
|
return self._ws.send_method(
|
|
"server.files.list",
|
|
{},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def get_dir_info(self, callback=None, directory='gcodes', *args):
|
|
logging.debug(f"Sending server.files.get_directory {directory}")
|
|
return self._ws.send_method(
|
|
"server.files.get_directory",
|
|
{"path": directory},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def get_file_metadata(self, filename, callback=None, *args):
|
|
return self._ws.send_method(
|
|
"server.files.metadata",
|
|
{"filename": filename},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def get_database_list(self, callback=None, *args):
|
|
return self._ws.send_method(
|
|
"server.database.list",
|
|
{},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def add_database_item(self, namespace, key=None, val=None, callback=None, *args):
|
|
return self._ws.send_method(
|
|
"server.database.post_item",
|
|
{"namespace": namespace,
|
|
"key": key,
|
|
"value": val},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def get_database_item(self, namespace, key=None, callback=None, *args):
|
|
return self._ws.send_method(
|
|
"server.database.get_item",
|
|
{"namespace": namespace,
|
|
"key": key},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def del_database_item(self, namespace, key=None, callback=None, *args):
|
|
return self._ws.send_method(
|
|
"server.database.delete_item",
|
|
{"namespace": namespace,
|
|
"key": key},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def reset_job_history_totals(self, callback=None, *args):
|
|
return self._ws.send_method(
|
|
"server.history.reset_totals",
|
|
{},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def del_all_job_history(self, callback=None, *args):
|
|
return self._ws.send_method(
|
|
"server.history.delete_job",
|
|
{"all": "true"},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def object_subscription(self, updates):
|
|
logging.debug("Sending printer.objects.subscribe")
|
|
return self._ws.send_method(
|
|
"printer.objects.subscribe",
|
|
updates
|
|
)
|
|
|
|
def power_device_off(self, device, callback=None, *args):
|
|
logging.debug(f"Sending machine.device_power.off: {device}")
|
|
return self._ws.send_method(
|
|
"machine.device_power.off",
|
|
{device: False},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def power_device_on(self, device, callback=None, *args):
|
|
logging.debug("Sending machine.device_power.on {device}")
|
|
return self._ws.send_method(
|
|
"machine.device_power.on",
|
|
{device: False},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def print_cancel(self, callback=None, *args):
|
|
logging.debug("Sending printer.print.cancel")
|
|
return self._ws.send_method(
|
|
"printer.print.cancel",
|
|
{},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def print_pause(self, callback=None, *args):
|
|
logging.debug("Sending printer.print.pause")
|
|
return self._ws.send_method(
|
|
"printer.print.pause",
|
|
{},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def print_resume(self, callback=None, *args):
|
|
logging.debug("Sending printer.print.resume")
|
|
return self._ws.send_method(
|
|
"printer.print.resume",
|
|
{},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def print_start(self, filename, callback=None, *args):
|
|
logging.debug("Sending printer.print.start")
|
|
return self._ws.send_method(
|
|
"printer.print.start",
|
|
{
|
|
"filename": filename
|
|
},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def set_bed_temp(self, target, callback=None, *args):
|
|
logging.debug(f"Sending set_bed_temp: {KlippyGcodes.set_bed_temp(target)}")
|
|
return self._ws.send_method(
|
|
"printer.gcode.script",
|
|
{
|
|
"script": KlippyGcodes.set_bed_temp(target)
|
|
},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def set_heater_temp(self, heater, target, callback=None, *args):
|
|
logging.debug(f"Sending heater {heater} to temp: {target}")
|
|
return self._ws.send_method(
|
|
"printer.gcode.script",
|
|
{
|
|
"script": KlippyGcodes.set_heater_temp(heater, target)
|
|
},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def set_temp_fan_temp(self, temp_fan, target, callback=None, *args):
|
|
logging.debug(f"Sending temperature fan {temp_fan} to temp: {target}")
|
|
return self._ws.send_method(
|
|
"printer.gcode.script",
|
|
{
|
|
"script": KlippyGcodes.set_temp_fan_temp(temp_fan, target)
|
|
},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def set_tool_temp(self, tool, target, callback=None, *args):
|
|
logging.debug(f"Sending set_tool_temp: {KlippyGcodes.set_ext_temp(target, tool)}")
|
|
return self._ws.send_method(
|
|
"printer.gcode.script",
|
|
{
|
|
"script": KlippyGcodes.set_ext_temp(target, tool)
|
|
},
|
|
callback,
|
|
*args
|
|
)
|
|
|
|
def restart(self):
|
|
logging.debug("Sending printer.restart")
|
|
return self._ws.send_method(
|
|
"printer.restart"
|
|
)
|
|
|
|
def restart_firmware(self):
|
|
logging.debug("Sending printer.firmware_restart")
|
|
return self._ws.send_method(
|
|
"printer.firmware_restart"
|
|
)
|
|
|
|
def identify_client(self, version, api_key):
|
|
logging.debug("Sending server.connection.identify")
|
|
return self._ws.send_method(
|
|
"server.connection.identify",
|
|
{
|
|
"client_name": "KlipperScreen",
|
|
"version": f"{version}",
|
|
"type": "display",
|
|
"url": "https://github.com/KlipperScreen/KlipperScreen",
|
|
"api_key": f"{api_key}"
|
|
},
|
|
)
|