import os, time, zipfile from pymxs import runtime as rt try: from PySide6 import QtWidgets except ImportError: from PySide2 import QtWidgets def mock_connect_api(ui): if len(ui.e_hash.text()) < 5: QtWidgets.QMessageBox.warning(ui, "Acesso Negado", "Hash inválida."); return ui.b_conn.setText("Conectando..."); QtWidgets.QApplication.processEvents(); time.sleep(1) for w in [ui.c_chn, ui.c_pst, ui.e_tit, ui.e_mp3, ui.b_mp3]: w.setEnabled(True); w.setStyleSheet("background: white; color: black; font-weight: bold;") ui.c_chn.clear(); ui.c_chn.addItems(["Canal: Imóveis Virtuais", "Canal: Showroom Zombisco"]) ui.c_pst.clear(); ui.c_pst.addItems(["[ + Cadastrar Nova Postagem ]", "Editar: Apartamento V1"]) ui.chk_up.setEnabled(True); ui.chk_up.setStyleSheet("color: #FFD700; font-weight: bold;"); ui.chk_up.setChecked(True) ui.b_conn.setText("✅ Conectado!"); ui.b_conn.setStyleSheet("background: #000; color: #0F0; font-weight: bold; border: 1px solid #0F0;") on_post_changed(ui, 0) def on_post_changed(ui, idx): if idx == 0: ui.e_tit.setText(""); ui.e_tit.setPlaceholderText("Título do projeto...") ui.chk_tmb.setChecked(True); ui.chk_tmb.setEnabled(False); ui.chk_tmb.setStyleSheet("color: #00FF00; font-weight: bold;") else: ui.e_tit.setText(ui.c_pst.currentText().replace("Editar: ", "")) ui.chk_tmb.setEnabled(True); ui.chk_tmb.setStyleSheet("color: white; font-weight: bold;") def finalize_export(ui, p_bk, p_glb): if not os.path.exists(p_glb): os.makedirs(p_glb) fo = []; rt.execute("max modify mode"); tgs = ui.get_processable_items() ui.pb.setFormat("Preparando materiais VR (Multi-ID)..."); QtWidgets.QApplication.processEvents() for d in tgs: if ui._is_cancelled: break if "Bake" in d['item'].text(1) or "Já existe" in d['item'].text(1) or "Mat" in d['item'].text(1): try: max_id = rt.execute(f"""(local o = getNodeByName "{d['name']}"; local mid = 1; if o != undefined and isKindOf o Editable_Poly do ( for f = 1 to (polyop.getNumFaces o) do ( local id = polyop.getFaceMatID o f; if id > mid do mid = id ) ); mid)""") if max_id > 1: # CONSTRUTOR DE MULTI-MATERIAL (Para objetos que foram processados pelo P4: Multi-UV) ms_mat = f"""( local o = getNodeByName "{d['name']}" local mm = MultiMaterial numsubs:{int(max_id)} name:(o.name + "_MultiMat") for i = 1 to {int(max_id)} do ( local ip = @"{p_bk}/{d['name']}_B_ID" + (i as string) + ".jpg" if doesFileExist ip do ( local m = PhysicalMaterial name:(o.name + "_MatID_" + (i as string)) m.base_color_map = BitmapTexture filename:ip m.roughness = 1.0; m.metalness = 0.0; try(m.reflectivity = 0.0; m.reflection_weight = 0.0)catch() mm[i] = m ) ) o.material = mm )""" rt.execute(ms_mat) d['item'].setText(1, "Mat OK (Multi)") fo.append(rt.getNodeByName(d['name'])) else: # MATERIAL PADRÃO (Para objetos normais que cabem numa textura só) ip = os.path.join(p_bk, f"{d['name']}_B.jpg").replace("\\", "/") if os.path.exists(ip): rt.execute(f"""( local o = getNodeByName "{d['name']}" if o != undefined do ( local m = PhysicalMaterial name:(o.name + "_Mat"); m.base_color_map = BitmapTexture filename:@"{ip}" m.roughness = 1.0; m.metalness = 0.0; try(m.reflectivity = 0.0; m.reflection_weight = 0.0)catch() o.material = m ) )""") fo.append(rt.getNodeByName(d['name'])); d['item'].setText(1, "Mat OK") except Exception as e: d['item'].setText(1, "Erro Mat"); print(f"Erro Material em {d['name']}: {e}") if not fo and not ui._is_cancelled: QtWidgets.QMessageBox.warning(ui, "Exportação Interrompida", "Nenhum objeto validado para exportação.") ui.pb.setFormat("Pronto"); ui.pb.setValue(0) return if fo and not ui._is_cancelled: rt.select(fo); sn = rt.maxFileName.split(".")[0] if rt.maxFileName else "Cena_VR4LIFE" og = os.path.join(p_glb, f"{sn}.glb").replace("\\", "/"); oz = os.path.join(p_glb, f"{sn}.zip").replace("\\", "/") tp = os.path.join(p_glb, f"{sn}_Thumb.jpg").replace("\\", "/"); mp = ui.e_mp3.text(); hm = os.path.exists(mp) to = False if ui.chk_tmb.isChecked() and ui.chk_tmb.isEnabled(): ui.pb.setFormat("📸 Thumbnail (1280x720)..."); QtWidgets.QApplication.processEvents() rt.execute(f"""( local c = getActiveCamera(); if c == undefined do ( local ac = for cam in cameras where (classof cam != TargetObject) collect cam; if ac.count > 0 do c = ac[1] ); try ( if c != undefined then ( render camera:c outputwidth:1280 outputheight:720 outputfile:@"{tp}" vfb:false quiet:true ) else ( render outputwidth:1280 outputheight:720 outputfile:@"{tp}" vfb:false quiet:true ) ) catch() )""") if os.path.exists(tp): to = True ui.pb.setFormat("Limpando Hierarquia e XForm..."); QtWidgets.QApplication.processEvents() rt.execute(""" ( local sel = selection as array for o in sel do ( o.parent = undefined ResetXForm o collapseStack o if not isKindOf o Editable_Poly do try(convertToPoly o)catch() ) ) """) ui.pb.setFormat("Montando GLB..."); QtWidgets.QApplication.processEvents() try: rt.exportFile(og, rt.name("noPrompt"), selectedOnly=True, using=rt.GLTFExporter) except: rt.execute(f'exportFile "{og}" #noPrompt selectedOnly:true') zo = False if os.path.exists(og): ui.pb.setFormat("🗜️ ZIPando GLB..."); QtWidgets.QApplication.processEvents() try: with zipfile.ZipFile(oz, 'w', zipfile.ZIP_DEFLATED) as zf: zf.write(og, os.path.basename(og)) zo = True except: pass if ui.chk_up.isChecked() and ui.chk_up.isEnabled() and zo: cs = ui.c_chn.currentText(); ac = "NOVO" if ui.c_pst.currentIndex() == 0 else "ATUALIZAR"; tit = ui.e_tit.text() ui.pb.setFormat(f"📡 API: Enviando..."); QtWidgets.QApplication.processEvents(); time.sleep(1.5) msg = f"Sincronizado!\n\n📡 Canal: {cs}\n📝 Ação: {ac}\n🏷️ Titulo: {tit}\n\n📦 ZIP: {os.path.basename(oz)}\n📸 Thumb: {'Sim' if to else 'Não'}\n🎵 Áudio: {'Sim' if hm else 'Não'}" QtWidgets.QMessageBox.information(ui, "Cloud API", msg) else: msg = f"Exportado:\n{og}"; if zo: msg += f"\n\n🗜️ ZIP:\n{oz}" if to: msg += f"\n\n📸 Thumb:\n{tp}" QtWidgets.QMessageBox.information(ui, "Sucesso", msg) ui.pb.setFormat("Pronto"); ui.pb.setValue(0)