1
0
Fork 0
mirror of https://github.com/plankanban/planka.git synced 2025-07-18 20:59:44 +02:00

fix: Add icons to popup menu items

This commit is contained in:
Maksim Eltyshev 2025-07-17 15:23:53 +02:00
parent adf60c0c74
commit e659ed4a2d
13 changed files with 81 additions and 21 deletions

View file

@ -7,7 +7,7 @@ import React, { useCallback } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { useDispatch } from 'react-redux'; import { useDispatch } from 'react-redux';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Menu } from 'semantic-ui-react'; import { Icon, Menu } from 'semantic-ui-react';
import { FilePicker, Popup } from '../../../lib/custom-ui'; import { FilePicker, Popup } from '../../../lib/custom-ui';
import entryActions from '../../../entry-actions'; import entryActions from '../../../entry-actions';
@ -47,6 +47,7 @@ const AddAttachmentStep = React.memo(({ onClose }) => {
<Menu secondary vertical className={styles.menu}> <Menu secondary vertical className={styles.menu}>
<FilePicker multiple onSelect={handleFilesSelect}> <FilePicker multiple onSelect={handleFilesSelect}>
<Menu.Item className={styles.menuItem}> <Menu.Item className={styles.menuItem}>
<Icon name="computer" className={styles.menuItemIcon} />
{t('common.fromComputer', { {t('common.fromComputer', {
context: 'title', context: 'title',
})} })}

View file

@ -21,6 +21,11 @@
padding-left: 14px; padding-left: 14px;
} }
.menuItemIcon {
float: left;
margin: 0 0.5em 0 0;
}
.tip { .tip {
opacity: 0.5; opacity: 0.5;
} }

View file

@ -23,6 +23,6 @@
.menuItemIcon { .menuItemIcon {
float: left; float: left;
margin-right: 0.5em; margin: 0 0.5em 0 0;
} }
} }

View file

@ -7,7 +7,7 @@ import React, { useCallback, useMemo } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { shallowEqual, useDispatch, useSelector } from 'react-redux'; import { shallowEqual, useDispatch, useSelector } from 'react-redux';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Menu } from 'semantic-ui-react'; import { Icon, Menu } from 'semantic-ui-react';
import { Popup } from '../../../lib/custom-ui'; import { Popup } from '../../../lib/custom-ui';
import selectors from '../../../selectors'; import selectors from '../../../selectors';
@ -290,6 +290,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
<Menu secondary vertical className={styles.menu}> <Menu secondary vertical className={styles.menu}>
{canEditName && ( {canEditName && (
<Menu.Item className={styles.menuItem} onClick={handleEditNameClick}> <Menu.Item className={styles.menuItem} onClick={handleEditNameClick}>
<Icon name="edit outline" className={styles.menuItemIcon} />
{t('action.editTitle', { {t('action.editTitle', {
context: 'title', context: 'title',
})} })}
@ -297,6 +298,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{!board.limitCardTypesToDefaultOne && canEditType && ( {!board.limitCardTypesToDefaultOne && canEditType && (
<Menu.Item className={styles.menuItem} onClick={handleEditTypeClick}> <Menu.Item className={styles.menuItem} onClick={handleEditTypeClick}>
<Icon name="map outline" className={styles.menuItemIcon} />
{t('action.editType', { {t('action.editType', {
context: 'title', context: 'title',
})} })}
@ -304,6 +306,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{card.type === CardTypes.PROJECT && canUseMembers && ( {card.type === CardTypes.PROJECT && canUseMembers && (
<Menu.Item className={styles.menuItem} onClick={handleUsersClick}> <Menu.Item className={styles.menuItem} onClick={handleUsersClick}>
<Icon name="user outline" className={styles.menuItemIcon} />
{t('common.members', { {t('common.members', {
context: 'title', context: 'title',
})} })}
@ -311,6 +314,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{canUseLabels && ( {canUseLabels && (
<Menu.Item className={styles.menuItem} onClick={handleLabelsClick}> <Menu.Item className={styles.menuItem} onClick={handleLabelsClick}>
<Icon name="bookmark outline" className={styles.menuItemIcon} />
{t('common.labels', { {t('common.labels', {
context: 'title', context: 'title',
})} })}
@ -318,6 +322,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{card.type === CardTypes.STORY && canUseMembers && ( {card.type === CardTypes.STORY && canUseMembers && (
<Menu.Item className={styles.menuItem} onClick={handleUsersClick}> <Menu.Item className={styles.menuItem} onClick={handleUsersClick}>
<Icon name="user outline" className={styles.menuItemIcon} />
{t('common.members', { {t('common.members', {
context: 'title', context: 'title',
})} })}
@ -325,6 +330,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{card.type === CardTypes.PROJECT && canEditDueDate && ( {card.type === CardTypes.PROJECT && canEditDueDate && (
<Menu.Item className={styles.menuItem} onClick={handleEditDueDateClick}> <Menu.Item className={styles.menuItem} onClick={handleEditDueDateClick}>
<Icon name="calendar check outline" className={styles.menuItemIcon} />
{t('action.editDueDate', { {t('action.editDueDate', {
context: 'title', context: 'title',
})} })}
@ -332,6 +338,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{card.type === CardTypes.PROJECT && canEditStopwatch && ( {card.type === CardTypes.PROJECT && canEditStopwatch && (
<Menu.Item className={styles.menuItem} onClick={handleEditStopwatchClick}> <Menu.Item className={styles.menuItem} onClick={handleEditStopwatchClick}>
<Icon name="clock outline" className={styles.menuItemIcon} />
{t('action.editStopwatch', { {t('action.editStopwatch', {
context: 'title', context: 'title',
})} })}
@ -339,6 +346,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{canDuplicate && ( {canDuplicate && (
<Menu.Item className={styles.menuItem} onClick={handleDuplicateClick}> <Menu.Item className={styles.menuItem} onClick={handleDuplicateClick}>
<Icon name="copy outline" className={styles.menuItemIcon} />
{t('action.duplicateCard', { {t('action.duplicateCard', {
context: 'title', context: 'title',
})} })}
@ -346,6 +354,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{canMove && ( {canMove && (
<Menu.Item className={styles.menuItem} onClick={handleMoveClick}> <Menu.Item className={styles.menuItem} onClick={handleMoveClick}>
<Icon name="share square outline" className={styles.menuItemIcon} />
{t('action.moveCard', { {t('action.moveCard', {
context: 'title', context: 'title',
})} })}
@ -353,6 +362,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{prevList && canRestore && ( {prevList && canRestore && (
<Menu.Item className={styles.menuItem} onClick={handleRestoreClick}> <Menu.Item className={styles.menuItem} onClick={handleRestoreClick}>
<Icon name="undo alternate" className={styles.menuItemIcon} />
{t('action.restoreToList', { {t('action.restoreToList', {
context: 'title', context: 'title',
list: prevList.name || t(`common.${prevList.type}`), list: prevList.name || t(`common.${prevList.type}`),
@ -361,6 +371,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{list.type !== ListTypes.ARCHIVE && canArchive && ( {list.type !== ListTypes.ARCHIVE && canArchive && (
<Menu.Item className={styles.menuItem} onClick={handleArchiveClick}> <Menu.Item className={styles.menuItem} onClick={handleArchiveClick}>
<Icon name="folder open outline" className={styles.menuItemIcon} />
{t('action.archiveCard', { {t('action.archiveCard', {
context: 'title', context: 'title',
})} })}
@ -368,6 +379,7 @@ const ActionsStep = React.memo(({ cardId, onNameEdit, onClose }) => {
)} )}
{canDelete && ( {canDelete && (
<Menu.Item className={styles.menuItem} onClick={handleDeleteClick}> <Menu.Item className={styles.menuItem} onClick={handleDeleteClick}>
<Icon name="trash alternate outline" className={styles.menuItemIcon} />
{isInTrashList {isInTrashList
? t('action.deleteForever', { ? t('action.deleteForever', {
context: 'title', context: 'title',

View file

@ -13,4 +13,9 @@
margin: 0; margin: 0;
padding-left: 14px; padding-left: 14px;
} }
.menuItemIcon {
float: left;
margin: 0 0.5em 0 0;
}
} }

View file

@ -7,7 +7,7 @@ import React, { useCallback, useMemo } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { useDispatch, useSelector } from 'react-redux'; import { useDispatch, useSelector } from 'react-redux';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Menu } from 'semantic-ui-react'; import { Icon, Menu } from 'semantic-ui-react';
import { Popup } from '../../../../lib/custom-ui'; import { Popup } from '../../../../lib/custom-ui';
import selectors from '../../../../selectors'; import selectors from '../../../../selectors';
@ -180,12 +180,14 @@ const ActionsStep = React.memo(({ userId, onClose }) => {
<Popup.Content> <Popup.Content>
<Menu secondary vertical className={styles.menu}> <Menu secondary vertical className={styles.menu}>
<Menu.Item className={styles.menuItem} onClick={handleEditInformationClick}> <Menu.Item className={styles.menuItem} onClick={handleEditInformationClick}>
<Icon name="info" className={styles.menuItemIcon} />
{t('action.editInformation', { {t('action.editInformation', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
{!user.lockedFieldNames.includes('username') && ( {!user.lockedFieldNames.includes('username') && (
<Menu.Item className={styles.menuItem} onClick={handleEditUsernameClick}> <Menu.Item className={styles.menuItem} onClick={handleEditUsernameClick}>
<Icon name="at" className={styles.menuItemIcon} />
{t('action.editUsername', { {t('action.editUsername', {
context: 'title', context: 'title',
})} })}
@ -193,6 +195,7 @@ const ActionsStep = React.memo(({ userId, onClose }) => {
)} )}
{!user.lockedFieldNames.includes('email') && ( {!user.lockedFieldNames.includes('email') && (
<Menu.Item className={styles.menuItem} onClick={handleEditEmailClick}> <Menu.Item className={styles.menuItem} onClick={handleEditEmailClick}>
<Icon name="mail outline" className={styles.menuItemIcon} />
{t('action.editEmail', { {t('action.editEmail', {
context: 'title', context: 'title',
})} })}
@ -200,6 +203,7 @@ const ActionsStep = React.memo(({ userId, onClose }) => {
)} )}
{!user.lockedFieldNames.includes('password') && ( {!user.lockedFieldNames.includes('password') && (
<Menu.Item className={styles.menuItem} onClick={handleEditPasswordClick}> <Menu.Item className={styles.menuItem} onClick={handleEditPasswordClick}>
<Icon name="keyboard outline" className={styles.menuItemIcon} />
{t('action.editPassword', { {t('action.editPassword', {
context: 'title', context: 'title',
})} })}
@ -207,6 +211,7 @@ const ActionsStep = React.memo(({ userId, onClose }) => {
)} )}
{!user.lockedFieldNames.includes('role') && ( {!user.lockedFieldNames.includes('role') && (
<Menu.Item className={styles.menuItem} onClick={handleEditRoleClick}> <Menu.Item className={styles.menuItem} onClick={handleEditRoleClick}>
<Icon name="sun outline" className={styles.menuItemIcon} />
{t('action.editRole', { {t('action.editRole', {
context: 'title', context: 'title',
})} })}
@ -221,6 +226,7 @@ const ActionsStep = React.memo(({ userId, onClose }) => {
className={styles.menuItem} className={styles.menuItem}
onClick={user.isDeactivated ? handleActivateClick : handleDeactivateClick} onClick={user.isDeactivated ? handleActivateClick : handleDeactivateClick}
> >
<Icon name={user.isDeactivated ? 'plus' : 'close'} className={styles.menuItemIcon} />
{user.isDeactivated {user.isDeactivated
? t('action.activateUser', { ? t('action.activateUser', {
context: 'title', context: 'title',
@ -231,6 +237,7 @@ const ActionsStep = React.memo(({ userId, onClose }) => {
</Menu.Item> </Menu.Item>
{user.isDeactivated && !user.isDefaultAdmin && ( {user.isDeactivated && !user.isDefaultAdmin && (
<Menu.Item className={styles.menuItem} onClick={handleDeleteClick}> <Menu.Item className={styles.menuItem} onClick={handleDeleteClick}>
<Icon name="trash alternate outline" className={styles.menuItemIcon} />
{t('action.deleteUser', { {t('action.deleteUser', {
context: 'title', context: 'title',
})} })}

View file

@ -13,4 +13,9 @@
margin: 0; margin: 0;
padding-left: 14px; padding-left: 14px;
} }
.menuItemIcon {
float: left;
margin: 0 0.5em 0 0;
}
} }

View file

@ -7,7 +7,7 @@ import React, { useCallback, useMemo } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { useDispatch, useSelector } from 'react-redux'; import { useDispatch, useSelector } from 'react-redux';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Menu } from 'semantic-ui-react'; import { Icon, Menu } from 'semantic-ui-react';
import { Popup } from '../../../lib/custom-ui'; import { Popup } from '../../../lib/custom-ui';
import selectors from '../../../selectors'; import selectors from '../../../selectors';
@ -128,38 +128,45 @@ const ActionsStep = React.memo(({ listId, onNameEdit, onCardAdd, onClose }) => {
<Popup.Content> <Popup.Content>
<Menu secondary vertical className={styles.menu}> <Menu secondary vertical className={styles.menu}>
<Menu.Item className={styles.menuItem} onClick={handleEditNameClick}> <Menu.Item className={styles.menuItem} onClick={handleEditNameClick}>
<Icon name="edit outline" className={styles.menuItemIcon} />
{t('action.editTitle', { {t('action.editTitle', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
<Menu.Item className={styles.menuItem} onClick={handleEditTypeClick}> <Menu.Item className={styles.menuItem} onClick={handleEditTypeClick}>
<Icon name="map outline" className={styles.menuItemIcon} />
{t('action.editType', { {t('action.editType', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
<Menu.Item className={styles.menuItem} onClick={handleEditColorClick}> <Menu.Item className={styles.menuItem} onClick={handleEditColorClick}>
<Icon name="dot circle outline" className={styles.menuItemIcon} />
{t('action.editColor', { {t('action.editColor', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
<Menu.Item className={styles.menuItem} onClick={handleAddCardClick}> <Menu.Item className={styles.menuItem} onClick={handleAddCardClick}>
<Icon name="list alternate outline" className={styles.menuItemIcon} />
{t('action.addCard', { {t('action.addCard', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
<Menu.Item className={styles.menuItem} onClick={handleSortClick}> <Menu.Item className={styles.menuItem} onClick={handleSortClick}>
<Icon name="sort amount down" className={styles.menuItemIcon} />
{t('action.sortList', { {t('action.sortList', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
{list.type === ListTypes.CLOSED && ( {list.type === ListTypes.CLOSED && (
<Menu.Item className={styles.menuItem} onClick={handleArchiveCardsClick}> <Menu.Item className={styles.menuItem} onClick={handleArchiveCardsClick}>
<Icon name="folder open outline" className={styles.menuItemIcon} />
{t('action.archiveCards', { {t('action.archiveCards', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
)} )}
<Menu.Item className={styles.menuItem} onClick={handleDeleteClick}> <Menu.Item className={styles.menuItem} onClick={handleDeleteClick}>
<Icon name="trash alternate outline" className={styles.menuItemIcon} />
{t('action.deleteList', { {t('action.deleteList', {
context: 'title', context: 'title',
})} })}

View file

@ -13,4 +13,9 @@
margin: 0; margin: 0;
padding-left: 14px; padding-left: 14px;
} }
.menuItemIcon {
float: left;
margin: 0 0.5em 0 0;
}
} }

View file

@ -7,7 +7,7 @@ import React, { useCallback, useMemo } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { useDispatch, useSelector } from 'react-redux'; import { useDispatch, useSelector } from 'react-redux';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Menu } from 'semantic-ui-react'; import { Icon, Menu } from 'semantic-ui-react';
import { Popup } from '../../../../lib/custom-ui'; import { Popup } from '../../../../lib/custom-ui';
import selectors from '../../../../selectors'; import selectors from '../../../../selectors';
@ -66,12 +66,14 @@ const ActionsStep = React.memo(({ taskId, onNameEdit, onClose }) => {
<Menu secondary vertical className={styles.menu}> <Menu secondary vertical className={styles.menu}>
{!task.linkedCardId && ( {!task.linkedCardId && (
<Menu.Item className={styles.menuItem} onClick={handleEditNameClick}> <Menu.Item className={styles.menuItem} onClick={handleEditNameClick}>
<Icon name="align left" className={styles.menuItemIcon} />
{t('action.editDescription', { {t('action.editDescription', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
)} )}
<Menu.Item className={styles.menuItem} onClick={handleDeleteClick}> <Menu.Item className={styles.menuItem} onClick={handleDeleteClick}>
<Icon name="trash alternate outline" className={styles.menuItemIcon} />
{t('action.deleteTask', { {t('action.deleteTask', {
context: 'title', context: 'title',
})} })}

View file

@ -13,4 +13,9 @@
margin: 0; margin: 0;
padding-left: 14px; padding-left: 14px;
} }
.menuItemIcon {
float: left;
margin: 0 0.5em 0 0;
}
} }

View file

@ -7,7 +7,7 @@ import React, { useCallback } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { useDispatch, useSelector } from 'react-redux'; import { useDispatch, useSelector } from 'react-redux';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Button, Menu } from 'semantic-ui-react'; import { Button, Icon, Menu } from 'semantic-ui-react';
import { Popup } from '../../../lib/custom-ui'; import { Popup } from '../../../lib/custom-ui';
import selectors from '../../../selectors'; import selectors from '../../../selectors';
@ -31,15 +31,15 @@ const UserStep = React.memo(({ onClose }) => {
onClose(); onClose();
}, [onClose, dispatch]); }, [onClose, dispatch]);
const handleLogoutClick = useCallback(() => {
dispatch(entryActions.logout());
}, [dispatch]);
const handleAdministrationClick = useCallback(() => { const handleAdministrationClick = useCallback(() => {
dispatch(entryActions.openAdministrationModal()); dispatch(entryActions.openAdministrationModal());
onClose(); onClose();
}, [onClose, dispatch]); }, [onClose, dispatch]);
const handleLogoutClick = useCallback(() => {
dispatch(entryActions.logout());
}, [dispatch]);
let logoutMenuItemProps; let logoutMenuItemProps;
if (isLogouting) { if (isLogouting) {
logoutMenuItemProps = { logoutMenuItemProps = {
@ -61,29 +61,30 @@ const UserStep = React.memo(({ onClose }) => {
<Popup.Content> <Popup.Content>
<Menu secondary vertical className={styles.menu}> <Menu secondary vertical className={styles.menu}>
<Menu.Item className={styles.menuItem} onClick={handleSettingsClick}> <Menu.Item className={styles.menuItem} onClick={handleSettingsClick}>
<Icon name="user" className={styles.menuItemIcon} />
{t('common.settings', { {t('common.settings', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
{withAdministration && (
<Menu.Item className={styles.menuItem} onClick={handleAdministrationClick}>
<Icon name="setting" className={styles.menuItemIcon} />
{t('common.administration', {
context: 'title',
})}
</Menu.Item>
)}
<hr className={styles.divider} />
<Menu.Item <Menu.Item
{...logoutMenuItemProps} // eslint-disable-line react/jsx-props-no-spreading {...logoutMenuItemProps} // eslint-disable-line react/jsx-props-no-spreading
className={styles.menuItem} className={styles.menuItem}
onClick={handleLogoutClick} onClick={handleLogoutClick}
> >
<Icon name="log out" className={styles.menuItemIcon} />
{t('action.logOut', { {t('action.logOut', {
context: 'title', context: 'title',
})} })}
</Menu.Item> </Menu.Item>
{withAdministration && (
<>
<hr className={styles.divider} />
<Menu.Item className={styles.menuItem} onClick={handleAdministrationClick}>
{t('common.administration', {
context: 'title',
})}
</Menu.Item>
</>
)}
</Menu> </Menu>
</Popup.Content> </Popup.Content>
</> </>

View file

@ -20,4 +20,9 @@
margin: 0; margin: 0;
padding-left: 14px; padding-left: 14px;
} }
.menuItemIcon {
float: left;
margin: 0 0.5em 0 0;
}
} }