Flipbook animation script for ImageLabels or Buttons

Copy and paste the following code into a LocalScript that is a child of an ImageLabel or ImageButton.

(or you can just insert this model: Flipbook Script - Roblox)

-- Created by Noobot9k

if script.Parent and (script.Parent:IsA("ImageLabel") or script.Parent:IsA("ImageButton")) then else error(script:GetFullName().. " must be a child of a ImageLabel or ImageButton.") end

local RunService = game:GetService("RunService")

local CurrentFrame = 0
local _nextFrame = tick()
local itterateConnection : RBXScriptConnection?

if script:GetAttribute("RandomStartFrame") then
	local TileCount : Vector2 = script:GetAttribute("TileCount")
	script:SetAttribute("CurrentFrame", math.random(0, TileCount.X * TileCount.Y))
end

function itterate(_, deltaTime)
	
	local Framerate : number =			script:GetAttribute("Framerate")
	local LoopStartFrame : number =		script:GetAttribute("LoopStartFrame") or 0
	local TileCount : Vector2 =			script:GetAttribute("TileCount")
	local TotalTiles : number =			script:GetAttribute("TotalTiles") or TileCount.X * TileCount.Y
	local StartTile : number =			script:GetAttribute("StartTile") or 0
	local TilePadding : Vector2 =		script:GetAttribute("TilePadding")
	local TileResolution : Vector2 =	script:GetAttribute("TileResolution")
	CurrentFrame = 						math.max(script:GetAttribute("CurrentFrame"), StartTile)
	
	local Y = math.floor(CurrentFrame / TileCount.X)
	local X = CurrentFrame - (Y * TileCount.X)
	
	script.Parent.ImageRectSize = TileResolution
	script.Parent.ImageRectOffset = (TileResolution + TilePadding) * Vector2.new(X, Y)
	
	if tick() >= _nextFrame then
		_nextFrame = tick() + (1 / Framerate)
		CurrentFrame = CurrentFrame + 1
	end
	if CurrentFrame >= TotalTiles then CurrentFrame = math.max(LoopStartFrame, StartTile) end -- CurrentFrame >= TileCount.X * TileCount.Y
	if script:GetAttribute("CurrentFrame") ~= CurrentFrame then script:SetAttribute("CurrentFrame", CurrentFrame) end
	
end
function disable()
	if itterateConnection then itterateConnection:Disconnect() itterateConnection = nil end
end
function enable()
	disable()
	itterateConnection = RunService.Stepped:Connect(itterate)
end
enable()

Your hierarchy should look something like this when you’re done:
image

Add the following attributes to the LocalScript:
image
CurrentFrame, Framerate, LoopStartFrame, StartTile, and TotalTiles are numbers.
RandomStartFrame is a boolean.
TileCount, TilePadding, and TileResolution are all Vector2s.

Set the image of the ImageLabel or ImageButton to be one with a grid of smaller images on it that are each similar looking but slightly different.

Set the TileCount attribute to be how many smaller images are in the larger image along the X-axis and the Y-axis. X is how many columns and Y is how many rows.

Set the TileResolution to the resolution of each tile in the source image. If your image is 1024x1024 and your TileCount is 4x4 then your TileResolution should be 256x256.
Note that the max image resolution on Roblox is 1024x1024 so if you upload a 2048x2048 image with a TileCount of 4x4, your TileResolution wouldn’t be 512x512, it would still be 256x256 as Roblox would downscale your texture to 1024².

Set TotalTiles to the total number of tiles in your animation across all rows and columns. If TileCount is 4x4 but the bottom row only has 3 images and then an empty space, then you should set TotalTiles to 15 instead of 16.

The Framerate attribute is how fast your animation plays. Higher values are faster and lower ones are slower. 16 is usually a good starting value.

If this animation isn’t always visible, it would be a good idea to make another script disable this one to save on performance so it’s not running in the background. Roblox unfortunately doesn’t provide an easy way to check if a UI item is visible or not.

See my reply below for an example of it in action.

23 Likes

Not exactly sure what you mean by “flipbook animation”, since alone it’s pretty vague on what it could look like. Perhaps a video or gif showing it could help?

Here’s an example:

This flipbook tileset image is 896x896.

Its TileCount would be 7x7.
TileResolution would be 128x128.
Framerate should be 60 in this case.
TotalTiles should be 49 and not 50 since there’s an empty spot in the bottom-right corner.

In-game, it would look something like this:
download

You can use this website to preview what your flipbook animation will look like before you upload it to Roblox:
https://is.si/animator/

This is mostly meant to be used for UI items like a saving or loading animation.

2 Likes

You can make Never Gonna with that lmao

What programm can i use to generate such a flipbook image. For example from a video

If you make the animation with Blender you can render each frame out as its own PNG and then use a program to combine them into a single tileset like the example I gave above. This is the program I use.

If you have a video you’d like to do this with then you’ll need to find a program that can split it into individual PNG files of each frame.

1 Like

I modulated it for everyone who wants to have this code in a more OOP-friendly way! Now you can call the module and enable the animation by passing the imagelabel or imagebutton as the first parameter!

Make Sure to move the attributes from the module script to the Object (imagelabel/Imagebutton) you are passing through the function!
Note: this is also only set up to be a CLIENT module

-- Created by Noobot9k
-- Modulated by ImPremiumJay
local RunService = game:GetService("RunService")
local Animate = {}

local CurrentFrame = {}
local _nextFrame = {}
local itterateConnection = {}

local function itterate(Obj,_, deltaTime)

	local Framerate : number =			Obj:GetAttribute("Framerate")
	local LoopStartFrame : number =		Obj:GetAttribute("LoopStartFrame") or 0
	local TileCount : Vector2 =			Obj:GetAttribute("TileCount")
	local TotalTiles : number =			Obj:GetAttribute("TotalTiles") or TileCount.X * TileCount.Y
	local StartTile : number =			Obj:GetAttribute("StartTile") or 0
	local TilePadding : Vector2 =		Obj:GetAttribute("TilePadding")
	local TileResolution : Vector2 =	Obj:GetAttribute("TileResolution")
	CurrentFrame[Obj] = 						math.max(Obj:GetAttribute("CurrentFrame"), StartTile)

	local Y = math.floor(CurrentFrame[Obj] / TileCount.X)
	local X = CurrentFrame[Obj] - (Y * TileCount.X)

	Obj.ImageRectSize = TileResolution
	Obj.ImageRectOffset = (TileResolution + TilePadding) * Vector2.new(X, Y)

	if tick() >= _nextFrame[Obj] then
		_nextFrame[Obj] = tick() + (1 / Framerate)
		CurrentFrame[Obj] = CurrentFrame[Obj] + 1
	end
	if CurrentFrame[Obj] >= TotalTiles then CurrentFrame[Obj] = math.max(LoopStartFrame, StartTile) end -- CurrentFrame >= TileCount.X * TileCount.Y
	if Obj:GetAttribute("CurrentFrame") ~= CurrentFrame[Obj] then Obj:SetAttribute("CurrentFrame", CurrentFrame[Obj]) end

end

function Animate.disable(Obj)
	if itterateConnection[Obj] then itterateConnection[Obj]:Disconnect() itterateConnection[Obj] = nil end
end
function Animate.enable(Obj)
	Animate.disable(Obj)
	
	CurrentFrame[Obj] = 0
	_nextFrame[Obj] = tick()
	itterateConnection[Obj] = {}


	if Obj:GetAttribute("RandomStartFrame") then
		local TileCount : Vector2 = Obj:GetAttribute("TileCount")
		Obj:SetAttribute("CurrentFrame", math.random(0, TileCount.X * TileCount.Y))
	end
	
	itterateConnection[Obj] = RunService.Stepped:Connect(function(T, deltaTime)
		itterate(Obj,T,deltaTime)
	end)
end

return Animate
4 Likes

this does not work. When uploading a spritesheet at 2100x1800 res but using TileResolution = Vector2.new(2100/5, 1800/6) results in incorrect output (the sheet layout is 5 by 6)

Roblox has a max texture resolution of 1024x1024. You’ll need to adjust your settings for that.

You are right! This fixed my issue. Thanks a ton