-
-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathbrowseasset.py
191 lines (147 loc) · 5.88 KB
/
browseasset.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
import sys
from tkinter import EventType, Tk, ttk # type: ignore
from typing import Any, Dict, Optional
from automate.ark import ArkSteamManager
from config import get_global_config
from ue.base import UEBase
from ue.loader import AssetNotFound
from ue.paths import find_asset_from_external_path
root: Optional[Tk] = None
tree: Optional[ttk.Treeview] = None
treenodes: Dict[str, Any] = {}
LEAF_TYPES = (
'str',
'StringProperty',
'Guid',
)
def create_ui():
global root, tree # pylint: disable=global-statement
# Window
root = Tk()
root.columnconfigure(0, weight=1)
root.rowconfigure(0, weight=1)
root.minsize(width=600, height=400)
root.geometry('1100x900')
# Grid-based layout frame
frame = ttk.Frame(root, padding="6 6 6 6")
frame.grid(column=0, row=0, sticky='nsew')
frame.columnconfigure(0, weight=1)
frame.rowconfigure(0, weight=1)
# The tree view
tree = ttk.Treeview(frame, columns=('type', 'value'))
tree.grid(column=0, row=0, sticky='nsew')
tree.columnconfigure(0, weight=1)
tree.rowconfigure(0, weight=1)
tree.column('#0', width=260, stretch=0)
tree.heading('type', text='Type')
tree.heading('value', text='Value')
tree.column('type', width=110, stretch=False, anchor='e')
tree.column('value', anchor='w')
# Styles for some different types
# https://lospec.com/palette-list/island-joy-16
tree.tag_configure('int', foreground='#cb4d68')
tree.tag_configure('bool', foreground='#393457')
tree.tag_configure('Guid', foreground='#1e8875')
tree.tag_configure('NameIndex', foreground='#11adc1')
tree.tag_configure('StringProperty', foreground='#11adc1')
tree.tag_configure('Property', foreground='#393457')
# Support virtual tree items with the 'placeholdered' tag
tree.tag_bind('placeholdered', '<<TreeviewOpen>>', on_tree_open)
# Scroll bar to control the treeview
vsb = ttk.Scrollbar(frame, orient="vertical", command=tree.yview)
vsb.grid(row=0, column=1, sticky='nse')
tree.configure(yscrollcommand=vsb.set)
def on_tree_open(evt: EventType):
assert tree
# This is called when a node containing a placeholder is opened for the first time
itemId = tree.selection()[0]
# node = treenodes[itemId]
# Remove the placeholder item under this item
tree.delete(itemId + '_placeholder')
# Remove placeholdered tag
tags = list(tree.item(itemId, 'tags'))
tags.remove('placeholdered')
tree.item(itemId, tags=tags)
# Convert and insert the proper nodes
insert_fields_for_node(itemId)
def node_id(node, parentId=None):
'''Create a string ID for a node.'''
if isinstance(node, UEBase):
newId = str(id(node))
if parentId:
return parentId + '_' + newId
return newId
return None
def type_name(value):
'''Return the type of a value as a string.'''
return str(type(value).__name__)
def has_children(value):
valueType = type_name(value)
if valueType in LEAF_TYPES:
return False
if isinstance(value, UEBase) or type(value) == list: # isinstance(value, Iterable)
return True
return False
def get_node_iterator(node):
if isinstance(node, list):
return ((f'0x{i:X} ({i})', value) for i, value in enumerate(node))
if isinstance(node, UEBase):
return ((name, node.field_values[name]) for name in getattr(node, 'field_order', None) or node.field_values.keys())
raise TypeError("Invalid node type for iterator")
def add_placeholder_node(itemId):
# Add a tag to specifiy this node as having a placeholder (so it will raise an event when opened)
tags = tree.item(itemId)['tags']
if not hasattr(tags, 'append'):
tags = [tags]
tags.append('placeholdered')
tree.item(itemId, tags=tags)
# Add a dummy node to it so it look slike it can be opened
placeholderId = tree.insert(itemId, 'end', itemId + '_placeholder', text='<virtual tree placeholder>')
return placeholderId
def add_asset_to_root(asset):
itemId = tree.insert('', 'end', node_id(asset), text=asset.name, values=(type_name(asset), ))
treenodes[itemId] = asset
insert_fields_for_node(itemId)
tree.item(itemId, open=True)
def get_value_as_string(value):
if isinstance(value, list):
return f'{len(value)} entries'
if isinstance(value, type):
return value.__name__
return str(value)
def insert_fields_for_node(parentId):
node = treenodes[parentId]
skip_level_name = getattr(node, 'skip_level_field', None)
if skip_level_name:
node = node.field_values.get(skip_level_name, node)
fields = get_node_iterator(node)
for name, value in fields:
typeName = str(type(value).__name__)
newId = node_id(value, parentId)
strValue = get_value_as_string(value)
itemId = tree.insert(parentId, 'end', newId, text=name, values=(typeName, strValue), tags=(typeName, ))
if has_children(value):
treenodes[itemId] = value
add_placeholder_node(itemId)
def load_asset(assetname):
assetname = loader.clean_asset_name(assetname)
root.title("Asset Browser : " + assetname)
asset = loader[assetname]
add_asset_to_root(asset)
def find_asset(assetname, loader):
try:
assetname = find_asset_from_external_path(assetname, loader, True)
except AssetNotFound:
print(f'Not found: {assetname}', file=sys.stderr)
sys.exit(404)
return assetname
if __name__ == '__main__':
arkman = ArkSteamManager()
loader = arkman.getLoader()
config = get_global_config()
assetname = sys.argv[1] if len(sys.argv) > 1 else None
create_ui()
assert root
assetname = find_asset(assetname, loader)
load_asset(assetname)
root.mainloop()