Scopes
In Fusion, you create a lot of objects. These objects need to be destroyed when you're done with them.
Fusion has some coding conventions to make large quantities of objects easier to manage.
Scopes¶
When you create many objects at once, you often want to destroy them together later.
To make this easier, some people add their objects to an array. Arrays that group together objects like this are given a special name: scopes.
To create a new scope, create an empty array.
2 3 4 |
|
Later, when you create objects, they will ask for a scope as the first argument.
2 3 4 5 |
|
That object will add its destroy()
function to the scope:
2 3 4 5 6 7 |
|
Repeat as many times as you like. Objects appear in order of creation.
2 3 4 5 6 7 |
|
Later, destroy the scope by using the doCleanup()
function. The contents are
destroyed in reverse order.
2 3 4 5 6 7 8 9 10 11 12 13 |
|
Scopes passed to doCleanup
can contain:
- Functions to be run (like those
destroy()
functions above) - Roblox instances to destroy
- Roblox event connections to disconnect
- Your own objects with
:destroy()
or:Destroy()
methods to be called - Other nested scopes to be cleaned up
You can add these manually using table.insert
if you need custom behaviour,
or if you are working with objects that don't add themselves to scopes.
That's all there is to scopes. They are arrays of objects which later get passed to a cleanup function.
Improved Scopes¶
This syntax is recommended
From now on, you'll see this syntax used throughout the tutorials.
Fusion can help manage your scopes for you. This unlocks convenient syntax, and allows Fusion to optimise your code.
You can call scoped()
to obtain a new scope.
2 3 4 5 6 7 8 9 10 |
|
Unlike {}
(which always creates a new array), scoped
can re-use old arrays.
This helps keep your program running smoothly.
Beyond making your code more efficient, you can also use scoped
for convenient
syntax.
You can pass a table of functions into scoped
:
2 3 4 5 6 7 8 9 10 11 12 13 |
|
If those functions take scope
as their first argument, you can use them as
methods directly on the scope:
2 3 4 5 6 7 8 9 10 11 12 13 |
|
This makes it harder to mess up writing scopes. Your code reads more naturally, too.
For convenience, Fusion exposes the insert
method to insert destruction tasks
into a scope:
local conn, ins = scope:insert(
RunService.Heartbeat:Connnect(doUpdate),
Instance.new("Part", workspace)
)
Adding Methods In Bulk¶
Try passing Fusion
to scoped()
- it's a table with functions, too.
local scope = scoped(Fusion)
-- all still works!
local thing1 = scope:Value("i am thing 1")
local thing2 = scope:Value("i am thing 2")
local thing3 = scope:Value("i am thing 3")
scope:doCleanup()
This gives you access to all of Fusion's functions without having to import each one manually.
If you need to mix in other things, you can pass in another table.
local scope = scoped(Fusion, {
Foo = ...,
Bar = ...
})
You can do this for as many tables as you need.
Conflicting names
If you pass in two tables that contain things with the same name, scoped()
will error.
Reusing Methods From Other Scopes¶
Sometimes, you'll want to make a new scope with the same methods as an existing scope.
local foo = scoped({
Foo = Foo,
Bar = Bar,
Baz = Baz
})
-- `bar` should have the same methods as `foo`
-- it'd be nice to define this once only...
local bar = scoped({
Foo = Foo,
Bar = Bar,
Baz = Baz
})
print(foo.Baz == bar.Baz) --> true
bar:doCleanup()
foo:doCleanup()
To do this, Fusion provides a deriveScope
function. It behaves like scoped
but lets you skip defining the methods. Instead, you give it an example of what
the scope should look like.
local foo = scoped({
Foo = Foo,
Bar = Bar,
Baz = Baz
})
-- `bar` should have the same methods as `foo`
-- now, it's only defined once!
local bar = foo:deriveScope()
print(foo.Baz == bar.Baz) --> true
bar:doCleanup()
foo:doCleanup()
Deriving scopes like this is highly efficient because Fusion can re-use the same information for both scopes. It also helps keep your definitions all in one place.
You can also add more method tables if you'd like to.
local foo = scoped({
Foo = Foo,
Bar = Bar,
Baz = Baz
})
-- `bar` should have the same methods as `foo`
-- now, it's only defined once!
local bar = foo:deriveScope({
Garb = Garb
})
print(bar.Garb) --> function: 0x123456789abcdef
print(foo.Garb) --> nil
Inner Scopes¶
The main reason you would want to create a new scope is to create things that get destroyed at different times.
For example, imagine you're creating a dropdown menu. You create a new scope for the menu, which you clean up when the menu is closed.
local uiScope = scoped(Fusion)
-- ... create the ui ...
uiScope:insert(
dropdownOpened:Connect(function()
local dropdownScope = uiScope:deriveScope()
-- ... create the dropdown ...
dropdownScope:insert(
dropdownClosed:Connect(function()
dropdownScope:doCleanup()
end)
)
end)
)
This ordinarily works just fine; when the dropdown is opened, the new scope is created, and when the dropdown is closed, the new scope is destroyed.
However, what if the UI gets cleaned up while the dropdown is open? The
uiScope
will get cleaned up, but the dropdownScope
will not.
To help with this, Fusion provides an innerScope
method. It works just like
deriveScope
, but it adds in extra logic:
- When the original scope is cleaned up, the 'inner scope' is cleaned up too
- You can still call
doCleanup()
to clean the inner scope up early
local uiScope = scoped(Fusion)
-- ... create the ui ...
uiScope:insert(
dropdownOpened:Connect(function()
local dropdownScope = uiScope:innerScope()
-- ... create the dropdown ...
dropdownScope:insert(
dropdownClosed:Connect(function()
dropdownScope:doCleanup()
end)
)
end)
)
Now, the dropdown scope is guaranteed to be cleaned up if the UI it came from is cleaned up. This strictly limits how long the dropdown can exist for.
Inner scopes are often the safest choice for creating new scopes. They let you
call doCleanup
whenever you like, but guarantee that they won't stick around
beyond the rest of the code they're in.
When You'll Use This¶
Scopes might sound like a lot of upfront work. However, you'll find in practice that Fusion manages a lot of this for you, and it makes your code much more resilient to memory leaks and other kinds of memory management issues.
You'll need to create and destroy your own scopes manually sometimes. For example, you'll need to create a scope in your main code file to start using Fusion, and you might want to make a few more in other parts of your code.
However, Fusion manages most of your scopes for you, so for large parts of your codebase, you won't have to consider scopes and destruction at all.