diff --git a/webpack/assets/javascripts/react_app/components/HostsIndex/ActionKebab.js b/webpack/assets/javascripts/react_app/components/HostsIndex/ActionKebab.js
index 0af1524b467..1c9fd8bbb7c 100644
--- a/webpack/assets/javascripts/react_app/components/HostsIndex/ActionKebab.js
+++ b/webpack/assets/javascripts/react_app/components/HostsIndex/ActionKebab.js
@@ -1,6 +1,7 @@
-import React, { useState } from 'react';
+import React from 'react';
import PropTypes from 'prop-types';
-import { Dropdown, KebabToggle } from '@patternfly/react-core';
+import { Menu, MenuToggle, Popper } from '@patternfly/react-core';
+import { EllipsisVIcon } from '@patternfly/react-icons';
/**
* Generate a button or a dropdown of buttons
@@ -8,31 +9,48 @@ import { Dropdown, KebabToggle } from '@patternfly/react-core';
* @param {Object} action action to preform when the button is click can be href with data-method or Onclick
* @return {Function} button component or splitbutton component
*/
-export const ActionKebab = ({ items }) => {
- const [isOpen, setIsOpen] = useState(false);
+export const ActionKebab = ({ items, menuOpen, setMenuOpen }) => {
+ const containerRef = React.useRef();
if (!items.length) return null;
+ const menu = (
+
+ );
+
+ const menuToggle = (
+ setMenuOpen(prev => !prev)}
+ isExpanded={menuOpen}
+ >
+
+
+ );
+
return (
- <>
- {items.length > 0 && (
-
- }
- isOpen={isOpen}
- isPlain
- dropdownItems={items}
- />
- )}
- >
+
);
};
ActionKebab.propTypes = {
items: PropTypes.arrayOf(PropTypes.node),
+ menuOpen: PropTypes.bool.isRequired,
+ setMenuOpen: PropTypes.func.isRequired,
};
ActionKebab.defaultProps = {
diff --git a/webpack/assets/javascripts/react_app/components/HostsIndex/TableRowActions/core.js b/webpack/assets/javascripts/react_app/components/HostsIndex/TableRowActions/core.js
new file mode 100644
index 00000000000..8085c282c62
--- /dev/null
+++ b/webpack/assets/javascripts/react_app/components/HostsIndex/TableRowActions/core.js
@@ -0,0 +1,45 @@
+import forceSingleton from '../../../common/forceSingleton';
+
+const coreTableRowActionsRegistry = forceSingleton(
+ 'coreTableRowActionsRegistry',
+ () => ({})
+);
+
+// Unlike the column registry which is collecting objects that describe table columns,
+// here we collect an object containing a single getActions funtion, which returns an array of kebab action items.
+export const registerGetActions = ({
+ pluginName,
+ getActionsFunc,
+ tableName = 'hosts',
+}) => {
+ if (!coreTableRowActionsRegistry[pluginName])
+ coreTableRowActionsRegistry[pluginName] = {};
+ coreTableRowActionsRegistry[pluginName][tableName] = {
+ getActions: getActionsFunc,
+ };
+};
+
+export const registeredTableRowActions = ({ tableName = 'hosts' }) => {
+ const result = {};
+ Object.keys(coreTableRowActionsRegistry).forEach(pluginName => {
+ if (coreTableRowActionsRegistry[pluginName]?.[tableName]) {
+ result[pluginName] = coreTableRowActionsRegistry[pluginName][tableName];
+ }
+ });
+ // { katello: { getActions: [Function: getActions] } }
+ return result;
+};
+
+export const getActions = (hostDetailsResult, { tableName = 'hosts' } = {}) => {
+ const result = [];
+ const allGetActionsFuncs = registeredTableRowActions({ tableName });
+ Object.values(allGetActionsFuncs).forEach(
+ ({ getActions: getActionsFunc }) => {
+ if (typeof getActionsFunc !== 'function') return;
+ result.push(...getActionsFunc(hostDetailsResult));
+ }
+ );
+ return result;
+};
+
+export default getActions;
diff --git a/webpack/assets/javascripts/react_app/components/HostsIndex/index.js b/webpack/assets/javascripts/react_app/components/HostsIndex/index.js
index ab45dbda913..ea721b2c8d8 100644
--- a/webpack/assets/javascripts/react_app/components/HostsIndex/index.js
+++ b/webpack/assets/javascripts/react_app/components/HostsIndex/index.js
@@ -4,8 +4,10 @@ import { useSelector, useDispatch, shallowEqual } from 'react-redux';
import { Tr, Td, ActionsColumn } from '@patternfly/react-table';
import {
ToolbarItem,
+ Divider,
Dropdown,
DropdownItem,
+ MenuItem,
KebabToggle,
Flex,
FlexItem,
@@ -52,6 +54,7 @@ import getColumnData from './Columns/core';
import { categoriesFromFrontendColumnData } from '../ColumnSelector/helpers';
import ColumnSelector from '../ColumnSelector';
import { ForemanActionsBarContext } from '../HostDetails/ActionsBar';
+import { registerGetActions, getActions } from './TableRowActions/core';
export const ForemanHostsIndexActionsBarContext = forceSingleton(
'ForemanHostsIndexActionsBarContext',
@@ -59,6 +62,7 @@ export const ForemanHostsIndexActionsBarContext = forceSingleton(
);
const HostsIndex = () => {
+ const [menuOpen, setMenuOpen] = useState(false);
const [allColumns, setAllColumns] = useState(
getColumnData({ tableName: 'hosts' })
);
@@ -205,48 +209,75 @@ const HostsIndex = () => {
});
const dropdownItems = [
-
- {__('Delete')}
- ,
-
{__('Build management')}
- ,
- ,
+ ,
+ ,
+ ];
+
+ const dangerZoneItems = [
+ ,
+ ,
];
const registeredItems = useSelector(selectKebabItems, shallowEqual);
const pluginToolbarItems = jsReady => (
-
+
{jsReady && }
);
- const rowKebabItems = ({
+ const coreRowKebabItems = ({
id,
name: hostName,
compute_id: computeId,
can_delete: canDelete,
+ can_edit: canEdit,
}) => [
+ {
+ title: __('Edit'),
+ onClick: () => {
+ window.location.href = foremanUrl(`/hosts/${id}/edit`);
+ },
+ isDisabled: !canEdit,
+ },
+ {
+ title: __('Clone'),
+ onClick: () => {
+ window.location.href = foremanUrl(`/hosts/${id}/clone`);
+ },
+ isDisabled: !canEdit,
+ },
{
title: __('Delete'),
onClick: () => deleteHostHandler({ id, hostName, computeId }),
@@ -254,6 +285,11 @@ const HostsIndex = () => {
},
];
+ registerGetActions({
+ pluginName: 'core',
+ getActionsFunc: coreRowKebabItems,
+ });
+
const [legacyUIKebabOpen, setLegacyUIKebabOpen] = useState(false);
const legacyUIKebab = (
{
ouiaId="hosts-index-table"
params={params}
setParams={setParamsAndAPI}
- getActions={rowKebabItems}
+ getActions={getActions}
itemCount={subtotal}
results={results}
url={HOSTS_API_PATH}
@@ -364,7 +400,7 @@ const HostsIndex = () => {
isPending={status === STATUS.PENDING}
>
{results?.map((result, rowIndex) => {
- const rowActions = rowKebabItems(result);
+ const rowActions = getActions(result);
return (
{}
diff --git a/webpack/assets/javascripts/react_app/components/PF4/TableIndexPage/Table/Table.js b/webpack/assets/javascripts/react_app/components/PF4/TableIndexPage/Table/Table.js
index f93621b9808..da311e02960 100644
--- a/webpack/assets/javascripts/react_app/components/PF4/TableIndexPage/Table/Table.js
+++ b/webpack/assets/javascripts/react_app/components/PF4/TableIndexPage/Table/Table.js
@@ -76,14 +76,22 @@ export const Table = ({
setSelectedItem({ id, name });
setDeleteModalOpen(true);
};
- const actions = ({ can_delete: canDelete, id, name, ...item }) =>
+ const actions = ({
+ can_delete: canDelete,
+ can_edit: canEdit,
+ id,
+ name,
+ ...item
+ }) =>
[
isDeleteable && {
title: __('Delete'),
onClick: () => onDeleteClick({ id, name }),
isDisabled: !canDelete,
},
- ...((getActions && getActions({ id, name, canDelete, ...item })) ?? []),
+ ...((getActions &&
+ getActions({ id, name, canDelete, canEdit, ...item })) ??
+ []),
].filter(Boolean);
const RowSelectTd = rowSelectTd;
return (