Blender-Pipeline/studio/studio_asset_configureLayer.py

757 lines
26 KiB
Python
Raw Normal View History

2020-12-19 10:20:34 +00:00
# THIS FILE IS A PART OF VCStudio
# PYTHON 3
import os
import datetime
from subprocess import *
# GTK module ( Graphical interface
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
from gi.repository import GLib
from gi.repository import Gdk
import cairo
# Own modules
from settings import settings
from settings import talk
from settings import fileformats
from settings import oscalls
from project_manager import pm_project
#UI modules
from UI import UI_elements
from UI import UI_color
# Studio
from studio import studio_dialogs
from studio import analytics
from studio import story
2020-12-25 14:20:34 +00:00
from studio import history
2020-12-19 10:20:34 +00:00
def layer(win, call):
##########################################################################
# This file will do the asset configuration for linking assets into anima-
# tion files. The ways it's going to be performed is following:
# 0. User creates the asset. The /dev/ folder for the asset is created.
# 1. User finished the asset and the checklists is now 100%.
# 2. User Clicks on a configure button which launches this UI.
# 3. User choses the main blendfile. It is copied to the /ast/ folder.
# 4. User choses the collection and the rig with in the file. and it's
# saved into autolink.data
# 5. User creates a shot in the script editor. And creates a blend-file.
# 6. Use presses the link button on the blend file. And it loads the assets
# mentioned in the script.
# 7. User presses the okay button and a script is running inside blender
# that will read autolink.data files. And open /ast/ blend files. From
# which it will loaded the right collection and make library override or
# proxy to the rig. Both specified here by the user.
# See:
# studio/studio_shot_linkLayer.py
# studio/bpy_do_linking.py
# studio/studio_assetLayer.py
# The way this file going to work will be like a step by step wizzard.
# This file will be able to be openned from studio_shot_linkLayer dialog so
# there will be 3 different UIs in this file.
# 0. If checklist is not yet 100%. There will be a message to finish the
# asset first.
# 1. When the checklist is 100%. There will UI to select / copy /ast/file
# 2. When that's done. The program going to launch a script to read data
# from the newly created /ast/ file. And will present the user with
# selection of collections and objects from the collections. So to
# what to link and what to proxy.
##########################################################################
# Making the layer
surface = cairo.ImageSurface(cairo.FORMAT_ARGB32, win.current['w'],
win.current['h'])
layer = cairo.Context(surface)
#text setting
layer.select_font_face("Monospace", cairo.FONT_SLANT_NORMAL, cairo.FONT_WEIGHT_NORMAL)
UI_color.set(layer, win, "dark_overdrop")
layer.rectangle(
0,
0,
win.current["w"],
win.current["h"],
)
layer.fill()
UI_color.set(layer, win, "node_background")
UI_elements.roundrect(layer, win,
win.current["w"]/2-250,
100,
500,
win.current["h"]-200,
10)
2020-12-25 14:20:34 +00:00
# Documentation entry
def do():
def after(win, var):
pass
studio_dialogs.help(win, "help", after, SEARCH=talk.text("documentation_link_assets"))
UI_elements.roundrect(layer, win,
win.current["w"]/2-250,
win.current["h"]-140,
40,
40,
10,
do,
"question")
2020-12-19 10:20:34 +00:00
# Exit button
def do():
win.current["calls"][call]["var"] = False
UI_elements.roundrect(layer, win,
win.current["w"]/2+210,
win.current["h"]-140,
40,
40,
10,
button=do,
icon="cancel",
tip=talk.text("cancel"))
# Okay so before we going to clip everything. Let's make a little widget.
# this will be similar to the thing on top of shots. With the little round
# icons. And a some kind of progress bar with fat points. But this one
# will have only 3 points.
# Let's simplify the process a little bit
x = win.current["w"]/2-250
y = 100
width = 500
height = win.current["h"]-200
# Okay. let's draw the icons.
threeicons = {
"edit_mode":False,
"blender":False,
"link_configure":False
}
for numb, icon in enumerate(threeicons):
UI_elements.image(layer, win,"settings/themes/"\
+win.settings["Theme"]+"/icons/"+icon+".png",
x+(width/4)*numb+(width/8)+40,
y+20,
40,
40)
# Before we can do progress bar and or progress dots we need to check
# a few things about the asset.
# First is whether it's done.
threeicons["edit_mode"] = story.get_asset_data(win, win.current["asset_configure"]["asset"])["fraction"] == 1
# Then whether the ast exists.
threeicons["blender"] = os.path.exists(win.project+"/ast"+win.current["asset_configure"]["asset"]+".blend")
# Then whether the autolink.data exists.
threeicons["link_configure"] = os.path.exists(win.project+"/dev"+win.current["asset_configure"]["asset"]+"/autolink.data")
# Now let's manually make the fraction.
fraction = 0.0
if threeicons["edit_mode"]:
fraction = 0.33
if threeicons["blender"]:
fraction = 0.66
if threeicons["link_configure"]:
fraction = 1
UI_color.set(layer, win, "progress_background")
UI_elements.roundrect(layer, win,
x+20,
y+80,
width-40,
0,
5)
UI_color.set(layer, win, "progress_active")
UI_elements.roundrect(layer, win,
x+20,
y+80,
(width-40)*fraction,
0,
5)
# And the 3 dots
for numb, icon in enumerate(threeicons):
UI_color.set(layer, win, "progress_background")
if threeicons[icon]: # If this folder has any files.
UI_color.set(layer, win, "progress_active")
UI_elements.roundrect(layer, win,
x+(width/4)*numb+(width/8)+50,
y+75,
0,
0,
10)
# Now let's do 3 UIs.
################## STEP 1: MAKE ASSET ######################
if not threeicons["edit_mode"]:
fraction = story.get_asset_data(win, win.current["asset_configure"]["asset"])["fraction"]
UI_elements.text(layer, win, "asset_configuration_step",
x,
y+150,
width,
30,
10,
fill=False,
centered=True,
editable=False)
win.text["asset_configuration_step"]["text"] = talk.text("asset_configuration_step_1")
UI_elements.text(layer, win, "asset_configuration_step1",
x,
y+200,
width,
30,
10,
fill=False,
centered=True,
editable=False)
win.text["asset_configuration_step1"]["text"] = str(int(round(fraction*100)))+"%"
UI_color.set(layer, win, "progress_background")
UI_elements.roundrect(layer, win,
x+20,
y+250,
width-40,
0,
5)
UI_color.set(layer, win, "progress_active")
UI_elements.roundrect(layer, win,
x+20,
y+250,
(width-40)*fraction,
0,
5)
####################### STEP 2 : COPY BLEND-FILE TO AST ####################
# Setting up the scroll
if "asset_configure" not in win.scroll:
win.scroll["asset_configure"] = 0
current_Y = 310 # The max scroll value
if threeicons["edit_mode"] and not threeicons["blender"]:
UI_elements.text(layer, win, "asset_configuration_step",
x,
y+150,
width,
30,
10,
fill=False,
centered=True,
editable=False)
win.text["asset_configuration_step"]["text"] = talk.text("asset_configuration_step_2")
# So if the file is selected I want to have an apply button.
if win.current["asset_configure"]["blend_to_copy"]:
UI_color.set(layer, win, "text_normal")
layer.set_font_size(20)
layer.move_to(x+40,
y+height-20)
layer.show_text(win.current["asset_configure"]["blend_to_copy"])
def do():
copy_from = win.project+"/dev"+win.current["asset_configure"]["asset"]+"/"+win.current["asset_configure"]["blend_to_copy"]
copy_to = win.project+"/ast"+win.current["asset_configure"]["asset"]+".blend"
newname = copy_to[copy_to.rfind("/")+1:]
copy_to = copy_to[:copy_to.rfind("/")]
oscalls.copy_file(
win,
copy_from,
copy_to,
newname)
UI_elements.roundrect(layer, win,
win.current["w"]/2+170,
win.current["h"]-140,
40,
40,
10,
button=do,
icon="ok",
tip=talk.text("checked"))
# Here I want to draw a whole window with blend file selection and stuff.
# so here we go.
# Clipping everything
UI_elements.roundrect(layer, win,
win.current["w"]/2-250,
300,
500,
win.current["h"]-460,
10,
fill=False)
layer.clip()
clip = [
win.current["w"]/2-250,
300,
500,
win.current["h"]-460]
# Background
#UI_color.set(layer, win, "dark_overdrop")
#layer.rectangle(x, y,width,height)
#layer.fill()
# Now we draw all the blend files of the file. I know it's redundant but
# I want to be able to configure an asset from the linking window.
tileX = x+70
for filename in sorted(os.listdir(win.project+"/dev"+win.current["asset_configure"]["asset"])):
if filename.endswith(".blend"):
if int(current_Y + win.scroll["asset_configure"]) in range(0-200, height):
# Making the layer
node2surface = cairo.ImageSurface(cairo.FORMAT_ARGB32, 170, 200)
node2 = cairo.Context(node2surface)
node2.select_font_face("Monospace", cairo.FONT_SLANT_NORMAL, cairo.FONT_WEIGHT_NORMAL)
UI_elements.roundrect(node2, win,
0,
0,
170,
200,
10,
fill=False)
node2.clip()
# Background
UI_color.set(node2, win, "dark_overdrop")
node2.rectangle(0,0,170, 200)
node2.fill()
# Banner
UI_color.set(node2, win, "node_blendfile")
node2.rectangle(0,0,170, 20)
node2.fill()
# Outputting the layer
layer.set_source_surface(node2surface,
tileX-10,
current_Y + win.scroll["asset_configure"] )
layer.paint()
UI_elements.image(layer, win, win.project+"/dev"+win.current["asset_configure"]["asset"]+"/"+filename,
tileX,
current_Y + win.scroll["asset_configure"] + 30,
150,
150)
UI_color.set(layer, win, "text_normal")
layer.set_font_size(12)
layer.move_to(tileX,
current_Y + win.scroll["asset_configure"]+15)
layer.show_text(filename[filename.rfind("/")+1:][:22])
# If selected
layer.set_line_width(4)
if win.current["asset_configure"]["blend_to_copy"] == filename:
UI_color.set(layer, win, "progress_background")
UI_elements.roundrect(layer, win,
tileX-10,
current_Y + win.scroll["asset_configure"],
170,
200,
10,
fill=False)
layer.stroke()
# Button to activate it
def do():
win.current["asset_configure"]["blend_to_copy"] = filename
UI_elements.roundrect(layer, win,
tileX-10,
current_Y + win.scroll["asset_configure"],
170,
200,
10,
button=do,
tip=filename,
fill=False,
clip=clip)
layer.stroke()
layer.set_line_width(2)
tileX += 200
if tileX > 1000:
tileX = x+70
current_Y += 230
####################### STEP 3 : CONFIGURE AUTOLINK.DATA ####################
if threeicons["edit_mode"] and threeicons["blender"]:
UI_elements.text(layer, win, "asset_configuration_step",
x,
y+150,
width,
30,
10,
fill=False,
centered=True,
editable=False)
win.text["asset_configuration_step"]["text"] = talk.text("asset_configuration_step_3")
if win.current["asset_configure"]["apply"]:
def do():
# Here I'm going to actually write stuff to the file. It's not hard.
autolink = open(win.project+"/dev"+win.current["asset_configure"]["asset"]+"/autolink.data", "w")
cols = win.current["asset_configure"]["collections"]
for col in cols:
# If the collection is selected we write it.
if cols[col]["selected"]:
autolink.write("Link : "+col+"\n")
# Also we need to then look throught the collection to
# see if there are any proxy objects. Aramtures...
objs = cols[col]["objects"]
for obj in objs:
# Same thing if it's selected. Then write it to file.
if objs[obj]["selected"]:
autolink.write("Proxy : "+obj+"\n")
autolink.close()
2020-12-25 14:20:34 +00:00
# And a history entry real quick
history.record(win, "/dev"+win.current["asset_configure"]["asset"]+"/autolink.data", "[Updated]")
2020-12-19 10:20:34 +00:00
# And we need to quit the window. So...
win.current["calls"][call]["var"] = False
UI_elements.roundrect(layer, win,
win.current["w"]/2+170,
win.current["h"]-140,
40,
40,
10,
button=do,
icon="ok",
tip=talk.text("checked"))
# For this step I can't just copy stuff from assed layer and pretend that everything
# is good. Because this part requires a whole new script to write and test.
# See:
# studio/bpy_get_blend_content.py
# Now let's actually run this script quickly. Basically we need to get data out of
# the asset blend file. But of course not on every frame. It's a bit too much. So
# here is how we do it.
if not win.current["asset_configure"]["collections"]:
# If in the current data the data about the file still doesn't exists we load
# it.
blenderpath = oscalls.get_current_blender(win)
assetblend = win.project+"/ast"+win.current["asset_configure"]["asset"]+".blend"
checkframes = Popen([blenderpath, "-b", assetblend , "-P",
os.getcwd()+"/studio/bpy_get_blend_content.py"],stdout=PIPE, universal_newlines=True)
checkframes.wait()
checkstring = checkframes.stdout.read()
# Let's also see if the file autolink.data already there. Meaning I can get
# data that is already saved there.
linkcols = []
linkobjs = []
if threeicons["link_configure"]:
autolink = open(win.project+"/dev"+win.current["asset_configure"]["asset"]+"/autolink.data")
autolink = autolink.read()
autolink = autolink.split("\n")
for line in autolink:
if line.startswith("Link : "):
linkcols.append(line[7:])
elif line.startswith("Proxy : "):
linkobjs.append(line[8:])
else:
# If there is no file. I want to start the applying process already:
win.current["asset_configure"]["apply"] = True
# Now let's parse this crazy stuff that we've got from the blender script.
for line in checkstring.split("\n"):
if line.startswith(">>> "):
line = line[4:]
if "<==" in line:
col = line[:line.find(" <==")]
obj = line[line.find(" <==")+5:].split(" <== ")
else:
col = line
obj = False
selected = False
if col in win.current["asset_configure"]["asset"] and not threeicons["link_configure"]:
selected = True
elif col in linkcols:
selected = True
# So we found a collection. Let's write it in
if col not in win.current["asset_configure"]["collections"]:
win.current["asset_configure"]["collections"][col] = {
"selected":selected,
"objects":{}
}
# We also got the object. Let's put it inside the collection.
# But first. If it's an armature we want it to be selected.
if obj:
selected = False
if "Armature" in obj[1] and not threeicons["link_configure"]:
selected = True
elif obj[0] in linkobjs:
selected = True
win.current["asset_configure"]["collections"][col]["objects"][obj[0]] = {
"selected":selected,
"type":obj[1]
}
# Now let's draw the findings to the screen. But first we need
# buttons to see in what catergory are we.
buttons = ["collection", "rig"]
for num, button in enumerate(buttons):
if win.current["asset_configure"]["step3_button"] == button:
UI_color.set(layer, win, "progress_time")
UI_elements.roundrect(layer, win,
x+10+(40*num),
y+200,
40,
40,
10)
def do():
win.current["asset_configure"]["step3_button"] = button
UI_elements.roundrect(layer, win,
x+10+(40*num),
y+200,
40,
40,
10,
do,
button)
# Now we can clip the bastard
UI_elements.roundrect(layer, win,
win.current["w"]/2-250,
350,
500,
win.current["h"]-510,
10,
fill=False)
layer.clip()
clip = [
win.current["w"]/2-250,
350,
500,
win.current["h"]-510]
# Background
#UI_color.set(layer, win, "dark_overdrop")
#layer.rectangle(x, y,width,height)
#layer.fill()
# Let's start by drawing the collections first since it's the first thing
# that the user will be doing. Is selecting the collections.
current_Y = current_Y - 50
if win.current["asset_configure"]["step3_button"] == "collection":
for collection in win.current["asset_configure"]["collections"]:
# We are going to draw a little collection icon. And put a text
# with it's name.
UI_elements.image(layer, win,
"settings/themes/"+win.settings["Theme"]+"/icons/collection.png",
x+60, y+current_Y+win.scroll["asset_configure"], 40, 40)
UI_color.set(layer, win, "text_normal")
layer.set_font_size(20)
layer.move_to(x+120,
y+current_Y + win.scroll["asset_configure"]+30)
layer.show_text(collection)
# Now let's make a checkbox button beside each collection.
# Mimicing the Blender UI so to speak.
icon = "unchecked"
if win.current["asset_configure"]["collections"][collection]["selected"]:
icon = "checked"
def do():
win.current["asset_configure"]["collections"][collection]["selected"]\
= not win.current["asset_configure"]["collections"][collection]["selected"]
win.current["asset_configure"]["apply"] = True
UI_elements.roundrect(layer, win,
x+10,
y+current_Y + win.scroll["asset_configure"],
40,
40,
10,
do,
icon)
current_Y = current_Y + 50
else:
# If we are not doing the collection but instead we are selecting the
# rigs. We need to draw different stuff.
for collection in win.current["asset_configure"]["collections"]:
# But we want to draw only objects that are in selected colllections
# so here what we do.
if win.current["asset_configure"]["collections"][collection]["selected"]:
for obj in win.current["asset_configure"]["collections"][collection]["objects"]:
name = obj
obj = win.current["asset_configure"]["collections"][collection]["objects"][obj]
# Now first thing is we need to find an icon to represent
# this object.
icon = "obj"
if "Mesh" in obj["type"]:
icon = "mesh"
elif "Armature" in obj["type"]:
icon = "rig"
elif "Camera" in obj["type"]:
icon = "shot"
UI_elements.image(layer, win,
"settings/themes/"+win.settings["Theme"]+"/icons/"+icon+".png",
x+60, y+current_Y+win.scroll["asset_configure"], 40, 40)
UI_color.set(layer, win, "text_normal")
layer.set_font_size(20)
layer.move_to(x+120,
y+current_Y + win.scroll["asset_configure"]+30)
layer.show_text(name)
icon = "link"
if obj["selected"]:
icon = "override"
def do():
obj["selected"]\
= not obj["selected"]
win.current["asset_configure"]["apply"] = True
UI_elements.roundrect(layer, win,
x+10,
y+current_Y + win.scroll["asset_configure"],
40,
40,
10,
do,
icon)
current_Y = current_Y + 50
current_Y = current_Y - 300
###########################
UI_elements.scroll_area(layer, win, "asset_configure",
x,
y+300,
width,
height-400,
current_Y,
bar=True,
mmb=True,
url="asset_configure"
)
return surface