I am a fairly intermediate/advanced "scripter" in RBX.Lua and I have been wanting to move on to more advanced topics such as environments and metatable manipulation. Though, everywhere I look, it's mostly just the bare essentials, it doesn't teach you what it's useful for or how to use it in general. So, I was just hoping someone here could point me in the direction of something that's actually worthwhile. Thanks!
Roblox wiki, Youtube videos like Friaza Peaspod and many more. Even try understanding free model scripts.
If you are looking for some good places to learn environments and metatable manipulation, these are some good links
Environments http://lua-users.org/wiki/EnvironmentsTutorial
Metatable Manipulation http://lua-users.org/wiki/MetamethodsTutorial
PS: This site is also good for learning any level of Lua (Beginners - Professional)
I might tack on a few more resources if I recall them.
On metatables and the environment. First, we'll start with metatables.
A metatable is a table, just like any other, except that it can be assigned to another table to give the table more functionality. Metatables can be given specific keys called metamethods to control what happens when you, say, try to get something from the table that isn't there. That can be achieved with the __index metamethod. The __index method can be a table/userdata, or it can be a function.
Example 1: __index as a table
1 | t = { a = 1 , b = 2 , c = 3 } |
2 |
3 | mt = { |
4 | __index = { d = 4 } |
5 | } |
6 |
7 | setmetatable (t, mt) |
8 |
9 | print (t.a, t.b, t.c, t.d) --"d" isn't in the table, but it's in the __index, so we can still print it. |
Example 2: __index as a userdata
1 | t = { a = 1 , b = 2 , c = 3 } |
2 | part = Instance.new( "Part" ) |
3 | mt = { __index = part } |
4 | setmetatable (t, mt) |
5 | print (t.BrickColor) --There's no BrickColor in t, but there's a BrickColor in part. |
Example 3: __index as a function with the table and key as arguments
1 | t = { a = 1 , b = 2 , c = 3 } |
2 | mt = { |
3 | __index = function (self, key) |
4 | return "Key: " ..key.. " was not found." |
5 | end |
6 | } |
7 | setmetatable (t, mt) |
8 | print (t.d) |
There's also __newindex, a function that controls what happens when you try to set a new key in the table.
01 | t = { a = 1 , b = 2 , c = 3 } |
02 |
03 | mt = { |
04 | __newindex = function (self, key, value) |
05 | --[[Since doing "self[key] = value" would trigger __newindex again |
06 | and create an infinite loop, we have the rawget/rawset functions to |
07 | prevent the above from happening. |
08 | --]] |
09 | rawset (self, key, value) |
10 | print (key.. " was appended with a value of " ..value) |
11 | end |
12 | } |
13 |
14 | setmetatable (t, mt) |
15 |
16 | t.a = 5 |
17 | t.b = 4 |
18 | t.c = 3 |
19 | t.d = 2 |
20 | t.e = 1 |
The environment is a table of every global variable that the current stack level, or a given function, has access to. When you create a (non-localized) variable, it gets stored in the environment. You can access the script's environment with getfenv
.
1 | a = 1 |
2 | print (a) |
3 | print ( getfenv ().a) |
4 |
5 | getfenv ().b = 2 |
6 | print ( getfenv ().b) |
7 | print (b) |
We can alter the environment with setfenv
.
1 | setfenv ( 1 , { a = 1 , b = 2 , c = 3 } ) |
2 | print (a) |
Notice how we can't print anything, because print
was in our old environment, and now we have a new environment. How do we fix that? We use __index.
1 | env = getfenv () |
2 | mt = { __index = env } |
3 | setfenv ( 1 , setmetatable ( { a = 1 , b = 2 , c = 3 } , mt)) |
4 | print (a) --Now it works, since we brought the old environment with us in the __index. |
And that's the basics of it. From there, you just build on it with your creativity.
Closed as Too Broad by Unclear
This question has been closed because it is too broad and is generally unanswerable. Please ask a more specific question.
Why was this question closed?