zkk f892315588 优化状态栏显示网络图标,优化IP信息更新不及时问题,实现注册码页面
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
2025-03-03 15:28:36 +08:00

501 lines
22 KiB
Python

import logging
import os
import gi
gi.require_version("Gtk", "3.0")
from gi.repository import Gtk, Pango
from datetime import datetime
from ks_includes.screen_panel import ScreenPanel
from ks_includes.KlippyGtk import find_widget
from ks_includes.widgets.flowboxchild_extended import PrintListItem
def format_label(widget):
label = find_widget(widget, Gtk.Label)
if label is not None:
label.set_line_wrap_mode(Pango.WrapMode.CHAR)
label.set_line_wrap(True)
label.set_ellipsize(Pango.EllipsizeMode.END)
label.set_lines(2)
class Panel(ScreenPanel):
def __init__(self, screen, title):
title = title or (_("Print") if self._printer.extrudercount > 0 else _("Gcodes"))
super().__init__(screen, title)
sortdir = self._config.get_main_config().get("print_sort_dir", "name_asc")
sortdir = sortdir.split('_')
self.sort_items = {
"name": _("Name"),
"date": _("Date"),
"size": _("Size"),
}
if sortdir[0] not in self.sort_items or sortdir[1] not in ["asc", "desc"]:
sortdir = ["name", "asc"]
self.sort_current = [sortdir[0], 0 if sortdir[1] == "asc" else 1] # 0 for asc, 1 for desc
self.sort_icon = ["arrow-up", "arrow-down"]
self.source = ""
self.time_24 = self._config.get_main_config().getboolean("24htime", True)
self.showing_rename = False
self.loading = False
self.cur_directory = 'gcodes'
self.list_button_size = self._gtk.img_scale * self.bts
self.headerbox = Gtk.Box(hexpand=True, vexpand=False)
n = 0
for name, val in self.sort_items.items():
s = self._gtk.Button(None, val, f"color{n % 4 + 1}", .5, Gtk.PositionType.RIGHT, 1)
s.get_style_context().add_class("buttons_slim")
if name == self.sort_current[0]:
s.set_image(self._gtk.Image(self.sort_icon[self.sort_current[1]], self._gtk.img_scale * self.bts))
s.connect("clicked", self.change_sort, name)
self.labels[f'sort_{name}'] = s
self.headerbox.add(s)
n += 1
self.refresh = self._gtk.Button("refresh", style=f"color{n % 4 + 1}", scale=self.bts)
self.refresh.get_style_context().add_class("buttons_slim")
self.refresh.connect('clicked', self._refresh_files)
n += 1
self.headerbox.add(self.refresh)
self.switch_mode = self._gtk.Button("fine-tune", style=f"color{n % 4 + 1}", scale=self.bts)
self.switch_mode.get_style_context().add_class("buttons_slim")
self.switch_mode.connect('clicked', self.switch_view_mode)
n += 1
self.headerbox.add(self.switch_mode)
self.loading_msg = _('Loading...')
self.labels['path'] = Gtk.Label(label=self.loading_msg, vexpand=True, no_show_all=True)
self.labels['path'].set_ellipsize(Pango.EllipsizeMode.END)
self.labels['path'].show()
self.thumbsize = self._gtk.img_scale * self._gtk.button_image_scale * 2.5
logging.info(f"Thumbsize: {self.thumbsize}")
self.flowbox = Gtk.FlowBox(selection_mode=Gtk.SelectionMode.NONE,
column_spacing=0, row_spacing=0)
list_mode = self._config.get_main_config().get("print_view", 'thumbs')
logging.info(list_mode)
self.list_mode = list_mode == 'list'
if self.list_mode:
self.flowbox.set_min_children_per_line(1)
self.flowbox.set_max_children_per_line(1)
else:
columns = 3 if self._screen.vertical_mode else 4
self.flowbox.set_min_children_per_line(columns)
self.flowbox.set_max_children_per_line(columns)
self.scroll = self._gtk.ScrolledWindow()
self.scroll.add(self.flowbox)
self.main = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, vexpand=True)
self.main.add(self.headerbox)
self.main.add(self.labels['path'])
self.main.add(self.scroll)
self.content.add(self.main)
self.set_loading(True)
self._screen._ws.klippy.get_dir_info(self.load_files, self.cur_directory)
def switch_view_mode(self, widget):
self.list_mode ^= True
logging.info(f"lista {self.list_mode}")
if self.list_mode:
self.flowbox.set_min_children_per_line(1)
self.flowbox.set_max_children_per_line(1)
else:
columns = 3 if self._screen.vertical_mode else 4
self.flowbox.set_min_children_per_line(columns)
self.flowbox.set_max_children_per_line(columns)
self._config.set("main", "print_view", 'list' if self.list_mode else 'thumbs')
self._config.save_user_config_options()
self._refresh_files()
def activate(self):
if self.cur_directory != "gcodes":
self.change_dir()
self._screen.files.add_callback(self._callback)
def deactivate(self):
self._screen.files.remove_callback(self._callback)
def create_item(self, item):
fbchild = PrintListItem()
fbchild.set_date(item['modified'])
fbchild.set_size(item['size'])
if 'dirname' in item:
if item['dirname'].startswith("."):
return
name = item['dirname']
path = f"{self.cur_directory}/{name}"
fbchild.set_as_dir(True)
elif 'filename' in item:
if (item['filename'].startswith(".") or
os.path.splitext(item['filename'])[1] not in {'.gcode', '.gco', '.g'}):
return
name = item['filename']
path = f"{self.cur_directory}/{name}"
path = path.replace('gcodes/', '')
else:
logging.error(f"Unknown item {item}")
return
basename = os.path.splitext(name)[0]
fbchild.set_path(path)
fbchild.set_name(basename.casefold())
if self.list_mode:
label = Gtk.Label(label=basename, hexpand=True, vexpand=False)
format_label(label)
info = Gtk.Label(hexpand=True, halign=Gtk.Align.START, wrap=True, wrap_mode=Pango.WrapMode.WORD_CHAR)
info.get_style_context().add_class("print-info")
info.set_markup(self.get_info_str(item, path))
delete = Gtk.Button(hexpand=False, vexpand=False, can_focus=False, always_show_image=True)
delete.get_style_context().add_class("custom-icon-button")
delete.set_image(self._gtk.Image("delete", self.list_button_size * 2, self.list_button_size * 2))
rename = Gtk.Button(hexpand=False, vexpand=False, can_focus=False, always_show_image=True)
rename.get_style_context().add_class("custom-icon-button")
rename.set_image(self._gtk.Image("rename", self.list_button_size * 2, self.list_button_size * 2))
itemname = Gtk.Label(hexpand=True, halign=Gtk.Align.START, ellipsize=Pango.EllipsizeMode.END)
itemname.get_style_context().add_class("print-filename")
itemname.set_markup(f"<big><b>{basename}</b></big>")
icon = Gtk.Button()
icon.get_style_context().add_class("custom-icon-button")
row = Gtk.Grid(hexpand=True, vexpand=True, valign=Gtk.Align.CENTER)
row.get_style_context().add_class("frame-item")
row.attach(icon, 0, 0, 1, 2)
row.attach(itemname, 1, 0, 3, 1)
row.attach(info, 1, 1, 1, 1)
row.attach(rename, 2, 1, 1, 1)
row.attach(delete, 3, 1, 1, 1)
if 'filename' in item:
icon.connect("clicked", self.confirm_print, path)
image_args = (path, icon, self.thumbsize / 2, True, "file")
delete.connect("clicked", self.confirm_delete_file, f"gcodes/{path}")
rename.connect("clicked", self.show_rename, f"gcodes/{path}")
action_icon = "printer" if self._printer.extrudercount > 0 else "load"
action = self._gtk.Button(action_icon, style="custom-icon-button")
action.connect("clicked", self.confirm_print, path)
action.set_hexpand(False)
action.set_vexpand(False)
action.set_halign(Gtk.Align.END)
row.attach(action, 4, 0, 1, 2)
elif 'dirname' in item:
icon.connect("clicked", self.change_dir, path)
image_args = (None, icon, self.thumbsize / 2, True, "folder")
delete.connect("clicked", self.confirm_delete_directory, path)
rename.connect("clicked", self.show_rename, path)
action = self._gtk.Button("load", style="custom-icon-button")
action.connect("clicked", self.change_dir, path)
action.set_hexpand(False)
action.set_vexpand(False)
action.set_halign(Gtk.Align.END)
row.attach(action, 4, 0, 1, 2)
else:
return
fbchild.add(row)
else: # Thumbnail view
icon = self._gtk.Button(label=basename)
icon.get_style_context().add_class("custom-icon-button")
if 'filename' in item:
icon.connect("clicked", self.confirm_print, path)
image_args = (path, icon, self.thumbsize, False, "file")
elif 'dirname' in item:
icon.connect("clicked", self.change_dir, path)
image_args = (None, icon, self.thumbsize, False, "folder")
else:
return
fbchild.add(icon)
self.image_load(*image_args)
return fbchild
def show_path(self):
self.labels['path'].set_vexpand(False)
if self.cur_directory == 'gcodes':
self.labels['path'].hide()
else:
self.labels['path'].set_text(self.cur_directory)
self.labels['path'].show()
def image_load(self, filepath, widget, size=-1, small=True, iconname=None):
pixbuf = self.get_file_image(filepath, size, size, small)
if pixbuf is not None:
widget.set_image(Gtk.Image.new_from_pixbuf(pixbuf))
elif iconname is not None:
widget.set_image(self._gtk.Image(iconname, size, size))
format_label(widget)
def confirm_delete_file(self, widget, filepath):
logging.debug(f"Sending delete_file {filepath}")
params = {"path": f"{filepath}"}
self._screen._confirm_send_action(
None,
_("Delete File?") + "\n\n" + filepath,
"server.files.delete_file",
params
)
def confirm_delete_directory(self, widget, dirpath):
logging.debug(f"Sending delete_directory {dirpath}")
params = {"path": f"{dirpath}", "force": True}
self._screen._confirm_send_action(
None,
_("Delete Directory?") + "\n\n" + dirpath,
"server.files.delete_directory",
params
)
def back(self):
if self.showing_rename:
self.hide_rename()
return True
if self.cur_directory != 'gcodes':
self.change_dir(None, os.path.dirname(self.cur_directory))
return True
return False
def change_dir(self, widget=None, directory='gcodes'):
if directory == '':
directory = 'gcodes'
if directory != self.cur_directory:
logging.info(f'Changing directory to: {directory}')
self.cur_directory = directory
self.show_path()
self._refresh_files()
def change_sort(self, widget, key):
if self.sort_current[0] == key:
self.sort_current[1] = (self.sort_current[1] + 1) % 2
else:
oldkey = self.sort_current[0]
logging.info(f"Changing from {oldkey} to {key}")
self.labels[f'sort_{oldkey}'].set_image(None)
self.labels[f'sort_{oldkey}'].show_all()
self.sort_current = [key, 0]
self.labels[f'sort_{key}'].set_image(self._gtk.Image(self.sort_icon[self.sort_current[1]],
self._gtk.img_scale * self.bts))
self.labels[f'sort_{key}'].show()
self.set_sort()
self._config.set("main", "print_sort_dir", f'{key}_{"asc" if self.sort_current[1] == 0 else "desc"}')
self._config.save_user_config_options()
def set_sort(self):
reverse = self.sort_current[1] != 0
if self.sort_current[0] == "name":
self.flowbox.set_sort_func(self.sort_names, reverse)
elif self.sort_current[0] == "date":
self.flowbox.set_sort_func(self.sort_dates, reverse)
elif self.sort_current[0] == "size":
self.flowbox.set_sort_func(self.sort_sizes, reverse)
@staticmethod
def sort_names(a: PrintListItem, b: PrintListItem, reverse):
if a.get_is_dir() - b.get_is_dir() != 0:
return a.get_is_dir() - b.get_is_dir()
if a.get_name() < b.get_name():
return 1 if reverse else -1
if a.get_name() > b.get_name():
return -1 if reverse else 1
return 0
@staticmethod
def sort_sizes(a: PrintListItem, b: PrintListItem, reverse):
if a.get_is_dir() - b.get_is_dir() != 0:
return a.get_is_dir() - b.get_is_dir()
return b.get_size() - a.get_size() if reverse else a.get_size() - b.get_size()
@staticmethod
def sort_dates(a: PrintListItem, b: PrintListItem, reverse):
if a.get_is_dir() - b.get_is_dir() != 0:
return a.get_is_dir() - b.get_is_dir()
return b.get_date() - a.get_date() if reverse else a.get_date() - b.get_date()
def confirm_print(self, widget, filename):
action = _("Print") if self._printer.extrudercount > 0 else _("Start")
buttons = [
{"name": action, "response": Gtk.ResponseType.OK},
{"name": _("Cancel"), "response": Gtk.ResponseType.CANCEL, "style": 'dialog-error'}
]
label = Gtk.Label(hexpand=True, vexpand=True, wrap=True, wrap_mode=Pango.WrapMode.WORD_CHAR)
label.set_markup(f"<b>{filename}</b>\n")
box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, vexpand=True)
box.pack_start(label, False, False, 0)
height = (self._screen.height - self._gtk.dialog_buttons_height - self._gtk.font_size * 5) * .75
pixbuf = self.get_file_image(filename, self._screen.width * .9, height)
if pixbuf is not None:
image = Gtk.Image.new_from_pixbuf(pixbuf)
box.pack_start(image, True, True, 0)
fileinfo = self._screen.files.get_file_info(filename)
if "estimated_time" in fileinfo:
box.pack_start(
Gtk.Label(label=_("Estimated Time") + f': {self.format_time(fileinfo["estimated_time"])}'),
False, False, 2
)
self._gtk.Dialog(f'{action} {filename}', buttons, box, self.confirm_print_response, filename)
def confirm_print_response(self, dialog, response_id, filename):
self._gtk.remove_dialog(dialog)
if response_id == Gtk.ResponseType.OK:
if self._screen.license.is_interface_valid() and not self._screen.license.is_active():
if not self._screen.license.is_time_sufficient():
if "license" not in self._screen._cur_panels:
self._screen.show_panel(
"license",
title="license",
remove_all=False,
func=self._screen._ws.klippy.print_start,
file=filename,
)
return
logging.info(f"Starting print: {filename}")
self._screen._ws.klippy.print_start(filename)
def get_info_str(self, item, path):
info = ""
if "modified" in item:
info += _("Modified") if 'dirname' in item else _("Uploaded")
if self.time_24:
info += f':<b> {datetime.fromtimestamp(item["modified"]):%Y/%m/%d %H:%M}</b>\n'
else:
info += f':<b> {datetime.fromtimestamp(item["modified"]):%Y/%m/%d %I:%M %p}</b>\n'
if "size" in item:
info += _("Size") + f': <b>{self.format_size(item["size"])}</b>\n'
if 'filename' in item:
fileinfo = self._screen.files.get_file_info(path)
if "layer_height" in fileinfo:
info += _("Layer Height") + f': <b>{fileinfo["layer_height"]}</b>' + _("mm") + '\n'
if "filament_type" in fileinfo:
info += _("Filament") + f': <b>{fileinfo["filament_type"]}</b>\n'
if "filament_name" in fileinfo:
info += f'<b>{fileinfo["filament_name"]}</b>\n'
if "estimated_time" in fileinfo:
info += _("Estimated Time") + f': <b>{self.format_time(fileinfo["estimated_time"])}</b>'
return info
def load_files(self, result, method, params):
start = datetime.now()
self.set_loading(True)
if not result.get("result") or not isinstance(result["result"], dict):
logging.info(result)
return
items = [self.create_item(item) for item in [*result["result"]["dirs"], *result["result"]["files"]]]
for item in filter(None, items):
self.flowbox.add(item)
self.set_sort()
self.set_loading(False)
logging.info(f"Loaded in {(datetime.now() - start).total_seconds():.3f} seconds")
def delete_from_list(self, path):
logging.info(f"deleting {path}")
for item in self.flowbox.get_children():
if item.get_path() in {path, f"gcodes/{path}"}:
logging.info("found removing")
self.flowbox.remove(item)
return True
def add_item_from_callback(self, action, data):
item = data['item']
if 'source_item' in data:
self.delete_from_list(data['source_item']['path'])
else:
self.delete_from_list(item['path'])
path = os.path.join("gcodes", item["path"])
if self.cur_directory != os.path.dirname(path):
return
if action in {"create_dir", "move_dir"}:
item.update({"path": path, "dirname": os.path.split(item["path"])[1]})
else:
item.update({"path": path, "filename": os.path.split(item["path"])[1]})
fbchild = self.create_item(item)
if fbchild:
self.flowbox.add(fbchild)
self.flowbox.invalidate_sort()
self.flowbox.show_all()
def _callback(self, action, data):
logging.info(f"{action}: {data}")
if action in {"create_dir", "create_file"}:
self.add_item_from_callback(action, data)
elif action == "delete_file":
self.delete_from_list(data['item']["path"])
elif action == "delete_dir":
self.delete_from_list(os.path.join("gcodes", data['item']["path"]))
elif action in {"modify_file", "move_file", "move_dir"}:
if "path" in data['item'] and data['item']["path"].startswith("gcodes/"):
data['item']["path"] = data['item']["path"][7:]
self.add_item_from_callback(action, data)
def _refresh_files(self, *args):
logging.info("Refreshing")
self.set_loading(True)
for child in self.flowbox.get_children():
self.flowbox.remove(child)
self._screen._ws.klippy.get_dir_info(self.load_files, self.cur_directory)
def set_loading(self, loading):
self.loading = loading
for child in self.headerbox.get_children():
child.set_sensitive(not loading)
self._gtk.Button_busy(self.refresh, loading)
if loading:
self.labels['path'].set_text(self.loading_msg)
self.labels['path'].show()
return
self.show_path()
self.content.show_all()
def show_rename(self, widget, fullpath):
self.source = fullpath
logging.info(self.source)
for child in self.content.get_children():
self.content.remove(child)
if "rename_file" not in self.labels:
self._create_rename_box(fullpath)
self.content.add(self.labels['rename_file'])
self.labels['new_name'].set_text(fullpath[7:])
self.labels['new_name'].grab_focus_without_selecting()
self.showing_rename = True
def _create_rename_box(self, fullpath):
lbl = Gtk.Label(label=_("Rename/Move:"), halign=Gtk.Align.START, hexpand=False)
self.labels['new_name'] = Gtk.Entry(text=fullpath, hexpand=True)
self.labels['new_name'].connect("activate", self.rename)
self.labels['new_name'].connect("focus-in-event", self._screen.show_keyboard)
save = self._gtk.Button("complete", _("Save"), "color3")
save.set_hexpand(False)
save.connect("clicked", self.rename)
box = Gtk.Box()
box.pack_start(self.labels['new_name'], True, True, 5)
box.pack_start(save, False, False, 5)
self.labels['rename_file'] = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, spacing=5,
hexpand=True, vexpand=True, valign=Gtk.Align.CENTER)
self.labels['rename_file'].pack_start(lbl, True, True, 5)
self.labels['rename_file'].pack_start(box, True, True, 5)
def hide_rename(self):
self._screen.remove_keyboard()
for child in self.content.get_children():
self.content.remove(child)
self.content.add(self.main)
self.content.show()
self.showing_rename = False
def rename(self, widget):
params = {"source": self.source, "dest": f"gcodes/{self.labels['new_name'].get_text()}"}
self._screen._send_action(
widget,
"server.files.move",
params
)
self.back()