diff --git a/src/ImageRunModal.jsx b/src/ImageRunModal.jsx
index 9e334b73..411b12fb 100644
--- a/src/ImageRunModal.jsx
+++ b/src/ImageRunModal.jsx
@@ -9,11 +9,12 @@ import { Modal } from "@patternfly/react-core/dist/esm/components/Modal";
import { NumberInput } from "@patternfly/react-core/dist/esm/components/NumberInput";
import { Popover } from "@patternfly/react-core/dist/esm/components/Popover";
import { Radio } from "@patternfly/react-core/dist/esm/components/Radio";
+import { Spinner } from "@patternfly/react-core/dist/esm/components/Spinner/index.js";
import { Tab, TabTitleText, Tabs } from "@patternfly/react-core/dist/esm/components/Tabs";
import { Text, TextContent, TextList, TextListItem, TextVariants } from "@patternfly/react-core/dist/esm/components/Text";
import { TextInput } from "@patternfly/react-core/dist/esm/components/TextInput";
import { ToggleGroup, ToggleGroupItem } from "@patternfly/react-core/dist/esm/components/ToggleGroup";
-import { Select, SelectGroup, SelectOption, SelectVariant } from "@patternfly/react-core/dist/esm/deprecated/components/Select";
+import { Bullseye } from "@patternfly/react-core/dist/esm/layouts/Bullseye/index.js";
import { Flex, FlexItem } from "@patternfly/react-core/dist/esm/layouts/Flex";
import { Grid, GridItem } from "@patternfly/react-core/dist/esm/layouts/Grid";
import { OutlinedQuestionCircleIcon } from '@patternfly/react-icons';
@@ -23,6 +24,7 @@ import { debounce } from 'throttle-debounce';
import cockpit from 'cockpit';
import { DynamicListForm } from 'cockpit-components-dynamic-list.jsx';
+import { TypeaheadSelect } from 'cockpit-components-typeahead-select';
import { onDownloadContainer, onDownloadContainerFinished } from './Containers.jsx';
import { EnvVar, validateEnvVar } from './Env.jsx';
@@ -583,44 +585,33 @@ export class ImageRunModal extends React.Component {
const input = this.buildFilterRegex(searchText, false);
- const results = imageRegistries
- .map((reg, index) => {
- const filtered = (reg in images ? images[reg] : [])
- .filter(image => {
- if (image.isSystem && !isSystem) {
- return false;
- }
- if ('isSystem' in image && !image.isSystem && isSystem) {
- return false;
- }
- return image.Name.search(input) !== -1;
- })
- .map((image, index) => {
- return (
-
- );
- });
-
- if (filtered.length === 0) {
- return [];
- } else {
- return (
-
- {filtered}
-
- );
- }
- })
- .filter(group => group.length !== 0); // filter out empty groups
+ const results = [];
+ imageRegistries.forEach(reg => {
+ let need_header = this.state.searchByRegistry == 'all';
+ (reg in images ? images[reg] : [])
+ .filter(image => {
+ if (image.isSystem && !isSystem) {
+ return false;
+ }
+ if ('isSystem' in image && !image.isSystem && isSystem) {
+ return false;
+ }
+ return image.Name.search(input) !== -1;
+ })
+ .forEach(image => {
+ if (need_header) {
+ results.push({ key: results.length, decorator: "header", content: reg });
+ need_header = false;
+ }
- // Remove when there is a filter selected.
- if (this.state.searchByRegistry !== 'all' && imageRegistries.length === 1 && results.length === 1) {
- return results[0].props.children;
- }
+ results.push({
+ key: results.length,
+ value: image,
+ content: image.toString(),
+ description: image.Description
+ });
+ });
+ });
return results;
};
@@ -804,6 +795,12 @@ export class ImageRunModal extends React.Component {
);
+ const spinnerOptions = (
+ this.state.searchInProgress
+ ? [{ value: "_searching", content: , isDisabled: true }]
+ : []
+ );
+
/* ignore Enter key, it otherwise opens the first popover help; this clears
* the search input and is still irritating from other elements like check boxes */
const defaultBody = (
@@ -901,30 +898,22 @@ export class ImageRunModal extends React.Component {
}
>
-
+ // We do our own filtering when producing imageListOptions
+ filterFunction={(filterValue, options) => options}
+ selectOptions={imageListOptions.concat(spinnerOptions)}
+ footer={footer} />
{(image || localImage) &&
diff --git a/test/check-application b/test/check-application
index 2e867b31..7e5c22ad 100755
--- a/test/check-application
+++ b/test/check-application
@@ -504,28 +504,29 @@ class TestApplication(testlib.MachineCase):
# Check showing of entrypoint
b.click("#containers-containers-create-container-btn")
- b.click("#create-image-image-select-typeahead")
- b.click(f'button.pf-v5-c-select__menu-item:contains("{IMG_REGISTRY}")')
+ b.click("#create-image-image input")
+ b.click(f'button.pf-v5-c-menu__item:contains("{IMG_REGISTRY}")')
b.wait_val("#run-image-dialog-command", '/etc/docker/registry/config.yml')
b.wait_text("#run-image-dialog-entrypoint", '/entrypoint.sh')
# Deleting image will cleanup both command and entrypoint
- b.click("button.pf-v5-c-select__toggle-clear")
+ b.click("button[aria-label='Clear input value']")
b.wait_val("#run-image-dialog-command", '')
b.wait_not_present("#run-image-dialog-entrypoint")
# Edited command will not be cleared
- b.click("#create-image-image-select-typeahead")
- b.click(f'button.pf-v5-c-select__menu-item:contains("{IMG_REGISTRY}")')
+ b.click("#create-image-image input")
+ time.sleep(1) # wait for TypeaheadSelect focus shenanigans
+ b.click(f'button.pf-v5-c-menu__item:contains("{IMG_REGISTRY}")')
b.wait_val("#run-image-dialog-command", '/etc/docker/registry/config.yml')
b.set_input_text("#run-image-dialog-command", '/etc/docker/registry/config.yaml')
- b.click("button.pf-v5-c-select__toggle-clear")
+ b.click("button[aria-label='Clear input value']")
b.wait_not_present("#run-image-dialog-entrypoint")
b.wait_val("#run-image-dialog-command", '/etc/docker/registry/config.yaml')
# Setting a new image will still keep the old command and not prefill it
- b.click("#create-image-image-select-typeahead")
- b.click(f'button.pf-v5-c-select__menu-item:contains({IMG_ALPINE})')
+ b.click("#create-image-image input")
+ b.click(f'button.pf-v5-c-menu__item:contains({IMG_ALPINE})')
b.wait_visible("#run-image-dialog-pull-latest-image")
b.wait_val("#run-image-dialog-command", '/etc/docker/registry/config.yaml')
@@ -553,8 +554,8 @@ class TestApplication(testlib.MachineCase):
# Test the isSystem boolean for searching
# https://github.com/cockpit-project/cockpit-podman/pull/891
b.click("#containers-containers-create-container-btn")
- b.set_input_text("#create-image-image-select-typeahead", "registry")
- b.wait_visible('button.pf-v5-c-select__menu-item:contains("registry")')
+ b.set_input_text("#create-image-image input", "registry")
+ b.wait_visible('button.pf-v5-c-menu__item:contains("registry")')
self.confirm_modal("Cancel")
def testBasicUser(self):
@@ -875,9 +876,9 @@ class TestApplication(testlib.MachineCase):
# Intermediate images are not shown in create container dialog
b.click("#containers-containers-create-container-btn")
b.wait_visible('div.pf-v5-c-modal-box header:contains("Create container")')
- b.click("#create-image-image-select-typeahead")
- b.wait_visible(f".pf-v5-c-select__menu-item:contains('{IMG_REGISTRY}')")
- b.wait_not_present(".pf-v5-c-select__menu-item:contains('none')")
+ b.click("#create-image-image input")
+ b.wait_visible(f".pf-v5-c-menu__item:contains('{IMG_REGISTRY}')")
+ b.wait_not_present(".pf-v5-c-menu__item:contains('none')")
b.click(".pf-v5-c-modal-box .btn-cancel")
b.wait_not_present(".pf-v5-c-modal-box")
@@ -1750,40 +1751,40 @@ class TestApplication(testlib.MachineCase):
# Test special characters.
# Initially both the registry image and Alpine are shown
- b.click("#create-image-image")
- b.wait_visible(f'button.pf-v5-c-select__menu-item:contains("{IMG_ALPINE_LATEST}")')
- b.wait_visible(f'button.pf-v5-c-select__menu-item:contains("{IMG_REGISTRY_LATEST}")')
+ b.click("#create-image-image input")
+ b.wait_visible(f'button.pf-v5-c-menu__item:contains("{IMG_ALPINE_LATEST}")')
+ b.wait_visible(f'button.pf-v5-c-menu__item:contains("{IMG_REGISTRY_LATEST}")')
# Filter it down to just Alpine
- b.set_input_text("#create-image-image-select-typeahead", "|alpi*ne?\\")
- b.wait_not_present(f'button.pf-v5-c-select__menu-item:contains("{IMG_REGISTRY_LATEST}")')
- b.wait_visible(f'button.pf-v5-c-select__menu-item:contains("{IMG_ALPINE_LATEST}")')
+ b.set_input_text("#create-image-image input", "|alpi*ne?\\")
+ b.wait_not_present(f'button.pf-v5-c-menu__item:contains("{IMG_REGISTRY_LATEST}")')
+ b.wait_visible(f'button.pf-v5-c-menu__item:contains("{IMG_ALPINE_LATEST}")')
# No local results found
- b.set_input_text("#create-image-image-select-typeahead", "notfound")
+ b.set_input_text("#create-image-image input", "notfound")
b.click('button.pf-v5-c-toggle-group__button:contains("Local")')
- b.wait_text("button.pf-v5-c-select__menu-item.pf-m-disabled", "No images found")
+ b.wait_text("button.pf-v5-c-menu__item[disabled]", "No images found")
# Local results found
- b.set_input_text("#create-image-image-select-typeahead", "registry")
- b.wait_text("button.pf-v5-c-select__menu-item", IMG_REGISTRY_LATEST)
+ b.set_input_text("#create-image-image input", "registry")
+ b.wait_text("button.pf-v5-c-menu__item:not([disabled])", IMG_REGISTRY_LATEST)
if auth:
b.assert_pixels(".pf-v5-c-modal-box", "image-select", skip_layouts=["rtl"])
# Local registry
b.click('button.pf-v5-c-toggle-group__button:contains("localhost:5000")')
- b.set_input_text("#create-image-image-select-typeahead", "my-busybox", blur=False)
- b.wait_text("button.pf-v5-c-select__menu-item:not(.pf-m-disabled)", "localhost:5000/my-busybox")
+ b.set_input_text("#create-image-image input", "my-busybox", blur=False)
+ b.wait_text("button.pf-v5-c-menu__item:not([disabled])", "localhost:5000/my-busybox")
# pressing Enter does not disturb the dialog or open popup help
- b.wait_val("#create-image-image-select-typeahead", "my-busybox")
+ b.wait_val("#create-image-image input", "my-busybox")
b.key("Enter")
time.sleep(0.3)
self.assertFalse(b.is_present(".pf-v5-c-popover"))
- b.wait_val("#create-image-image-select-typeahead", "my-busybox")
+ b.wait_val("#create-image-image input", "my-busybox")
# Select image
- b.click('button.pf-v5-c-select__menu-item:contains("localhost:5000/my-busybox")')
+ b.click('button.pf-v5-c-menu__item:contains("localhost:5000/my-busybox")')
# Remote image, no pull latest image option
b.wait_not_present("#run-image-dialog-pull-latest-image")
@@ -1807,14 +1808,14 @@ class TestApplication(testlib.MachineCase):
b.set_input_text("#run-image-dialog-name", container_name)
# Local registry
- b.set_input_text("#create-image-image-select-typeahead", "no-container")
- b.wait_text("button.pf-v5-c-select__menu-item.pf-m-disabled", "No images found")
+ b.set_input_text("#create-image-image input", "no-container")
+ b.wait_text("button.pf-v5-c-menu__item[disabled]", "No images found")
# Search with full url
- b.set_input_text("#create-image-image-select-typeahead", "localhost:5000/my-busybox")
+ b.set_input_text("#create-image-image input", "localhost:5000/my-busybox")
b.click('button.pf-v5-c-toggle-group__button:contains("Local")')
# Select image
- b.click('button.pf-v5-c-select__menu-item:contains("localhost:5000/my-busybox")')
+ b.click('button.pf-v5-c-menu__item:contains("localhost:5000/my-busybox")')
# Pull the latest image
b.set_checked("#run-image-dialog-pull-latest-image", True)
@@ -1839,20 +1840,20 @@ class TestApplication(testlib.MachineCase):
b.set_input_text("#run-image-dialog-name", container_name)
# Open image dropdown, it should list both Alpine and Busybox from the local repository
- b.click("#create-image-image")
- b.wait_visible(f'button.pf-v5-c-select__menu-item:contains("{IMG_ALPINE_LATEST}")')
- b.wait_visible(f'button.pf-v5-c-select__menu-item:contains("{IMG_BUSYBOX_LATEST}")')
+ b.click("#create-image-image input")
+ b.wait_visible(f'button.pf-v5-c-menu__item:contains("{IMG_ALPINE_LATEST}")')
+ b.wait_visible(f'button.pf-v5-c-menu__item:contains("{IMG_BUSYBOX_LATEST}")')
# Filter it down to just Busybox
- b.set_input_text("#create-image-image-select-typeahead", IMG_BUSYBOX_LATEST)
- b.wait_visible(f'button.pf-v5-c-select__menu-item:contains("{IMG_BUSYBOX_LATEST}")')
- b.wait_not_present(f'button.pf-v5-c-select__menu-item:contains("{IMG_ALPINE_LATEST}")')
+ b.set_input_text("#create-image-image input", IMG_BUSYBOX_LATEST)
+ b.wait_visible(f'button.pf-v5-c-menu__item:contains("{IMG_BUSYBOX_LATEST}")')
+ b.wait_not_present(f'button.pf-v5-c-menu__item:contains("{IMG_ALPINE_LATEST}")')
# Switch to another view; different images have different registries, so don't assume their name, just
# that at least one more exists
b.click('.image-search-footer .pf-v5-c-toggle-group__item:nth-of-type(3) button')
- b.wait_in_text(".pf-v5-c-select__menu-list", "No images found")
+ b.wait_in_text(".pf-v5-c-menu", "No images found")
# Back to local and let's select Busybox and create the container
b.click('button.pf-v5-c-toggle-group__button:contains("Local")')
- b.click(f'button.pf-v5-c-select__menu-item:contains("{IMG_BUSYBOX_LATEST}")')
+ b.click(f'button.pf-v5-c-menu__item:contains("{IMG_BUSYBOX_LATEST}")')
b.click('.pf-v5-c-modal-box__footer #create-image-create-run-btn')
b.wait(lambda: self.getContainerAttr(container_name, "State", sel) in 'Running')
@@ -1896,7 +1897,8 @@ class TestApplication(testlib.MachineCase):
b.wait_visible('div.pf-v5-c-modal-box header:contains("Create container")')
# Inspect and fill modal dialog
- b.wait_val("#create-image-image-select-typeahead", IMG_BUSYBOX_LATEST)
+ b.wait_val("#create-image-image input", IMG_BUSYBOX_LATEST)
+ time.sleep(1) # wait for TypeaheadSelect focus shenanigans
# Check that there is autogenerated name and then overwrite it
b.wait_not_val("#run-image-dialog-name", "")
@@ -2189,7 +2191,7 @@ class TestApplication(testlib.MachineCase):
b.click(f'#containers-images tbody tr:contains("{IMG_BUSYBOX}") .ct-container-create')
b.wait_visible('div.pf-v5-c-modal-box header:contains("Create container")')
- b.wait_val("#create-image-image-select-typeahead", IMG_BUSYBOX_LATEST)
+ b.wait_val("#create-image-image input", IMG_BUSYBOX_LATEST)
b.set_input_text("#run-image-dialog-name", "busybox-without-publish")
# Set up command line
@@ -2272,7 +2274,7 @@ class TestApplication(testlib.MachineCase):
b.click(f'#containers-images tbody tr:contains("{IMG_BUSYBOX}") .ct-container-create')
b.wait_visible('div.pf-v5-c-modal-box header:contains("Create container")')
- b.wait_val("#create-image-image-select-typeahead", IMG_BUSYBOX_LATEST)
+ b.wait_val("#create-image-image input", IMG_BUSYBOX_LATEST)
b.set_input_text("#run-image-dialog-name", container_name)
b.set_input_text("#run-image-dialog-command", "sh -c sleep infinity")
@@ -2813,20 +2815,20 @@ class TestApplication(testlib.MachineCase):
b.set_input_text("#run-image-dialog-name", container_name)
# Open image dropdown, it should list both Alpine and Busybox from the local repository
- b.click("#create-image-image")
- b.wait_visible(f'button.pf-v5-c-select__menu-item:contains("{IMG_ALPINE_LATEST}")')
- b.wait_visible(f'button.pf-v5-c-select__menu-item:contains("{IMG_BUSYBOX_LATEST}")')
+ b.click("#create-image-image input")
+ b.wait_visible(f'button.pf-v5-c-menu__item:contains("{IMG_ALPINE_LATEST}")')
+ b.wait_visible(f'button.pf-v5-c-menu__item:contains("{IMG_BUSYBOX_LATEST}")')
# Filter it down to just Busybox
- b.set_input_text("#create-image-image-select-typeahead", IMG_BUSYBOX_LATEST)
- b.wait_visible(f'button.pf-v5-c-select__menu-item:contains("{IMG_BUSYBOX_LATEST}")')
- b.wait_not_present(f'button.pf-v5-c-select__menu-item:contains("{IMG_ALPINE_LATEST}")')
+ b.set_input_text("#create-image-image input", IMG_BUSYBOX_LATEST)
+ b.wait_visible(f'button.pf-v5-c-menu__item:contains("{IMG_BUSYBOX_LATEST}")')
+ b.wait_not_present(f'button.pf-v5-c-menu__item:contains("{IMG_ALPINE_LATEST}")')
# Switch to another view; different images have different registries, so don't assume their name, just
# that at least one more exists
b.click('.image-search-footer .pf-v5-c-toggle-group__item:nth-of-type(3) button')
- b.wait_in_text(".pf-v5-c-select__menu-list", "No images found")
+ b.wait_in_text(".pf-v5-c-menu", "No images found")
# Back to local and let's select Busybox and create the container
b.click('button.pf-v5-c-toggle-group__button:contains("Local")')
- b.click(f'button.pf-v5-c-select__menu-item:contains("{IMG_BUSYBOX_LATEST}")')
+ b.click(f'button.pf-v5-c-menu__item:contains("{IMG_BUSYBOX_LATEST}")')
b.click('.pf-v5-c-modal-box__footer #create-image-create-run-btn')
b.wait_not_present("#run-image-dialog-name")
@@ -3131,15 +3133,15 @@ class TestApplication(testlib.MachineCase):
b.click("#containers-containers button.pf-v5-c-button.pf-m-primary")
b.set_input_text("#run-image-dialog-name", "new-busybox")
# Searching for container by prefix fails
- b.set_input_text("#create-image-image-select-typeahead", "localhost:80/my-busy")
- b.wait_text("button.pf-v5-c-select__menu-item.pf-m-disabled", "No images found")
+ b.set_input_text("#create-image-image input", "localhost:80/my-busy")
+ b.wait_text("button.pf-v5-c-menu__item[disabled]", "No images found")
b.wait_in_text(".pf-v5-c-alert.pf-m-danger", "couldn't search registry")
# Giving full image name finds valid manifest therefore image is pullable
- b.set_input_text("#create-image-image-select-typeahead", container_name)
- b.wait_text("button.pf-v5-c-select__menu-item:not(.pf-m-disabled)", container_name)
+ b.set_input_text("#create-image-image input", container_name)
+ b.wait_text("button.pf-v5-c-menu__item:not([disabled])", container_name)
# Select image and create a container
- b.click(f"button.pf-v5-c-select__menu-item:contains({container_name})")
+ b.click(f"button.pf-v5-c-menu__item:contains({container_name})")
b.click("#create-image-create-btn")
# Wait for download to finish
@@ -3150,25 +3152,25 @@ class TestApplication(testlib.MachineCase):
b.click("#containers-containers button.pf-v5-c-button.pf-m-primary")
# "couldn't search registry" error is hidden when some local image is found
- b.set_input_text("#create-image-image-select-typeahead", "")
- b.set_input_text("#create-image-image-select-typeahead", "localhost:80/my-busy")
- b.wait_text("button.pf-v5-c-select__menu-item:not(.pf-m-disabled)", f"{container_name}:latest")
+ b.set_input_text("#create-image-image input", "")
+ b.set_input_text("#create-image-image input", "localhost:80/my-busy")
+ b.wait_text("button.pf-v5-c-menu__item:not([disabled])", f"{container_name}:latest")
b.wait_not_present(".pf-v5-c-alert.pf-m-danger")
b.click('button.pf-v5-c-toggle-group__button:contains("Local")')
# Error is shown again when no image was found locally
- b.set_input_text("#create-image-image-select-typeahead", "localhost:80/their-busy")
- b.wait_text("button.pf-v5-c-select__menu-item.pf-m-disabled", "No images found")
+ b.set_input_text("#create-image-image input", "localhost:80/their-busy")
+ b.wait_text("button.pf-v5-c-menu__item[disabled]", "No images found")
b.wait_in_text(".pf-v5-c-alert.pf-m-danger", "couldn't search registry")
# Searching for full container name with tag finds the image
b.set_input_text("#run-image-dialog-name", "tagged-busybox")
# Search should still work with spaces around image name
- b.set_input_text("#create-image-image-select-typeahead", " localhost:80/my-busybox:nosearch-tag ")
+ b.set_input_text("#create-image-image input", " localhost:80/my-busybox:nosearch-tag ")
b.click('button.pf-v5-c-toggle-group__button:contains("All")')
- b.wait_text("button.pf-v5-c-select__menu-item:not(.pf-m-disabled)", container_name + ":nosearch-tag")
- b.wait_js_cond("document.getElementsByClassName('pf-v5-c-select__menu-item').length === 1")
- b.click(f"button.pf-v5-c-select__menu-item:contains({container_name + ":nosearch-tag"})")
+ b.wait_text("button.pf-v5-c-menu__item:not([disabled])", container_name + ":nosearch-tag")
+ b.wait_js_cond("document.querySelectorAll('.pf-v5-c-menu__item:not([disabled])').length === 1")
+ b.click(f"button.pf-v5-c-menu__item:contains({container_name + ":nosearch-tag"})")
b.click("#create-image-create-btn")
# Wait for download to finish
@@ -3178,9 +3180,9 @@ class TestApplication(testlib.MachineCase):
# Check that manifest search with tag also works on searchable repository
b.click("#containers-containers button.pf-v5-c-button.pf-m-primary")
- b.set_input_text("#create-image-image-select-typeahead", "localhost:5000/my-busybox:search-tag")
- b.wait_text("button.pf-v5-c-select__menu-item:not(.pf-m-disabled)", "localhost:5000/my-busybox:search-tag")
- b.wait_js_cond("document.getElementsByClassName('pf-v5-c-select__menu-item').length === 1")
+ b.set_input_text("#create-image-image input", "localhost:5000/my-busybox:search-tag")
+ b.wait_text("button.pf-v5-c-menu__item:not([disabled])", "localhost:5000/my-busybox:search-tag")
+ b.wait_js_cond("document.querySelectorAll('.pf-v5-c-menu__item:not([disabled])').length === 1")
if __name__ == '__main__':