| mNo edit summary | No edit summary | ||
| Line 4: | Line 4: | ||
| local ProtofluxColor = require("Module:ProtoFlux_Type_Color") | local ProtofluxColor = require("Module:ProtoFlux_Type_Color") | ||
| local css = { | |||
| 	attachment_point = function(self, color, order) | |||
| 		return string.format( | |||
| 			"width: 30px; background-color: black; fill: %s; stroke: %s; order: %s;", | |||
| 			ColorToCss(MultA(color, 0.15)), | |||
| 			ColorToCss(color), | |||
| 			order | |||
| 		) | |||
| 	end, | |||
| } | |||
| local fieldHandlers = {} | local fieldHandlers = {} | ||
| Line 170: | Line 181: | ||
| function CreateFieldLabel(Container, Field) | function CreateFieldLabel(Container, Field) | ||
| 	Container | 	local c = Container | ||
| 		:tag('div') | 		:tag('div') | ||
| 			:attr('title', Field and (Field.Name .. ' ' .. Field.FieldType .. '<' .. Field.Type .. '>') or '') -- Add a basic mouseover description | 			:attr('title', Field and (Field.Name .. ' ' .. Field.FieldType .. '<' .. Field.Type .. '>') or '') -- Add a basic mouseover description | ||
| 			:cssText("display:flex; width:30%; align-items:center;") | 			:cssText("display:flex; width:30%; align-items:center;") | ||
| 	local color, is_impulse = GetColor(Field) | |||
| 	local segments = GetTypeSegments(Field) | |||
| 	CreateConnectorAttachmentPoint(extrasButton, "hover text", color, segments, false, true) | |||
| 	local extrasButton = c:tag('div'):cssText('display:flex;') | |||
| 	extrasButton:tag('div') -- HTML div to contain the extras button | |||
| 		:cssText('align-self:stretch; width:32px; margin-right:2px; background-color: #2B2F35; /* Resonite Mid */ text-align:center; border-radius: 8px;display:flex;align-items:center;justify-content:center;') | |||
| 		:wikitext("☰") | |||
| 		:done() -- Close extras button div | |||
| 	c:tag('div') -- HTML div to contain the field label | |||
| 		:cssText('text-align: left; overflow: hidden; text-overflow: ellipsis; flex: 1;') | |||
| 		:wikitext( (Field and Field.Name or '') .. ':' ) | |||
| 		:done() -- Close field label div | |||
| end | end | ||
| Line 306: | Line 325: | ||
| end | end | ||
| --- Returns an  | --- Returns a CSS color for the data type or impulse and whether the type is an impulse | ||
| -- @param  | -- @param connector Table containing a Name (Optional) and Type (Required) for the input/output. | ||
| function  | function GetColor(connector) | ||
| 	if connector == nil then error("connector is nil") end | |||
| 	if connector == nil or connector.Type == nil then error("Missing Type") end | |||
| 	local color = ProtofluxColor.get_impulse_color(connector.Type) | |||
| 	if color ~= nil then | |||
| 		return color, true | |||
| 	end | |||
| 	return GetTypeColor(connector), false | |||
| end | |||
| --- Returns a CSS color for the data type | |||
| -- @param connector Table containing a Name (Optional) and Type (Required) for the input/output or global. | |||
| function GetTypeColor(connector) | |||
| 	if connector == nil then error("connector is nil") end | |||
| 	if connector == nil or connector.Type == nil then error("Missing Type") end | |||
| 	local color = ProtofluxColor.get_type_color(connector.Type) | |||
| 	return MultRGB(color, 1.5) | |||
| end | |||
| function MultRGB(color, fact) | |||
| 	return { | |||
| 		r=color.r*fact, | |||
| 		g=color.g*fact, | |||
| 		b=color.b*fact, | |||
| 		a=color.a, | |||
| 	} | |||
| end | |||
| --- Creates a new input or output attachement point for a connector row. | |||
| --	@param container The Scribunto HTML node we'll be creating this row inside of. Should be a container of some sort. | |||
| --  @param hover_text Text to show when hovering, possibly nil | |||
| --	@param color The color of this attachment point (table with r, g, b, optionally a). | |||
| --	@param segments The number of segments for the type, used for vectors or matrices. | |||
| --	@param is_impulse True if this is an impulse, false if it is a data type. | |||
| --	@param is_input True if this is an input side attachment point, false if it is an output side attachment point. | |||
| function CreateConnectorAttachmentPoint(container, hover_text, color, segments, is_impulse, is_input) | |||
| 	local connector_type = nil | |||
| 	if is_impulse then connector_type = "InputArrow" | |||
| 	elseif is_input then connector_type = "InputBox" | |||
| 	else connector_type = "OutputBox" | |||
| 	end | |||
| 	container | |||
| 		:tag( 'div' ) | |||
| 			:attr('title', hover_text or "") -- Add a basic mouseover description | |||
| 			:cssText(css:attachment_point(color, is_input and 1 or 3)) | |||
| 			:wikitext(mw.getCurrentFrame():expandTemplate({title = 'ProtoFluxConnector', args = {Type=connector_type, Segments=segments}})) | |||
| end | |||
| local type_segments = { | |||
| 	enabled = { | |||
| 		bool=true, | |||
| 		byte=true, | |||
| 		sbyte=true, | |||
| 		ushort=true, | |||
| 		short=true, | |||
| 		uint=true, | |||
| 		int=true, | |||
| 		ulong=true, | |||
| 		long=true, | |||
| 		float=true, | |||
| 		double=true, | |||
| 		decimal=true, | |||
| 	}, | |||
| 	suffix = { | |||
| 		["2"]=2, | |||
| 		["3"]=3, | |||
| 		["4"]=4, | |||
| 		--["2x2"]=2, | |||
| 		--["3x3"]=3, | |||
| 		--["4x4"]=4, | |||
| 	}, | |||
| } | |||
| --- Returns a CSS color for the data type | |||
| -- @param connector Table containing a Name (Optional) and Type (Required) for the input/output. | |||
| function GetTypeSegments(connector) | |||
| 	if connector == nil then error("connector is nil") end | 	if connector == nil then error("connector is nil") end | ||
| 	if connector == nil or connector.Type == nil then error("Missing Type") end | 	if connector == nil or connector.Type == nil then error("Missing Type") end | ||
| 	local base = connector.Type:match("^%a+") | |||
| 	if not type_segments.enabled[base] then return 1 end | |||
| 	local suffix = connector.Type:sub(#base+1) | |||
| 	return type_segments.suffix[suffix] or 1 | |||
| end | end | ||
| return p | return p | ||
Revision as of 17:43, 11 February 2024
The Text Component
Lua error at line 373: attempt to index local 'container' (a nil value).
The RelativePositioner Component
Lua error at line 373: attempt to index local 'container' (a nil value).
The AssetMultiplexer<ITexture2D> Component
Lua error at line 373: attempt to index local 'container' (a nil value).
-- Package definition to return to Scribunto - this allows us to define methods that can be called
-- when this module is targeted.
local p = {}
local ProtofluxColor = require("Module:ProtoFlux_Type_Color")
local css = {
	attachment_point = function(self, color, order)
		return string.format(
			"width: 30px; background-color: black; fill: %s; stroke: %s; order: %s;",
			ColorToCss(MultA(color, 0.15)),
			ColorToCss(color),
			order
		)
	end,
}
local fieldHandlers = {}
--- Method that generates the HTML output
--	@param frame A Scribunto frame instance. frame.args contains the parameters passed into the module call
function p.GenerateUI( frame )
	-- Parse the JSON input, returning an empty array if no argument was passed
	local fields = mw.text.jsonDecode(frame.args.Fields or '[]')
	
	-- Create an HTML div element to contain our component UI
	local componentContainer = mw.html.create( 'div' )
	componentContainer
		:attr('class', frame.args.Inline and '' or 'floatright')
		:cssText('color: rgb(224, 224, 224); background-color: rgb(18, 20, 28); width: 512px; display: flex; flex-direction: column; align-items: stretch; gap: 6px; padding: 8px;')
		:tag('div') -- HTML div to contain the component title and buttons
			:cssText('display:flex;gap:4px;')
			:tag('div') -- component title
				:cssText('color: #f8f770; /* Resonite Yellow */ border-radius: 8px; background-color: #2b2f35; /* Resonite Mid */ text-align:center; font-size: 1.25rem; font-weight: bold; flex-grow: 1;')
				:wikitext(frame.args.Name)
				:done() -- Close component title div
			:tag('div') -- duplicate button
				:cssText('background-color: #3f9e44; /* Resonite Sub-Green */ border-radius: 8px; width: 2.5rem; display: flex; align-items:center; justify-content:center;')
				:wikitext("[[File:Duplicate.svg|Duplicate|28px]]")
				:done()
			:tag('div') -- destroy button
				:cssText('background-color: #ae5458; /* Resonite Sub-Red */ border-radius: 8px; width: 2.5rem; display: flex; align-items:center; justify-content:center;')
				:wikitext("[[File:Destroy.svg|Destroy|28px]]")
				:done()
			:done() -- Close title and buttons div
	
	-- Iterate over each row, and populate it with the inputs/outputs. If the node is asymmetric,
	-- the value passed for either input or output might be nil.
	for i=1,#fields do
		CreateField(componentContainer, fields[i])
	end
			
	-- Return the HTML generated above to the wiki page this script is invoked from.
	return tostring(componentContainer) .. '[[Category:Components:All]]'
end
--- Creates a new Component field row in the output
-- @param Container The Scribunto HTML node we'll be creating this row inside of. Should be a container of some sort.
-- @param Field Table containing a Name, FieldType, and Type for the field on this row.
function CreateField(Container, Field)
	local fieldContentContainer = Container
		:tag('div') -- HTML div to contain the field.
			:cssText('display: flex;')
			:tag('div') -- HTML div to contain the field
				:cssText( 'flex-grow: 2; overflow: hidden; display: flex; flex-direction: row;')
	
	if fieldHandlers[Field.FieldType] == nil then
		fieldContentContainer:wikitext("ERROR: No handler for field type " .. Field.FieldType)
		return
	end
	fieldHandlers[Field.FieldType](fieldContentContainer, Field)
end
local enumTypes = {
	["TextHorizontalAlignment"] = true,
	["TextVerticalAlignment"] = true,
	["Elements.Assets.AlignmentMode"] = true,
	["Alignment"] = true,
	["DefaultSpace"] = true
}
function fieldHandlers.Sync(Container, Field)
	CreateFieldLabel(Container,Field)
	if Field.Type == 'Bool' then
		CreateUICheckbox(Container, Field.Value)
	elseif Field.Type == 'Int' or Field.Type == 'String' or Field.Type == 'Float' or Field.Type == 'Double' then
		CreateUIInputBox(Container, Field.Value)
	elseif Field.Type == 'Float2' or Field.Type == 'Float3' then
		CreateUIVector(Container, Field.Value)
	elseif enumTypes[Field.Type] then
		CreateUIForwardBackwards(Container, Field.Value)
	elseif Field.Type == "ColorX" then
		local c = Container
			:tag('div')
				:cssText('display:flex; flex-direction: column; align-items:stretch; flex-grow:1; gap: 4px;')
		CreateUIColor(c, Field.Value)
		local c2 = c:tag('div')
					:cssText('display:flex; flex-direction: row; align-items: stretch; flex-grow: 1; gap: 4px;')
					:tag('div')
						:cssText('display:flex; justify-content: center; align-items:center;')
						:wikitext('Profile:')
						:done()
		CreateUIForwardBackwards(c2, Field.Value[5])
	else
		Container
			:wikitext( Field.FieldType .. '<' .. Field.Type .. '>')
	end
end
function CreateUIVector(Container, Vector)
	local c = Container
		:tag('div')
			:cssText('display:flex; height: 1.5rem; flex-direction: row; align-items:stretch; flex-grow:1; gap: 4px;')
	
	local letters = {"x", "y", "z", "w"}
	for i=1,#Vector do
		CreateUIBasicText(c,letters[i] or "?")
		CreateUIInputBox(c, Vector[1])
	end
end
function fieldHandlers.SyncRef(Container,Field)
	CreateFieldLabel(Container,Field)
	CreateUISyncRefEditor(Container, Field.Value)
end
function fieldHandlers.AssetRef(Container,Field)
	CreateFieldLabel(Container,Field)
	
	CreateUISyncRefEditor(Container, Field.Value)
end
function fieldHandlers.SyncAssetList(Container,Field)
	local c = Container
		:tag('div')
			:cssText('display: flex; flex-direction: column; flex-grow: 1; gap: 4px;')
			:tag('div') -- HTML div to contain the field label
				:attr('title', Field and (Field.Name .. ' ' .. Field.FieldType .. '<' .. Field.Type .. '>') or '') -- Add a basic mouseover description
				:cssText('text-align: center;')
				:wikitext( (Field and Field.Name or '') .. ' (list):' )
				:done() -- Close field label div
	
	for i=1,#Field.Value do
		local ref = CreateUISyncRefEditor(c, Field.Value[i])
		CreateUISmallButton(ref, 'X')
	end
	CreateUIButton(c, 'Add')
end
function fieldHandlers.SyncObject(Container, Field)
	Container = Container
		:tag('div')
			:cssText('display:flex; flex-direction:column;gap:6px;width:100%;')
	
	CreateFieldLabel(Container, Field)
	local c = Container
		:tag('div') -- Indented contianer
			:cssText('display:flex;flex:1;')
	c:tag('div') -- Black line
		:cssText('background-color:black;width:4px;margin: 0 6px;')
		:done() -- Close Black line
	fieldContainer = c:tag('div'):cssText("display:flex;flex-direction:column;gap:6px;flex:1;")
	
	for i=1,#Field.Fields do
		CreateField(fieldContainer, Field.Fields[i])
	end
end
function fieldHandlers.DelegateButton(Container, Field)
	CreateUIRefInputBox(Container, Field.Name)
end
function CreateFieldLabel(Container, Field)
	local c = Container
		:tag('div')
			:attr('title', Field and (Field.Name .. ' ' .. Field.FieldType .. '<' .. Field.Type .. '>') or '') -- Add a basic mouseover description
			:cssText("display:flex; width:30%; align-items:center;")
	
	
	local color, is_impulse = GetColor(Field)
	local segments = GetTypeSegments(Field)
	CreateConnectorAttachmentPoint(extrasButton, "hover text", color, segments, false, true)
	
	local extrasButton = c:tag('div'):cssText('display:flex;')
	extrasButton:tag('div') -- HTML div to contain the extras button
		:cssText('align-self:stretch; width:32px; margin-right:2px; background-color: #2B2F35; /* Resonite Mid */ text-align:center; border-radius: 8px;display:flex;align-items:center;justify-content:center;')
		:wikitext("☰")
		:done() -- Close extras button div
			
	c:tag('div') -- HTML div to contain the field label
		:cssText('text-align: left; overflow: hidden; text-overflow: ellipsis; flex: 1;')
		:wikitext( (Field and Field.Name or '') .. ':' )
		:done() -- Close field label div
end
function CreateUIColor(Container, Color)
	local ColorNoAlpha = {Color[1], Color[2], Color[3], 1}
	local c = Container
		:tag('div')
			:cssText('display:flex; height: 1.5rem; flex-direction: row; align-items:stretch; flex-grow:1; gap: 4px;')
	CreateUIBasicText(c, 'R')
	CreateUIInputBox(c, Color[1])
	CreateUIBasicText(c, 'G')
	CreateUIInputBox(c, Color[2])
	CreateUIBasicText(c, 'B')
	CreateUIInputBox(c, Color[3])
	CreateUIBasicText(c, 'A')
	CreateUIInputBox(c, Color[4])
	c:tag('div')
		:cssText('display:flex; height: 1.5rem; flex-direction: row; align-items:stretch; flex-grow:1;')
			:tag('div')
				:cssText('flex-grow: 1; background-color: ' .. ColorToCss(ColorNoAlpha) .. ';')
				:done()
			:tag('div')
				:cssText('flex-grow: 1; background-color: ' .. ColorToCss(Color) .. ';')
				:done()
		:done()
end
function CreateUIBasicText(Container, Text)
	Container
		:wikitext(Text)
		:done()
	return Container;
end
function CreateUIForwardBackwards(Container, Text)
	local c = Container
		:tag('div')
			:cssText('display:flex; height: 1.5rem; flex-direction: row; align-items:stretch; flex-grow:1; gap: 4px;')
	CreateUISmallButton(c, '<<')
		:tag('div')
			:cssText('border-radius: 8px; background: rgb(8, 8, 10); flex-grow:1; display:flex; justify-content: center; align-items:center;')
			:wikitext(Text)
			:done()
	CreateUISmallButton(c, '>>')
	return Container
end
function CreateUICheckbox(Container, Checked)
	local c = Container
		:tag('div')
			:cssText('box-sizing: border-box; background-color: #2b2f35; /* Resonite Mid */ border-radius: 8px; width: 1.5rem; height: 1.5rem; padding: 2px;')
	if Checked then
		c:wikitext('[[File:Checkmark.png|checkmark|20px]]')
	end
end
function CreateUISmallButton(Container, Text)
	Container
		:tag('div')
			:cssText('display:flex; flex-direction: column; align-items:center; justify-content: center;')
			:tag('div')
				:cssText('width: 1.5rem; height: 1.5rem; background-color: rgb(43, 46, 54); border-radius: 8px; flex-grow:0; display:flex; justify-content: center; align-items:center;')
				:wikitext(Text)
				:done()
	return Container
end
function CreateUIButton(Container, Text)
	Container
		:tag('div')
			:cssText('display:flex; flex-direction: column; align-items:stretch; justify-content: center;')
			:tag('div')
				:cssText('height: 1.5rem; background-color: rgb(43, 46, 54); border-radius: 8px; flex-grow:0; display:flex; justify-content: center; align-items:center;')
				:wikitext(Text)
				:done()
	return Container
end
function CreateUIInputBox(Container, Text)
	local Italics = false
	if not Text then
		Text = 'null'
		Italics = true
	end
	
	Container
		:tag('div')
			:cssText('box-sizing: border-box; height: 1.5rem; border: 3px solid rgb(43, 46, 54); border-radius: 8px; background: rgb(30, 33, 38); flex-grow:1; display:flex; justify-content: center; align-items:center;'  ..  (Italics and 'font-style:italic;' or ''))
			:wikitext(Text)
			:done()
end
function CreateUIRefInputBox(Container, Text)
	local Italics = false
	if not Text then
		Text = 'null'
		Italics = true
	end
	
	Container
		:tag('div')
			:cssText('height: 1.5rem; background: rgb(43, 46, 54); border-radius: 8px; flex-grow:1; display:flex; justify-content: center; align-items:center;' ..  (Italics and 'font-style:italic;' or ''))
			:wikitext(Text)
			:done()
	return Container
end
function CreateUISyncRefEditor(Container, Text)
	local c = Container
	:tag('div')
		:cssText('display:flex; flex-direction: row; align-items:stretch; flex-grow:1; gap: 4px;')
	
	CreateUISmallButton(c, '⤴')
	CreateUISmallButton(c, '↑')
	CreateUIRefInputBox(c, Text)
	CreateUISmallButton(c, '∅')
	return c
end
function ColorToCss(color)
	return ("rgba(%s,%s,%s,%s)"):format(color[1]*255, color[2]*255, color[3]*255, color[4] or 1)
end
--- Returns a CSS color for the data type or impulse and whether the type is an impulse
-- @param connector Table containing a Name (Optional) and Type (Required) for the input/output.
function GetColor(connector)
	if connector == nil then error("connector is nil") end
	if connector == nil or connector.Type == nil then error("Missing Type") end
	local color = ProtofluxColor.get_impulse_color(connector.Type)
	if color ~= nil then
		return color, true
	end
	return GetTypeColor(connector), false
end
--- Returns a CSS color for the data type
-- @param connector Table containing a Name (Optional) and Type (Required) for the input/output or global.
function GetTypeColor(connector)
	if connector == nil then error("connector is nil") end
	if connector == nil or connector.Type == nil then error("Missing Type") end
	local color = ProtofluxColor.get_type_color(connector.Type)
	return MultRGB(color, 1.5)
end
function MultRGB(color, fact)
	return {
		r=color.r*fact,
		g=color.g*fact,
		b=color.b*fact,
		a=color.a,
	}
end
--- Creates a new input or output attachement point for a connector row.
--	@param container The Scribunto HTML node we'll be creating this row inside of. Should be a container of some sort.
--  @param hover_text Text to show when hovering, possibly nil
--	@param color The color of this attachment point (table with r, g, b, optionally a).
--	@param segments The number of segments for the type, used for vectors or matrices.
--	@param is_impulse True if this is an impulse, false if it is a data type.
--	@param is_input True if this is an input side attachment point, false if it is an output side attachment point.
function CreateConnectorAttachmentPoint(container, hover_text, color, segments, is_impulse, is_input)
	local connector_type = nil
	if is_impulse then connector_type = "InputArrow"
	elseif is_input then connector_type = "InputBox"
	else connector_type = "OutputBox"
	end
	container
		:tag( 'div' )
			:attr('title', hover_text or "") -- Add a basic mouseover description
			:cssText(css:attachment_point(color, is_input and 1 or 3))
			:wikitext(mw.getCurrentFrame():expandTemplate({title = 'ProtoFluxConnector', args = {Type=connector_type, Segments=segments}}))
end
local type_segments = {
	enabled = {
		bool=true,
		byte=true,
		sbyte=true,
		ushort=true,
		short=true,
		uint=true,
		int=true,
		ulong=true,
		long=true,
		float=true,
		double=true,
		decimal=true,
	},
	suffix = {
		["2"]=2,
		["3"]=3,
		["4"]=4,
		--["2x2"]=2,
		--["3x3"]=3,
		--["4x4"]=4,
	},
}
--- Returns a CSS color for the data type
-- @param connector Table containing a Name (Optional) and Type (Required) for the input/output.
function GetTypeSegments(connector)
	if connector == nil then error("connector is nil") end
	if connector == nil or connector.Type == nil then error("Missing Type") end
	local base = connector.Type:match("^%a+")
	if not type_segments.enabled[base] then return 1 end
	local suffix = connector.Type:sub(#base+1)
	return type_segments.suffix[suffix] or 1
end
return p
