diff --git a/.gitignore b/.gitignore
index d86da6a..1642218 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,5 +1,7 @@
.env
__pycache__/
+.agents/
+.codex/
*.pyc
build/
dist/
@@ -7,3 +9,4 @@ dist/
releases/*/kx-bridge
releases/*/extract_credentials
releases/*/extract_credentials.exe
+config/config.ini
diff --git a/kobrax_moonraker_bridge.py b/kobrax_moonraker_bridge.py
index 841e9b1..e711fdb 100644
--- a/kobrax_moonraker_bridge.py
+++ b/kobrax_moonraker_bridge.py
@@ -26,6 +26,8 @@ import sys
import tempfile
import time
import threading
+import io
+import zipfile
# Bei PyInstaller-Binary liegt alles neben sys.executable, sonst neben __file__
_BASE = os.path.dirname(sys.executable) if getattr(sys, "frozen", False) else os.path.dirname(os.path.abspath(__file__))
@@ -132,6 +134,135 @@ def _parse_gcode_estimated_time(data: bytes) -> int:
return secs
+_FILAMENT_COLOR_KEYS = (
+ "filament_colour", "filament_color", "filament_colours", "filament_colors",
+ "extruder_colour", "extruder_color",
+)
+_FILAMENT_MATERIAL_KEYS = (
+ "filament_type", "filament_types", "filament_settings_id", "filament_preset",
+)
+_KNOWN_MATERIALS = (
+ "PLA-CF", "PETG-CF", "PA-CF", "PLA SILK", "PETG", "PLA", "ABS", "ASA",
+ "TPU", "PA", "PC", "HIPS", "PVA",
+)
+
+
+def _normalize_material(value) -> str:
+ text = str(value or "").upper().replace("_", " ").replace("-", "-").strip()
+ for material in _KNOWN_MATERIALS:
+ if re.search(rf"(^|[^A-Z0-9]){re.escape(material)}([^A-Z0-9]|$)", text):
+ return material
+ return text.split()[0] if text else "PLA"
+
+
+def _color_to_rgb(value, default=None):
+ if default is None:
+ default = [255, 255, 255]
+ if isinstance(value, list) and len(value) >= 3:
+ try:
+ return [max(0, min(255, int(value[0]))),
+ max(0, min(255, int(value[1]))),
+ max(0, min(255, int(value[2])))]
+ except Exception:
+ return default
+ if isinstance(value, str):
+ m = re.search(r"#?([0-9a-fA-F]{6})(?:[0-9a-fA-F]{2})?", value)
+ if m:
+ raw = m.group(1)
+ return [int(raw[0:2], 16), int(raw[2:4], 16), int(raw[4:6], 16)]
+ return default
+
+
+def _rgba(color) -> list[int]:
+ rgb = _color_to_rgb(color)
+ return [rgb[0], rgb[1], rgb[2], 255]
+
+
+def _rgb_distance(a, b) -> int:
+ ar, ag, ab = _color_to_rgb(a)
+ br, bg, bb = _color_to_rgb(b)
+ return (ar - br) ** 2 + (ag - bg) ** 2 + (ab - bb) ** 2
+
+
+def _parse_colors_from_text(text: str) -> list[list[int]]:
+ colors = []
+ for match in re.finditer(r"#?([0-9a-fA-F]{6})(?:[0-9a-fA-F]{2})?", text or ""):
+ raw = match.group(1)
+ colors.append([int(raw[0:2], 16), int(raw[2:4], 16), int(raw[4:6], 16)])
+ return colors
+
+
+def _split_config_values(value: str) -> list[str]:
+ value = (value or "").strip().strip("[]")
+ parts = re.split(r"[;,]", value)
+ return [p.strip().strip("\"' ") for p in parts if p.strip().strip("\"' ")]
+
+
+def _extract_config_value(line: str) -> str:
+ attr = re.search(r"value=[\"']([^\"']+)[\"']", line)
+ if attr:
+ return attr.group(1)
+ if "=" in line:
+ return line.split("=", 1)[1].strip()
+ if ":" in line:
+ return line.split(":", 1)[1].strip()
+ return line
+
+
+def _parse_filament_metadata_text(text: str) -> dict:
+ colors: list[list[int]] = []
+ materials: list[str] = []
+ for line in (text or "").splitlines():
+ low = line.lower()
+ if any(k in low for k in _FILAMENT_COLOR_KEYS):
+ value = _extract_config_value(line)
+ for color in _parse_colors_from_text(value):
+ if color not in colors:
+ colors.append(color)
+ if any(k in low for k in _FILAMENT_MATERIAL_KEYS):
+ value = _extract_config_value(line)
+ for part in _split_config_values(value):
+ mat = _normalize_material(part)
+ if mat and mat not in materials:
+ materials.append(mat)
+ return {"colors": colors, "materials": materials}
+
+
+def _merge_filament_metadata(target: dict, source: dict):
+ for key in ("colors", "materials"):
+ for value in source.get(key, []):
+ if value not in target[key]:
+ target[key].append(value)
+
+
+def _parse_uploaded_filament_metadata(data: bytes, filename: str) -> dict:
+ """Best-effort extraction of slicer filament colors/materials from G-code or 3MF."""
+ meta = {"colors": [], "materials": [], "source": ""}
+ suffix = pathlib.Path(filename or "").suffix.lower()
+ if suffix == ".3mf" or data[:4] == b"PK\x03\x04":
+ try:
+ with zipfile.ZipFile(io.BytesIO(data)) as zf:
+ for name in zf.namelist():
+ lname = name.lower()
+ if not lname.endswith((".config", ".json", ".model", ".xml", ".gcode")):
+ continue
+ if zf.getinfo(name).file_size > 2_000_000:
+ continue
+ text = zf.read(name).decode("utf-8", errors="ignore")
+ before = len(meta["colors"]) + len(meta["materials"])
+ _merge_filament_metadata(meta, _parse_filament_metadata_text(text))
+ if len(meta["colors"]) + len(meta["materials"]) > before and not meta["source"]:
+ meta["source"] = name
+ except Exception as e:
+ log.warning(f"3MF metadata could not be parsed: {e}")
+ else:
+ search = (data[:131072] + data[-262144:]).decode("utf-8", errors="ignore")
+ _merge_filament_metadata(meta, _parse_filament_metadata_text(search))
+ if meta["colors"] or meta["materials"]:
+ meta["source"] = "gcode"
+ return meta
+
+
class KobraXBridge:
def __init__(self, client: KobraXClient, args=None):
self.client = client
@@ -167,6 +298,7 @@ class KobraXBridge:
self._ams_slots: list[dict] = []
self._ams_loaded_slot: int = -1
self._last_uploaded_file: str = ""
+ self._uploaded_filament_metadata: dict[str, dict] = {}
self._serve_dir = tempfile.TemporaryDirectory(prefix="kobrax_serve_")
self._serve_dir_path: str = self._serve_dir.name
@@ -357,6 +489,118 @@ class KobraXBridge:
"tray_exist_bits": format(tray_exist_bits, "X"),
}
+ def _uploaded_metadata_for(self, filename: str) -> dict:
+ if not filename:
+ return {"colors": [], "materials": [], "source": ""}
+ return (self._uploaded_filament_metadata.get(filename)
+ or self._uploaded_filament_metadata.get(os.path.basename(filename))
+ or {"colors": [], "materials": [], "source": ""})
+
+ def _loaded_ams_slots(self) -> list[tuple[int, dict]]:
+ return [(i, s) for i, s in enumerate(self._ams_slots) if s.get("status") == 5]
+
+ def _filtered_loaded_ams_slots(self) -> list[tuple[int, dict]]:
+ loaded = self._loaded_ams_slots()
+ default_slot = getattr(self._args, "default_ams_slot", "auto")
+ if default_slot == "auto":
+ return loaded
+ try:
+ slot_idx = int(default_slot)
+ except ValueError:
+ return loaded
+ selected = [(i, s) for i, s in loaded if i == slot_idx]
+ if selected:
+ return selected
+ log.warning(f"Standard-Slot {slot_idx} ist leer – fallback auf Auto")
+ return loaded
+
+ def _build_ams_assignments(self, filename: str) -> list[dict]:
+ loaded = self._filtered_loaded_ams_slots()
+ if not loaded:
+ return []
+
+ metadata = self._uploaded_metadata_for(filename)
+ colors = metadata.get("colors") or []
+ materials = metadata.get("materials") or []
+ target_count = max(len(colors), len(materials))
+
+ # Without slicer metadata, preserve the previous behavior: advertise all
+ # occupied slots and let the printer/firmware use its normal fallback.
+ if target_count == 0:
+ return [{
+ "paint_index": i,
+ "slot_index": i,
+ "paint_color": _rgba(slot.get("color", [255, 255, 255])),
+ "ams_color": _rgba(slot.get("color", [255, 255, 255])),
+ "material_type": _normalize_material(slot.get("type", "PLA")),
+ "reason": "loaded",
+ } for i, slot in loaded]
+
+ assignments = []
+ used_slots: set[int] = set()
+ for paint_index in range(target_count):
+ target_color = colors[paint_index] if paint_index < len(colors) else None
+ target_material = (_normalize_material(materials[paint_index])
+ if paint_index < len(materials) else "")
+ candidates = loaded
+ if target_material:
+ material_matches = [
+ (i, s) for i, s in candidates
+ if _normalize_material(s.get("type", "PLA")) == target_material
+ ]
+ if material_matches:
+ candidates = material_matches
+
+ unused = [(i, s) for i, s in candidates if i not in used_slots]
+ if unused:
+ candidates = unused
+
+ def _score(item):
+ slot_index, slot = item
+ score = 0
+ if target_color is not None:
+ score += _rgb_distance(target_color, slot.get("color", [255, 255, 255]))
+ if target_material and _normalize_material(slot.get("type", "PLA")) != target_material:
+ score += 200000
+ return score, slot_index
+
+ slot_index, slot = min(candidates, key=_score)
+ used_slots.add(slot_index)
+ assignments.append({
+ "paint_index": paint_index,
+ "slot_index": slot_index,
+ "paint_color": _rgba(target_color or slot.get("color", [255, 255, 255])),
+ "ams_color": _rgba(slot.get("color", [255, 255, 255])),
+ "material_type": _normalize_material(slot.get("type", target_material or "PLA")),
+ "target_material": target_material,
+ "reason": "metadata",
+ })
+
+ summary = [
+ f"T{a['paint_index']}→S{a['slot_index']} {a['material_type']}"
+ for a in assignments
+ ]
+ log.info(f"AMS metadata mapping for {filename}: {', '.join(summary)}")
+ return assignments
+
+ def _build_anycubic_ams_mapping(self, filename: str) -> list[dict]:
+ return [{
+ "paint_index": a["paint_index"],
+ "ams_index": a["slot_index"],
+ "paint_color": a["paint_color"],
+ "ams_color": a["ams_color"],
+ "material_type": a["material_type"],
+ } for a in self._build_ams_assignments(filename)]
+
+ def _build_simple_ams_mapping(self, filename: str) -> list[dict]:
+ return [{
+ "slot_index": a["slot_index"],
+ "material_type": a["material_type"],
+ "color": a["ams_color"][:3],
+ "paint_index": a["paint_index"],
+ "paint_color": a["paint_color"][:3],
+ } for a in self._build_ams_assignments(filename)]
+
# -------------------------------------------------------------------------
# WebSocket push
# -------------------------------------------------------------------------
@@ -545,6 +789,16 @@ class KobraXBridge:
# Slicer-Zeitschätzung aus GCode-Header auslesen
self._state["slicer_time"] = _parse_gcode_estimated_time(file_data)
+ filament_meta = _parse_uploaded_filament_metadata(file_data, remote_filename)
+ self._uploaded_filament_metadata[remote_filename] = filament_meta
+ self._uploaded_filament_metadata[os.path.basename(remote_filename)] = filament_meta
+ if filament_meta.get("colors") or filament_meta.get("materials"):
+ log.info(
+ "Upload filament metadata: "
+ f"colors={filament_meta.get('colors', [])} "
+ f"materials={filament_meta.get('materials', [])} "
+ f"source={filament_meta.get('source', '')}"
+ )
# Datei auf Disk ablegen (temp-Verzeichnis) damit Drucker sie per HTTP abrufen kann
safe_name = os.path.basename(remote_filename) # keine Pfad-Traversal
@@ -612,31 +866,12 @@ class KobraXBridge:
def _start_print(self, filename: str, url: str = "", md5: str = "", filesize: int = 0):
self._state["file_ready"] = ""
- default_slot = getattr(self._args, "default_ams_slot", "auto")
- all_loaded = [(i, s) for i, s in enumerate(self._ams_slots) if s.get("status") == 5]
- if default_slot != "auto":
- try:
- slot_idx = int(default_slot)
- loaded = [(i, s) for i, s in all_loaded if i == slot_idx]
- if not loaded:
- log.warning(f"Standard-Slot {slot_idx} ist leer – fallback auf Auto")
- loaded = all_loaded
- except ValueError:
- loaded = all_loaded
- else:
- loaded = all_loaded
- use_ams = len(loaded) > 0
- ams_box_mapping = [
- {
- "paint_index": i,
- "ams_index": i,
- "paint_color": [255, 255, 255, 255],
- "ams_color": [255, 255, 255, 255],
- "material_type": s.get("type", "PLA"),
- }
- for i, s in loaded
- ]
- log.info(f"AMS-Slots: {len(loaded)}/{len(self._ams_slots)} belegt → {[i for i,_ in loaded]}")
+ ams_box_mapping = self._build_anycubic_ams_mapping(filename)
+ use_ams = len(ams_box_mapping) > 0
+ log.info(
+ f"AMS-Slots: {len(self._loaded_ams_slots())}/{len(self._ams_slots)} belegt "
+ f"→ {[m['ams_index'] for m in ams_box_mapping]}"
+ )
auto_leveling = getattr(self._args, "auto_leveling", 1)
payload = {
"taskid": "-1",
@@ -682,36 +917,7 @@ class KobraXBridge:
log.info(f"Druck starten: {filename}")
- # AMS-Mapping aus gecachtem State — leere Slots (status != 5) überspringen
- default_slot = getattr(self._args, "default_ams_slot", "auto")
- ams_box_mapping = []
- for i, slot in enumerate(self._ams_slots):
- if slot.get("status") != 5:
- log.info(f"AMS-Slot {i} leer (status={slot.get('status')}) – übersprungen")
- continue
- if default_slot != "auto":
- try:
- if i != int(default_slot):
- continue
- except ValueError:
- pass
- ams_box_mapping.append({
- "slot_index": i,
- "material_type": slot.get("type", "PLA"),
- "color": slot.get("color", [255, 255, 255]),
- })
- # Fallback auf alle belegten Slots wenn gewählter Slot leer war
- if default_slot != "auto" and not ams_box_mapping:
- log.warning(f"Standard-Slot {default_slot} leer – fallback auf alle belegten Slots")
- for i, slot in enumerate(self._ams_slots):
- if slot.get("status") != 5:
- continue
- ams_box_mapping.append({
- "slot_index": i,
- "material_type": slot.get("type", "PLA"),
- "color": slot.get("color", [255, 255, 255]),
- })
-
+ ams_box_mapping = self._build_simple_ams_mapping(filename)
use_ams = len(ams_box_mapping) > 0
payload = {
@@ -1358,7 +1564,7 @@ nav.bottom-nav{display:none;position:fixed;bottom:0;left:0;right:0;
0
-
+
@@ -1395,23 +1601,23 @@ nav.bottom-nav{display:none;position:fixed;bottom:0;left:0;right:0;
≡ Ereignis-Log
⬇ Download
+ style="font-size:12px;padding:4px 10px;background:var(--raised);border-radius:6px;color:var(--txt2);text-decoration:none">⬇
Download
+ style="font-size:12px;padding:5px 10px;border-radius:6px;border:1px solid var(--border);background:var(--accent);color:#fff;cursor:pointer;white-space:nowrap">⬇ Auto
+ style="font-size:12px;padding:5px 10px;border-radius:6px;border:1px solid var(--border);background:var(--raised);color:var(--txt2);cursor:pointer">✕ Clear
- Dir:
+ Dir:
- Topic:
+ Topic:
@@ -1469,16 +1675,20 @@ var LANG_DE={
confirm_cancel:'Druck wirklich abbrechen?',
settings_title:'Einstellungen',settings_connection:'Verbindung',settings_print:'Druckeinstellungen',settings_poll:'Poll-Intervall',settings_version:'Version',
settings_save:'Speichern & Neustart',settings_printer_ip:'Drucker-IP',settings_mqtt_port:'MQTT-Port',
- settings_username:'MQTT-Benutzername',settings_password:'MQTT-Passwort',settings_device_id:'Device-ID',settings_mode_id:'Mode-ID',hint_ip_no_port:'Nur IP-Adresse, kein Port (z.B. 192.168.1.102)',
+ settings_username:'MQTT-Benutzername',settings_password:'MQTT-Passwort',settings_device_id:'Device-ID',settings_mode_id:'Mode-ID',settings_device_placeholder:'32 Hex-Zeichen',hint_ip_no_port:'Nur IP-Adresse, kein Port (z.B. 192.168.1.102)',
settings_default_slot:'Standard-Slot (Einfarbdruck)',settings_slot_auto:'Auto (alle belegten Slots)',settings_auto_leveling:'Auto-Leveling vor Druck',
update_check:'Auf Updates prüfen',update_checking:'Prüfe...',update_available:'verfügbar',update_none:'Bereits aktuell',
update_apply:'Jetzt installieren',update_applying:'Lade herunter...',update_restarting:'Starte neu...',update_error:'Fehler',
btn_connect:'⚡ Verbinden',btn_disconnect:'✕ Trennen',
lbl_conn_error:'Verbindungsfehler:',
+ settings_button_title:'Einstellungen',
slot_edit_title:'Slot bearbeiten',slot_edit_color:'Farbe',slot_edit_material:'Material',
slot_edit_save:'💾 Speichern',slot_edit_custom:'z.B. PLA, PETG, ABS…',
slot_edit_ok:'AMS Slot',
- log_dir_all:'Alle',
+ ams_slot_select:'Slot auswählen',
+ log_dir_all:'Alle',log_download:'Download',log_auto:'Auto',log_clear:'Leeren',log_dir:'Richtung:',log_topic:'Topic:',
+ log_settings_error:'Settings-Fehler:',log_axis_error:'Achsen-Fehler:',log_home_error:'Home-Fehler:',log_motors_error:'Motoren-Fehler:',log_temp_error:'Temp-Fehler:',log_light_error:'Licht-Fehler:',log_speed_error:'Speed-Fehler:',log_fan_error:'Lüfter-Fehler:',log_ams_error:'AMS-Fehler:',log_stream_unavailable:'Stream nicht verfügbar',
+ print_action_pause:'Pause',print_action_resume:'Fortsetzen',print_action_cancel:'Abbrechen',
file_ready_btn:'▶ Druck starten',
file_cancel_btn:'✕ Abbrechen'
};
@@ -1503,16 +1713,20 @@ var LANG_EN={
confirm_cancel:'Really cancel the print?',
settings_title:'Settings',settings_connection:'Connection',settings_print:'Print Settings',settings_poll:'Poll Interval',settings_version:'Version',
settings_save:'Save & Restart',settings_printer_ip:'Printer IP',settings_mqtt_port:'MQTT Port',
- settings_username:'MQTT Username',settings_password:'MQTT Password',settings_device_id:'Device ID',settings_mode_id:'Mode ID',hint_ip_no_port:'IP address only, no port (e.g. 192.168.1.102)',
+ settings_username:'MQTT Username',settings_password:'MQTT Password',settings_device_id:'Device ID',settings_mode_id:'Mode ID',settings_device_placeholder:'32 hex characters',hint_ip_no_port:'IP address only, no port (e.g. 192.168.1.102)',
settings_default_slot:'Default Slot (single color)',settings_slot_auto:'Auto (all loaded slots)',settings_auto_leveling:'Auto-Leveling before print',
update_check:'Check for Updates',update_checking:'Checking...',update_available:'available',update_none:'Already up to date',
update_apply:'Install Now',update_applying:'Downloading...',update_restarting:'Restarting...',update_error:'Error',
btn_connect:'⚡ Connect',btn_disconnect:'✕ Disconnect',
lbl_conn_error:'Connection error:',
+ settings_button_title:'Settings',
slot_edit_title:'Edit Slot',slot_edit_color:'Color',slot_edit_material:'Material',
slot_edit_save:'💾 Save',slot_edit_custom:'e.g. PLA, PETG, ABS…',
slot_edit_ok:'AMS Slot',
- log_dir_all:'All',
+ ams_slot_select:'Select slot',
+ log_dir_all:'All',log_download:'Download',log_auto:'Auto',log_clear:'Clear',log_dir:'Dir:',log_topic:'Topic:',
+ log_settings_error:'Settings error:',log_axis_error:'Axis error:',log_home_error:'Home error:',log_motors_error:'Motor error:',log_temp_error:'Temperature error:',log_light_error:'Light error:',log_speed_error:'Speed error:',log_fan_error:'Fan error:',log_ams_error:'AMS error:',log_stream_unavailable:'Stream unavailable',
+ print_action_pause:'Pause',print_action_resume:'Resume',print_action_cancel:'Cancel',
file_ready_btn:'▶ Start Print',
file_cancel_btn:'✕ Cancel'
};
@@ -1580,12 +1794,14 @@ function applyLang(){
setText('lbl-password',T.settings_password);
setText('lbl-device-id',T.settings_device_id);
setText('lbl-mode-id',T.settings_mode_id);
+ var did=document.getElementById('s-device-id');if(did)did.setAttribute('placeholder',T.settings_device_placeholder);
setText('lbl-default-slot',T.settings_default_slot);
setText('opt-slot-auto',T.settings_slot_auto);
setText('lbl-auto-leveling',T.settings_auto_leveling);
setText('lbl-update-check',T.update_check);
setText('lbl-update-apply',T.update_apply);
+ var sb=document.getElementById('settings-btn');if(sb)sb.setAttribute('title',T.settings_button_title);
// Speed buttons
setText('d-spd-lbl-1',T.speed_silent.replace(/^\S+\s/,''));
setText('d-spd-lbl-2',T.speed_normal.replace(/^\S+\s/,''));
@@ -1593,6 +1809,8 @@ function applyLang(){
// AMS feed/unload
document.querySelectorAll('.lbl-feed').forEach(e=>e.textContent=T.lbl_feed);
document.querySelectorAll('.lbl-unload').forEach(e=>e.textContent=T.lbl_unload);
+ setText('ams-no-data',T.ams_no_data);
+ setText('ams-slot-lbl',T.ams_slot_select);
// conn-btn text (nur wenn nicht im Übergangszustand)
updateConnBtn();
// Slot-Edit-Dialog
@@ -1601,6 +1819,11 @@ function applyLang(){
setText('btn-slot-edit-save',T.slot_edit_save);
var mi=document.getElementById('slot-edit-mat');if(mi)mi.setAttribute('placeholder',T.slot_edit_custom);
setText('logdir-all',T.log_dir_all);
+ setText('lbl-log-download',T.log_download);
+ setText('lbl-log-auto',T.log_auto);
+ setText('lbl-log-clear',T.log_clear);
+ setText('lbl-log-dir',T.log_dir);
+ setText('lbl-log-topic',T.log_topic);
setText('file-ready-btn',T.file_ready_btn);
setText('file-cancel-btn',T.file_cancel_btn);
}
@@ -1639,7 +1862,7 @@ var logTopicFilter=''; // '' = no topic filter
function clog(msg,cls){
cls=cls||'msg-info';
- var ts=new Date().toLocaleTimeString('de',{hour:'2-digit',minute:'2-digit',second:'2-digit'});
+ var ts=new Date().toLocaleTimeString(currentLang==='de'?'de':'en',{hour:'2-digit',minute:'2-digit',second:'2-digit'});
_appendLog({ts:ts,lvl:'',name:'ui',msg:msg},cls);
}
function _lvlCls(lvl){
@@ -1851,10 +2074,10 @@ function updateConnBtn(){
var offline=S.kobra_state==='offline';
if(offline){
btn.className='conn-btn disconnected';
- btn.textContent=T.btn_connect||'⚡ Verbinden';
+ btn.textContent=T.btn_connect||'⚡ Connect';
} else {
btn.className='conn-btn connected';
- btn.textContent=T.btn_disconnect||'✕ Trennen';
+ btn.textContent=T.btn_disconnect||'✕ Disconnect';
}
}
@@ -1996,7 +2219,7 @@ function saveSlotEdit(){
closeSlotEdit();
clog((T.slot_edit_ok||'AMS Slot')+' '+(_slotEditIndex+1)+': '+mat+' '+hex,'msg-ok');
})
- .catch(function(e){clog('Fehler: '+e,'msg-err');});
+ .catch(function(e){clog((T.log_error||'Error:')+' '+e,'msg-err');});
}
document.addEventListener('DOMContentLoaded',function(){
document.getElementById('s-printer-ip').addEventListener('input',function(){
@@ -2035,7 +2258,7 @@ function saveSettings(){
},4000);
}).catch(function(e){
btn.disabled=false;setText('btn-save-settings',T.settings_save);
- clog('Settings-Fehler: '+e,'msg-err');
+ clog((T.log_settings_error||'Settings error:')+' '+e,'msg-err');
});
}
function checkUpdate(){
@@ -2082,7 +2305,7 @@ async function poll(){
Object.assign(S,d);
applyState();
updateHistory();
- }catch(e){clog('Poll-Fehler: '+e,'msg-err')}
+ }catch(e){clog((T.log_poll_error||'Poll error:')+' '+e,'msg-err')}
}
var pollTimer;
(function(){
@@ -2092,10 +2315,10 @@ var pollTimer;
// ── Print actions ──
function printAction(a){
- post('/printer/print/'+a,{}).then(function(){clog('Druck: '+a,'msg-ok');poll()})
- .catch(function(e){clog('Fehler: '+e,'msg-err')});
+ post('/printer/print/'+a,{}).then(function(){clog((T.nav_print||'Print')+': '+(T['print_action_'+a]||a),'msg-ok');poll()})
+ .catch(function(e){clog((T.log_error||'Error:')+' '+e,'msg-err')});
}
-function confirmCancel(){if(confirm('Druck wirklich abbrechen?'))printAction('cancel')}
+function confirmCancel(){if(confirm(T.confirm_cancel||'Really cancel the print?'))printAction('cancel')}
// ── Axis motion ──
// axis codes: 0=X, 1=Y, 2=Z
@@ -2111,28 +2334,28 @@ function move(axis,dir,dist){
// axis: 0=X,1=Y,2=Z → printer axis codes: 1=X,2=Y,3=Z
var axisMap={0:1,1:2,2:3};
post('/api/axis',{axis:axisMap[axis],move_type:1,distance:dir*dist})
- .then(function(){clog('Achse '+(axis===0?'X':axis===1?'Y':'Z')+' '+(dir>0?'+':'')+dir*dist+'mm','msg-ok')})
- .catch(function(e){clog('Achse-Fehler: '+e,'msg-err')});
+ .then(function(){clog((T.log_axis||'Axis')+' '+(axis===0?'X':axis===1?'Y':'Z')+' '+(dir>0?'+':'')+dir*dist+'mm','msg-ok')})
+ .catch(function(e){clog((T.log_axis_error||'Axis error:')+' '+e,'msg-err')});
}
function homeAll(){
post('/api/axis',{axis:5,move_type:2,distance:0})
.then(function(){clog('Home All','msg-ok')})
- .catch(function(e){clog('Home-Fehler: '+e,'msg-err')});
+ .catch(function(e){clog((T.log_home_error||'Home error:')+' '+e,'msg-err')});
}
function homeXY(){
post('/api/axis',{axis:4,move_type:2,distance:0})
.then(function(){clog('Home XY','msg-ok')})
- .catch(function(e){clog('Home-Fehler: '+e,'msg-err')});
+ .catch(function(e){clog((T.log_home_error||'Home error:')+' '+e,'msg-err')});
}
function homeZ(){
post('/api/axis',{axis:3,move_type:2,distance:0})
.then(function(){clog('Home Z','msg-ok')})
- .catch(function(e){clog('Home-Fehler: '+e,'msg-err')});
+ .catch(function(e){clog((T.log_home_error||'Home error:')+' '+e,'msg-err')});
}
function disableMotors(){
post('/api/axis',{action:'turnOff'})
.then(function(){clog('Motors Off','msg-ok')})
- .catch(function(e){clog('Motors-Fehler: '+e,'msg-err')});
+ .catch(function(e){clog((T.log_motors_error||'Motor error:')+' '+e,'msg-err')});
}
// ── Temperature ──
@@ -2140,21 +2363,21 @@ function setNozzle(){
var v=parseFloat(document.getElementById('p-nozzle-inp').value||0);
post('/api/temperature',{nozzle:v,bed:S.bed_target})
.then(function(){clog('Nozzle → '+v+'°C','msg-ok')})
- .catch(function(e){clog('Temp-Fehler: '+e,'msg-err')});
+ .catch(function(e){clog((T.log_temp_error||'Temperature error:')+' '+e,'msg-err')});
}
function setBed(){
var v=parseFloat(document.getElementById('p-bed-inp').value||0);
post('/api/temperature',{nozzle:S.nozzle_target,bed:v})
.then(function(){clog(T.label_bed+' → '+v+'°C','msg-ok')})
- .catch(function(e){clog('Temp-Fehler: '+e,'msg-err')});
+ .catch(function(e){clog((T.log_temp_error||'Temperature error:')+' '+e,'msg-err')});
}
// ── Light ──
function setLight(){
var on=document.getElementById('d-light-toggle').checked;
post('/api/light',{on:on,brightness:80})
- .then(function(){clog('Licht '+(on?'an, '+br+'%':'aus'),'msg-ok')})
- .catch(function(e){clog('Licht-Fehler: '+e,'msg-err')});
+ .then(function(){clog(on?(T.log_light_on||'Light on'):(T.log_light_off||'Light off'),'msg-ok')})
+ .catch(function(e){clog((T.log_light_error||'Light error:')+' '+e,'msg-err')});
}
// ── Print Speed ──
@@ -2165,7 +2388,7 @@ function setSpeed(mode){
if(b) b.classList.toggle('spd-active',m===mode);
});
post('/api/speed',{mode:mode})
- .catch(function(e){clog('Speed-Fehler: '+e,'msg-err')});
+ .catch(function(e){clog((T.log_speed_error||'Speed error:')+' '+e,'msg-err')});
}
// ── Fan ──
@@ -2173,15 +2396,15 @@ function setFan(){
var v=parseInt(document.getElementById('d-fan').value);
document.getElementById('d-fan-val').textContent=v;
post('/api/fan',{speed:v})
- .then(function(){clog('Lüfter → '+v+'%','msg-ok')})
- .catch(function(e){clog('Lüfter-Fehler: '+e,'msg-err')});
+ .then(function(){clog((T.log_fan||'Fan →')+' '+v+'%','msg-ok')})
+ .catch(function(e){clog((T.log_fan_error||'Fan error:')+' '+e,'msg-err')});
}
function quickFan(v){
document.getElementById('d-fan').value=v;
document.getElementById('d-fan-val').textContent=v;
post('/api/fan',{speed:v})
- .then(function(){clog('Lüfter → '+v+'%','msg-ok')})
- .catch(function(e){clog('Lüfter-Fehler: '+e,'msg-err')});
+ .then(function(){clog((T.log_fan||'Fan →')+' '+v+'%','msg-ok')})
+ .catch(function(e){clog((T.log_fan_error||'Fan error:')+' '+e,'msg-err')});
}
// ── AMS ──
@@ -2189,7 +2412,7 @@ function amsFeed(type){
var slot=parseInt(document.getElementById('ams-slot-sel').value);
post('/api/ams/feed',{slot_index:slot,type:type})
.then(function(){clog((type===1?T.lbl_feed:T.lbl_unload)+' Slot '+(slot+1),'msg-ok')})
- .catch(function(e){clog('AMS-Fehler: '+e,'msg-err')});
+ .catch(function(e){clog((T.log_ams_error||'AMS error:')+' '+e,'msg-err')});
}
// ── Camera ──
@@ -2206,13 +2429,13 @@ function camStart(){
img.style.display='none';
ph.style.display='flex';
camOn=false;
- document.getElementById('cam-toggle-btn').textContent=T.btn_cam_start||'▶ Kamera';
- clog((T.log_error||'Fehler:')+' Stream nicht verfügbar','msg-err');
+ document.getElementById('cam-toggle-btn').textContent=T.btn_cam_start||'▶ Camera';
+ clog((T.log_error||'Error:')+' '+(T.log_stream_unavailable||'Stream unavailable'),'msg-err');
};
img.src='/api/camera/stream?t='+Date.now();
camOn=true;
- document.getElementById('cam-toggle-btn').textContent=T.btn_cam_stop||'◼ Kamera';
- clog((T.log_cam_start||'Kamera gestartet'),'msg-ok');
+ document.getElementById('cam-toggle-btn').textContent=T.btn_cam_stop||'◼ Camera';
+ clog((T.log_cam_start||'Camera started'),'msg-ok');
// MJPEG liefert kein onload – Spinner nach kurzem Timeout ausblenden
setTimeout(function(){
sp.style.display='none';
@@ -2221,7 +2444,7 @@ function camStart(){
}).catch(function(e){
sp.style.display='none';
ph.style.display='flex';
- clog((T.log_error||'Fehler:')+' '+e,'msg-err');
+ clog((T.log_error||'Error:')+' '+e,'msg-err');
});
}
function camStop(){
@@ -2232,9 +2455,9 @@ function camStop(){
document.getElementById('cam-spinner').style.display='none';
document.getElementById('cam-placeholder').style.display='flex';
camOn=false;
- document.getElementById('cam-toggle-btn').textContent=T.btn_cam_start||'▶ Kamera';
- clog(T.log_cam_stop||'Kamera gestoppt','msg-ok');
- }).catch(function(e){clog((T.log_error||'Fehler:')+' '+e,'msg-err')});
+ document.getElementById('cam-toggle-btn').textContent=T.btn_cam_start||'▶ Camera';
+ clog(T.log_cam_stop||'Camera stopped','msg-ok');
+ }).catch(function(e){clog((T.log_error||'Error:')+' '+e,'msg-err')});
}
function toggleCam(){if(camOn)camStop();else camStart()}
@@ -2597,6 +2820,7 @@ function toggleCam(){if(camOn)camStop();else camStart()}
"thumbnail": self._thumbnail_b64,
"connection_error": s["connection_error"],
"file_ready": s["file_ready"],
+ "uploaded_filaments": self._uploaded_metadata_for(s["file_ready"] or self._last_uploaded_file),
"version": self._read_version(),
})
@@ -2955,11 +3179,8 @@ function toggleCam(){if(camOn)camStop();else camStart()}
elif method == "printer.print.start":
filename = params.get("filename", self._last_uploaded_file)
loop = asyncio.get_event_loop()
- resp = await loop.run_in_executor(
- None, lambda: self.client.publish("print", "start",
- {"filename": filename, "use_ams": False}, timeout=15.0)
- )
- result = "ok" if resp else "timeout"
+ await loop.run_in_executor(None, lambda: self._start_print(filename))
+ result = "ok"
elif method == "printer.print.pause":
loop = asyncio.get_event_loop()
await loop.run_in_executor(None, self.client.pause_print)