471 lines
18 KiB
Python
471 lines
18 KiB
Python
|
|
"""
|
|
Plugin for making an asset library of all blender file found in a folder
|
|
"""
|
|
|
|
|
|
from asset_library.adapters.adapter import AssetLibraryAdapter
|
|
from asset_library.common.bl_utils import load_datablocks
|
|
from asset_library.common.template import Template
|
|
|
|
import bpy
|
|
from bpy.props import (StringProperty, IntProperty, BoolProperty)
|
|
import re
|
|
from pathlib import Path
|
|
from itertools import groupby
|
|
import uuid
|
|
import os
|
|
import shutil
|
|
import json
|
|
import time
|
|
|
|
|
|
class ScanFolderLibrary(AssetLibraryAdapter):
|
|
|
|
name = "Scan Folder"
|
|
source_directory : StringProperty(subtype='DIR_PATH')
|
|
template_file : StringProperty()
|
|
template_image : StringProperty()
|
|
template_video : StringProperty()
|
|
template_description : StringProperty()
|
|
#blend_depth : IntProperty()
|
|
#externalize_preview : BoolProperty(default=True)
|
|
|
|
#def draw_header(self, layout):
|
|
# '''Draw the header of the Asset Browser Window'''
|
|
# layout.separator()
|
|
# layout.operator("actionlib.store_anim_pose", text='Add Action', icon='FILE_NEW')
|
|
|
|
#def update(self):
|
|
#
|
|
def get_asset_path(self, name, catalog, directory=None):
|
|
directory = directory or self.source_directory
|
|
catalog = self.norm_file_name(catalog)
|
|
name = self.norm_file_name(name)
|
|
|
|
return Path(directory, self.get_asset_relative_path(name, catalog))
|
|
|
|
def get_image_path(self, name, catalog, filepath):
|
|
catalog = self.norm_file_name(catalog)
|
|
name = self.norm_file_name(name)
|
|
return self.format_path(self.template_image, dict(name=name, catalog=catalog, filepath=filepath))
|
|
|
|
def get_video_path(self, name, catalog, filepath):
|
|
catalog = self.norm_file_name(catalog)
|
|
name = self.norm_file_name(name)
|
|
return self.format_path(self.template_video, dict(name=name, catalog=catalog, filepath=filepath))
|
|
|
|
'''
|
|
def get_asset_description(self, asset, catalog, modified):
|
|
|
|
asset_path = self.get_asset_relative_path(name=asset.name, catalog=catalog)
|
|
asset_name = self.norm_file_name(asset.name)
|
|
|
|
asset_description = dict(
|
|
filepath='{source_directory}/' + asset_path.as_posix(),
|
|
modified=modified,
|
|
library_id=self.library.id,
|
|
assets=[]
|
|
)
|
|
|
|
asset_description['assets'].append(dict(
|
|
catalog=catalog,
|
|
metadata=dict(asset.asset_data),
|
|
tags=asset.asset_data.tags.keys(),
|
|
type=self.data_type,
|
|
image=str(self.template_image.format(name=asset_name)),
|
|
video=str(self.template_video.format(name=asset_name)),
|
|
name=asset.name)
|
|
)
|
|
|
|
return asset_description
|
|
'''
|
|
|
|
def get_asset_description(self, data, asset_path):
|
|
|
|
asset_path = self.prop_rel_path(asset_path, 'source_directory')
|
|
modified = data.get('modified', time.time_ns())
|
|
|
|
if self.data_type == 'FILE':
|
|
return dict(
|
|
filepath=asset_path,
|
|
author=data.get('author'),
|
|
modified=modified,
|
|
catalog=data['catalog'],
|
|
tags=[],
|
|
type=self.data_type,
|
|
image=self.template_image,
|
|
name=data['name']
|
|
)
|
|
|
|
return dict(
|
|
filepath=asset_path,
|
|
modified=modified,
|
|
library_id=self.library.id,
|
|
assets=[dict(
|
|
catalog=asset_data['catalog'],
|
|
author=data.get('author'),
|
|
metadata=asset_data.get('metadata', {}),
|
|
description=asset_data.get('description'),
|
|
tags=asset_data.get('tags', []),
|
|
type=self.data_type,
|
|
image=self.template_image,
|
|
video=self.template_video,
|
|
name=asset_data['name']) for asset_data in data['assets']
|
|
]
|
|
)
|
|
|
|
def _find_blend_files(self):
|
|
'''Get a sorted list of all blender files found matching the template'''
|
|
template = Template(self.template)
|
|
|
|
print(f'Search for blend using glob template: {template.glob_pattern}')
|
|
|
|
source_directory = Path(os.path.expandvars(self.source_directory))
|
|
print(f'Scanning Folder {source_directory}...')
|
|
blend_files = list(source_directory.glob(template.glob_pattern))
|
|
|
|
blend_files.sort()
|
|
|
|
return blend_files
|
|
|
|
'''
|
|
def _group_key(self, asset_data):
|
|
"""Group assets inside one blend"""
|
|
|
|
catalog_parts = asset_data['catalog'].split('/') + [asset_data['name']]
|
|
|
|
return catalog_parts[:self.blend_depth]
|
|
|
|
def bundle(self, cache_diff=None):
|
|
"""Group all asset in one or multiple blends for the asset browser"""
|
|
|
|
if self.data_type not in ('FILE', 'ACTION'):
|
|
print(f'{self.data_type} is not supported yet')
|
|
return
|
|
|
|
lib_path = self.library_path
|
|
catalog_data = self.read_catalog() # TODO remove unused catalog
|
|
|
|
#asset_file_datas = self.fetch() # TODO replace to only change new assets
|
|
|
|
if not cache_diff:
|
|
# Get list of all modifications
|
|
cache, cache_diff = self.diff()
|
|
self.write_cache(cache)
|
|
|
|
elif isinstance(cache_diff, (Path, str)):
|
|
cache_diff = json.loads(Path(cache_diff).read_text(encoding='utf-8'))
|
|
|
|
if self.blend_depth == 0:
|
|
groups = [(cache_diff)]
|
|
else:
|
|
cache_diff.sort(key=self._group_key)
|
|
groups = groupby(cache_diff, key=self._group_key)
|
|
|
|
# #print(cache_diff)
|
|
# print('\n')
|
|
# for sub_path, asset_datas in groups:
|
|
|
|
# print('\n')
|
|
# print(f'{sub_path=}')
|
|
# print(f'asset_datas={list(asset_datas)}')
|
|
|
|
# raise Exception()
|
|
|
|
#progress = 0
|
|
total_assets = len(cache_diff)
|
|
print(f'total_assets={total_assets}')
|
|
|
|
if total_assets == 0:
|
|
print('No assets found')
|
|
return
|
|
|
|
i = 0
|
|
for sub_path, asset_datas in groups:
|
|
|
|
# print('\n')
|
|
# print(f'{sub_path=}')
|
|
# print(f'asset_datas={list(asset_datas)}')
|
|
|
|
# print('\n')
|
|
|
|
blend_name = sub_path[-1].replace(' ', '_').lower()
|
|
blend_path = Path(lib_path, *sub_path, blend_name).with_suffix('.blend')
|
|
|
|
if blend_path.exists():
|
|
print(f'Opening existing bundle blend: {blend_path}')
|
|
bpy.ops.wm.open_mainfile(filepath=str(blend_path))
|
|
else:
|
|
print(f'Create new bundle blend to: {blend_path}')
|
|
bpy.ops.wm.read_homefile(use_empty=True)
|
|
|
|
for asset_data in asset_datas:
|
|
if total_assets <= 100 or i % int(total_assets / 10) == 0:
|
|
print(f'Progress: {int(i / total_assets * 100)+1}')
|
|
|
|
operation = asset_data.get('operation', 'ADD')
|
|
asset = getattr(bpy.data, self.data_types).get(asset_data['name'])
|
|
|
|
if operation == 'REMOVE':
|
|
if asset:
|
|
getattr(bpy.data, self.data_types).remove(asset)
|
|
else:
|
|
print(f'ERROR : Remove Asset: {asset_data["name"]} not found in {blend_path}')
|
|
continue
|
|
|
|
elif operation == 'MODIFY':
|
|
if not asset:
|
|
print(f'WARNING: Modifiy Asset: {asset_data["name"]} not found in {blend_path} it will be created')
|
|
|
|
elif operation == 'ADD' or not asset:
|
|
if asset:
|
|
#raise Exception(f"Asset {asset_data['name']} Already in Blend")
|
|
getattr(bpy.data, self.data_types).remove(asset)
|
|
|
|
#print(f"INFO: Add new asset: {asset_data['name']}")
|
|
asset = getattr(bpy.data, self.data_types).new(name=asset_data['name'])
|
|
else:
|
|
print(f'operation {operation} not supported should be in (ADD, REMOVE, MODIFIED)')
|
|
continue
|
|
|
|
asset.asset_mark()
|
|
|
|
# Load external preview if exists
|
|
#template_image = Template(asset_data['preview'])
|
|
image_path = Path(asset_data['image'])
|
|
if not image_path.is_absolute():
|
|
image_path = Path(asset_data['filepath'], image_path)
|
|
|
|
image_path = self.format_path(image_path.as_posix())
|
|
|
|
if image_path and image_path.exists():
|
|
with bpy.context.temp_override(id=asset):
|
|
bpy.ops.ed.lib_id_load_custom_preview(
|
|
filepath=str(image_path)
|
|
)
|
|
#else:
|
|
# print(f'Preview {image_path} not found for asset {asset}')
|
|
|
|
asset.asset_data.description = asset_data.get('description', '')
|
|
|
|
catalog_name = asset_data['catalog']
|
|
catalog = catalog_data.get(catalog_name)
|
|
if not catalog:
|
|
catalog = {'id': str(uuid.uuid4()), 'name': catalog_name}
|
|
catalog_data[catalog_name] = catalog
|
|
|
|
asset.asset_data.catalog_id = catalog['id']
|
|
|
|
metadata = asset_data.get('metadata', {})
|
|
|
|
library_id = self.library.id
|
|
if 'library_id' in asset_data:
|
|
library_id = asset_data['library_id']
|
|
|
|
metadata['.library_id'] = library_id
|
|
|
|
#print(metadata)
|
|
|
|
metadata['filepath'] = asset_data['filepath']
|
|
for k, v in metadata.items():
|
|
asset.asset_data[k] = v
|
|
|
|
# Set tags if specified the asset_description
|
|
tags = asset_data.get('tags', [])
|
|
if tags:
|
|
for tag in asset.asset_data.tags[:]:
|
|
asset.asset_data.tags.remove(tag)
|
|
|
|
for tag in tags:
|
|
if not tag:
|
|
continue
|
|
asset.asset_data.tags.new(tag, skip_if_exists=True)
|
|
|
|
i += 1
|
|
|
|
print(f'Saving Blend to {blend_path}')
|
|
|
|
blend_path.parent.mkdir(exist_ok=True, parents=True)
|
|
bpy.ops.wm.save_as_mainfile(filepath=str(blend_path), compress=True)
|
|
|
|
self.write_catalog(catalog_data)
|
|
|
|
bpy.ops.wm.quit_blender()
|
|
'''
|
|
|
|
'''
|
|
def conform(self, directory, templates):
|
|
"""Split each assets per blend and externalize preview"""
|
|
|
|
print(f'Conforming {self.library.name} to {directory}')
|
|
|
|
if self.data_type not in ('FILE', 'ACTION'):
|
|
print(f'{self.data_type} is not supported yet')
|
|
return
|
|
|
|
#lib_path = self.library_path
|
|
source_directory = Path(os.path.expandvars(self.source_directory))
|
|
catalog_data = self.read_catalog(filepath=source_directory)
|
|
catalog_ids = {v['id']: {'path': k, 'name': v['name']} for k,v in catalog_data.items()}
|
|
directory = Path(directory).resolve()
|
|
|
|
template_image = templates.get('image') or self.template_image
|
|
template_video = templates.get('video') or self.template_video
|
|
|
|
# Get list of all modifications
|
|
for blend_file in self._find_blend_files():
|
|
|
|
modified = blend_file.stat().st_mtime_ns
|
|
|
|
print(f'Scanning blendfile {blend_file}...')
|
|
with bpy.data.libraries.load(str(blend_file), link=True, assets_only=True) as (data_from, data_to):
|
|
asset_names = getattr(data_from, self.data_types)
|
|
print(f'Found {len(asset_names)} {self.data_types} inside')
|
|
|
|
setattr(data_to, self.data_types, asset_names)
|
|
|
|
assets = getattr(data_to, self.data_types)
|
|
#print('assets', assets)
|
|
|
|
for asset in assets:
|
|
#TODO options for choose beetween asset catalog and filepath directory
|
|
asset_catalog_data = catalog_ids.get(asset.asset_data.catalog_id)
|
|
|
|
if not asset_catalog_data:
|
|
print(f'No catalog found for asset {asset.name}')
|
|
asset_catalog_data = {"path": blend_file.parent.relative_to(source_directory).as_posix()}
|
|
|
|
catalog_path = asset_catalog_data['path']
|
|
|
|
asset_path = self.get_asset_path(name=asset.name, catalog=catalog_path, directory=directory)
|
|
asset_description = self.get_asset_description(asset, catalog=catalog_path, modified=modified)
|
|
|
|
self.write_description_file(asset_description, asset_path)
|
|
#Write blend file containing only one asset
|
|
self.write_asset(asset=asset, asset_path=asset_path)
|
|
|
|
# Copy image if source image found else write the asset preview
|
|
src_image_path = self.get_path('image', name=asset.name, asset_path=blend_file, template=template_image)
|
|
dst_image_path = self.get_path('image', name=asset.name, asset_path=asset_path)
|
|
|
|
if src_image_path.exists():
|
|
self.copy_file(src_image_path, dst_image_path)
|
|
else:
|
|
self.write_preview(asset.preview, dst_image_path)
|
|
|
|
# Copy video if source video found
|
|
src_video_path = self.get_path('video', name=asset.name, asset_path=blend_file, template=template_video)
|
|
|
|
#print('src_video_path', src_video_path)
|
|
if src_video_path.exists():
|
|
dst_video_path = self.get_path('video', name=asset.name, asset_path=asset_path)
|
|
self.copy_file(src_video_path, dst_video_path)
|
|
|
|
self.write_catalog(catalog_data, filepath=directory)
|
|
'''
|
|
def fetch(self):
|
|
"""Gather in a list all assets found in the folder"""
|
|
|
|
print(f'Fetch Assets for {self.library.name}')
|
|
|
|
source_directory = Path(self.source_directory)
|
|
template_file = Template(self.template_file)
|
|
catalog_data = self.read_catalog(directory=source_directory)
|
|
catalog_ids = {v['id']: k for k, v in catalog_data.items()}
|
|
|
|
cache = self.read_cache() or []
|
|
|
|
print(f'Search for blend using glob template: {template_file.glob_pattern}')
|
|
|
|
print(f'Scanning Folder {source_directory}...')
|
|
#blend_files = list(source_directory.glob(template.glob_pattern))
|
|
|
|
# Remove delete blends for the list
|
|
#blend_paths = [self.prop_rel_path(f, 'source_directory') for f in blend_files]
|
|
#print('blend_paths', blend_paths)
|
|
|
|
|
|
#cache = []
|
|
#blend_paths = []
|
|
new_cache = []
|
|
|
|
for asset_path in template_file.glob(source_directory):#sorted(blend_files):
|
|
|
|
source_rel_path = self.prop_rel_path(asset_path, 'source_directory')
|
|
modified = asset_path.stat().st_mtime_ns
|
|
|
|
# Check if the asset description as already been cached
|
|
asset_description = next((a for a in cache if a['filepath'] == source_rel_path), None)
|
|
|
|
if asset_description and asset_description['modified'] >= modified:
|
|
print(asset_path, 'is skipped because not modified')
|
|
new_cache.append(asset_description)
|
|
continue
|
|
|
|
rel_path = asset_path.relative_to(source_directory).as_posix()
|
|
field_data = template_file.parse(rel_path)
|
|
|
|
catalogs = [v for k,v in sorted(field_data.items()) if k.isdigit()]
|
|
catalogs = [c.replace('_', ' ').title() for c in catalogs]
|
|
|
|
asset_name = field_data.get('asset_name', asset_path.stem)
|
|
|
|
asset_datas = {
|
|
"name": asset_name,
|
|
"catalog": '/'.join(catalogs),
|
|
"assets": [],
|
|
'modified': modified
|
|
}
|
|
|
|
if self.data_type == 'FILE':
|
|
asset_description = self.get_asset_description(asset_datas, asset_path)
|
|
new_cache.append(asset_description)
|
|
continue
|
|
|
|
# Now check if there is a asset description file
|
|
asset_description_path = self.find_path(self.template_description, asset_datas, filepath=asset_path)
|
|
if asset_description_path:
|
|
new_cache.append(self.read_file(asset_description_path))
|
|
continue
|
|
|
|
# Scan the blend file for assets inside and write a custom asset description for info found
|
|
print(f'Scanning blendfile {asset_path}...')
|
|
assets = self.load_datablocks(asset_path, type=self.data_types, link=True, assets_only=True)
|
|
print(f'Found {len(assets)} {self.data_types} inside')
|
|
|
|
for asset in assets:
|
|
#catalog_path = catalog_ids.get(asset.asset_data.catalog_id)
|
|
|
|
#if not catalog_path:
|
|
# print(f'No catalog found for asset {asset.name}')
|
|
catalog_path = asset_datas['catalog']#asset_path.relative_to(self.source_directory).as_posix()
|
|
|
|
asset_datas['assets'] += [dict(
|
|
catalog=catalog_path,
|
|
tags=asset.asset_data.tags.keys(),
|
|
metadata=dict(asset.asset_data),
|
|
type=self.data_type,
|
|
name=asset.name
|
|
)]
|
|
|
|
getattr(bpy.data, self.data_types).remove(asset)
|
|
|
|
asset_description = self.get_asset_description(asset_datas, asset_path)
|
|
|
|
new_cache.append(asset_description)
|
|
|
|
|
|
#cache = [a for a in cache if a['filepath'] in blend_paths]
|
|
|
|
#for a in asset_data:
|
|
# print(a)
|
|
|
|
#print(asset_data)
|
|
new_cache.sort(key=lambda x:x['filepath'])
|
|
|
|
return new_cache
|
|
|
|
# Write json data file to store all asset found
|
|
#print(f'Writing asset data file to, {asset_data_path}')
|
|
#asset_data_path.write_text(json.dumps(asset_data, indent=4)) |