158 lines
5.4 KiB
GDScript
158 lines
5.4 KiB
GDScript
# MIT License
|
|
#
|
|
# Copyright (c) 2023 Mark McKay
|
|
# https://github.com/blackears/cyclopsLevelBuilder
|
|
#
|
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
# of this software and associated documentation files (the "Software"), to deal
|
|
# in the Software without restriction, including without limitation the rights
|
|
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
# copies of the Software, and to permit persons to whom the Software is
|
|
# furnished to do so, subject to the following conditions:
|
|
#
|
|
# The above copyright notice and this permission notice shall be included in all
|
|
# copies or substantial portions of the Software.
|
|
#
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
# SOFTWARE.
|
|
|
|
@tool
|
|
class_name CommandSubtractBlock
|
|
extends CyclopsCommand
|
|
|
|
class NewBlockInfo extends RefCounted:
|
|
var data:MeshVectorData
|
|
var materials:Array[Material]
|
|
var path:NodePath
|
|
#var centroid:Vector3
|
|
var xform:Transform3D
|
|
|
|
#Public
|
|
var block_paths:Array[NodePath]
|
|
var block_to_subtract_path:NodePath
|
|
var block_name_prefix:String = "Block_"
|
|
|
|
#Private
|
|
var start_blocks:Array[TrackedBlock]
|
|
var subtracted_block_cache:TrackedBlock
|
|
var added_blocks:Array[NewBlockInfo]
|
|
|
|
func _init():
|
|
command_name = "Subtract blocks"
|
|
|
|
func restore_tracked_block(tracked:TrackedBlock)->CyclopsBlock:
|
|
var parent = builder.get_node(tracked.path_parent)
|
|
|
|
var block:CyclopsBlock = preload("../nodes/cyclops_block.gd").new()
|
|
block.mesh_vector_data = tracked.data
|
|
block.materials = tracked.materials
|
|
block.name = tracked.name
|
|
# block.selected = tracked.selected
|
|
block.global_transform = tracked.world_xform
|
|
block.collision_type = tracked.collision_type
|
|
block.collision_layer = tracked.collision_layers
|
|
block.collision_mask = tracked.collision_mask
|
|
|
|
parent.add_child(block)
|
|
block.owner = builder.get_editor_interface().get_edited_scene_root()
|
|
|
|
if tracked.selected:
|
|
var selection:EditorSelection = builder.get_editor_interface().get_selection()
|
|
selection.add_node(block)
|
|
|
|
return block
|
|
|
|
func will_change_anything()->bool:
|
|
var subtrahend_block:CyclopsBlock = builder.get_node(block_to_subtract_path)
|
|
var subtrahend_vol:ConvexVolume = subtrahend_block.control_mesh
|
|
subtrahend_vol = subtrahend_vol.transformed(subtrahend_block.global_transform)
|
|
|
|
if block_paths.is_empty():
|
|
return false
|
|
|
|
for minuend_path in block_paths:
|
|
var minuend_block:CyclopsBlock = builder.get_node(minuend_path)
|
|
var minuend_vol:ConvexVolume = minuend_block.control_mesh
|
|
minuend_vol = minuend_vol.transformed(minuend_block.global_transform)
|
|
|
|
if minuend_vol.intersects_convex_volume(subtrahend_vol):
|
|
return true
|
|
|
|
return false
|
|
|
|
func do_it():
|
|
var subtrahend_block:CyclopsBlock = builder.get_node(block_to_subtract_path)
|
|
var snap_to_grid_util:SnapToGridUtil = CyclopsAutoload.calc_snap_to_grid_util()
|
|
|
|
if start_blocks.is_empty():
|
|
var subtrahend_vol:ConvexVolume = subtrahend_block.control_mesh
|
|
subtracted_block_cache = TrackedBlock.new(subtrahend_block)
|
|
subtrahend_vol = subtrahend_vol.transformed(subtrahend_block.global_transform)
|
|
var subtra_xform_inv:Transform3D = subtrahend_block.global_transform.affine_inverse()
|
|
|
|
for path in block_paths:
|
|
var block:CyclopsBlock = builder.get_node(path)
|
|
|
|
var minuend_vol:ConvexVolume = block.control_mesh
|
|
minuend_vol = minuend_vol.transformed(block.global_transform)
|
|
if !minuend_vol.intersects_convex_volume(subtrahend_vol):
|
|
continue
|
|
|
|
var tracker:TrackedBlock = TrackedBlock.new(block)
|
|
start_blocks.append(tracker)
|
|
|
|
var fragments:Array[ConvexVolume] = minuend_vol.subtract(subtrahend_vol)
|
|
|
|
for f in fragments:
|
|
f.copy_face_attributes(minuend_vol)
|
|
#var centroid:Vector3 = f.get_centroid()
|
|
#centroid = snap_to_grid_util.snap_point(centroid)
|
|
#f.translate(-centroid)
|
|
f = f.transformed(block.global_transform.affine_inverse())
|
|
|
|
var block_info:NewBlockInfo = NewBlockInfo.new()
|
|
block_info.data = f.to_mesh_vector_data()
|
|
block_info.materials = block.materials
|
|
block_info.xform = block.global_transform
|
|
#block_info.centroid = centroid
|
|
added_blocks.append(block_info)
|
|
|
|
#Delete source blocks
|
|
for block_info in start_blocks:
|
|
var del_block:CyclopsBlock = builder.get_node(block_info.path)
|
|
del_block.queue_free()
|
|
|
|
subtrahend_block.queue_free()
|
|
|
|
#Create blocks
|
|
for info in added_blocks:
|
|
var block:CyclopsBlock = preload("../nodes/cyclops_block.gd").new()
|
|
var parent:Node = builder.get_node(start_blocks[0].path_parent)
|
|
parent.add_child(block)
|
|
block.owner = builder.get_editor_interface().get_edited_scene_root()
|
|
block.name = GeneralUtil.find_unique_name(parent, block_name_prefix)
|
|
block.mesh_vector_data = info.data
|
|
block.materials = info.materials
|
|
# block.global_transform = Transform3D.IDENTITY.translated(info.centroid)
|
|
block.global_transform = info.xform
|
|
|
|
info.path = block.get_path()
|
|
|
|
|
|
|
|
func undo_it():
|
|
|
|
for info in added_blocks:
|
|
var added_block:CyclopsBlock = builder.get_node(info.path)
|
|
added_block.queue_free()
|
|
|
|
restore_tracked_block(subtracted_block_cache)
|
|
|
|
for tracked in start_blocks:
|
|
restore_tracked_block(tracked)
|
|
|