Instance behaviors
Opened this issue · 0 comments
Each non-property field on an instance can be thought of as a "behavior". These behaviors can be configured. When an instance is indexed, the the combination of the instance's class and the given index can be used to look up a behavior.
For example, when an instance of class "DataModel" is indexed with the string "GetService", this is checked against a behavior descriptor. If the descriptor has a behavior for that combination, that behavior is used. A descriptor might define instances of class DataModel with field GetService to have a behavior "ServerProvider.GetService". This behavior causes the GetService method to be returned, which then behaves as the GetService method does currently.
This feature extends descriptors with the $rbxmkBehavior
field. This is separate from descriptor extensions (#65) because the content of of the field has its own distinct structure:
// Fields with "Foo" are stand-ins for some defined value.
{
"$rbxmkBehavior": {
"Classes": { // Set of classes that define behaviors.
"FooInstance": { // Name of a class to which behaviors will be applied.
// If false, prevent behaviors being applied to subclasses of
// FooInstance. Defaults to true.
"Inheritable": false,
// Map of fields to behaviors. Each key is the name of a field that
// indexes an instance of FooInstance. The field need not exist, or
// have a corresponding descriptor. The "sym." prefix causes a
// symbol to be used. Each value is the name of a predefined
// behavior. While it may not always be the case, behaviors are
// usually derived from some instance property or method, in which
// case their name will have the form "<class>.<member>".
// Regardless, this is only a name.
"Behaviors": {
// Properties can be mapped to property behaviors.
"FooProperty": "Instance.Property",
// Methods can be mapped to method behaviors.
"FooMethod": "Instance.Method",
// A field name can be a symbol.
"sym.FooSymbol": "Instance.Symbol",
},
}
}
}
}
Most of the features planned to use descriptor extensions are more suited to use behaviors instead. The following list gives an overview of possible behaviors that could be implemented from current and planned features:
- Basic
Instance.Name
: Property used by methods that query the name of an instance.Instance.Parent
: Property used to get or set parent instance.Instance.ClearAllChildren
: Behaves as ClearAllChildren method.Instance.Clone
: Behaves as Clone method.Instance.Destroy
: Behaves as Destroy method.Instance.FindFirstAncestor
: Behaves as FindFirstAncestor method.Instance.FindFirstAncestorOfClass
: Behaves as FindFirstAncestorOfClass method.Instance.FindFirstAncestorWhichIsA
: Behaves as FindFirstAncestorWhichIsA method.Instance.FindFirstChild
: Behaves as FindFirstChild method.Instance.FindFirstChildOfClass
: Behaves as FindFirstChildOfClass method.Instance.FindFirstChildWhichIsA
: Behaves as FindFirstChildWhichIsA method.Instance.GetChildren
: Behaves as GetChildren method.Instance.GetDescendants
: Behaves as GetDescendants method.Instance.GetFullName
: Behaves as GetFullName method.Instance.IsA
: Behaves as IsA method.Instance.IsAncestorOf
: Behaves as IsAncestorOf method.Instance.IsDescendantOf
: Behaves as IsDescendantOf method.
- Symbolic
Desc
IsService
Properties
RawDesc
Reference
- Descend
Descend
: Behaves as Descend method.
- DataModel
ServiceProvider.GetService
: Behaves as GetService method.DataModel.Metadata
: Get/set root metadata.
- Attributes
Instance.AttributesSerialize
: Property used to serialize attributes.Instance.GetAttribute
: Behaves as GetAttribute method.Instance.GetAttributes
Behaves as GetAttributes method.Instance.SetAttribute
Behaves as SetAttribute method.Instance.SetAttributes
Behaves as SetAttributes method.
- Tags
Instance.Tags
: Property used to serialize tags.CollectionService.AddTag
: Behaves as AddTag method.CollectionService.GetAllTags
: Behaves as GetAllTags method.CollectionService.GetTagged
: Behaves as GetTagged method.CollectionService.GetTags
: Behaves as GetTags method.CollectionService.HasTag
: Behaves as HasTag method.CollectionService.RemoveTag
: Behaves as RemoveTag method.
- Pivots
PVInstance.GetPivot
: Behaves as GetPivot method.PVInstance.PivotTo
: Behaves as PivotTo method.Model.WorldPivotData
Model.WorldPivot
Model.PrimaryPart
Model.GetPrimaryPartCFrame
Model.SetPrimaryPartCFrame
Model.GetBoundingBox
BasePart.CFrame
BasePart.Orientation
BasePart.Position
BasePart.Rotation
BasePart.Size
Attachment.CFrame
: The primary property from which other Attachment properties are derived.Attachment.Axis
Attachment.Orientation
Attachment.Position
Attachment.SecondaryAxis
Attachment.WorldAxis
Attachment.WorldCFrame
Attachment.WorldOrientation
Attachment.WorldPosition
Attachment.WorldSecondaryAxis
Bone.Transform
Bone.TransformedCFrame
Bone.TransformedWorldCFrame
Notably, Instance.ClassName
cannot be a behavior, because the class is needed to resolve which behavior to select. It could be defined under $rbxmkConfig
instead, though.
The default global descriptor would define the expected class/fields to map to their expected behaviors. This would make all behaviors available by default, while giving the user the option to fully customize instance behavior.
Symbols
If symbols can be mapped, then should the implementation of the sym library be revised? Instead of having a defined set of symbols, the sym library would generate a symbol for the indexed name on the fly, allowing any symbol name to be used. Then behaviors can be mapped to any arbitrary symbol instead of from a limited baked-in set.
On the other hand, behaviors could eliminate the need for symbols entirely. The purpose of symbols is to have fields that are forward-compatible with any additions Roblox might make to instances. If all behaviors are user-definable, then they can just be redefined to make way for any additions from Roblox.