|
|
|
# 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 bpy.props import *
|
|
|
|
from PyHSPlasma import *
|
|
|
|
|
|
|
|
from .node_core import PlasmaNodeBase, PlasmaNodeSocketBase
|
|
|
|
from ..properties.modifiers.physics import bounds_types
|
|
|
|
|
|
|
|
class PlasmaConditionSocket(PlasmaNodeSocketBase, bpy.types.NodeSocket):
|
|
|
|
bl_color = (0.188, 0.086, 0.349, 1.0)
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaVolumeReportNode(PlasmaNodeBase, bpy.types.Node):
|
|
|
|
bl_category = "CONDITIONS"
|
|
|
|
bl_idname = "PlasmaVoumeReportNode"
|
|
|
|
bl_label = "Region Trigger Settings"
|
|
|
|
|
|
|
|
report_when = EnumProperty(name="When",
|
|
|
|
description="When the region should trigger",
|
|
|
|
items=[("each", "Each Event", "The region will trigger on every enter/exit"),
|
|
|
|
("first", "First Event", "The region will trigger on the first event only"),
|
|
|
|
("count", "Population", "When the region has a certain number of objects inside it")])
|
|
|
|
threshold = IntProperty(name="Threshold",
|
|
|
|
description="How many objects should be in the region for it to trigger",
|
|
|
|
min=1)
|
|
|
|
|
|
|
|
def init(self, context):
|
|
|
|
self.outputs.new("PlasmaVolumeSettingsSocketOut", "Trigger Settings")
|
|
|
|
|
|
|
|
def draw_buttons(self, context, layout):
|
|
|
|
layout.prop(self, "report_when")
|
|
|
|
if self.report_when == "count":
|
|
|
|
row = layout.row()
|
|
|
|
row.label("Threshold: ")
|
|
|
|
row.prop(self, "threshold", text="")
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaVolumeSensorNode(PlasmaNodeBase, bpy.types.Node):
|
|
|
|
bl_category = "CONDITIONS"
|
|
|
|
bl_idname = "PlasmaVolumeSensorNode"
|
|
|
|
bl_label = "Region Sensor"
|
|
|
|
bl_width_default = 190
|
|
|
|
|
|
|
|
# Region Mesh
|
|
|
|
region = StringProperty(name="Region",
|
|
|
|
description="Object that defines the region mesh")
|
|
|
|
bounds = EnumProperty(name="Bounds",
|
|
|
|
description="Physical object's bounds",
|
|
|
|
items=bounds_types)
|
|
|
|
|
|
|
|
# Detector Properties
|
|
|
|
report_on = EnumProperty(name="Triggerers",
|
|
|
|
description="What triggers this region?",
|
|
|
|
options={"ANIMATABLE", "ENUM_FLAG"},
|
|
|
|
items=[("avatar", "Avatars", "Avatars trigger this region"),
|
|
|
|
("dynamics", "Dynamics", "Any non-avatar dynamic physical object (eg kickables)")],
|
|
|
|
default={"avatar"})
|
|
|
|
|
|
|
|
def init(self, context):
|
|
|
|
self.inputs.new("PlasmaVolumeSettingsSocketIn", "Trigger on Enter", "enter")
|
|
|
|
self.inputs.new("PlasmaVolumeSettingsSocketIn", "Trigger on Exit", "exit")
|
|
|
|
self.outputs.new("PlasmaConditionSocket", "Satisfies", "satisfies")
|
|
|
|
|
|
|
|
def draw_buttons(self, context, layout):
|
|
|
|
layout.prop(self, "report_on")
|
|
|
|
|
|
|
|
# Okay, if they changed the name of the ObData, that's THEIR problem...
|
|
|
|
layout.prop_search(self, "region", bpy.data, "objects", icon="MESH_DATA")
|
|
|
|
layout.prop(self, "bounds")
|
|
|
|
|
|
|
|
def export(self, exporter, tree, bo, so):
|
|
|
|
interface = exporter.mgr.add_object(plInterfaceInfoModifier, name=self.create_key_name(tree), so=so)
|
|
|
|
|
|
|
|
# Region Enters
|
|
|
|
enter_simple = self.find_input_socket("enter").allow
|
|
|
|
enter_settings = self.find_input("enter", "PlasmaVolumeReportNode")
|
|
|
|
if enter_simple or enter_settings is not None:
|
|
|
|
key = self._export_volume_event(exporter, tree, bo, so, plVolumeSensorConditionalObject.kTypeEnter, enter_settings)
|
|
|
|
interface.addIntfKey(key)
|
|
|
|
|
|
|
|
# Region Exits
|
|
|
|
exit_simple = self.find_input_socket("exit").allow
|
|
|
|
exit_settings = self.find_input("exit", "PlasmaVolumeReportNode")
|
|
|
|
if exit_simple or exit_settings is not None:
|
|
|
|
key = self._export_volume_event(exporter, tree, bo, so, plVolumeSensorConditionalObject.kTypeExit, exit_settings)
|
|
|
|
interface.addIntfKey(key)
|
|
|
|
|
|
|
|
# Don't forget to export the physical object itself!
|
|
|
|
# [trollface.jpg]
|
|
|
|
phys_bo = bpy.data.objects[self.region]
|
|
|
|
simIface, physical = exporter.physics.generate_physical(phys_bo, so, self.bounds, "{}_VolumeSensor".format(bo.name))
|
|
|
|
|
|
|
|
physical.memberGroup = plSimDefs.kGroupDetector
|
|
|
|
if "avatar" in self.report_on:
|
|
|
|
physical.reportGroup |= 1 << plSimDefs.kGroupAvatar
|
|
|
|
if "dynamics" in self.report_on:
|
|
|
|
physical.reportGroup |= 1 << plSimDefs.kGroupDynamic
|
|
|
|
|
|
|
|
def _export_volume_event(self, exporter, tree, bo, so, event, settings):
|
|
|
|
if event == plVolumeSensorConditionalObject.kTypeEnter:
|
|
|
|
suffix = "Enter"
|
|
|
|
else:
|
|
|
|
suffix = "Exit"
|
|
|
|
|
|
|
|
theName = "{}_{}_{}".format(tree.name, self.name, suffix)
|
|
|
|
print(" [LogicModifier '{}']".format(theName))
|
|
|
|
logicKey = exporter.mgr.find_create_key(plLogicModifier, name=theName, so=so)
|
|
|
|
logicmod = logicKey.object
|
|
|
|
logicmod.setLogicFlag(plLogicModifier.kMultiTrigger, True)
|
|
|
|
|
|
|
|
# LogicMod notification... This is one of the cases where the linked node needs to match
|
|
|
|
# exactly one key...
|
|
|
|
notify = plNotifyMsg()
|
|
|
|
notify.BCastFlags = (plMessage.kNetPropagate | plMessage.kLocalPropagate)
|
|
|
|
for i in self.find_outputs("satisfies"):
|
|
|
|
key = i.get_key(exporter, tree, so)
|
|
|
|
if key is None:
|
|
|
|
print(" WARNING: '{}' Node '{}' doesn't expose a key. It won't be triggered!".format(i.bl_idname, i.name))
|
|
|
|
else:
|
|
|
|
notify.addReceiver(key)
|
|
|
|
logicmod.notify = notify
|
|
|
|
|
|
|
|
# Now, the detector objects
|
|
|
|
print(" [ObjectInVolumeDetector '{}']".format(theName))
|
|
|
|
detKey = exporter.mgr.find_create_key(plObjectInVolumeDetector, name=theName, so=so)
|
|
|
|
det = detKey.object
|
|
|
|
|
|
|
|
print(" [VolumeSensorConditionalObject '{}']".format(theName))
|
|
|
|
volKey = exporter.mgr.find_create_key(plVolumeSensorConditionalObject, name=theName, so=so)
|
|
|
|
volsens = volKey.object
|
|
|
|
|
|
|
|
volsens.type = event
|
|
|
|
if settings is not None:
|
|
|
|
if settings.report_when == "first":
|
|
|
|
volsens.first = True
|
|
|
|
elif settings.report_when == "threshold":
|
|
|
|
volsens.trigNum = settings.threshold
|
|
|
|
|
|
|
|
# There appears to be a mandatory order for these keys...
|
|
|
|
det.addReceiver(volKey)
|
|
|
|
det.addReceiver(logicKey)
|
|
|
|
|
|
|
|
# End mandatory order
|
|
|
|
logicmod.addCondition(volKey)
|
|
|
|
return logicKey
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaVolumeSettingsSocket(PlasmaNodeSocketBase):
|
|
|
|
bl_color = (43.1, 24.7, 0.0, 1.0)
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaVolumeSettingsSocketIn(PlasmaVolumeSettingsSocket, bpy.types.NodeSocket):
|
|
|
|
allow = BoolProperty()
|
|
|
|
|
|
|
|
def draw(self, context, layout, node, text):
|
|
|
|
if not self.is_linked:
|
|
|
|
layout.prop(self, "allow", text="")
|
|
|
|
layout.label(text)
|
|
|
|
|
|
|
|
|
|
|
|
class PlasmaVolumeSettingsSocketOut(PlasmaVolumeSettingsSocket, bpy.types.NodeSocket):
|
|
|
|
pass
|