Squashed commit of the following: commit 69bcfa4bd00fc1af8da8df191b6a4d86309c7d23 Merge: 2dba9f8e 964a81c3 Author: zkk <1007518571@qq.com> Date: Mon Mar 3 15:25:49 2025 +0800 Merge commit '964a81c37cd165b9f8df20db87fd915ba03d10b5' into release commit 964a81c37cd165b9f8df20db87fd915ba03d10b5 Author: zkk <1007518571@qq.com> Date: Fri Feb 28 16:27:57 2025 +0800 状态栏增加网络连接图标 commit 9c99cd91250c7aeb3d9728e540590299a1caf150 Author: zkk <1007518571@qq.com> Date: Fri Feb 28 16:20:52 2025 +0800 修复网络页面IP更新不及时的问题 commit 842a616c2558757d19b7553d23aee9f1daffdc80 Author: zkk <1007518571@qq.com> Date: Tue Feb 25 14:53:31 2025 +0800 注册码功能的实现 commit 2dba9f8ee2725847f31c891b19df6d9b81e03568 Merge: 03cd3722 18f05cc5 Author: zkk <1007518571@qq.com> Date: Fri Feb 14 15:21:36 2025 +0800 Merge commit '18f05cc52dfb7889853a4f84aad975309ab7dbbe' into release commit 18f05cc52dfb7889853a4f84aad975309ab7dbbe Author: zkk <1007518571@qq.com> Date: Wed Feb 5 10:59:00 2025 +0800 修复连接不了未加密网络的旧问题 commit 025a26209cbe15a0b7f5f1aba5166da63d6eaebf Author: zkk <1007518571@qq.com> Date: Fri Jan 24 15:48:52 2025 +0800 重构网络页面,解决卡顿问题 commit fdc851c4e66e3b872eb2564442f0c67369b9e3e2 Author: zkk <1007518571@qq.com> Date: Fri Jan 24 15:13:41 2025 +0800 优化菜单页面的点击效果和增加列表风格属性 commit e6cfa874208f8d98d0360fb073e4da15c935b202 Author: zkk <1007518571@qq.com> Date: Tue Jan 21 15:11:28 2025 +0800 优化进入工厂模式的方式为10s内点击5次 commit 03cd37229ee6c606e9814af602709d81b2c79bec Merge: cf7cafef adcbaa17 Author: zkk <1007518571@qq.com> Date: Fri Jan 17 14:38:18 2025 +0800 Merge commit 'adcbaa1795bffedc3000bc9e59eabba04d946dd4' into release commit adcbaa1795bffedc3000bc9e59eabba04d946dd4 Author: zkk <1007518571@qq.com> Date: Tue Jan 14 15:34:10 2025 +0800 实现探针偏移值校准保存前范围检测,避免保存错误值无法启动问题 commit cf7cafefbcd2274c9a481e58042d11db8cd79961 Merge: dc3478b4 8d259cea Author: zkk <1007518571@qq.com> Date: Tue Jan 7 17:26:59 2025 +0800 Merge commit '8d259ceadd5026cecc1dc61224d91aef4f3877b4' into release commit 8d259ceadd5026cecc1dc61224d91aef4f3877b4 Author: zkk <1007518571@qq.com> Date: Tue Jan 7 17:15:46 2025 +0800 实现 打印中恢复喷头温度 提示语的翻译 commit dd11c9683cb5af5407e650eea9b683b1066256c7 Author: zkk <1007518571@qq.com> Date: Tue Jan 7 16:52:04 2025 +0800 优化单个喷头时不显示喷头偏移菜单 commit c3ec7a81bbe1490b76202485d0502a4829a3cee5 Author: zkk <1007518571@qq.com> Date: Fri Jan 3 17:52:24 2025 +0800 修复屏幕耗材检测显示偶尔不准确的问题 commit 413bf6fc4a740f53d265c63abe7c51eaad483cb0 Author: zkk <1007518571@qq.com> Date: Fri Jan 3 15:00:54 2025 +0800 修复耗材检测弹窗异常问题 commit dc3478b48c7da1dbae120d1270c9ab64646d6647 Merge: 02c7556c 3bfc1aa7 Author: zkk <1007518571@qq.com> Date: Thu Jan 2 11:18:07 2025 +0800 Merge commit '3bfc1aa714e282d74b801155d830377ca58d8f59' into release commit 3bfc1aa714e282d74b801155d830377ca58d8f59 Author: zkk <1007518571@qq.com> Date: Thu Jan 2 11:17:15 2025 +0800 修复移轴页面移轴功能失效问题 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: # ks_includes/locales/KlipperScreen.pot resolved by 69bcfa4bd00fc1af8da8df191b6a4d86309c7d23 version # ks_includes/locales/zh_CN/LC_MESSAGES/KlipperScreen.mo resolved by 69bcfa4bd00fc1af8da8df191b6a4d86309c7d23 version # ks_includes/locales/zh_CN/LC_MESSAGES/KlipperScreen.po resolved by 69bcfa4bd00fc1af8da8df191b6a4d86309c7d23 version # ks_includes/locales/zh_TW/LC_MESSAGES/KlipperScreen.mo resolved by 69bcfa4bd00fc1af8da8df191b6a4d86309c7d23 version # ks_includes/locales/zh_TW/LC_MESSAGES/KlipperScreen.po resolved by 69bcfa4bd00fc1af8da8df191b6a4d86309c7d23 version # panels/factory_settings.py resolved by 69bcfa4bd00fc1af8da8df191b6a4d86309c7d23 version # panels/network.py resolved by 69bcfa4bd00fc1af8da8df191b6a4d86309c7d23 version
235 lines
10 KiB
Python
235 lines
10 KiB
Python
import logging
|
|
import gi
|
|
|
|
gi.require_version("Gtk", "3.0")
|
|
from gi.repository import Gtk, Pango
|
|
from ks_includes.screen_panel import ScreenPanel
|
|
|
|
|
|
class Panel(ScreenPanel):
|
|
|
|
def __init__(self, screen, title, **kwargs):
|
|
title = title or _("license")
|
|
super().__init__(screen, title)
|
|
self.title_text = (
|
|
_("This device is not activated and is available for trial use only")
|
|
+ "\n"
|
|
+ _("Please enter a key to activate")
|
|
)
|
|
self.key_len = 15
|
|
self.full = False
|
|
self.interface = screen.license
|
|
self.serial_num = self.interface.get_unique_id() or _("Unknown")
|
|
self.is_active = self.interface.is_active()
|
|
self.args = {}
|
|
|
|
def update_time(self):
|
|
total_printed_time = max(0, self.interface.get_total_printed_time())
|
|
trial_time = max(0, self.interface.get_trial_time())
|
|
remaining_time = max(0, trial_time - total_printed_time)
|
|
|
|
self.license_box["elapsed_trial_time_value"].set_text(self.seconds_to_hms(total_printed_time))
|
|
self.license_box["trial_time_value"].set_text(self.seconds_to_hms(trial_time))
|
|
self.license_box["remain_time_value"].set_text(self.seconds_to_hms(remaining_time))
|
|
|
|
def seconds_to_hms(self, seconds):
|
|
|
|
if not isinstance(seconds, (int, float)) or seconds < 0:
|
|
raise ValueError(f"seconds must be a non-negative number, got {seconds}")
|
|
|
|
hours, remainder = divmod(int(seconds), 3600)
|
|
minutes, secs = divmod(remainder, 60)
|
|
|
|
return f"{hours:03}:{minutes:02}:{secs:02}"
|
|
|
|
def state_update(self, text):
|
|
self.license_box["state_text_value"].set_text(text)
|
|
|
|
def verify_key(self, key):
|
|
try:
|
|
res = self.interface.verify_activation_code(key)
|
|
if res:
|
|
if self.interface.is_active():
|
|
self.is_active = True
|
|
self.state_update(_("Permanent Activation"))
|
|
else:
|
|
self.state_update(_("Key is valid"))
|
|
else:
|
|
self.state_update(_("Key is invalid"))
|
|
except Exception as e:
|
|
logging.exception(e)
|
|
|
|
def active_refresh(self, **args):
|
|
self.args["full"] = args.get("full")
|
|
self.args["callback"] = args.get("func")
|
|
self.args["file"] = args.get("file")
|
|
self.args["onboarding"] = args.get("onboarding")
|
|
self.display_dialog()
|
|
|
|
def deactivate(self):
|
|
self._screen.remove_keyboard()
|
|
for child in self.content.get_children():
|
|
self.content.remove(child)
|
|
|
|
def display_dialog(self, full=False, key=""):
|
|
BUTTON_CONFIGS = {
|
|
"trial_with_callback": [
|
|
{"name": _("Activate"), "response": Gtk.ResponseType.OK, "style": "dialog-info"},
|
|
{"name": _("Skip"), "response": Gtk.ResponseType.CANCEL, "style": "dialog-error"},
|
|
],
|
|
"full_features": [
|
|
{"name": _("Reset"), "response": Gtk.ResponseType.APPLY, "style": "dialog-secondary"},
|
|
{"name": _("Activate"), "response": Gtk.ResponseType.OK, "style": "dialog-info"},
|
|
{"name": _("Close"), "response": Gtk.ResponseType.CLOSE, "style": "dialog-error"},
|
|
],
|
|
"Trial": [
|
|
{"name": _("Activate"), "response": Gtk.ResponseType.OK, "style": "dialog-info"},
|
|
{"name": _("Trial"), "response": Gtk.ResponseType.CLOSE, "style": "dialog-error"},
|
|
],
|
|
"default": [
|
|
{"name": _("Activate"), "response": Gtk.ResponseType.OK, "style": "dialog-info"},
|
|
{"name": _("Close"), "response": Gtk.ResponseType.CLOSE, "style": "dialog-error"},
|
|
],
|
|
}
|
|
|
|
if self.args.get("callback") and self.interface.is_trial_active():
|
|
buttons = BUTTON_CONFIGS["trial_with_callback"]
|
|
elif self.args.get("full"):
|
|
buttons = BUTTON_CONFIGS["full_features"]
|
|
elif self.args.get("onboarding"):
|
|
buttons = BUTTON_CONFIGS["Trial"]
|
|
else:
|
|
buttons = BUTTON_CONFIGS["default"]
|
|
|
|
self.create_license_key_dialog(buttons=buttons, key=key)
|
|
|
|
def create_license_key_dialog(self, buttons=None, key=""):
|
|
|
|
if buttons is None:
|
|
buttons = [
|
|
{"name": _("Activate"), "response": Gtk.ResponseType.OK, "style": "dialog-info"},
|
|
{"name": _("Close"), "response": Gtk.ResponseType.CLOSE, "style": "dialog-error"},
|
|
]
|
|
|
|
self.title_label = Gtk.Label(hexpand=True, vexpand=False, wrap=True, wrap_mode=Pango.WrapMode.WORD_CHAR)
|
|
self.title_label.set_markup(f"<big>{self.title_text}</big>\n")
|
|
self.title_label.set_margin_top(50)
|
|
self.title_label.set_margin_start(20)
|
|
self.title_label.set_halign(Gtk.Align.START)
|
|
|
|
self.license_box = {}
|
|
|
|
self.grid = Gtk.Grid(column_spacing=20, row_spacing=20, hexpand=True, vexpand=True)
|
|
|
|
def add_labeled_value(row, label_text, value_text):
|
|
label = Gtk.Label(label=label_text, use_markup=True, xalign=0, wrap=True)
|
|
value = Gtk.Label(label=value_text, use_markup=True, xalign=0, wrap=True)
|
|
self.grid.attach(label, 0, row, 1, 1)
|
|
self.grid.attach(value, 1, row, 1, 1)
|
|
return label, value
|
|
|
|
status_text = _("Not activated")
|
|
if not self.interface.is_interface_valid():
|
|
status_text = _("Unknown")
|
|
elif self.is_active:
|
|
status_text = _("Permanent Activation")
|
|
self.license_box["state_text"], self.license_box["state_text_value"] = add_labeled_value(
|
|
0, _("State:"), status_text
|
|
)
|
|
self.license_box["serial_num_text"], self.license_box["serial_num_value"] = add_labeled_value(
|
|
1, _("Serial Number:"), self.serial_num
|
|
)
|
|
self.license_box["trial_time_text"], self.license_box["trial_time_value"] = add_labeled_value(
|
|
2, _("Trial Time:"), "000:00:00"
|
|
)
|
|
self.license_box["elapsed_trial_time_text"], self.license_box["elapsed_trial_time_value"] = add_labeled_value(
|
|
3, _("Elapsed trial time:"), "000:00:00"
|
|
)
|
|
self.license_box["remain_time_text"], self.license_box["remain_time_value"] = add_labeled_value(
|
|
4, _("Remaining Time:"), "000:00:00"
|
|
)
|
|
|
|
self.license_box["key_text"] = Gtk.Label(label=_("Key:"), use_markup=True, xalign=0, wrap=True)
|
|
self.grid.attach(self.license_box["key_text"], 0, 5, 1, 1)
|
|
|
|
self.license_box["key_input"] = Gtk.Entry(hexpand=False, vexpand=False)
|
|
self.license_box["key_input"].set_max_length(self.key_len)
|
|
self.license_box["key_input"].set_text(key)
|
|
self.license_box["key_input"].connect("button-press-event", self.on_show_keyboard)
|
|
self.grid.attach(self.license_box["key_input"], 1, 5, 1, 1)
|
|
|
|
image = self._gtk.Image("license", self._gtk.content_width * 0.4, self._gtk.content_height * 0.4)
|
|
image.set_margin_start(60)
|
|
image.set_margin_end(20)
|
|
|
|
main_box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, vexpand=True)
|
|
horizontal_box = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, vexpand=True)
|
|
horizontal_box.set_margin_top(20)
|
|
|
|
self.grid.set_margin_start(60)
|
|
horizontal_box.pack_start(image, False, True, 0)
|
|
horizontal_box.pack_start(self.grid, True, True, 0)
|
|
|
|
main_box.pack_start(self.title_label, False, True, 0)
|
|
main_box.pack_start(horizontal_box, True, True, 0)
|
|
|
|
self.dialog = self._gtk.Dialog("License", buttons, main_box, self.confirm_license_response)
|
|
self.update_time()
|
|
|
|
def confirm_license_response(self, dialog, response_id):
|
|
if response_id == Gtk.ResponseType.YES:
|
|
if self.interface.enabled_registration():
|
|
self.state_update(_("Enabled successfully"))
|
|
elif response_id == Gtk.ResponseType.APPLY:
|
|
if len(self.license_box["key_input"].get_text()) == 0:
|
|
self.state_update(_("Key is empty"))
|
|
return
|
|
else:
|
|
if self.interface.reset_registration(self.license_box["key_input"].get_text()):
|
|
self.update_time()
|
|
self.state_update(_("Reset successfully"))
|
|
else:
|
|
self.state_update(_("Key is invalid"))
|
|
elif response_id == Gtk.ResponseType.CLOSE:
|
|
self._gtk.remove_dialog(dialog)
|
|
self._screen._menu_go_back()
|
|
elif response_id == Gtk.ResponseType.CANCEL:
|
|
self._gtk.remove_dialog(dialog)
|
|
self._screen._menu_go_back()
|
|
if not self.interface.is_active() and self.interface.is_trial_active():
|
|
if self.args.get("callback"):
|
|
self.args["callback"](self.args["file"])
|
|
elif response_id == Gtk.ResponseType.OK:
|
|
if len(self.license_box["key_input"].get_text()) == 0:
|
|
self.state_update(_("Key is empty"))
|
|
return
|
|
self.verify_key(self.license_box["key_input"].get_text())
|
|
self.update_time()
|
|
|
|
def on_show_keyboard(self, entry=None, event=None):
|
|
self._gtk.remove_dialog(self.dialog)
|
|
lbl = Gtk.Label(_("Please enter a key to activate"), halign=Gtk.Align.START, hexpand=False)
|
|
self.labels["entry"] = Gtk.Entry(hexpand=True)
|
|
self.labels["entry"].set_max_length(self.key_len)
|
|
self.labels["entry"].connect("focus-in-event", self._screen.show_keyboard)
|
|
save = self._gtk.Button("complete", _("Save"), "color3")
|
|
save.set_hexpand(False)
|
|
save.connect("clicked", self.on_save_key)
|
|
input_box = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, spacing=5)
|
|
input_box.pack_start(self.labels["entry"], True, True, 5)
|
|
input_box.pack_start(save, False, False, 5)
|
|
self.labels["input_box"] = Gtk.Box(
|
|
orientation=Gtk.Orientation.VERTICAL, spacing=5, hexpand=True, vexpand=True, valign=Gtk.Align.CENTER
|
|
)
|
|
self.labels["input_box"].pack_start(lbl, True, True, 5)
|
|
self.labels["input_box"].pack_start(input_box, True, True, 5)
|
|
self.content.add(self.labels["input_box"])
|
|
self.labels["entry"].grab_focus_without_selecting()
|
|
|
|
def on_save_key(self, dialog):
|
|
key_text = self.labels["entry"].get_text()
|
|
self._screen.remove_keyboard()
|
|
for child in self.content.get_children():
|
|
self.content.remove(child)
|
|
self.display_dialog(self.full, key=key_text)
|