You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

315 lines
12 KiB

# 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")
col.prop_menu_enum(active_page, "version")
# 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")