# 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 . 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")