cleanup
This commit is contained in:
590
testing/__tests__/33_input_field_tests.lua
Normal file
590
testing/__tests__/33_input_field_tests.lua
Normal file
@@ -0,0 +1,590 @@
|
||||
-- ====================
|
||||
-- Input Field Tests
|
||||
-- ====================
|
||||
-- Test suite for text input functionality in FlexLove
|
||||
|
||||
local lu = require("testing.luaunit")
|
||||
local loveStub = require("testing.loveStub")
|
||||
|
||||
-- Setup LÖVE environment
|
||||
_G.love = loveStub
|
||||
|
||||
-- Load FlexLove after setting up love stub
|
||||
local FlexLove = require("FlexLove")
|
||||
|
||||
-- Test fixtures
|
||||
local testElement
|
||||
|
||||
TestInputField = {}
|
||||
|
||||
function TestInputField:setUp()
|
||||
-- Reset FlexLove state
|
||||
FlexLove.Gui.topElements = {}
|
||||
FlexLove.Gui._focusedElement = nil
|
||||
|
||||
-- Create a test input element
|
||||
testElement = FlexLove.Element.new({
|
||||
x = 100,
|
||||
y = 100,
|
||||
width = 200,
|
||||
height = 40,
|
||||
editable = true,
|
||||
text = "Hello World",
|
||||
})
|
||||
end
|
||||
|
||||
function TestInputField:tearDown()
|
||||
testElement = nil
|
||||
FlexLove.Gui.topElements = {}
|
||||
FlexLove.Gui._focusedElement = nil
|
||||
end
|
||||
|
||||
-- ====================
|
||||
-- Focus Management Tests
|
||||
-- ====================
|
||||
|
||||
function TestInputField:testFocusElement()
|
||||
-- Initially not focused
|
||||
lu.assertFalse(testElement:isFocused())
|
||||
|
||||
-- Focus element
|
||||
testElement:focus()
|
||||
|
||||
-- Should be focused
|
||||
lu.assertTrue(testElement:isFocused())
|
||||
lu.assertEquals(FlexLove.Gui._focusedElement, testElement)
|
||||
end
|
||||
|
||||
function TestInputField:testBlurElement()
|
||||
-- Focus element first
|
||||
testElement:focus()
|
||||
lu.assertTrue(testElement:isFocused())
|
||||
|
||||
-- Blur element
|
||||
testElement:blur()
|
||||
|
||||
-- Should not be focused
|
||||
lu.assertFalse(testElement:isFocused())
|
||||
lu.assertNil(FlexLove.Gui._focusedElement)
|
||||
end
|
||||
|
||||
function TestInputField:testFocusSwitchBetweenElements()
|
||||
local element1 = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 30,
|
||||
editable = true,
|
||||
text = "Element 1",
|
||||
})
|
||||
|
||||
local element2 = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 50,
|
||||
width = 100,
|
||||
height = 30,
|
||||
editable = true,
|
||||
text = "Element 2",
|
||||
})
|
||||
|
||||
-- Focus element1
|
||||
element1:focus()
|
||||
lu.assertTrue(element1:isFocused())
|
||||
lu.assertFalse(element2:isFocused())
|
||||
|
||||
-- Focus element2 (should blur element1)
|
||||
element2:focus()
|
||||
lu.assertFalse(element1:isFocused())
|
||||
lu.assertTrue(element2:isFocused())
|
||||
lu.assertEquals(FlexLove.Gui._focusedElement, element2)
|
||||
end
|
||||
|
||||
function TestInputField:testSelectOnFocus()
|
||||
local element = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 30,
|
||||
editable = true,
|
||||
text = "Test Text",
|
||||
selectOnFocus = true,
|
||||
})
|
||||
|
||||
-- Focus element with selectOnFocus enabled
|
||||
element:focus()
|
||||
|
||||
-- Should select all text
|
||||
lu.assertTrue(element:hasSelection())
|
||||
local startPos, endPos = element:getSelection()
|
||||
lu.assertEquals(startPos, 0)
|
||||
lu.assertEquals(endPos, 9) -- "Test Text" has 9 characters
|
||||
end
|
||||
|
||||
-- ====================
|
||||
-- Cursor Management Tests
|
||||
-- ====================
|
||||
|
||||
function TestInputField:testSetCursorPosition()
|
||||
testElement:focus()
|
||||
|
||||
-- Set cursor to position 5
|
||||
testElement:setCursorPosition(5)
|
||||
|
||||
lu.assertEquals(testElement:getCursorPosition(), 5)
|
||||
end
|
||||
|
||||
function TestInputField:testCursorPositionBounds()
|
||||
testElement:focus()
|
||||
|
||||
-- Try to set cursor beyond text length
|
||||
testElement:setCursorPosition(999)
|
||||
|
||||
-- Should clamp to text length
|
||||
lu.assertEquals(testElement:getCursorPosition(), 11) -- "Hello World" has 11 characters
|
||||
|
||||
-- Try to set negative cursor position
|
||||
testElement:setCursorPosition(-5)
|
||||
|
||||
-- Should clamp to 0
|
||||
lu.assertEquals(testElement:getCursorPosition(), 0)
|
||||
end
|
||||
|
||||
function TestInputField:testMoveCursor()
|
||||
testElement:focus()
|
||||
testElement:setCursorPosition(5)
|
||||
|
||||
-- Move cursor right
|
||||
testElement:moveCursorBy(2)
|
||||
lu.assertEquals(testElement:getCursorPosition(), 7)
|
||||
|
||||
-- Move cursor left
|
||||
testElement:moveCursorBy(-3)
|
||||
lu.assertEquals(testElement:getCursorPosition(), 4)
|
||||
end
|
||||
|
||||
function TestInputField:testMoveCursorToStartEnd()
|
||||
testElement:focus()
|
||||
testElement:setCursorPosition(5)
|
||||
|
||||
-- Move to end
|
||||
testElement:moveCursorToEnd()
|
||||
lu.assertEquals(testElement:getCursorPosition(), 11)
|
||||
|
||||
-- Move to start
|
||||
testElement:moveCursorToStart()
|
||||
lu.assertEquals(testElement:getCursorPosition(), 0)
|
||||
end
|
||||
|
||||
-- ====================
|
||||
-- Text Buffer Management Tests
|
||||
-- ====================
|
||||
|
||||
function TestInputField:testGetText()
|
||||
lu.assertEquals(testElement:getText(), "Hello World")
|
||||
end
|
||||
|
||||
function TestInputField:testSetText()
|
||||
testElement:setText("New Text")
|
||||
|
||||
lu.assertEquals(testElement:getText(), "New Text")
|
||||
lu.assertEquals(testElement.text, "New Text")
|
||||
end
|
||||
|
||||
function TestInputField:testInsertTextAtCursor()
|
||||
testElement:focus()
|
||||
testElement:setCursorPosition(5) -- After "Hello"
|
||||
|
||||
testElement:insertText(" Beautiful")
|
||||
|
||||
lu.assertEquals(testElement:getText(), "Hello Beautiful World")
|
||||
lu.assertEquals(testElement:getCursorPosition(), 15) -- Cursor after inserted text
|
||||
end
|
||||
|
||||
function TestInputField:testInsertTextAtSpecificPosition()
|
||||
testElement:insertText("Super ", 6) -- Before "World"
|
||||
|
||||
lu.assertEquals(testElement:getText(), "Hello Super World")
|
||||
end
|
||||
|
||||
function TestInputField:testDeleteText()
|
||||
testElement:deleteText(0, 6) -- Delete "Hello "
|
||||
|
||||
lu.assertEquals(testElement:getText(), "World")
|
||||
end
|
||||
|
||||
function TestInputField:testReplaceText()
|
||||
testElement:replaceText(0, 5, "Hi") -- Replace "Hello" with "Hi"
|
||||
|
||||
lu.assertEquals(testElement:getText(), "Hi World")
|
||||
end
|
||||
|
||||
function TestInputField:testMaxLengthConstraint()
|
||||
local element = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 30,
|
||||
editable = true,
|
||||
text = "Test",
|
||||
maxLength = 10,
|
||||
})
|
||||
|
||||
element:focus()
|
||||
element:moveCursorToEnd()
|
||||
|
||||
-- Try to insert text that would exceed maxLength
|
||||
element:insertText(" Very Long Text")
|
||||
|
||||
-- Should not insert text that exceeds maxLength
|
||||
lu.assertEquals(element:getText(), "Test")
|
||||
|
||||
-- Insert text that fits within maxLength
|
||||
element:insertText(" Text")
|
||||
lu.assertEquals(element:getText(), "Test Text")
|
||||
end
|
||||
|
||||
-- ====================
|
||||
-- Selection Management Tests
|
||||
-- ====================
|
||||
|
||||
function TestInputField:testSetSelection()
|
||||
testElement:setSelection(0, 5) -- Select "Hello"
|
||||
|
||||
lu.assertTrue(testElement:hasSelection())
|
||||
local startPos, endPos = testElement:getSelection()
|
||||
lu.assertEquals(startPos, 0)
|
||||
lu.assertEquals(endPos, 5)
|
||||
end
|
||||
|
||||
function TestInputField:testGetSelectedText()
|
||||
testElement:setSelection(0, 5) -- Select "Hello"
|
||||
|
||||
local selectedText = testElement:getSelectedText()
|
||||
lu.assertEquals(selectedText, "Hello")
|
||||
end
|
||||
|
||||
function TestInputField:testSelectAll()
|
||||
testElement:selectAll()
|
||||
|
||||
lu.assertTrue(testElement:hasSelection())
|
||||
local startPos, endPos = testElement:getSelection()
|
||||
lu.assertEquals(startPos, 0)
|
||||
lu.assertEquals(endPos, 11) -- Full text length
|
||||
end
|
||||
|
||||
function TestInputField:testClearSelection()
|
||||
testElement:setSelection(0, 5)
|
||||
lu.assertTrue(testElement:hasSelection())
|
||||
|
||||
testElement:clearSelection()
|
||||
|
||||
lu.assertFalse(testElement:hasSelection())
|
||||
local startPos, endPos = testElement:getSelection()
|
||||
lu.assertNil(startPos)
|
||||
lu.assertNil(endPos)
|
||||
end
|
||||
|
||||
function TestInputField:testDeleteSelection()
|
||||
testElement:focus()
|
||||
testElement:setSelection(0, 6) -- Select "Hello "
|
||||
|
||||
local deleted = testElement:deleteSelection()
|
||||
|
||||
lu.assertTrue(deleted)
|
||||
lu.assertEquals(testElement:getText(), "World")
|
||||
lu.assertFalse(testElement:hasSelection())
|
||||
lu.assertEquals(testElement:getCursorPosition(), 0)
|
||||
end
|
||||
|
||||
-- ====================
|
||||
-- Text Input Tests
|
||||
-- ====================
|
||||
|
||||
function TestInputField:testTextInput()
|
||||
testElement:focus()
|
||||
testElement:setCursorPosition(5) -- After "Hello"
|
||||
|
||||
-- Simulate text input
|
||||
testElement:textinput(",")
|
||||
|
||||
lu.assertEquals(testElement:getText(), "Hello, World")
|
||||
lu.assertEquals(testElement:getCursorPosition(), 6)
|
||||
end
|
||||
|
||||
function TestInputField:testTextInputWithSelection()
|
||||
testElement:focus()
|
||||
testElement:setSelection(0, 5) -- Select "Hello"
|
||||
|
||||
-- Simulate text input (should replace selection)
|
||||
testElement:textinput("Hi")
|
||||
|
||||
lu.assertEquals(testElement:getText(), "Hi World")
|
||||
lu.assertFalse(testElement:hasSelection())
|
||||
lu.assertEquals(testElement:getCursorPosition(), 2)
|
||||
end
|
||||
|
||||
function TestInputField:testTextInputCallbacks()
|
||||
local inputCalled = false
|
||||
local changeCalled = false
|
||||
local inputText = nil
|
||||
local newText = nil
|
||||
local oldText = nil
|
||||
|
||||
local element = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 30,
|
||||
editable = true,
|
||||
text = "Test",
|
||||
onTextInput = function(_, text)
|
||||
inputCalled = true
|
||||
inputText = text
|
||||
end,
|
||||
onTextChange = function(_, new, old)
|
||||
changeCalled = true
|
||||
newText = new
|
||||
oldText = old
|
||||
end,
|
||||
})
|
||||
|
||||
element:focus()
|
||||
element:moveCursorToEnd()
|
||||
element:textinput("!")
|
||||
|
||||
lu.assertTrue(inputCalled)
|
||||
lu.assertTrue(changeCalled)
|
||||
lu.assertEquals(inputText, "!")
|
||||
lu.assertEquals(newText, "Test!")
|
||||
lu.assertEquals(oldText, "Test")
|
||||
end
|
||||
|
||||
-- ====================
|
||||
-- Keyboard Input Tests
|
||||
-- ====================
|
||||
|
||||
function TestInputField:testBackspaceKey()
|
||||
testElement:focus()
|
||||
testElement:setCursorPosition(5) -- After "Hello"
|
||||
|
||||
-- Simulate backspace key
|
||||
testElement:keypressed("backspace", nil, false)
|
||||
|
||||
lu.assertEquals(testElement:getText(), "Hell World")
|
||||
lu.assertEquals(testElement:getCursorPosition(), 4)
|
||||
end
|
||||
|
||||
function TestInputField:testDeleteKey()
|
||||
testElement:focus()
|
||||
testElement:setCursorPosition(5) -- After "Hello", before " "
|
||||
|
||||
-- Simulate delete key
|
||||
testElement:keypressed("delete", nil, false)
|
||||
|
||||
lu.assertEquals(testElement:getText(), "HelloWorld")
|
||||
lu.assertEquals(testElement:getCursorPosition(), 5)
|
||||
end
|
||||
|
||||
function TestInputField:testArrowKeys()
|
||||
testElement:focus()
|
||||
testElement:setCursorPosition(5)
|
||||
|
||||
-- Right arrow
|
||||
testElement:keypressed("right", nil, false)
|
||||
lu.assertEquals(testElement:getCursorPosition(), 6)
|
||||
|
||||
-- Left arrow
|
||||
testElement:keypressed("left", nil, false)
|
||||
lu.assertEquals(testElement:getCursorPosition(), 5)
|
||||
end
|
||||
|
||||
function TestInputField:testHomeEndKeys()
|
||||
testElement:focus()
|
||||
testElement:setCursorPosition(5)
|
||||
|
||||
-- End key
|
||||
testElement:keypressed("end", nil, false)
|
||||
lu.assertEquals(testElement:getCursorPosition(), 11)
|
||||
|
||||
-- Home key
|
||||
testElement:keypressed("home", nil, false)
|
||||
lu.assertEquals(testElement:getCursorPosition(), 0)
|
||||
end
|
||||
|
||||
function TestInputField:testEscapeKey()
|
||||
testElement:focus()
|
||||
testElement:setSelection(0, 5)
|
||||
lu.assertTrue(testElement:hasSelection())
|
||||
|
||||
-- Escape should clear selection
|
||||
testElement:keypressed("escape", nil, false)
|
||||
|
||||
lu.assertFalse(testElement:hasSelection())
|
||||
lu.assertTrue(testElement:isFocused()) -- Still focused
|
||||
|
||||
-- Another escape should blur
|
||||
testElement:keypressed("escape", nil, false)
|
||||
|
||||
lu.assertFalse(testElement:isFocused())
|
||||
end
|
||||
|
||||
function TestInputField:testCtrlA()
|
||||
testElement:focus()
|
||||
|
||||
-- Simulate Ctrl+A (need to mock modifiers)
|
||||
local oldIsDown = _G.love.keyboard.isDown
|
||||
_G.love.keyboard.isDown = function(...)
|
||||
local keys = {...}
|
||||
for _, key in ipairs(keys) do
|
||||
if key == "lctrl" or key == "rctrl" then
|
||||
return true
|
||||
end
|
||||
end
|
||||
return false
|
||||
end
|
||||
|
||||
testElement:keypressed("a", "", false)
|
||||
|
||||
lu.assertTrue(testElement:hasSelection())
|
||||
local startPos, endPos = testElement:getSelection()
|
||||
lu.assertEquals(startPos, 0)
|
||||
lu.assertEquals(endPos, 11)
|
||||
|
||||
-- Reset mock
|
||||
_G.love.keyboard.isDown = oldIsDown
|
||||
end
|
||||
|
||||
function TestInputField:testEnterKeyMultiline()
|
||||
local element = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 60,
|
||||
editable = true,
|
||||
multiline = true,
|
||||
text = "Line 1",
|
||||
})
|
||||
|
||||
element:focus()
|
||||
element:moveCursorToEnd()
|
||||
|
||||
-- Simulate Enter key
|
||||
element:keypressed("return", nil, false)
|
||||
|
||||
-- Should insert newline
|
||||
lu.assertEquals(element:getText(), "Line 1\n")
|
||||
end
|
||||
|
||||
function TestInputField:testEnterKeySingleline()
|
||||
local enterCalled = false
|
||||
|
||||
local element = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 30,
|
||||
editable = true,
|
||||
multiline = false,
|
||||
text = "Test",
|
||||
onEnter = function()
|
||||
enterCalled = true
|
||||
end,
|
||||
})
|
||||
|
||||
element:focus()
|
||||
element:moveCursorToEnd()
|
||||
|
||||
-- Simulate Enter key
|
||||
element:keypressed("return", nil, false)
|
||||
|
||||
-- Should trigger onEnter callback, not insert newline
|
||||
lu.assertTrue(enterCalled)
|
||||
lu.assertEquals(element:getText(), "Test")
|
||||
end
|
||||
|
||||
-- ====================
|
||||
-- Multi-line Tests
|
||||
-- ====================
|
||||
|
||||
function TestInputField:testMultilineTextSplitting()
|
||||
local element = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 80,
|
||||
editable = true,
|
||||
multiline = true,
|
||||
text = "Line 1\nLine 2\nLine 3",
|
||||
})
|
||||
|
||||
-- Trigger line splitting
|
||||
element:_splitLines()
|
||||
|
||||
lu.assertEquals(#element._lines, 3)
|
||||
lu.assertEquals(element._lines[1], "Line 1")
|
||||
lu.assertEquals(element._lines[2], "Line 2")
|
||||
lu.assertEquals(element._lines[3], "Line 3")
|
||||
end
|
||||
|
||||
-- ====================
|
||||
-- UTF-8 Support Tests
|
||||
-- ====================
|
||||
|
||||
function TestInputField:testUTF8Characters()
|
||||
local element = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 30,
|
||||
editable = true,
|
||||
text = "Hello 世界",
|
||||
})
|
||||
|
||||
element:focus()
|
||||
element:moveCursorToEnd()
|
||||
|
||||
-- Insert UTF-8 character
|
||||
element:insertText("!")
|
||||
|
||||
lu.assertEquals(element:getText(), "Hello 世界!")
|
||||
end
|
||||
|
||||
function TestInputField:testUTF8Selection()
|
||||
local element = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 30,
|
||||
editable = true,
|
||||
text = "Hello 世界",
|
||||
})
|
||||
|
||||
-- Select UTF-8 characters
|
||||
element:setSelection(6, 8) -- Select "世界"
|
||||
|
||||
local selected = element:getSelectedText()
|
||||
lu.assertEquals(selected, "世界")
|
||||
end
|
||||
|
||||
-- ====================
|
||||
-- Password Mode Tests
|
||||
-- ====================
|
||||
|
||||
function TestInputField:testPasswordModeDisablesMultiline()
|
||||
local element = FlexLove.Element.new({
|
||||
x = 10,
|
||||
y = 10,
|
||||
width = 100,
|
||||
height = 30,
|
||||
editable = true,
|
||||
multiline = true,
|
||||
passwordMode = true,
|
||||
text = "password",
|
||||
})
|
||||
|
||||
-- Password mode should override multiline
|
||||
lu.assertFalse(element.multiline)
|
||||
end
|
||||
|
||||
-- Run tests
|
||||
lu.LuaUnit.run()
|
||||
Reference in New Issue
Block a user