|
|
|
# This file is part of Korman.
|
|
|
|
#
|
|
|
|
# Korman 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 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# Korman 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 Korman. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
import bpy
|
|
|
|
from pathlib import Path
|
|
|
|
|
|
|
|
from .. import korlib
|
|
|
|
|
|
|
|
|
|
|
|
class AgeButtonsPanel:
|
|
|
|
bl_space_type = "PROPERTIES"
|
|
|
|
bl_region_type = "WINDOW"
|
|
|
|
bl_context = "world"
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
|
|
|
return context.world and context.scene.render.engine == "PLASMA_GAME"
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaGameHelper:
|
|
|
|
@property
|
|
|
|
def active_game(self):
|
|
|
|
games = bpy.context.world.plasma_games
|
|
|
|
prefs = bpy.context.user_preferences.addons["korman"].preferences
|
|
|
|
active_game_index = games.active_game_index
|
|
|
|
if active_game_index < len(prefs.games):
|
|
|
|
return prefs.games[active_game_index]
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def format_path(self, dirname="dat", ext=".age"):
|
|
|
|
active_game = self.active_game
|
|
|
|
if active_game is None:
|
|
|
|
return ""
|
|
|
|
age_name = bpy.context.world.plasma_age.age_name
|
|
|
|
return Path(active_game.path, dirname, age_name).with_suffix(ext)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def legal_game(self):
|
|
|
|
if self.active_game is not None:
|
|
|
|
return bool(bpy.context.world.plasma_age.age_name.strip())
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaGameExportMenu(PlasmaGameHelper, bpy.types.Menu):
|
|
|
|
bl_label = "Plasma Export Menu"
|
|
|
|
bl_description = "Additional export methods"
|
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
age_name = context.world.plasma_age.age_name
|
|
|
|
active_game = self.active_game
|
|
|
|
legal_game = self.legal_game
|
|
|
|
|
|
|
|
# Export Localization
|
|
|
|
row = layout.row()
|
|
|
|
row.operator_context = "EXEC_DEFAULT"
|
|
|
|
row.enabled = legal_game
|
|
|
|
op = row.operator("export.plasma_loc", icon="FILE_SCRIPT")
|
|
|
|
if active_game is not None:
|
|
|
|
op.filepath = active_game.path
|
|
|
|
op.version = active_game.version
|
|
|
|
|
|
|
|
# Export Python
|
|
|
|
row = layout.row()
|
|
|
|
row.operator_context = "EXEC_DEFAULT"
|
|
|
|
row.enabled = legal_game and active_game.version != "pvMoul"
|
|
|
|
op = row.operator("export.plasma_pak", icon="FILE_SCRIPT")
|
|
|
|
op.filepath = str(self.format_path("Python", ".pak"))
|
|
|
|
if active_game is not None:
|
|
|
|
op.version = active_game.version
|
|
|
|
|
|
|
|
# Launch Age
|
|
|
|
row = layout.row()
|
|
|
|
row.operator_context = "EXEC_DEFAULT"
|
|
|
|
age_path = self.format_path()
|
|
|
|
row.active = legal_game and active_game.can_launch and age_path.exists()
|
|
|
|
op = row.operator("export.plasma_age", icon="RENDER_ANIMATION", text="Launch Age")
|
|
|
|
if active_game is not None:
|
|
|
|
op.actions = {"LAUNCH"}
|
|
|
|
op.dat_only = False
|
|
|
|
op.filepath = str(age_path)
|
|
|
|
op.version = active_game.version
|
|
|
|
op.player = active_game.player
|
|
|
|
op.ki = active_game.ki
|
|
|
|
op.serverini = active_game.serverini
|
|
|
|
|
|
|
|
# Package Age
|
|
|
|
row = layout.row()
|
|
|
|
row.operator_context = "INVOKE_DEFAULT"
|
|
|
|
row.enabled = legal_game
|
|
|
|
op = row.operator("export.plasma_age", icon="PACKAGE", text="Package Age")
|
|
|
|
if active_game is not None:
|
|
|
|
op.actions = {"EXPORT"}
|
|
|
|
op.dat_only = False
|
|
|
|
op.filepath = "{}.zip".format(age_name)
|
|
|
|
op.version = active_game.version
|
|
|
|
|
|
|
|
# Profile Export
|
|
|
|
row = layout.row()
|
|
|
|
row.operator_context = "EXEC_DEFAULT"
|
|
|
|
age_path = self.format_path()
|
|
|
|
row.active = legal_game and active_game.can_launch and age_path.exists()
|
|
|
|
op = row.operator("export.plasma_age", icon="RADIO", text="Profile Export")
|
|
|
|
if active_game is not None:
|
|
|
|
op.actions = {"EXPORT", "PROFILE"}
|
|
|
|
op.dat_only = False
|
|
|
|
op.filepath = str(age_path)
|
|
|
|
op.version = active_game.version
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaGamePanel(AgeButtonsPanel, PlasmaGameHelper, bpy.types.Panel):
|
|
|
|
bl_label = "Plasma Games"
|
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
prefs = context.user_preferences.addons["korman"].preferences
|
|
|
|
games = context.world.plasma_games
|
|
|
|
age = context.world.plasma_age
|
|
|
|
active_game = self.active_game
|
|
|
|
legal_game = self.legal_game
|
|
|
|
|
|
|
|
row = layout.row()
|
|
|
|
# Remember: game storage moved to addon preferences!
|
|
|
|
row.template_list("PlasmaGameListRO", "games", prefs, "games", games,
|
|
|
|
"active_game_index", rows=2)
|
|
|
|
row.operator("ui.korman_open_prefs", icon="PREFERENCES", text="")
|
|
|
|
|
|
|
|
layout.separator()
|
|
|
|
row = layout.row(align=True)
|
|
|
|
|
|
|
|
# Export Age
|
|
|
|
row.operator_context = "EXEC_DEFAULT"
|
|
|
|
row.enabled = legal_game
|
|
|
|
op = row.operator("export.plasma_age", icon="EXPORT")
|
|
|
|
if active_game is not None:
|
|
|
|
op.actions = {"EXPORT"}
|
|
|
|
op.dat_only = False
|
|
|
|
op.filepath = str(self.format_path())
|
|
|
|
op.version = active_game.version
|
|
|
|
|
|
|
|
# Test Age (exports and tests the age)
|
|
|
|
row = row.row(align=True)
|
|
|
|
row.operator_context = "EXEC_DEFAULT"
|
|
|
|
# Sadly, if we nuke this row, the menu is nuked as well...
|
|
|
|
row.enabled = legal_game
|
|
|
|
op = row.operator("export.plasma_age", icon="RENDER_ANIMATION", text="Test Age")
|
|
|
|
if active_game is not None:
|
|
|
|
op.actions = {"EXPORT", "LAUNCH"}
|
|
|
|
op.dat_only = False
|
|
|
|
op.filepath = str(self.format_path())
|
|
|
|
op.version = active_game.version
|
|
|
|
op.player = active_game.player
|
|
|
|
op.ki = active_game.ki
|
|
|
|
op.serverini = active_game.serverini
|
|
|
|
|
|
|
|
# Special Menu
|
|
|
|
row = row.row(align=True)
|
|
|
|
row.enabled = True
|
|
|
|
row.menu("PlasmaGameExportMenu", icon='DOWNARROW_HLT', text="")
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaGameListRO(bpy.types.UIList):
|
|
|
|
def draw_item(self, context, layout, data, item, icon, active_data, active_property, index=0, flt_flag=0):
|
|
|
|
layout.label(item.name, icon="BOOKMARKS")
|
|
|
|
|
|
|
|
class PlasmaGameListRW(bpy.types.UIList):
|
|
|
|
def draw_item(self, context, layout, data, item, icon, active_data, active_property, index=0, flt_flag=0):
|
|
|
|
layout.prop(item, "name", text="", emboss=False, icon="BOOKMARKS")
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaPageList(bpy.types.UIList):
|
|
|
|
def draw_item(self, context, layout, data, item, icon, active_data, active_property, index=0, flt_flag=0):
|
|
|
|
layout.prop(item, "name", text="", emboss=False, icon="BOOKMARKS")
|
|
|
|
layout.prop(item, "enabled", text="")
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaAgePanel(AgeButtonsPanel, bpy.types.Panel):
|
|
|
|
bl_label = "Plasma Age"
|
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
age = context.world.plasma_age
|
|
|
|
|
|
|
|
# We want a list of pages and an editor below that
|
|
|
|
row = layout.row()
|
|
|
|
row.template_list("PlasmaPageList", "pages", age, "pages", age,
|
|
|
|
"active_page_index", rows=2)
|
|
|
|
col = row.column(align=True)
|
|
|
|
col.operator("world.plasma_page_add", icon="ZOOMIN", text="")
|
|
|
|
col.operator("world.plasma_page_remove", icon="ZOOMOUT", text="")
|
|
|
|
|
|
|
|
# Page Properties
|
|
|
|
if age.active_page_index < len(age.pages):
|
|
|
|
active_page = age.pages[age.active_page_index]
|
|
|
|
|
|
|
|
layout.separator()
|
|
|
|
box = layout.box()
|
|
|
|
|
|
|
|
split = box.split()
|
|
|
|
col = split.column()
|
|
|
|
col.label("Page Flags:")
|
|
|
|
col.prop(active_page, "auto_load")
|
|
|
|
col.prop(active_page, "local_only")
|
|
|
|
|
|
|
|
col = split.column()
|
|
|
|
col.label("Page Info:")
|
|
|
|
col.prop(active_page, "name", text="")
|
|
|
|
col.prop(active_page, "seq_suffix")
|
|
|
|
|
|
|
|
split = box.split()
|
|
|
|
col = split.column()
|
|
|
|
col.label("Export For:")
|
|
|
|
col.prop(active_page, "version")
|
|
|
|
|
|
|
|
col = split.column()
|
|
|
|
default_page = active_page.seq_suffix == 0
|
|
|
|
invalid_page_type = default_page and active_page.page_type != "room"
|
|
|
|
col.alert = invalid_page_type
|
|
|
|
col.label("Page Type:")
|
|
|
|
col.prop(active_page, "page_type", text="")
|
|
|
|
col.alert = False
|
|
|
|
if default_page:
|
|
|
|
col.label("This is the Default Page.", icon="BOOKMARKS")
|
|
|
|
if invalid_page_type:
|
|
|
|
col.label("Page must be a Room.", icon="ERROR")
|
|
|
|
|
|
|
|
# Age Names should really be legal Python 2.x identifiers for AgeSDLHooks
|
|
|
|
legal_identifier = korlib.is_legal_python2_identifier(age.age_name)
|
|
|
|
illegal_age_name = not legal_identifier or '_' in age.age_name
|
|
|
|
bad_prefix = age.seq_prefix >= age.MOUL_PREFIX_RANGE[1] or age.seq_prefix <= age.MOUL_PREFIX_RANGE[0]
|
|
|
|
|
|
|
|
# Core settings
|
|
|
|
layout.separator()
|
|
|
|
split = layout.split()
|
|
|
|
|
|
|
|
col = split.column()
|
|
|
|
col.label("Age Time:")
|
|
|
|
col.prop(age, "start_time", text="Epoch")
|
|
|
|
col.prop(age, "day_length")
|
|
|
|
|
|
|
|
col = split.column()
|
|
|
|
col.label("Age Settings:")
|
|
|
|
col.alert = bad_prefix
|
|
|
|
col.prop(age, "seq_prefix", text="ID")
|
|
|
|
col.alert = illegal_age_name
|
|
|
|
col.prop(age, "age_name", text="")
|
|
|
|
|
|
|
|
if age.seq_prefix >= age.MOUL_PREFIX_RANGE[1]:
|
|
|
|
layout.label(text="Your sequence prefix is too high for Myst Online: Uru Live", icon="ERROR")
|
|
|
|
elif age.seq_prefix <= age.MOUL_PREFIX_RANGE[0]:
|
|
|
|
# Unlikely.
|
|
|
|
layout.label(text="Your sequence prefix is too low for Myst Online: Uru Live", icon="ERROR")
|
|
|
|
|
|
|
|
# Display a hint if the identifier is illegal
|
|
|
|
if illegal_age_name:
|
|
|
|
if not age.age_name:
|
|
|
|
layout.label(text="Age names cannot be empty", icon="ERROR")
|
|
|
|
elif korlib.is_python_keyword(age.age_name):
|
|
|
|
layout.label(text="Ages should not be named the same as a Python keyword", icon="ERROR")
|
|
|
|
elif age.age_sdl:
|
|
|
|
fixed_identifier = korlib.replace_python2_identifier(age.age_name)
|
|
|
|
layout.label(text="Age's SDL will use the name '{}'".format(fixed_identifier), icon="ERROR")
|
|
|
|
if '_' in age.age_name:
|
|
|
|
layout.label(text="Age names should not contain underscores", icon="ERROR")
|
|
|
|
|
|
|
|
layout.separator()
|
|
|
|
split = layout.split()
|
|
|
|
|
|
|
|
col = split.column()
|
|
|
|
col.label("Export Settings:")
|
|
|
|
col.enabled = korlib.ConsoleToggler.is_platform_supported()
|
|
|
|
col.prop(age, "verbose")
|
|
|
|
col.prop(age, "show_console")
|
|
|
|
|
|
|
|
col = split.column()
|
|
|
|
col.label("Plasma Settings:")
|
|
|
|
col.prop(age, "age_sdl")
|
|
|
|
col.prop(age, "use_texture_page")
|
|
|
|
|
|
|
|
layout.separator()
|
|
|
|
layout.prop(age, "envmap_method")
|
|
|
|
layout.prop(age, "lighting_method")
|
|
|
|
layout.prop(age, "localization_method")
|
|
|
|
layout.prop(age, "python_method")
|
|
|
|
layout.prop(age, "texcache_method")
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaEnvironmentPanel(AgeButtonsPanel, bpy.types.Panel):
|
|
|
|
bl_label = "Plasma Environment"
|
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
fni = context.world.plasma_fni
|
|
|
|
|
|
|
|
# warn about reversed linear fog values
|
|
|
|
if fni.fog_method == "linear" and fni.fog_start >= fni.fog_end and (fni.fog_start + fni.fog_end) != 0:
|
|
|
|
layout.label(text="Fog Start Value should be less than the End Value", icon="ERROR")
|
|
|
|
|
|
|
|
# basic colors
|
|
|
|
split = layout.split()
|
|
|
|
col = split.column()
|
|
|
|
col.prop(fni, "fog_color")
|
|
|
|
col = split.column()
|
|
|
|
col.prop(fni, "clear_color")
|
|
|
|
|
|
|
|
split = layout.split()
|
|
|
|
col = split.column()
|
|
|
|
col.label("Fog Settings:")
|
|
|
|
col.prop_menu_enum(fni, "fog_method")
|
|
|
|
col.separator()
|
|
|
|
if fni.fog_method == "linear":
|
|
|
|
col.prop(fni, "fog_start")
|
|
|
|
if fni.fog_method != "none":
|
|
|
|
col.prop(fni, "fog_end")
|
|
|
|
col.prop(fni, "fog_density")
|
|
|
|
|
|
|
|
col = split.column()
|
|
|
|
col.label("Draw Settings:")
|
|
|
|
col.prop(fni, "yon", text="Clipping")
|