2016-07-20 16:32:01 +02:00
|
|
|
# ##### BEGIN GPL LICENSE BLOCK #####
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU General Public License
|
|
|
|
# as published by the Free Software Foundation; either version 2
|
|
|
|
# of the License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program; if not, write to the Free Software Foundation,
|
|
|
|
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
#
|
|
|
|
# ##### END GPL LICENSE BLOCK #####
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
"""Blender-specific code.
|
|
|
|
|
|
|
|
Separated from __init__.py so that we can import & run from non-Blender environments.
|
|
|
|
"""
|
|
|
|
|
2016-04-13 15:37:19 +02:00
|
|
|
import logging
|
2016-07-08 17:00:44 +02:00
|
|
|
import os.path
|
2016-04-13 15:37:19 +02:00
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
import bpy
|
2016-06-23 19:00:47 +02:00
|
|
|
from bpy.types import AddonPreferences, Operator, WindowManager, Scene, PropertyGroup
|
2016-07-07 11:43:01 +02:00
|
|
|
from bpy.props import StringProperty, EnumProperty, PointerProperty, BoolProperty
|
2016-07-20 16:54:06 +02:00
|
|
|
import rna_prop_ui
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-08-26 17:43:20 +02:00
|
|
|
from . import pillar, async_loop
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-08-30 15:31:48 +02:00
|
|
|
PILLAR_SERVER_URL = 'https://cloud.blender.org/api/'
|
|
|
|
# PILLAR_SERVER_URL = 'http://pillar:5001/api/'
|
2016-05-04 14:30:47 +02:00
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
ADDON_NAME = 'blender_cloud'
|
2016-04-13 15:37:19 +02:00
|
|
|
log = logging.getLogger(__name__)
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-07-08 17:00:44 +02:00
|
|
|
icons = None
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-06-23 19:00:47 +02:00
|
|
|
def redraw(self, context):
|
|
|
|
context.area.tag_redraw()
|
|
|
|
|
|
|
|
|
2016-10-18 11:20:06 +02:00
|
|
|
def pyside_cache(wrapped):
|
|
|
|
"""Stores the result of the callable in Python-managed memory.
|
|
|
|
|
|
|
|
This is to work around the warning at
|
|
|
|
https://www.blender.org/api/blender_python_api_master/bpy.props.html#bpy.props.EnumProperty
|
|
|
|
"""
|
|
|
|
|
|
|
|
import functools
|
|
|
|
|
|
|
|
@functools.wraps(wrapped)
|
|
|
|
# We can't use (*args, **kwargs), because EnumProperty explicitly checks
|
|
|
|
# for the number of fixed positional arguments.
|
|
|
|
def wrapper(self, context):
|
|
|
|
result = None
|
|
|
|
try:
|
|
|
|
result = wrapped(self, context)
|
|
|
|
return result
|
|
|
|
finally:
|
|
|
|
wrapped._cached_result = result
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
|
|
|
|
@pyside_cache
|
2016-06-24 12:53:49 +02:00
|
|
|
def blender_syncable_versions(self, context):
|
2016-08-26 17:43:20 +02:00
|
|
|
"""Returns the list of items used by SyncStatusProperties.version EnumProperty."""
|
|
|
|
|
2016-06-24 12:53:49 +02:00
|
|
|
bss = context.window_manager.blender_sync_status
|
|
|
|
versions = bss.available_blender_versions
|
|
|
|
if not versions:
|
2016-06-30 14:43:04 +02:00
|
|
|
return [('', 'No settings stored in your Blender Cloud', '')]
|
2016-06-24 12:53:49 +02:00
|
|
|
return [(v, v, '') for v in versions]
|
|
|
|
|
|
|
|
|
2016-06-23 19:00:47 +02:00
|
|
|
class SyncStatusProperties(PropertyGroup):
|
|
|
|
status = EnumProperty(
|
|
|
|
items=[
|
|
|
|
('NONE', 'NONE', 'We have done nothing at all yet.'),
|
|
|
|
('IDLE', 'IDLE', 'User requested something, which is done, and we are now idle.'),
|
|
|
|
('SYNCING', 'SYNCING', 'Synchronising with Blender Cloud.'),
|
|
|
|
],
|
|
|
|
name='status',
|
2016-07-19 18:13:09 +02:00
|
|
|
description='Current status of Blender Sync',
|
2016-06-23 19:00:47 +02:00
|
|
|
update=redraw)
|
2016-06-24 12:53:49 +02:00
|
|
|
|
|
|
|
version = EnumProperty(
|
|
|
|
items=blender_syncable_versions,
|
|
|
|
name='Version of Blender from which to pull',
|
|
|
|
description='Version of Blender from which to pull')
|
|
|
|
|
2016-06-23 19:00:47 +02:00
|
|
|
message = StringProperty(name='message', update=redraw)
|
|
|
|
level = EnumProperty(
|
|
|
|
items=[
|
|
|
|
('INFO', 'INFO', ''),
|
|
|
|
('WARNING', 'WARNING', ''),
|
|
|
|
('ERROR', 'ERROR', ''),
|
2016-06-24 15:22:12 +02:00
|
|
|
('SUBSCRIBE', 'SUBSCRIBE', ''),
|
2016-06-23 19:00:47 +02:00
|
|
|
],
|
|
|
|
name='level',
|
|
|
|
update=redraw)
|
|
|
|
|
|
|
|
def report(self, level: set, message: str):
|
|
|
|
assert len(level) == 1, 'level should be a set of one string, not %r' % level
|
|
|
|
self.level = level.pop()
|
|
|
|
self.message = message
|
2016-06-24 12:53:49 +02:00
|
|
|
|
|
|
|
# Message can also be empty, just to erase it from the GUI.
|
|
|
|
# No need to actually log those.
|
|
|
|
if message:
|
2016-06-24 15:22:12 +02:00
|
|
|
try:
|
|
|
|
loglevel = logging._nameToLevel[self.level]
|
|
|
|
except KeyError:
|
|
|
|
loglevel = logging.WARNING
|
|
|
|
log.log(loglevel, message)
|
2016-06-24 12:53:49 +02:00
|
|
|
|
|
|
|
# List of syncable versions is stored in 'available_blender_versions' ID property,
|
|
|
|
# because I don't know how to store a variable list of strings in a proper RNA property.
|
|
|
|
@property
|
|
|
|
def available_blender_versions(self) -> list:
|
|
|
|
return self.get('available_blender_versions', [])
|
|
|
|
|
|
|
|
@available_blender_versions.setter
|
|
|
|
def available_blender_versions(self, new_versions):
|
|
|
|
self['available_blender_versions'] = new_versions
|
2016-06-23 19:00:47 +02:00
|
|
|
|
|
|
|
|
2016-10-18 11:20:06 +02:00
|
|
|
@pyside_cache
|
2016-08-26 17:43:20 +02:00
|
|
|
def bcloud_available_projects(self, context):
|
|
|
|
"""Returns the list of items used by BlenderCloudProjectGroup.project EnumProperty."""
|
|
|
|
|
2016-08-30 10:39:09 +02:00
|
|
|
attr_proj = preferences().attract_project
|
|
|
|
projs = attr_proj.available_projects
|
2016-08-26 17:43:20 +02:00
|
|
|
if not projs:
|
|
|
|
return [('', 'No projects available in your Blender Cloud', '')]
|
|
|
|
return [(p['_id'], p['name'], '') for p in projs]
|
|
|
|
|
|
|
|
|
|
|
|
class BlenderCloudProjectGroup(PropertyGroup):
|
|
|
|
status = EnumProperty(
|
|
|
|
items=[
|
|
|
|
('NONE', 'NONE', 'We have done nothing at all yet'),
|
|
|
|
('IDLE', 'IDLE', 'User requested something, which is done, and we are now idle'),
|
|
|
|
('FETCHING', 'FETCHING', 'Fetching available projects from Blender Cloud'),
|
|
|
|
],
|
|
|
|
name='status',
|
|
|
|
update=redraw)
|
|
|
|
|
|
|
|
project = EnumProperty(
|
|
|
|
items=bcloud_available_projects,
|
|
|
|
name='Cloud project',
|
|
|
|
description='Which Blender Cloud project to work with')
|
|
|
|
|
|
|
|
# List of projects is stored in 'available_projects' ID property,
|
|
|
|
# because I don't know how to store a variable list of strings in a proper RNA property.
|
|
|
|
@property
|
|
|
|
def available_projects(self) -> list:
|
|
|
|
return self.get('available_projects', [])
|
|
|
|
|
|
|
|
@available_projects.setter
|
|
|
|
def available_projects(self, new_projects):
|
|
|
|
self['available_projects'] = new_projects
|
|
|
|
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
class BlenderCloudPreferences(AddonPreferences):
|
|
|
|
bl_idname = ADDON_NAME
|
|
|
|
|
2016-04-19 11:37:46 +02:00
|
|
|
# The following two properties are read-only to limit the scope of the
|
|
|
|
# addon and allow for proper testing within this scope.
|
2016-06-23 19:00:47 +02:00
|
|
|
pillar_server = StringProperty(
|
2016-03-18 16:53:52 +01:00
|
|
|
name='Blender Cloud Server',
|
|
|
|
description='URL of the Blender Cloud backend server',
|
2016-05-04 14:30:47 +02:00
|
|
|
default=PILLAR_SERVER_URL,
|
|
|
|
get=lambda self: PILLAR_SERVER_URL
|
2016-03-18 16:53:52 +01:00
|
|
|
)
|
|
|
|
|
2016-03-31 11:54:15 +02:00
|
|
|
local_texture_dir = StringProperty(
|
|
|
|
name='Default Blender Cloud texture storage directory',
|
|
|
|
subtype='DIR_PATH',
|
|
|
|
default='//textures')
|
|
|
|
|
2016-07-07 11:43:01 +02:00
|
|
|
open_browser_after_share = BoolProperty(
|
|
|
|
name='Open browser after sharing file',
|
|
|
|
description='When enabled, Blender will open a webbrowser',
|
|
|
|
default=True
|
|
|
|
)
|
|
|
|
|
2016-10-11 11:17:24 +02:00
|
|
|
# TODO: store local path with the Attract project, so that people
|
|
|
|
# can switch projects and the local path switches with it.
|
2016-08-30 10:39:09 +02:00
|
|
|
attract_project = PointerProperty(type=BlenderCloudProjectGroup)
|
2016-10-11 11:17:24 +02:00
|
|
|
attract_project_local_path = StringProperty(
|
|
|
|
name='Local project path',
|
|
|
|
description='Local path of your Attract project, used to search for blend files; '
|
|
|
|
'usually best to set to an absolute path',
|
|
|
|
subtype='DIR_PATH',
|
|
|
|
default='//../')
|
2016-08-26 17:43:20 +02:00
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
def draw(self, context):
|
2016-05-03 18:29:43 +02:00
|
|
|
import textwrap
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
layout = self.layout
|
|
|
|
|
|
|
|
# Carefully try and import the Blender ID addon
|
|
|
|
try:
|
2016-04-01 14:11:12 +02:00
|
|
|
import blender_id
|
2016-03-18 16:53:52 +01:00
|
|
|
except ImportError:
|
2016-04-01 14:11:12 +02:00
|
|
|
blender_id = None
|
2016-03-18 16:53:52 +01:00
|
|
|
blender_id_profile = None
|
|
|
|
else:
|
2016-04-01 14:11:12 +02:00
|
|
|
blender_id_profile = blender_id.get_active_profile()
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-04-01 14:11:12 +02:00
|
|
|
if blender_id is None:
|
2016-07-08 17:00:44 +02:00
|
|
|
msg_icon = 'ERROR'
|
2016-05-03 18:29:43 +02:00
|
|
|
text = 'This add-on requires Blender ID'
|
|
|
|
help_text = 'Make sure that the Blender ID add-on is installed and activated'
|
2016-03-18 16:53:52 +01:00
|
|
|
elif not blender_id_profile:
|
2016-07-08 17:00:44 +02:00
|
|
|
msg_icon = 'ERROR'
|
2016-05-03 18:29:43 +02:00
|
|
|
text = 'You are logged out.'
|
|
|
|
help_text = 'To login, go to the Blender ID add-on preferences.'
|
2016-06-24 14:46:13 +02:00
|
|
|
elif bpy.app.debug and pillar.SUBCLIENT_ID not in blender_id_profile.subclients:
|
2016-07-08 17:00:44 +02:00
|
|
|
msg_icon = 'QUESTION'
|
2016-05-03 18:29:43 +02:00
|
|
|
text = 'No Blender Cloud credentials.'
|
|
|
|
help_text = ('You are logged in on Blender ID, but your credentials have not '
|
|
|
|
'been synchronized with Blender Cloud yet. Press the Update '
|
|
|
|
'Credentials button.')
|
2016-03-18 16:53:52 +01:00
|
|
|
else:
|
2016-07-08 17:00:44 +02:00
|
|
|
msg_icon = 'WORLD_DATA'
|
2016-05-03 18:29:43 +02:00
|
|
|
text = 'You are logged in as %s.' % blender_id_profile.username
|
|
|
|
help_text = ('To logout or change profile, '
|
|
|
|
'go to the Blender ID add-on preferences.')
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-06-24 12:53:49 +02:00
|
|
|
# Authentication stuff
|
|
|
|
auth_box = layout.box()
|
2016-07-08 17:00:44 +02:00
|
|
|
auth_box.label(text=text, icon=msg_icon)
|
2016-05-03 18:29:43 +02:00
|
|
|
|
|
|
|
help_lines = textwrap.wrap(help_text, 80)
|
|
|
|
for line in help_lines:
|
2016-06-24 12:53:49 +02:00
|
|
|
auth_box.label(text=line)
|
2016-06-24 14:46:13 +02:00
|
|
|
if bpy.app.debug:
|
|
|
|
auth_box.operator("pillar.credentials_update")
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-06-24 12:53:49 +02:00
|
|
|
# Texture browser stuff
|
|
|
|
texture_box = layout.box()
|
2016-07-08 17:00:44 +02:00
|
|
|
texture_box.enabled = msg_icon != 'ERROR'
|
2016-06-24 12:53:49 +02:00
|
|
|
sub = texture_box.column()
|
2016-07-08 17:00:44 +02:00
|
|
|
sub.label(text='Local directory for downloaded textures', icon_value=icon('CLOUD'))
|
2016-03-31 11:54:15 +02:00
|
|
|
sub.prop(self, "local_texture_dir", text='Default')
|
|
|
|
sub.prop(context.scene, "local_texture_dir", text='Current scene')
|
|
|
|
|
2016-06-24 12:53:49 +02:00
|
|
|
# Blender Sync stuff
|
2016-06-23 19:00:47 +02:00
|
|
|
bss = context.window_manager.blender_sync_status
|
2016-06-24 12:53:49 +02:00
|
|
|
bsync_box = layout.box()
|
2016-07-08 17:00:44 +02:00
|
|
|
bsync_box.enabled = msg_icon != 'ERROR'
|
2016-06-24 12:53:49 +02:00
|
|
|
row = bsync_box.row().split(percentage=0.33)
|
2016-07-08 17:00:44 +02:00
|
|
|
row.label('Blender Sync with Blender Cloud', icon_value=icon('CLOUD'))
|
2016-06-23 19:00:47 +02:00
|
|
|
|
|
|
|
icon_for_level = {
|
|
|
|
'INFO': 'NONE',
|
|
|
|
'WARNING': 'INFO',
|
|
|
|
'ERROR': 'ERROR',
|
2016-06-24 15:22:12 +02:00
|
|
|
'SUBSCRIBE': 'ERROR',
|
2016-06-23 19:00:47 +02:00
|
|
|
}
|
2016-07-08 17:00:44 +02:00
|
|
|
msg_icon = icon_for_level[bss.level] if bss.message else 'NONE'
|
2016-06-23 19:00:47 +02:00
|
|
|
message_container = row.row()
|
2016-07-08 17:00:44 +02:00
|
|
|
message_container.label(bss.message, icon=msg_icon)
|
2016-06-23 19:00:47 +02:00
|
|
|
|
2016-06-24 12:53:49 +02:00
|
|
|
sub = bsync_box.column()
|
2016-06-23 19:00:47 +02:00
|
|
|
|
2016-06-24 15:22:12 +02:00
|
|
|
if bss.level == 'SUBSCRIBE':
|
|
|
|
self.draw_subscribe_button(sub)
|
2016-07-08 12:38:58 +02:00
|
|
|
self.draw_sync_buttons(sub, bss)
|
2016-06-24 15:22:12 +02:00
|
|
|
|
2016-07-07 11:43:01 +02:00
|
|
|
# Image Share stuff
|
|
|
|
share_box = layout.box()
|
2016-07-08 17:00:44 +02:00
|
|
|
share_box.label('Image Sharing on Blender Cloud', icon_value=icon('CLOUD'))
|
2016-08-26 17:43:20 +02:00
|
|
|
share_box.enabled = msg_icon != 'ERROR'
|
2016-07-07 11:43:01 +02:00
|
|
|
share_box.prop(self, 'open_browser_after_share')
|
|
|
|
|
2016-08-26 17:43:20 +02:00
|
|
|
# Attract stuff
|
|
|
|
attract_box = layout.box()
|
|
|
|
attract_box.enabled = msg_icon != 'ERROR'
|
2016-10-11 11:17:24 +02:00
|
|
|
self.draw_attract_buttons(attract_box, self.attract_project)
|
2016-08-26 17:43:20 +02:00
|
|
|
|
2016-06-24 15:22:12 +02:00
|
|
|
def draw_subscribe_button(self, layout):
|
|
|
|
layout.operator('pillar.subscribe', icon='WORLD')
|
|
|
|
|
|
|
|
def draw_sync_buttons(self, layout, bss):
|
|
|
|
layout.enabled = bss.status in {'NONE', 'IDLE'}
|
|
|
|
|
|
|
|
buttons = layout.column()
|
2016-06-24 12:53:49 +02:00
|
|
|
row_buttons = buttons.row().split(percentage=0.5)
|
|
|
|
row_push = row_buttons.row()
|
2016-06-28 16:55:35 +02:00
|
|
|
row_pull = row_buttons.row(align=True)
|
2016-06-24 12:53:49 +02:00
|
|
|
|
|
|
|
row_push.operator('pillar.sync',
|
2016-06-28 16:55:35 +02:00
|
|
|
text='Save %i.%i settings' % bpy.app.version[:2],
|
2016-06-24 12:53:49 +02:00
|
|
|
icon='TRIA_UP').action = 'PUSH'
|
|
|
|
|
|
|
|
versions = bss.available_blender_versions
|
|
|
|
version = bss.version
|
|
|
|
if bss.status in {'NONE', 'IDLE'}:
|
|
|
|
if not versions or not version:
|
|
|
|
row_pull.operator('pillar.sync',
|
2016-06-28 16:55:35 +02:00
|
|
|
text='Find version to load',
|
2016-06-24 12:53:49 +02:00
|
|
|
icon='TRIA_DOWN').action = 'REFRESH'
|
|
|
|
else:
|
|
|
|
props = row_pull.operator('pillar.sync',
|
2016-06-28 16:55:35 +02:00
|
|
|
text='Load %s settings' % version,
|
2016-06-24 12:53:49 +02:00
|
|
|
icon='TRIA_DOWN')
|
2016-06-23 19:00:47 +02:00
|
|
|
props.action = 'PULL'
|
|
|
|
props.blender_version = version
|
2016-06-24 12:53:49 +02:00
|
|
|
row_pull.operator('pillar.sync',
|
|
|
|
text='',
|
|
|
|
icon='DOTSDOWN').action = 'SELECT'
|
|
|
|
else:
|
|
|
|
row_pull.label('Cloud Sync is running.')
|
2016-06-23 19:00:47 +02:00
|
|
|
|
2016-10-11 11:17:24 +02:00
|
|
|
def draw_attract_buttons(self, attract_box, bcp: BlenderCloudProjectGroup):
|
|
|
|
attract_row = attract_box.row(align=True)
|
|
|
|
attract_row.label('Attract', icon_value=icon('CLOUD'))
|
|
|
|
|
|
|
|
attract_row.enabled = bcp.status in {'NONE', 'IDLE'}
|
|
|
|
row_buttons = attract_row.row(align=True)
|
2016-08-26 17:43:20 +02:00
|
|
|
|
|
|
|
projects = bcp.available_projects
|
|
|
|
project = bcp.project
|
|
|
|
if bcp.status in {'NONE', 'IDLE'}:
|
|
|
|
if not projects or not project:
|
|
|
|
row_buttons.operator('pillar.projects',
|
|
|
|
text='Find project to load',
|
|
|
|
icon='FILE_REFRESH')
|
|
|
|
else:
|
|
|
|
row_buttons.prop(bcp, 'project')
|
|
|
|
row_buttons.operator('pillar.projects',
|
|
|
|
text='',
|
|
|
|
icon='FILE_REFRESH')
|
|
|
|
else:
|
|
|
|
row_buttons.label('Fetching available projects.')
|
|
|
|
|
2016-10-11 11:17:24 +02:00
|
|
|
attract_box.prop(self, 'attract_project_local_path')
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-06-24 15:22:12 +02:00
|
|
|
class PillarCredentialsUpdate(pillar.PillarOperatorMixin,
|
|
|
|
Operator):
|
2016-03-18 16:53:52 +01:00
|
|
|
"""Updates the Pillar URL and tests the new URL."""
|
2016-04-12 16:59:34 +02:00
|
|
|
bl_idname = 'pillar.credentials_update'
|
|
|
|
bl_label = 'Update credentials'
|
2016-10-04 16:44:15 +02:00
|
|
|
bl_description = 'Resynchronises your Blender ID login with Blender Cloud'
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-07-14 11:47:50 +02:00
|
|
|
log = logging.getLogger('bpy.ops.%s' % bl_idname)
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
|
|
|
# Only allow activation when the user is actually logged in.
|
|
|
|
return cls.is_logged_in(context)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def is_logged_in(cls, context):
|
2016-04-01 17:16:29 +02:00
|
|
|
try:
|
|
|
|
import blender_id
|
|
|
|
except ImportError:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return blender_id.is_logged_in()
|
2016-03-18 16:53:52 +01:00
|
|
|
|
|
|
|
def execute(self, context):
|
2016-04-12 16:59:34 +02:00
|
|
|
import blender_id
|
|
|
|
import asyncio
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
# Only allow activation when the user is actually logged in.
|
|
|
|
if not self.is_logged_in(context):
|
2016-04-12 16:59:34 +02:00
|
|
|
self.report({'ERROR'}, 'No active profile found')
|
|
|
|
return {'CANCELLED'}
|
|
|
|
|
|
|
|
try:
|
2016-05-04 14:30:47 +02:00
|
|
|
loop = asyncio.get_event_loop()
|
2016-06-24 15:22:12 +02:00
|
|
|
loop.run_until_complete(self.check_credentials(context, set()))
|
2016-04-12 16:59:34 +02:00
|
|
|
except blender_id.BlenderIdCommError as ex:
|
2016-04-19 10:34:34 +02:00
|
|
|
log.exception('Error sending subclient-specific token to Blender ID')
|
2016-05-04 14:30:47 +02:00
|
|
|
self.report({'ERROR'}, 'Failed to sync Blender ID to Blender Cloud')
|
2016-03-18 16:53:52 +01:00
|
|
|
return {'CANCELLED'}
|
2016-04-12 16:59:34 +02:00
|
|
|
except Exception as ex:
|
2016-04-13 15:37:19 +02:00
|
|
|
log.exception('Error in test call to Pillar')
|
2016-05-04 14:30:47 +02:00
|
|
|
self.report({'ERROR'}, 'Failed test connection to Blender Cloud')
|
2016-04-12 16:59:34 +02:00
|
|
|
return {'CANCELLED'}
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-04-19 10:34:34 +02:00
|
|
|
self.report({'INFO'}, 'Blender Cloud credentials & endpoint URL updated.')
|
2016-03-18 16:53:52 +01:00
|
|
|
return {'FINISHED'}
|
|
|
|
|
|
|
|
|
2016-06-24 15:22:12 +02:00
|
|
|
class PILLAR_OT_subscribe(Operator):
|
|
|
|
"""Opens a browser to subscribe the user to the Cloud."""
|
|
|
|
bl_idname = 'pillar.subscribe'
|
|
|
|
bl_label = 'Subscribe to the Cloud'
|
2016-10-03 19:29:41 +02:00
|
|
|
bl_description = "Opens a page in a web browser to subscribe to the Blender Cloud"
|
2016-06-24 15:22:12 +02:00
|
|
|
|
|
|
|
def execute(self, context):
|
|
|
|
import webbrowser
|
|
|
|
|
|
|
|
webbrowser.open_new_tab('https://cloud.blender.org/join')
|
|
|
|
self.report({'INFO'}, 'We just started a browser for you.')
|
|
|
|
|
|
|
|
return {'FINISHED'}
|
|
|
|
|
|
|
|
|
2016-08-26 17:43:20 +02:00
|
|
|
class PILLAR_OT_projects(async_loop.AsyncModalOperatorMixin,
|
|
|
|
pillar.PillarOperatorMixin,
|
|
|
|
Operator):
|
|
|
|
"""Fetches the projects available to the user, and ."""
|
|
|
|
bl_idname = 'pillar.projects'
|
|
|
|
bl_label = 'Fetch available projects'
|
|
|
|
|
|
|
|
stop_upon_exception = True
|
2016-10-04 16:42:13 +02:00
|
|
|
_log = logging.getLogger('bpy.ops.%s' % bl_idname)
|
2016-08-26 17:43:20 +02:00
|
|
|
|
|
|
|
async def async_execute(self, context):
|
|
|
|
import pillarsdk
|
|
|
|
from .pillar import pillar_call
|
|
|
|
|
2016-10-04 16:42:13 +02:00
|
|
|
self._log.info('Checking credentials')
|
|
|
|
try:
|
|
|
|
db_user = await self.check_credentials(context, ())
|
|
|
|
except pillar.UserNotLoggedInError as ex:
|
|
|
|
self._log.info('Not logged in error raised: %s', ex)
|
|
|
|
self.report({'ERROR'}, 'Please log in on Blender ID first.')
|
|
|
|
self.quit()
|
|
|
|
return
|
|
|
|
|
2016-08-26 17:43:20 +02:00
|
|
|
user_id = db_user['_id']
|
|
|
|
self.log.info('Going to fetch projects for user %s', user_id)
|
|
|
|
|
2016-08-30 10:39:09 +02:00
|
|
|
preferences().attract_project.status = 'FETCHING'
|
2016-08-26 17:43:20 +02:00
|
|
|
|
|
|
|
# Get all projects, except the home project.
|
|
|
|
projects_user = await pillar_call(
|
|
|
|
pillarsdk.Project.all,
|
|
|
|
{'where': {'user': user_id,
|
|
|
|
'category': {'$ne': 'home'}},
|
|
|
|
'sort': '-_created',
|
|
|
|
'projection': {'_id': True,
|
|
|
|
'name': True},
|
|
|
|
})
|
|
|
|
|
|
|
|
projects_shared = await pillar_call(
|
|
|
|
pillarsdk.Project.all,
|
|
|
|
{'where': {'user': {'$ne': user_id},
|
2016-09-29 18:54:50 +02:00
|
|
|
'permissions.groups.group': {'$in': db_user.groups}},
|
2016-08-26 17:43:20 +02:00
|
|
|
'sort': '-_created',
|
|
|
|
'projection': {'_id': True,
|
|
|
|
'name': True},
|
|
|
|
})
|
|
|
|
|
|
|
|
# We need to convert to regular dicts before storing in ID properties.
|
|
|
|
# Also don't store more properties than we need.
|
|
|
|
projects = [{'_id': p['_id'], 'name': p['name']} for p in projects_user['_items']] + \
|
|
|
|
[{'_id': p['_id'], 'name': p['name']} for p in projects_shared['_items']]
|
|
|
|
|
2016-08-30 10:39:09 +02:00
|
|
|
preferences().attract_project.available_projects = projects
|
2016-08-26 17:43:20 +02:00
|
|
|
|
|
|
|
self.quit()
|
|
|
|
|
|
|
|
def quit(self):
|
2016-08-30 10:39:09 +02:00
|
|
|
preferences().attract_project.status = 'IDLE'
|
2016-08-26 17:43:20 +02:00
|
|
|
super().quit()
|
|
|
|
|
|
|
|
|
2016-07-20 16:54:06 +02:00
|
|
|
class PILLAR_PT_image_custom_properties(rna_prop_ui.PropertyPanel, bpy.types.Panel):
|
|
|
|
"""Shows custom properties in the image editor."""
|
|
|
|
|
|
|
|
bl_space_type = 'IMAGE_EDITOR'
|
|
|
|
bl_region_type = 'UI'
|
|
|
|
bl_label = 'Custom Properties'
|
|
|
|
|
|
|
|
_context_path = 'edit_image'
|
|
|
|
_property_type = bpy.types.Image
|
|
|
|
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
def preferences() -> BlenderCloudPreferences:
|
|
|
|
return bpy.context.user_preferences.addons[ADDON_NAME].preferences
|
|
|
|
|
|
|
|
|
2016-07-08 17:00:44 +02:00
|
|
|
def load_custom_icons():
|
|
|
|
global icons
|
|
|
|
|
|
|
|
if icons is not None:
|
|
|
|
# Already loaded
|
|
|
|
return
|
|
|
|
|
|
|
|
import bpy.utils.previews
|
|
|
|
icons = bpy.utils.previews.new()
|
|
|
|
my_icons_dir = os.path.join(os.path.dirname(__file__), 'icons')
|
|
|
|
icons.load('CLOUD', os.path.join(my_icons_dir, 'icon-cloud.png'), 'IMAGE')
|
|
|
|
|
|
|
|
|
|
|
|
def unload_custom_icons():
|
|
|
|
global icons
|
|
|
|
|
|
|
|
if icons is None:
|
|
|
|
# Already unloaded
|
|
|
|
return
|
|
|
|
|
|
|
|
bpy.utils.previews.remove(icons)
|
|
|
|
icons = None
|
|
|
|
|
|
|
|
|
|
|
|
def icon(icon_name: str) -> int:
|
|
|
|
"""Returns the icon ID for the named icon.
|
|
|
|
|
|
|
|
Use with layout.operator('pillar.image_share', icon_value=icon('CLOUD'))
|
|
|
|
"""
|
|
|
|
|
|
|
|
return icons[icon_name].icon_id
|
|
|
|
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
def register():
|
2016-08-30 10:39:09 +02:00
|
|
|
bpy.utils.register_class(BlenderCloudProjectGroup)
|
2016-03-18 16:53:52 +01:00
|
|
|
bpy.utils.register_class(BlenderCloudPreferences)
|
|
|
|
bpy.utils.register_class(PillarCredentialsUpdate)
|
2016-06-23 19:00:47 +02:00
|
|
|
bpy.utils.register_class(SyncStatusProperties)
|
2016-06-24 15:22:12 +02:00
|
|
|
bpy.utils.register_class(PILLAR_OT_subscribe)
|
2016-08-26 17:43:20 +02:00
|
|
|
bpy.utils.register_class(PILLAR_OT_projects)
|
2016-07-20 16:54:06 +02:00
|
|
|
bpy.utils.register_class(PILLAR_PT_image_custom_properties)
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-05-17 17:30:38 +02:00
|
|
|
addon_prefs = preferences()
|
|
|
|
|
2016-05-18 11:56:46 +02:00
|
|
|
WindowManager.last_blender_cloud_location = StringProperty(
|
|
|
|
name="Last Blender Cloud browser location",
|
|
|
|
default="/")
|
2016-03-31 11:54:15 +02:00
|
|
|
|
|
|
|
def default_if_empty(scene, context):
|
|
|
|
"""The scene's local_texture_dir, if empty, reverts to the addon prefs."""
|
|
|
|
|
|
|
|
if not scene.local_texture_dir:
|
|
|
|
scene.local_texture_dir = addon_prefs.local_texture_dir
|
|
|
|
|
|
|
|
Scene.local_texture_dir = StringProperty(
|
|
|
|
name='Blender Cloud texture storage directory for current scene',
|
2016-03-18 16:53:52 +01:00
|
|
|
subtype='DIR_PATH',
|
2016-03-31 11:54:15 +02:00
|
|
|
default=addon_prefs.local_texture_dir,
|
|
|
|
update=default_if_empty)
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-06-23 19:00:47 +02:00
|
|
|
WindowManager.blender_sync_status = PointerProperty(type=SyncStatusProperties)
|
|
|
|
|
2016-07-08 17:00:44 +02:00
|
|
|
load_custom_icons()
|
|
|
|
|
2016-03-18 16:53:52 +01:00
|
|
|
|
|
|
|
def unregister():
|
2016-07-08 17:00:44 +02:00
|
|
|
unload_custom_icons()
|
|
|
|
|
2016-08-30 10:39:09 +02:00
|
|
|
bpy.utils.unregister_class(BlenderCloudProjectGroup)
|
2016-03-18 16:53:52 +01:00
|
|
|
bpy.utils.unregister_class(PillarCredentialsUpdate)
|
|
|
|
bpy.utils.unregister_class(BlenderCloudPreferences)
|
2016-06-23 19:00:47 +02:00
|
|
|
bpy.utils.unregister_class(SyncStatusProperties)
|
2016-06-24 15:22:12 +02:00
|
|
|
bpy.utils.unregister_class(PILLAR_OT_subscribe)
|
2016-08-26 17:43:20 +02:00
|
|
|
bpy.utils.unregister_class(PILLAR_OT_projects)
|
2016-07-20 16:54:06 +02:00
|
|
|
bpy.utils.unregister_class(PILLAR_PT_image_custom_properties)
|
2016-03-18 16:53:52 +01:00
|
|
|
|
2016-06-23 19:00:47 +02:00
|
|
|
del WindowManager.last_blender_cloud_location
|
|
|
|
del WindowManager.blender_sync_status
|