A guide on how to achieve clipped bulletholes

Have you ever wondered how you can make bulletholes clip of walls?

Hello, I have the solution for anyone who is interested in this! In this topic, I will be covering a step by step guide on how to achieve the same result as the examples below.

Here is an example video showing the end result (focus on the bullethole part):


1 more example:

First of all we need a module called WorldPositionToGuiPosition by @orange451.
Put the module inside ReplicatedStorage

image

I won’t be covering on how to make a gun system, so instead I made a simple localscript that makes sure that I will be “shooting a bullet” from my mouse.
Here is the localscript:

local UIS = game:GetService("UserInputService")
local mouse = game:GetService("Players").LocalPlayer:GetMouse()
mouse.Button1Down:Connect(function()
	local target = mouse.Target
	local hitPosition = mouse.Hit
	print(hitPosition)
end)
Explanation

What this code does is it checks every time the mouse is pressed, where the mouse’s position is inside 3D space, and prints that position.

I also made a part, resized it to Vector3.new(2, 10, 10) and anchored it so we can create clipped bulletholes on it.

Now a disclaimer is, is that we have to expand on the functions that the WorldPositionToGuiPosition module already provides. This is because the code we are going to write is going to be lengthy and could be used everywhere in your project. Okay, lets begin.

The main function we are going to write we will call it “WorldToGui.SetImageSurfaceGui” and it is a global function (no local infront of it). The parameters we will need for now is the target and position. Then we will use the WorldPositionToGuiPosition function, and get values returned which we store inside worldPosition.
The values that get returned are Surface, LocalWidth, LocalHeight, RelativeWidth, RelativeHeight
We get the face we hit:

function WorldToGui.SetImageSurfaceGui(target, position)
	local worldPosition = {WorldToGui:WorldPositionToGuiPosition(target, position)}
	local Face = worldPosition[1] --the face you hit
end 

Then we make a new SurfaceGui instance. We set the CanvasSize based on the LocalWidth and LocalHeight we got provided which is inside the worldPosition variable. What causes the clipping is the SurfaceGui property called “ClipsDescendants” set to true, so make 100% sure you have coded that in. The scale is set to 50, I recommend keeping it that way and not making it a parameter.
We now have this:

function WorldToGui.SetImageSurfaceGui(target, position)
	local worldPosition = {WorldToGui:WorldPositionToGuiPosition(target, position)}
	local Face = worldPosition[1] --the face you hit'

    local SCALE = 50
	local surfaceGui = Instance.new("SurfaceGui")
	surfaceGui.CanvasSize = Vector2.new(SCALE*worldPosition[2], SCALE*worldPosition[3])
	surfaceGui.Face = Face
	surfaceGui.Parent = target
	surfaceGui.ClipsDescendants = true
	surfaceGui.LightInfluence = 1
end 

We then need a frame and an ImageLabel to be created. We determine the position of the ImageLabel based on the RelativeWidth and the RelativeHeight. We also add parameters that are called imageId for the image and duration to make sure the the ImageLabel gets deleted after the specified time.
We now should have:

function WorldToGui.SetImageSurfaceGui(target, position, imageId, duration)
	local worldPosition = {WorldToGui:WorldPositionToGuiPosition(target, position)}
	local Face = worldPosition[1] --the face you hit

    local SCALE = 50
	local surfaceGui = Instance.new("SurfaceGui")
	surfaceGui.CanvasSize = Vector2.new(SCALE*worldPosition[2], SCALE*worldPosition[3])
	surfaceGui.Face = Face
	surfaceGui.Parent = target
	surfaceGui.ClipsDescendants = true
	surfaceGui.LightInfluence = 1

	local frame = Instance.new("Frame")
	frame.Parent = surfaceGui
	frame.ClipsDescendants = true
	frame.BackgroundTransparency = 1
	frame.Size = UDim2.new(1,0,1,0)

	local image = Instance.new("ImageLabel")
	image.Parent = frame
	image.Size = UDim2.new(0, 25, 0, 25) --standard size
	image.Position = UDim2.new(worldPosition[4], 0, worldPosition[5], 0)
	image.AnchorPoint = Vector2.new(0.5, 0.5)
	image.ImageTransparency = .1 --standard ImageTransparency
	image.BackgroundTransparency = 1
	image.ClipsDescendants = true
	image.Image = imageId
	game:GetService("Debris"):AddItem(image, duration)
end

We now have a function that does it all automatically, good! We go back to the localscript and use our made function:

local UIS = game:GetService("UserInputService")
local WorldPositionToGui = require(game:GetService("ReplicatedStorage"):WaitForChild("WorldToGui"))
local mouse = game:GetService("Players").LocalPlayer:GetMouse()
local imageId = "http://www.roblox.com/asset/?id=2859765692"
mouse.Button1Down:Connect(function()
	local target = mouse.Target
	local hitPosition = mouse.Hit.Position
	WorldPositionToGui.SetImageSurfaceGui(target, hitPosition, imageId, 100)
end)

Use any image you like, as long as it works. If you hopefully get no errors, you should get something like this:

We are done, you might think. But there are some things that we should do before we can rely on this function. First of all, every time we use the function a new SurfaceGui gets created, even for the same face and object.

image

This is also true for the Frame.

This will obviously cause allot of lag if you shoot the object allot of times, because too many SurfaceGuis will exist in the workspace.

The fix
To fix this issue, we have to check if a SurfaceGui or Frame already exists on that face. If a SurfaceGui does exist on that face, then we use that SurfaceGui, if it doesn’t exist yet, we create a new one!

To make things simple, we are going to create a new function called “checkGuiOnFace”. It is a local function, and has 2 parameters: target and Face.

local function checkGuiOnFace(target, Face)
	local partInstances = target:GetDescendants()
	for	i, v in pairs(partInstances) do
		if v:IsA("SurfaceGui") and v.Face == Face then
			return v
		end
	end
end
Explanation

It will go through all the descendants of the target, and check if a SurfaceGui exists. If the found SurfaceGui its property called Face, matches the given Face value, the function returns the SurfaceGui.

We do the same but this time for Frame.

local function checkFrameOnFace(target, Face)
	local partInstances = target:GetDescendants()
	for	i, v in pairs(partInstances) do
		if v:IsA("Frame") and v.Parent.Face == Face then
			return v
		end
	end
end

Then we implement it:

function WorldToGui.SetImageSurfaceGui(target, position, imageId, duration)
	local worldPosition = {WorldToGui:WorldPositionToGuiPosition(target, position)}
	local Face = worldPosition[1] --the face you hit

    local SCALE = 50
	local surfaceGui = checkGuiOnFace(target, Face)
	if surfaceGui == nil then
		surfaceGui = Instance.new("SurfaceGui")
		surfaceGui.CanvasSize = Vector2.new(SCALE*worldPosition[2], SCALE*worldPosition[3])
		surfaceGui.Face = Face
		surfaceGui.Parent = target
		surfaceGui.ClipsDescendants = true
		surfaceGui.LightInfluence = 1
	end

	local frame = checkFrameOnFace(target, Face)
	if frame == nil then
		frame = Instance.new("Frame")
		frame.Parent = surfaceGui
		frame.ClipsDescendants = true
		frame.BackgroundTransparency = 1
		frame.Size = UDim2.new(1,0,1,0)
	end

	local image = Instance.new("ImageLabel")
	image.Parent = frame
	image.Size = UDim2.new(0, 25, 0, 25) --standard size
	image.Position = UDim2.new(worldPosition[4], 0, worldPosition[5], 0)
	image.AnchorPoint = Vector2.new(0.5, 0.5)
	image.ImageTransparency = .1 --standard ImageTransparency
	image.BackgroundTransparency = 1
	image.ClipsDescendants = true
	image.Image = imageId
	game:GetService("Debris"):AddItem(image, duration)
	return image
end

Now the same works, but it will only spawn one SurfaceGui per face:
image

Another matter that I find quite important to address is the fact that many SurfaceGuis will exist inside instances whilst there not being any ImageLabels in it. For example, a player might have gone around the map shooting every object, then every object has a SurfaceGui serving no purpose because the ImageLabel was deleted. This will accumulate hunderds or not thousands of SurfaceGuis in total, causing performance issues.

That is why I want to share the following.

First and foremost, I suggest making this inside a separate Utilities modulescript. Make an array called objectDebris.

local objectDebris = {
    --these are comments showing the structure of the array
	--[object] = coroutine,
	--[object] = coroutine,
	--[object] = coroutine,
	--[object] = coroutine,
	--[object] = coroutine
}

Make a local function called “customDebris” that receives two parameters, object and duration.
Here is the function:

function Utilities.customDebris(object, lifetime)
	if object == nil or type(object.Destroy) ~= "function" then return end
	local cor = task.spawn(function()
		task.wait(lifetime)
		if object then
			object:Destroy()
		end
	end)
	objectDebris[object] = cor
end

Now another called updateLifetimeDebris, it receives the same parameters as customDebris does, object and lifetime. Here is the function:

function Utilities.updateLifetimeDebris(object, lifetime)
	local cor = objectDebris[object]
	if cor ~= nil and (coroutine.status(cor) ~= "dead" or coroutine.status(cor) ~= "suspended") then
		coroutine.close(cor)
	end
	Utilities.customDebris(object, lifetime)
end

Now back to the WorldToGui.SetImageSurfaceGui function. Add Utilities.updateLifetimeDebris at the bottom of the function, the object as the SurfaceGui and the duration the same duration as the ImageLabel.

function WorldToGui.SetImageSurfaceGui(target, position, imageId, duration)
	local SCALE = 50
	local worldPosition = {WorldToGui:WorldPositionToGuiPosition(target, position)}
	local Face = worldPosition[1] --the face you hit
	local surfaceGui = checkGuiOnSurface(target, Face)
	if surfaceGui == nil then
		surfaceGui = Instance.new("SurfaceGui")
		surfaceGui.CanvasSize = Vector2.new(SCALE*worldPosition[2], SCALE*worldPosition[3])
		surfaceGui.Face = Face
		surfaceGui.Parent = target
		surfaceGui.ClipsDescendants = true
		surfaceGui.LightInfluence = 1
	end
	local frame = checkFrameOnSurface(target, Face)
	if frame == nil then
		frame = Instance.new("Frame")
		frame.Parent = surfaceGui
		frame.ClipsDescendants = true
		frame.BackgroundTransparency = 1
		frame.Size = UDim2.new(1,0,1,0)
	end
	local image = Instance.new("ImageLabel")
	image.Parent = frame
	image.Size = UDim2.new(0, 25, 0, 25) --standard size
	image.Position = UDim2.new(worldPosition[4], 0, worldPosition[5], 0)
	image.AnchorPoint = Vector2.new(0.5, 0.5)
	image.ImageTransparency = .1 --standard ImageTransparency
	image.BackgroundTransparency = 1
	image.ClipsDescendants = true
	image.Image = imageId
	game:GetService("Debris"):AddItem(image, duration)
	updateLifetimeDebris(surfaceGui, duration)
	return image
end

Now whenever the last ImageLabel of a face gets destroyed, the SurfaceGui gets destroyed with it, making sure it doesn’t exist unnecessarily.

Thank you for reading. I hope you have gained some more knowledge from this and that I was clear. If you have any feedback on my approach or my writing skills or just want to say something then please put them below.

Feel free to copy everything!

Here is the place:
clipping_bulletholes_explanation.rbxl (58.0 KB)

15 Likes

Now for a little variation, rotate the image of the hole by any degree. See what happens :smiley:

Hello, that is not possible because the “clipping” effect will go away if you rotate the ImageLabel by any degree. :frowning:

this wont be unoptimized? since its rendering it more than a decal but nice!!

Also just remembered this won’t work on anything else other than normal parts.

Hello, I will later be adding more explanation on how to avoid this issue, with using the normal decal approach, whilst also making sure that bulletholes clip on normal parts.

Hello, could you eleborate on your comment? How does using ImageLabels over decals affect performance? In the explanation, the system makes sure to delete any SurfaceGui’s that have no Imagelabels left in it.

this post kind of explains it. Also yeah they will not reflect light. I tought it would make much more difference since its rendering 2 Instances at the same time, my bad. What about the meshparts also does this work on them ?

Hello, it does not work on meshparts because they are not a part and the bulletholes will “float”, because they are hitting the box of the mesh. However, if you just detect if the raycastresult.Instance is a meshpart or contains child that is a mesh, the system should opt for using a part with a decal on top of it.

1 Like