diff --git a/deploy/images/SearchDropdownExampleOpen.png b/deploy/images/SearchDropdownExampleOpen.png
new file mode 100644
index 00000000..e671afc6
Binary files /dev/null and b/deploy/images/SearchDropdownExampleOpen.png differ
diff --git a/deploy/images/SearchDropdownExampleSearching.png b/deploy/images/SearchDropdownExampleSearching.png
new file mode 100644
index 00000000..d6f70136
Binary files /dev/null and b/deploy/images/SearchDropdownExampleSearching.png differ
diff --git a/deploy/widget_icons/SearchDropdown.png b/deploy/widget_icons/SearchDropdown.png
new file mode 100644
index 00000000..9ee04e0a
Binary files /dev/null and b/deploy/widget_icons/SearchDropdown.png differ
diff --git a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/JMJb_dCLyH7X_iHp2DLoMYNEdokd.xml b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/JMJb_dCLyH7X_iHp2DLoMYNEdokd.xml
index 99772b42..378b6137 100644
--- a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/JMJb_dCLyH7X_iHp2DLoMYNEdokd.xml
+++ b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/JMJb_dCLyH7X_iHp2DLoMYNEdokd.xml
@@ -1,6 +1,6 @@
-
+
\ No newline at end of file
diff --git a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/J_q8ERiDV2uofqHvpCu2mo8uB9Ad.xml b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/J_q8ERiDV2uofqHvpCu2mo8uB9Ad.xml
index 99772b42..378b6137 100644
--- a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/J_q8ERiDV2uofqHvpCu2mo8uB9Ad.xml
+++ b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/J_q8ERiDV2uofqHvpCu2mo8uB9Ad.xml
@@ -1,6 +1,6 @@
-
+
\ No newline at end of file
diff --git a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/S9wGd_pDd1D_9FNVqjfXodz45sod.xml b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/S9wGd_pDd1D_9FNVqjfXodz45sod.xml
index 99772b42..378b6137 100644
--- a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/S9wGd_pDd1D_9FNVqjfXodz45sod.xml
+++ b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/S9wGd_pDd1D_9FNVqjfXodz45sod.xml
@@ -1,6 +1,6 @@
-
+
\ No newline at end of file
diff --git a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/V0mMK9LtsBu9P5LKljXbNE-C5qwd.xml b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/V0mMK9LtsBu9P5LKljXbNE-C5qwd.xml
index 99772b42..378b6137 100644
--- a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/V0mMK9LtsBu9P5LKljXbNE-C5qwd.xml
+++ b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/V0mMK9LtsBu9P5LKljXbNE-C5qwd.xml
@@ -1,6 +1,6 @@
-
+
\ No newline at end of file
diff --git a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/YjX3mbPy2HN8LR96rq4Fzb372VUd.xml b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/YjX3mbPy2HN8LR96rq4Fzb372VUd.xml
index 99772b42..378b6137 100644
--- a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/YjX3mbPy2HN8LR96rq4Fzb372VUd.xml
+++ b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/YjX3mbPy2HN8LR96rq4Fzb372VUd.xml
@@ -1,6 +1,6 @@
-
+
\ No newline at end of file
diff --git a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/fgSmFlkgej181VWOoxvdiZJ1K9gd.xml b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/fgSmFlkgej181VWOoxvdiZJ1K9gd.xml
index 7a6326b9..378b6137 100644
--- a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/fgSmFlkgej181VWOoxvdiZJ1K9gd.xml
+++ b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/fgSmFlkgej181VWOoxvdiZJ1K9gd.xml
@@ -1,6 +1,6 @@
-
+
-
+
\ No newline at end of file
diff --git a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/qR28PGzPsksFVChmMp9R0Zbo0uYd.xml b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/qR28PGzPsksFVChmMp9R0Zbo0uYd.xml
index 99772b42..378b6137 100644
--- a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/qR28PGzPsksFVChmMp9R0Zbo0uYd.xml
+++ b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/qR28PGzPsksFVChmMp9R0Zbo0uYd.xml
@@ -1,6 +1,6 @@
-
+
\ No newline at end of file
diff --git a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/xTX_9QKVz_sRcYaTywWZA4aZm4kd.xml b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/xTX_9QKVz_sRcYaTywWZA4aZm4kd.xml
new file mode 100644
index 00000000..378b6137
--- /dev/null
+++ b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/xTX_9QKVz_sRcYaTywWZA4aZm4kd.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/xTX_9QKVz_sRcYaTywWZA4aZm4kp.xml b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/xTX_9QKVz_sRcYaTywWZA4aZm4kp.xml
new file mode 100644
index 00000000..fb20dac4
--- /dev/null
+++ b/resources/project/Xc4-tOl6vkwzCeVKSWRahPrXJiQ/xTX_9QKVz_sRcYaTywWZA4aZm4kp.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/resources/project/elO4D6tOV7Lp-Jfo7ptgr2NlB30/mBLWFebZLKEYjmQwRCNBG-dcvOEd.xml b/resources/project/elO4D6tOV7Lp-Jfo7ptgr2NlB30/mBLWFebZLKEYjmQwRCNBG-dcvOEd.xml
new file mode 100644
index 00000000..99772b42
--- /dev/null
+++ b/resources/project/elO4D6tOV7Lp-Jfo7ptgr2NlB30/mBLWFebZLKEYjmQwRCNBG-dcvOEd.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/resources/project/elO4D6tOV7Lp-Jfo7ptgr2NlB30/mBLWFebZLKEYjmQwRCNBG-dcvOEp.xml b/resources/project/elO4D6tOV7Lp-Jfo7ptgr2NlB30/mBLWFebZLKEYjmQwRCNBG-dcvOEp.xml
new file mode 100644
index 00000000..fb20dac4
--- /dev/null
+++ b/resources/project/elO4D6tOV7Lp-Jfo7ptgr2NlB30/mBLWFebZLKEYjmQwRCNBG-dcvOEp.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/resources/project/f1CasPtPWEKkULO6lQ_WTjau2Nw/thSBeYSSaW6FYs_WYLbsJvWC9Acd.xml b/resources/project/f1CasPtPWEKkULO6lQ_WTjau2Nw/thSBeYSSaW6FYs_WYLbsJvWC9Acd.xml
new file mode 100644
index 00000000..99772b42
--- /dev/null
+++ b/resources/project/f1CasPtPWEKkULO6lQ_WTjau2Nw/thSBeYSSaW6FYs_WYLbsJvWC9Acd.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/resources/project/f1CasPtPWEKkULO6lQ_WTjau2Nw/thSBeYSSaW6FYs_WYLbsJvWC9Acp.xml b/resources/project/f1CasPtPWEKkULO6lQ_WTjau2Nw/thSBeYSSaW6FYs_WYLbsJvWC9Acp.xml
new file mode 100644
index 00000000..8222afbe
--- /dev/null
+++ b/resources/project/f1CasPtPWEKkULO6lQ_WTjau2Nw/thSBeYSSaW6FYs_WYLbsJvWC9Acp.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/resources/project/qPaYXYIWInHoCm9ctG8-uEqdxlk/Kc8NyOxBHteHzaAs2Svjccxs8jsd.xml b/resources/project/qPaYXYIWInHoCm9ctG8-uEqdxlk/Kc8NyOxBHteHzaAs2Svjccxs8jsd.xml
new file mode 100644
index 00000000..a75f7a81
--- /dev/null
+++ b/resources/project/qPaYXYIWInHoCm9ctG8-uEqdxlk/Kc8NyOxBHteHzaAs2Svjccxs8jsd.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/resources/project/qPaYXYIWInHoCm9ctG8-uEqdxlk/Kc8NyOxBHteHzaAs2Svjccxs8jsp.xml b/resources/project/qPaYXYIWInHoCm9ctG8-uEqdxlk/Kc8NyOxBHteHzaAs2Svjccxs8jsp.xml
new file mode 100644
index 00000000..f1a9071d
--- /dev/null
+++ b/resources/project/qPaYXYIWInHoCm9ctG8-uEqdxlk/Kc8NyOxBHteHzaAs2Svjccxs8jsp.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/_08KYHmkQrwI4wltgg7WZLZLxR4d.xml b/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/_08KYHmkQrwI4wltgg7WZLZLxR4d.xml
new file mode 100644
index 00000000..a75f7a81
--- /dev/null
+++ b/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/_08KYHmkQrwI4wltgg7WZLZLxR4d.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/_08KYHmkQrwI4wltgg7WZLZLxR4p.xml b/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/_08KYHmkQrwI4wltgg7WZLZLxR4p.xml
new file mode 100644
index 00000000..08a2d7b2
--- /dev/null
+++ b/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/_08KYHmkQrwI4wltgg7WZLZLxR4p.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/rVbGKXEYIRCxCGbb4rKJ4OqWDVod.xml b/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/rVbGKXEYIRCxCGbb4rKJ4OqWDVod.xml
new file mode 100644
index 00000000..4356a6ae
--- /dev/null
+++ b/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/rVbGKXEYIRCxCGbb4rKJ4OqWDVod.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/rVbGKXEYIRCxCGbb4rKJ4OqWDVop.xml b/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/rVbGKXEYIRCxCGbb4rKJ4OqWDVop.xml
new file mode 100644
index 00000000..c2b58c46
--- /dev/null
+++ b/resources/project/sSpxFEkGAHXZFJ64I8GyrijBl1s/rVbGKXEYIRCxCGbb4rKJ4OqWDVop.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/test/+wt/+test/BaseTest.m b/test/+wt/+test/BaseTest.m
index 292fa722..4567a861 100644
--- a/test/+wt/+test/BaseTest.m
+++ b/test/+wt/+test/BaseTest.m
@@ -191,13 +191,34 @@ function verifyPropertyValue(testCase, component, property, expValue)
expValue
end
- import matlab.unittest.constraints.Eventually
- import matlab.unittest.constraints.IsEqualTo
+ import matlab.unittest.constraints.*
% Verify values
+ if isStringScalar(expValue)
+
+ constraint = IsEqualTo(expValue,...
+ "Using", StringComparator);
+
+ elseif islogical(expValue)
+
+ constraint = IsEqualTo(expValue,...
+ "Using", LogicalComparator);
+
+ elseif isnumeric(expValue)
+
+ constraint = IsEqualTo(expValue,...
+ "Using", NumericComparator);
+
+ else
+
+ constraint = IsEqualTo(expValue);
+
+ end
+
+ % Perform the verification
testCase.verifyThat(...
@()get(component, property),...
- Eventually(IsEqualTo(expValue), "WithTimeoutOf", 5));
+ Eventually(constraint, "WithTimeoutOf", 5));
end %function
diff --git a/test/+wt/+test/BaseWidgetTest.m b/test/+wt/+test/BaseWidgetTest.m
index c56c6008..ecb5707e 100644
--- a/test/+wt/+test/BaseWidgetTest.m
+++ b/test/+wt/+test/BaseWidgetTest.m
@@ -114,6 +114,10 @@ function verifySetProperty(testCase,propName,newValue,expValue)
% Verify new property value
actualValue = testCase.Widget.(propName);
testCase.verifyEquality(actualValue, expValue);
+
+ % Tried to use this one, but the above works better because
+ % strings can equate to enumeration strings, etc.
+ % testCase.verifyPropertyValue(testCase.Widget, propName, expValue);
end %function
@@ -134,7 +138,7 @@ function verifySetPropertyError(testCase,propName,newValue,errorId)
end %function
- function verifyTypeAction(testCase,control,newValue,propName,expValue)
+ function verifyTypeAction(testCase,component,newValue,propName,expValue)
% If no expected new value after set was given, assume it
% matches the newValue provided as usual
@@ -143,12 +147,16 @@ function verifyTypeAction(testCase,control,newValue,propName,expValue)
end
% Type the new value into the control
- testCase.type(control, newValue);
+ testCase.type(component, newValue);
% Verify new property value
actualValue = testCase.Widget.(propName);
testCase.verifyEquality(actualValue, expValue);
-
+
+ % Tried to use this one, but the above works better because
+ % strings can equate to enumeration strings, etc.
+ % testCase.verifyPropertyValue(component, propName, expValue);
+
end %function
diff --git a/test/+wt/+test/SearchDropDown.m b/test/+wt/+test/SearchDropDown.m
new file mode 100644
index 00000000..3120f1ae
--- /dev/null
+++ b/test/+wt/+test/SearchDropDown.m
@@ -0,0 +1,129 @@
+classdef SearchDropDown < wt.test.BaseWidgetTest
+ % Implements a unit test for a widget or component
+
+ % Copyright 2025 The MathWorks Inc.
+
+ %% Properties
+ properties
+ ItemNames = [
+ "Voltage A"
+ "Voltage B"
+ "Current A"
+ "Current B"
+ "Power A"
+ "Power B"
+ ];
+ ItemData = [
+ 11
+ 12
+ 21
+ 22
+ 31
+ 32
+ ];
+ end
+
+
+ %% Class Setup
+ methods (TestClassSetup)
+
+ function createFigure(testCase)
+
+ % Start with superclass
+ testCase.createFigure@wt.test.BaseWidgetTest();
+
+ % Adjust grid size
+ testCase.Figure.Position(3:4) = [1200 700];
+ testCase.Grid.RowHeight = repmat({'fit'},1,4);
+ testCase.Grid.ColumnWidth = repmat({'1x'},1,6);
+
+ end %function
+
+ end %methods
+
+
+ %% Test Method Setup
+ methods (TestMethodSetup)
+
+ function setup(testCase)
+
+ fcn = @()wt.SearchDropDown(testCase.Grid);
+ testCase.Widget = verifyWarningFree(testCase,fcn);
+ drawnow
+
+ % Set the initial items
+ testCase.verifySetProperty("Items", testCase.ItemNames);
+
+ % Verify empty default
+ expValue = "";
+ testCase.verifyPropertyValue(testCase.Widget, "Value", expValue)
+ testCase.verifyPropertyValue(testCase.Widget.EditField, "Value", char(expValue))
+
+ % Set callback
+ testCase.Widget.ValueChangedFcn = @(s,e)onCallbackTriggered(testCase,e);
+
+ end %function
+
+ end %methods
+
+
+ %% Unit Test
+ methods (Test)
+
+ function testValueProperty(testCase)
+
+ % Get the component
+ comp = testCase.Widget;
+
+ % Set a value from the list
+ expValue = testCase.ItemNames(3);
+ testCase.Widget.Value = expValue;
+ testCase.verifyPropertyValue(comp, "Value", expValue)
+ testCase.verifyPropertyValue(comp.EditField, "Value", char(expValue))
+
+ % Clear the value
+ expValue = "";
+ testCase.Widget.Value = expValue;
+ testCase.verifyPropertyValue(comp, "Value", expValue)
+ testCase.verifyPropertyValue(comp.EditField, "Value", char(expValue))
+
+ % Verify number of callbacks so far
+ testCase.verifyCallbackCount(0)
+
+ end %function
+
+
+
+ function testInteractivePartialSearch(testCase)
+
+ % Get the component
+ comp = testCase.Widget;
+
+ % Simulate typing to filter the dropdown items
+ testCase.type(comp.EditField, "urren")
+ testCase.verifyPropertyValue(comp.ListBox, "Items", cellstr(testCase.ItemNames(3:4)'));
+
+ % Verify number of callbacks so far
+ testCase.verifyCallbackCount(1)
+
+ % Ensure the listbox is showing
+ % It might not be due to the nature of the type command
+ comp.SearchPanel.Visible = true;
+
+ % Simulate selecting the second item from the filtered list
+ testCase.choose(comp.ListBox, 2);
+
+ testCase.verifyPropertyValue(comp, "Value", testCase.ItemNames(4));
+ testCase.verifyPropertyValue(comp.EditField, "Value", char(testCase.ItemNames(4)));
+
+ % Verify number of callbacks so far
+ testCase.verifyCallbackCount(2)
+
+ % Verify the list is no longer visible
+ testCase.verifyNotVisible(comp.SearchPanel)
+
+ end %function
+
+ end %methods (Test)
+
+end %classdef
\ No newline at end of file
diff --git a/widgets/+wt/SearchDropDown.m b/widgets/+wt/SearchDropDown.m
new file mode 100644
index 00000000..0add4a62
--- /dev/null
+++ b/widgets/+wt/SearchDropDown.m
@@ -0,0 +1,893 @@
+classdef SearchDropDown < wt.abstract.BaseWidget &...
+ wt.mixin.Enableable & ...
+ wt.mixin.FieldColorable & ...
+ wt.mixin.FontStyled & ...
+ wt.mixin.Tooltipable
+ % Searchable text field with drop down list
+
+ % Copyright 2025 The MathWorks Inc.
+
+ % This is a prototype widget that is like an editable dropdown, but it
+ % enables you to search/filter better as you type. It combines an edit
+ % field with a listbox below. Space beneath is required for it to work.
+
+ % Known Issues
+ % 1. If you are typing in edit field and hit down arrow, it should go
+ % into the list below, but instead it completes the edit. But if you
+ % tab or enter to finish editing, it should complete the edit.
+
+
+ %% Events
+ events (HasCallbackProperty, NotifyAccess = protected)
+
+ % Triggered on value changed, has companion callback
+ ValueChanged
+
+ end %events
+
+
+ %% Public properties
+ properties (AbortSet)
+
+ % Placeholder
+ Placeholder (1,1) string = ""
+
+ % List of items to choose from
+ Items (:,1) string
+
+ % Confirmed value, selected by the user
+ Value (1,1) string
+
+ end %properties
+
+
+ %% Read-only properties
+ properties (SetAccess = protected, UsedInUpdate = false)
+
+ % Current typed value, displayed in the edit field
+ EditingValue (1,1) string
+
+ end %properties
+
+
+
+ %% Internal Properties
+ properties (Hidden)
+
+ Debug (1,1) logical = false
+
+ end %properties
+
+
+ properties (Dependent, Hidden, SetAccess = immutable)
+
+ % Indicates search panel exists
+ SearchExists (1,1) logical
+
+ % Indicates search is open
+ SearchOpen (1,1) logical
+
+ end %properties
+
+
+ properties (Transient, NonCopyable, Hidden, ...
+ SetAccess = protected, UsedInUpdate = false)
+
+ % Edit field
+ EditField matlab.ui.control.EditField
+
+ % Search panel for drop down
+ SearchPanel matlab.ui.container.Panel
+
+ % Grid for drop down
+ SearchGrid matlab.ui.container.GridLayout
+
+ % List selection
+ ListBox matlab.ui.control.ListBox
+
+ % Event listeners
+ LocationChangedListener event.listener
+ FigureCurrentObjectListener event.proplistener
+ FigureActivatedListener event.listener
+ FigureDeactivatedListener event.listener
+ WindowKeyPressListener event.listener
+
+ ListButtonDownListener event.listener
+ WindowMouseDownListener event.listener
+ WindowMouseReleaseListener event.listener
+
+
+ LastClick (1,1) string
+
+ % Figure ancestor
+ Figure matlab.ui.Figure
+
+ end %properties
+
+ % Accessors
+ methods
+ function tf = get.SearchOpen(obj)
+ tf = obj.SearchExists && obj.SearchPanel.Visible;
+ end
+ function tf = get.SearchExists(obj)
+ tf = isscalar(obj.SearchPanel) && isvalid(obj.SearchPanel);
+ end
+ end
+
+
+ %% Protected methods
+ methods (Access = protected)
+
+ function setup(obj)
+
+ % Call superclass method
+ obj.setup@wt.abstract.BaseWidget()
+
+ % Set default size
+ obj.Position(3:4) = [100 30];
+
+ % Configure Main Grid
+ % obj.Grid.Padding = 2;
+
+ % Create the edit control for user input
+ obj.EditField = uieditfield(obj.Grid, 'text');
+ obj.EditField.ValueChangingFcn = @(~,evt)onEditValueChanging(obj,evt);
+ obj.EditField.ValueChangedFcn = @(~,evt)onEditValueChanged(obj,evt);
+
+ % Create the search pannel (hidden)
+ obj.createSearchPanel()
+
+ % Attach figure
+ obj.Figure = ancestor(obj,'figure');
+
+ % Configure listeners
+ obj.LocationChangedListener = listener(obj.EditField, ...
+ "LocationChanged", @(~,evt)onLocationChanged(obj,evt));
+
+ % Listen to figure change
+ obj.createFigureListeners();
+
+ % Update the internal component lists
+ % obj.BackgroundColorableComponents = obj.Grid;
+
+ end %function
+
+
+ function update(obj)
+
+ if obj.Debug
+ disp('update');
+ end
+
+ % Update the contents
+ obj.EditField.Placeholder = obj.Placeholder;
+ obj.EditField.Value = obj.Value;
+ % obj.EditField.Value = obj.EditingValue;
+
+ % Update filtered list
+ % obj.updateFilteredList()
+
+ % Close the search panel
+ % (without giving focus to the edit field)
+ obj.closeSearchPanel(false);
+
+ end %function
+
+
+ function createSearchPanel(obj)
+
+ if obj.Debug
+ disp('createSearchPanel');
+ end
+
+ % Create the search panel content
+ obj.SearchPanel = uipanel("Parent",[]);
+
+ % Create the grid layout for the search panel
+ obj.SearchGrid = uigridlayout(obj.SearchPanel);
+ obj.SearchGrid.RowHeight = {'1x'};
+ obj.SearchGrid.ColumnWidth = {'1x'};
+ obj.SearchGrid.Padding = 0;
+
+ % Create the list box for displaying search results
+ obj.ListBox = uilistbox(obj.SearchGrid);
+ obj.ListBox.ValueChangedFcn = @(~,evt)onListValueChanged(obj,evt);
+
+
+ % Check for clicks
+ if isMATLABReleaseOlderThan("R2022b")
+
+ else
+ % R2022b and later can use this
+ obj.ListBox.ClickedFcn = @(~,evt)onListClicked(obj,evt);
+ end
+
+ % Listen for button down on list
+ obj.ListButtonDownListener = listener(obj.ListBox, ...
+ "ButtonDown", @(~,evt)onListButtonDown(obj,evt));
+
+ % Later: Add more controls for case sensitive, etc.
+
+ % Update the internal component lists
+ obj.FontStyledComponents = [obj.EditField, obj.ListBox];
+ obj.FieldColorableComponents = [obj.EditField, obj.ListBox];
+ obj.EnableableComponents = [obj.EditField];
+ obj.TooltipableComponents = [obj.EditField];
+ obj.BackgroundColorableComponents = [obj.Grid, obj.SearchPanel, obj.SearchGrid];
+
+ end %function
+
+
+ function openSearchPanel(obj)
+
+ if obj.Debug
+ disp('openSearchPanel');
+ end
+
+ % Has figure changed?
+ currentFigure = ancestor(obj,'figure');
+ if ~isequal(currentFigure, obj.Figure)
+ obj.onLocationChanged();
+ end
+
+ % Does the search panel exist?
+ if ~obj.SearchExists
+ obj.createSearchPanel();
+ end
+
+ % Show the complete list initially
+ items = obj.Items;
+ obj.ListBox.Items = items;
+
+ % What to select as default in list?
+ if isempty(items)
+ % Empty list selection
+ obj.ListBox.Value = {};
+ else
+ % Default to first item
+ obj.ListBox.Value = items(1);
+ end
+ idxInit = find(matches(items, obj.Value), 1);
+ if isscalar(idxInit)
+ % Default to matching item
+ obj.ListBox.Value = items(idxInit);
+ else
+ % Empty list selection
+ obj.ListBox.Value = {};
+ end
+
+ % Show the search panel
+ obj.SearchPanel.Parent = obj.Figure;
+
+ % Update position
+ obj.updateSearchPanelPosition()
+
+ % Toggle visible
+ obj.SearchPanel.Visible = 'on';
+
+ % Focus back on edit field
+ % disp(" openSearchPanel: Focus EditField");
+ % focus(obj.EditField)
+
+ end %function
+
+
+ function updateSearchPanelPosition(obj)
+
+ if obj.Debug
+ disp('updateSearchPanelPosition');
+ end
+
+ % if obj.SearchOpen
+ if obj.SearchExists
+
+ % Calculate position
+ edPos = getpixelposition(obj.EditField, true);
+ maxHeight = 300;
+ x = edPos(1);
+ w = edPos(3);
+ h = edPos(2) - 10;
+ h = min(h, maxHeight);
+ y = edPos(2) - h;
+ searchPosition = [x y w h];
+
+ % Set position
+ obj.SearchPanel.Position = searchPosition;
+
+ end
+
+ end %function
+
+
+ function closeSearchPanel(obj, focusEdit)
+
+ arguments
+ obj
+ focusEdit (1,1) logical = false;
+ end
+
+ if obj.Debug
+ disp('closeSearchPanel');
+ end
+
+ % Is the search panel open?
+ if obj.SearchOpen
+
+ % Hide the search panel
+ obj.SearchPanel.Visible = 'off';
+
+ end %if
+
+ % Is it in focus? If so, focus to the edit field instead
+ if obj.SearchExists
+
+ if obj.Debug
+ disp(' closeSearchPanel: check current focus object');
+ end
+
+ curObj = obj.Figure.CurrentObject;
+
+ searchObjs = [
+ obj.SearchGrid
+ obj.SearchPanel
+ obj.ListBox
+ ];
+
+ if isscalar(curObj) && any(curObj == searchObjs) && focusEdit
+ if obj.Debug
+ disp(" closeSearchPanel: Focus EditField");
+ end
+ focus(obj.EditField)
+ end
+
+ end
+
+ % curObjNew = obj.Figure.CurrentObject;
+
+ end %function
+
+
+ function updateFilteredList(obj)
+ % display the filtered items in the listbox
+
+ if obj.Debug
+ disp('updateFilteredList');
+ end
+
+ if obj.SearchExists
+
+ allItems = obj.Items;
+ isMatch = contains(allItems, obj.EditingValue, 'IgnoreCase', true);
+ items = allItems(isMatch);
+
+ % Update the list contents
+ obj.ListBox.Items = items;
+
+ % What to select as default in list?
+ % if isempty(items)
+ % % Empty list selection
+ % obj.ListBox.Value = {};
+ % elseif isempty(obj.ListBox.Value)
+ % % Default to first item
+ % obj.ListBox.Value = items(1);
+ % else
+ % % Leave selection as-is
+ % end
+
+ end %if
+
+ end %function
+
+
+ function onListValueChanged(obj,evt)
+ % Triggered after an item in the list has been selected
+
+ if obj.Debug
+ fprintf("onListValueChanged EventValue: %s ListboxValue: %s LastClick: %s\n", evt.Value, string(obj.ListBox.Value), obj.LastClick);
+ end
+
+ newValue = evt.Value;
+ obj.EditingValue = newValue;
+
+ switch obj.LastClick
+
+ case "ListBox"
+
+ % Clear state
+ obj.LastClick = "";
+
+ if isMATLABReleaseOlderThan("R2022b")
+
+ % Accept the value and close the
+ obj.acceptEditValue();
+
+ % Close the search panel
+ % and give focus to the edit field
+ obj.closeSearchPanel(true);
+
+ end
+
+ end %switch
+
+ end %function
+
+
+ function onListClicked(obj,evt)
+ % Triggered after an item in the list has been clicked
+ % (R2022b and later only)
+
+ % Get the changed value
+ % newValue = string(obj.ListBox.Value)
+ idx = evt.InteractionInformation.Item;
+ newValue = string(obj.ListBox.Items(idx));
+
+ if obj.Debug
+ % disp("onListClicked");
+ fprintf("onListClicked Index: %d ItemValue: %s ListboxValue: %s\n", idx, newValue, string(obj.ListBox.Value));
+ end
+
+ % Was an item clicked (rather than empty space)?
+ if isscalar(newValue)
+
+ % Complete the action
+ obj.EditingValue = newValue;
+
+ % Accept the value
+ obj.acceptEditValue();
+
+ % Close the search panel
+ % Do not give focus to the edit field
+ obj.closeSearchPanel(false);
+
+ end %if
+
+ end %function
+
+
+ function onFigureButtonDown(obj,evt)
+ % Triggered on any click within the figure
+
+
+ if obj.Debug
+ fprintf("onFigureButtonDown ClickOn: %s ListboxValue: %s\n", class(evt.HitObject), string(obj.ListBox.Value));
+ end
+
+
+ switch evt.HitObject
+
+ case obj.ListBox
+
+ obj.LastClick = "ListBox";
+
+ if isMATLABReleaseOlderThan("R2022b")
+ % For R2022a
+
+ % % Get the changed value
+ % newValue = string(obj.ListBox.Value);
+ %
+ %
+ % fprintf(" onFigureButtonDown (R2022a) - Accepting ListboxValue: %s\n", newValue);
+ %
+ % % Complete the action
+ % obj.EditingValue = newValue;
+ %
+ % % Accept the value and close the
+ % obj.acceptEditValue();
+ %
+ % % Close the search panel
+ % obj.closeSearchPanel(true);
+
+
+ else
+ % R2022b and later can use this
+
+ end
+
+ case obj.EditField
+
+ obj.LastClick = "EditField";
+
+ % Do nothing
+ % The dropdown is activated elsewhere?
+
+ otherwise
+ % Clicked any other figure object
+
+ obj.LastClick = "";
+
+ % Close the search panel
+ % Do not give focus to the edit field
+ obj.closeSearchPanel(false);
+
+ end %switch
+
+ end %function
+
+
+ function onFigureButtonUp(obj,evt)
+ % Triggered after
+
+ if obj.Debug
+ fprintf("onFigureButtonUp Hit: %s ListboxValue: %s\n", class(evt.HitObject), string(obj.ListBox.Value));
+ end
+
+ % newValue = string(obj.ListBox.Value)
+
+ % % Close the search panel
+ % obj.closeSearchPanel();
+
+ end %function
+
+
+ function onListButtonDown(obj,evt)
+ % Triggered after a key press in the list field
+
+ % Get the changed value
+ % newValue = string(obj.ListBox.Value)
+ idx = evt.InteractionInformation.Item;
+ newValue = string(obj.ListBox.Items(idx));
+
+ if obj.Debug
+ fprintf("onListButtonDown Index: %d ItemValue: %s ListboxValue: %s\n", idx, newValue, string(obj.ListBox.Value));
+ end
+
+ % Complete the action
+ obj.EditingValue = newValue;
+
+ % Accept the value
+ obj.acceptEditValue();
+
+ % Close the search panel
+ % and give focus to the edit field
+ obj.closeSearchPanel(true);
+
+ end %function
+
+
+ function onEditValueChanging(obj,evt)
+ % Triggered on edit field typing
+
+ if obj.Debug
+ fprintf("onEditValueChanging New EditingValue: %s Existing Value: %s\n", evt.Value, obj.Value);
+ end
+
+ % Get the editing value
+ newValue = evt.Value;
+
+ % Update the value property
+ obj.EditingValue = newValue;
+
+ % Verify search is open
+ if ~obj.SearchOpen
+ obj.openSearchPanel();
+ end
+
+ % Update filtered list
+ obj.updateFilteredList()
+
+ end %function
+
+
+ function onEditValueChanged(obj,evt)
+ % Triggered on edit field enter or loss of focus
+
+ if obj.Debug
+ fprintf("onEditValueChanged New EditingValue: %s Existing Value: %s LastClick: %s\n", evt.Value, obj.Value, obj.LastClick);
+ end
+
+ switch obj.LastClick
+
+ case "EditField"
+
+ % Clear state
+ obj.LastClick = "";
+
+ % Are there items available?
+ if isempty(obj.ListBox.Items)
+ % Get the editing value
+ newValue = evt.Value;
+ else
+ % Get the first match in items
+ newValue = obj.ListBox.Items(1);
+ end
+
+ % Update the value property
+ obj.EditingValue = newValue;
+
+ % Accept the value
+ obj.acceptEditValue();
+
+ % Close the search panel
+ %obj.closeSearchPanel(true);
+ % Disabled this in favor of closing in the update method. The
+ % problem is in 22a if we start typing in the edit, then click
+ % an item in the list, this one fires first and hides the list
+ % before the list's selection callback can occur.
+
+ end %switch
+
+ end %function
+
+
+ function onLocationChanged(obj,~)
+ % Triggered on edit field location (figure) changed
+
+ if obj.Debug
+ disp("onLoctionChanged");
+ end
+
+ % Attach figure
+ obj.Figure = ancestor(obj,'figure');
+
+ % Move SearchPanel
+ if obj.SearchExists
+ obj.SearchPanel.Parent = obj.Figure;
+ end
+
+ % Listen to figure events
+ obj.createFigureListeners();
+
+ end %function
+
+
+ function createFigureListeners(obj)
+ % Create listeners to figure changes
+
+ if obj.Debug
+ disp("createFigureListeners");
+ end
+
+ % Get the figure ancestor
+ obj.Figure = ancestor(obj,'figure');
+
+ % Listen to current object (focus) changes
+ obj.FigureCurrentObjectListener = listener(obj.Figure, ...
+ "CurrentObject", "PostSet", ...
+ @(~,evt)onFigureCurrentObjectChanged(obj,evt));
+
+ % Listen to figure activation
+ obj.FigureActivatedListener = listener(obj.Figure, ...
+ "FigureActivated", ...
+ @(~,evt)onFigureActivated(obj,evt));
+
+ % Listen to figure deactivation
+ obj.FigureDeactivatedListener = listener(obj.Figure, ...
+ "FigureDeactivated", ...
+ @(~,evt)onFigureDeactivated(obj,evt));
+
+
+ obj.WindowMouseDownListener = listener(obj.Figure, ...
+ "WindowMousePress", ...
+ @(~,evt)onFigureButtonDown(obj,evt));
+
+ obj.WindowMouseReleaseListener = listener(obj.Figure, ...
+ "WindowMouseRelease", ...
+ @(~,evt)onFigureButtonUp(obj,evt));
+
+ % Listen to key presses
+ obj.WindowKeyPressListener = listener(obj.Figure, ...
+ "WindowKeyPress", ...
+ @(~,evt)onWindowKeyPress(obj,evt));
+
+
+ end %function
+
+
+ function onFigureCurrentObjectChanged(obj,~)
+ % Triggered after figure's current object (focus) changed
+
+
+ if obj.Debug
+ fprintf("onFigureCurrentObjectChanged CurrentObject: %s\n", class(obj.Figure.CurrentObject));
+ end
+
+ obj.checkForSearchActivation();
+
+ end %function
+
+
+ function onFigureActivated(obj,~)
+ % Triggered after figure is activated
+
+ if obj.Debug
+ disp("onFigureActivated");
+ end
+
+ % Close list
+ % obj.closeSearchPanel(false)
+
+ end %function
+
+
+ function onFigureDeactivated(obj,~)
+ % Triggered after figure is deactivated
+
+ if obj.Debug
+ disp("onFigureDeactivated");
+ end
+
+ % Close list
+ % obj.closeSearchPanel(false)
+
+ end %function
+
+
+ function onWindowKeyPress(obj,evt)
+ % Triggered after a key is pressed in the figure
+
+
+ if obj.Debug
+ fprintf("onWindowKeyPress Key: %s CurrentObject: %s LastClick: %s\n", evt.Key, class(obj.Figure.CurrentObject), obj.LastClick);
+ end
+
+ obj.LastClick = "";
+
+ % Button presses
+ escPressed = evt.Key == "escape" && isempty(evt.Modifier);
+ downArrowPressed = evt.Key == "downarrow" && isempty(evt.Modifier);
+ % upArrowPressed = evt.Key == "uparrow" && isempty(evt.Modifier);
+ enterPressed = evt.Key == "return" && isempty(evt.Modifier);
+
+ % Search state
+ searchOpen = obj.SearchOpen;
+ searchExists = obj.SearchExists;
+
+ % Edit state
+ curObj = obj.Figure.CurrentObject;
+
+ % Edit focus?
+ editFocus = isequal(curObj, obj.EditField);
+
+ % Search focus?
+ if searchExists
+ searchObjs = [
+ obj.SearchGrid
+ obj.SearchPanel
+ obj.ListBox
+ ];
+ searchFocus = isscalar(curObj) && any(curObj == searchObjs);
+ % searchIndexOne = isequal(obj.ListBox.ValueIndex, 1);
+ else
+ searchFocus = false;
+ % searchIndexOne = false;
+ end
+
+ if editFocus && enterPressed
+ % If enter pressed, accept the entered value
+
+ obj.acceptEditValue()
+
+ % Close list
+ % Do not give focus to the edit field
+ obj.closeSearchPanel(false);
+
+ elseif searchOpen && escPressed
+ % If ESC is pressed while list is open, close the list
+
+ % Close list
+ % and give focus to the edit field
+ obj.closeSearchPanel(true);
+
+ elseif searchOpen && editFocus && downArrowPressed
+ % Focus the search list
+
+ if obj.Debug
+ disp(" onWindowKeyPress: Focus ListBox 1");
+ end
+ focus(obj.ListBox)
+
+ % elseif searchOpen && searchFocus && upArrowPressed && searchIndexOne
+ %
+ % focus(obj.EditField)
+
+ elseif searchExists && editFocus && downArrowPressed
+ % Open and focus the search list
+
+ obj.openSearchPanel();
+
+ if obj.Debug
+ disp(" onWindowKeyPress: Focus ListBox 3");
+ end
+ focus(obj.ListBox)
+
+ elseif searchExists && ~searchOpen && editFocus
+ % Open the search panel
+
+ obj.openSearchPanel();
+
+ elseif searchExists && ~searchOpen && searchFocus
+ % If search is hidden but still focused, change to the edit
+ % field focus
+
+ % Don't know why search would be closed here
+
+ if obj.Debug
+ disp(" onWindowKeyPress: Focus ListBox 4");
+ end
+
+ if isMATLABReleaseOlderThan("R2022b")
+ % Do nothing
+ % The edit field might actually be focused and selected,
+ % causing typing to delete the existing text in the edit
+ % field.
+ else
+ focus(obj.EditField)
+ end
+
+ end %if
+
+ end %function
+
+
+ function checkForSearchActivation(obj)
+ % Check current object and activate search if conditions met
+
+ % What is in focus?
+ focusObject = obj.Figure.CurrentObject;
+
+ if focusObject == obj.EditField
+ % Open list
+ obj.openSearchPanel();
+ elseif isequal(focusObject, obj.ListBox)
+ % Ignore - do nothing
+ else
+ % Close list
+ % Do not give focus to the edit field
+ obj.closeSearchPanel(false)
+ end
+
+ end %function
+
+
+ function acceptEditValue(obj)
+ % Accept the currently selected value
+
+
+ % Get typed value
+ editValue = obj.EditingValue;
+ newValue = editValue;
+
+ if obj.Debug
+ fprintf("acceptEditValue NewValue: %s OldValue: %s\n", newValue, obj.Value);
+ end
+
+ % % Get list items
+ % if obj.SearchExists
+ % items = obj.ListBox.Items;
+ % else
+ % items = obj.Items;
+ % end
+ %
+ % % Select the new value intelligently
+ % if matches(editValue, items)
+ % newValue = editValue;
+ % elseif isempty(items)
+ % newValue = "";
+ % else
+ % newValue = items(1);
+ % end
+
+ % Prepare event data
+ oldValue = obj.Value;
+ evtOut = wt.eventdata.PropertyChangedData('Value', ...
+ newValue, oldValue);
+
+ % Update the value
+ obj.Value = newValue;
+
+ % Request update
+ obj.requestUpdate();
+
+ % Close the search panel
+ % obj.closeSearchPanel(false);
+
+ % Trigger event
+ notify(obj,"ValueChanged",evtOut);
+
+ end %function
+
+ end %methods
+
+
+end %classdef
+
diff --git a/widgets/doc/WidgetsList.mlx b/widgets/doc/WidgetsList.mlx
index 25b9206f..f1efaad6 100644
Binary files a/widgets/doc/WidgetsList.mlx and b/widgets/doc/WidgetsList.mlx differ
diff --git a/widgets/examples/SearchDropDownExample.mlx b/widgets/examples/SearchDropDownExample.mlx
new file mode 100644
index 00000000..25b93dbb
Binary files /dev/null and b/widgets/examples/SearchDropDownExample.mlx differ
diff --git a/widgets/resources/appDesigner.json b/widgets/resources/appDesigner.json
index 9af5f45b..3b1b603d 100644
--- a/widgets/resources/appDesigner.json
+++ b/widgets/resources/appDesigner.json
@@ -305,8 +305,25 @@
300,
200
]
+ },
+ {
+ "className": "wt.SearchDropDown",
+ "componentName": "Search Drop Down",
+ "description": "Editable test field that drops down a list of search matches",
+ "icon": "",
+ "category": "Widgets Toolbox",
+ "authorName": "Robyn Jackey - MathWorks Consulting",
+ "authorEmail": "",
+ "version": "1.0",
+ "avatar": "",
+ "defaultPosition": [
+ 1,
+ 1,
+ 100,
+ 30
+ ]
}
],
- "schema": 1,
- "MATLABRelease": "R2021a"
+ "schema": 2,
+ "MATLABRelease": "R2022a"
}
\ No newline at end of file