UI - МЕЖПРОЕКТНАЯ КОММУНИКАЦИЯ: приведение delete-модалок к black-glass канону

This commit is contained in:
DCCONSTRUCTIONS 2026-04-20 18:48:47 +03:00
parent cbd40791e4
commit 6cb0545957
20 changed files with 239 additions and 158 deletions

View File

@ -49,8 +49,8 @@ export const CycleDeleteModal = observer(function CycleDeleteModal(props: ICycle
if (cycleId || peekCycle) router.push(`/${workspaceSlug}/projects/${projectId}/cycles`);
setToast({
type: TOAST_TYPE.SUCCESS,
title: "Success!",
message: "Cycle deleted successfully.",
title: t("common.success"),
message: t("entity.delete.success", { entity: t("common.cycle").toLowerCase() }),
});
})
.catch((errors) => {
@ -68,8 +68,8 @@ export const CycleDeleteModal = observer(function CycleDeleteModal(props: ICycle
} catch {
setToast({
type: TOAST_TYPE.ERROR,
title: "Warning!",
message: "Something went wrong please try again later.",
title: t("common.warning"),
message: t("common.something_went_wrong"),
});
}
@ -82,14 +82,11 @@ export const CycleDeleteModal = observer(function CycleDeleteModal(props: ICycle
handleSubmit={formSubmit}
isSubmitting={loader}
isOpen={isOpen}
title="Delete cycle"
content={
<>
Are you sure you want to delete cycle{' "'}
<span className="font-medium break-words text-primary">{cycle?.name}</span>
{'"'}? All of the data related to the cycle will be permanently removed. This action cannot be undone.
</>
}
title={t("entity.delete.label", { entity: t("common.cycle") })}
content={t("entity.delete.confirmation", {
entity: t("common.cycle").toLowerCase(),
identifier: cycle?.name ? `"${cycle.name}"` : "",
})}
/>
);
});

View File

@ -7,9 +7,9 @@
import { useState } from "react";
import { observer } from "mobx-react";
// ui
import { Button } from "@plane/propel/button";
import { useTranslation } from "@plane/i18n";
import { TOAST_TYPE, setToast } from "@plane/propel/toast";
import { EModalPosition, EModalWidth, ModalCore } from "@plane/ui";
import { AlertModalCore } from "@plane/ui";
// hooks
import { useProjectEstimates } from "@/hooks/store/estimates";
import { useEstimate } from "@/hooks/store/estimates/use-estimate";
@ -26,6 +26,7 @@ type TDeleteEstimateModal = {
export const DeleteEstimateModal = observer(function DeleteEstimateModal(props: TDeleteEstimateModal) {
// props
const { workspaceSlug, projectId, estimateId, isOpen, handleClose } = props;
const { t } = useTranslation();
// hooks
const { areEstimateEnabledByProjectId, deleteEstimate } = useProjectEstimates();
const { asJson: estimate } = useEstimate(estimateId);
@ -44,46 +45,32 @@ export const DeleteEstimateModal = observer(function DeleteEstimateModal(props:
setButtonLoader(false);
setToast({
type: TOAST_TYPE.SUCCESS,
title: "Estimate deleted",
message: "Estimate has been removed from your project.",
title: t("project_settings.estimates.delete_modal.success_title"),
message: t("project_settings.estimates.delete_modal.success_message"),
});
handleClose();
} catch (_error) {
setButtonLoader(false);
setToast({
type: TOAST_TYPE.ERROR,
title: "Estimate creation failed",
message: "We were unable to delete the estimate, please try again.",
title: t("project_settings.estimates.delete_modal.error_title"),
message: t("project_settings.estimates.delete_modal.error_message"),
});
}
};
return (
<ModalCore isOpen={isOpen} position={EModalPosition.TOP} width={EModalWidth.XXL}>
<div className="relative space-y-6 py-5">
{/* heading */}
<div className="relative flex items-center justify-between gap-2 px-5">
<div className="text-18 font-medium text-primary">Delete Estimate System</div>
</div>
{/* estimate steps */}
<div className="px-5">
<div className="text-14 text-secondary">
Deleting the estimate <span className="font-bold text-primary">{estimate?.name}</span>
&nbsp;system will remove it from all work items permanently. This action cannot be undone. If you add
estimates again, you will need to update all the work items.
</div>
</div>
<div className="relative flex items-center justify-end gap-3 border-t border-subtle px-5 pt-5">
<Button variant="secondary" size="lg" onClick={handleClose} disabled={buttonLoader}>
Cancel
</Button>
<Button variant="error-fill" size="lg" onClick={handleDeleteEstimate} disabled={buttonLoader}>
{buttonLoader ? "Deleting" : "Delete Estimate"}
</Button>
</div>
</div>
</ModalCore>
<AlertModalCore
handleClose={handleClose}
handleSubmit={handleDeleteEstimate}
isSubmitting={buttonLoader}
isOpen={isOpen}
title={t("project_settings.estimates.delete_modal.title")}
content={t("project_settings.estimates.delete_modal.description", { value: estimate?.name ?? "" })}
primaryButtonText={{
loading: t("project_settings.estimates.delete_modal.loading"),
default: t("project_settings.estimates.delete_modal.submit"),
}}
/>
);
});

View File

@ -47,16 +47,9 @@ export function DeclineIssueModal(props: Props) {
isSubmitting={isDeclining}
isOpen={isOpen}
title={t("inbox_issue.modals.decline.title")}
// TODO: Need to translate the confirmation message
content={
<>
Are you sure you want to decline work item{" "}
<span className="font-medium break-words text-primary">
{projectDetails?.identifier}-{data?.sequence_id}
</span>
{""}? This action cannot be undone.
</>
}
content={t("inbox_issue.modals.decline.content", {
value: `${projectDetails?.identifier}-${data?.sequence_id}`,
})}
primaryButtonText={{
loading: t("declining"),
default: t("decline"),

View File

@ -63,14 +63,7 @@ export const IssueAttachmentDeleteModal = observer(function IssueAttachmentDelet
isSubmitting={loader}
isOpen={isOpen}
title={t("attachment.delete")}
content={
<>
{/* TODO: Translate here */}
Are you sure you want to delete attachment-{" "}
<span className="font-bold">{getFileName(attachment.attributes.name)}</span>? This attachment will be
permanently removed. This action cannot be undone.
</>
}
content={t("attachment.delete_confirmation", { value: getFileName(attachment.attributes.name) })}
/>
);
});

View File

@ -8,6 +8,7 @@ import { useState } from "react";
import { observer } from "mobx-react";
import { useParams } from "next/navigation";
// types
import { useTranslation } from "@plane/i18n";
import { TOAST_TYPE, setToast } from "@plane/propel/toast";
import type { IIssueLabel } from "@plane/types";
// ui
@ -23,6 +24,7 @@ type Props = {
export const DeleteLabelModal = observer(function DeleteLabelModal(props: Props) {
const { isOpen, onClose, data } = props;
const { t } = useTranslation();
// router
const { workspaceSlug, projectId } = useParams();
// store hooks
@ -46,10 +48,10 @@ export const DeleteLabelModal = observer(function DeleteLabelModal(props: Props)
})
.catch((err) => {
setIsDeleteLoading(false);
const error = err?.error || "Label could not be deleted. Please try again.";
const error = err?.error || t("label.delete_error");
setToast({
type: TOAST_TYPE.ERROR,
title: "Error!",
title: t("common.error.label"),
message: error,
});
});
@ -61,13 +63,8 @@ export const DeleteLabelModal = observer(function DeleteLabelModal(props: Props)
handleSubmit={handleDeletion}
isSubmitting={isDeleteLoading}
isOpen={isOpen}
title="Delete Label"
content={
<>
Are you sure you want to delete <span className="font-medium text-primary">{data?.name}</span>? This will
remove the label from all the work item and from any views where the label is being filtered upon.
</>
}
title={t("entity.delete.label", { entity: t("common.label") })}
content={t("label.delete_confirmation", { value: data?.name ?? "" })}
/>
);
});

View File

@ -52,8 +52,8 @@ export const DeleteModuleModal = observer(function DeleteModuleModal(props: Prop
handleClose();
setToast({
type: TOAST_TYPE.SUCCESS,
title: "Success!",
message: "Module deleted successfully.",
title: t("common.success"),
message: t("entity.delete.success", { entity: t("common.module").toLowerCase() }),
});
})
.catch((errors) => {
@ -76,14 +76,11 @@ export const DeleteModuleModal = observer(function DeleteModuleModal(props: Prop
handleSubmit={handleDeletion}
isSubmitting={isDeleteLoading}
isOpen={isOpen}
title="Delete module"
content={
<>
Are you sure you want to delete module-{" "}
<span className="font-medium break-all text-primary">{data?.name}</span>? All of the data related to the
module will be permanently removed. This action cannot be undone.
</>
}
title={t("entity.delete.label", { entity: t("common.module") })}
content={t("entity.delete.confirmation", {
entity: t("common.module").toLowerCase(),
identifier: data?.name ? `"${data.name}"` : "",
})}
/>
);
});

View File

@ -8,6 +8,7 @@ import { useState } from "react";
import { observer } from "mobx-react";
// ui
import { useParams } from "next/navigation";
import { useTranslation } from "@plane/i18n";
import { TOAST_TYPE, setToast } from "@plane/propel/toast";
import { AlertModalCore } from "@plane/ui";
import { getPageName } from "@plane/utils";
@ -28,6 +29,7 @@ type TConfirmPageDeletionProps = {
export const DeletePageModal = observer(function DeletePageModal(props: TConfirmPageDeletionProps) {
const { isOpen, onClose, page, storeType } = props;
const { t } = useTranslation();
// states
const [isDeleting, setIsDeleting] = useState(false);
// store hooks
@ -52,8 +54,8 @@ export const DeletePageModal = observer(function DeletePageModal(props: TConfirm
handleClose();
setToast({
type: TOAST_TYPE.SUCCESS,
title: "Success!",
message: "Page deleted successfully.",
title: t("project_page.delete_modal.success_title"),
message: t("project_page.delete_modal.success_message"),
});
if (routePageId) {
@ -63,8 +65,8 @@ export const DeletePageModal = observer(function DeletePageModal(props: TConfirm
.catch(() => {
setToast({
type: TOAST_TYPE.ERROR,
title: "Error!",
message: "Page could not be deleted. Please try again.",
title: t("project_page.delete_modal.error_title"),
message: t("project_page.delete_modal.error_message"),
});
});
@ -79,14 +81,8 @@ export const DeletePageModal = observer(function DeletePageModal(props: TConfirm
handleSubmit={handleDelete}
isSubmitting={isDeleting}
isOpen={isOpen}
title="Delete page"
content={
<>
Are you sure you want to delete page-{" "}
<span className="font-medium break-words break-all text-primary">{getPageName(name)}</span> ? The Page will be
deleted permanently. This action cannot be undone.
</>
}
title={t("project_page.delete_modal.title")}
content={t("project_page.delete_modal.content", { value: getPageName(name) })}
/>
);
});

View File

@ -7,6 +7,7 @@
import { useState } from "react";
import { observer } from "mobx-react";
import { Loader } from "lucide-react";
import { useTranslation } from "@plane/i18n";
import { CloseIcon } from "@plane/propel/icons";
// plane imports
import { TOAST_TYPE, setToast } from "@plane/propel/toast";
@ -26,6 +27,7 @@ type TStateDelete = {
export const StateDelete = observer(function StateDelete(props: TStateDelete) {
const { totalStates, state, deleteStateCallback } = props;
const { t } = useTranslation();
// hooks
const { isMobile } = usePlatformOS();
// states
@ -47,15 +49,14 @@ export const StateDelete = observer(function StateDelete(props: TStateDelete) {
if (errorStatus.status === 400) {
setToast({
type: TOAST_TYPE.ERROR,
title: "Error!",
message:
"This state contains some work items within it, please move them to some other state to delete this state.",
title: t("common.error.label"),
message: t("project_settings.states.delete.blocked"),
});
} else {
setToast({
type: TOAST_TYPE.ERROR,
title: "Error!",
message: "State could not be deleted. Please try again.",
title: t("common.error.label"),
message: t("project_settings.states.delete.error"),
});
}
setIsDelete(false);
@ -69,13 +70,11 @@ export const StateDelete = observer(function StateDelete(props: TStateDelete) {
handleSubmit={handleDeleteState}
isSubmitting={isDelete}
isOpen={isDeleteModal}
title="Delete State"
content={
<>
Are you sure you want to delete state- <span className="font-medium text-primary">{state?.name}</span>? All
of the data related to the state will be permanently removed. This action cannot be undone.
</>
}
title={t("entity.delete.label", { entity: t("common.state") })}
content={t("entity.delete.confirmation", {
entity: t("common.state").toLowerCase(),
identifier: state?.name ? `"${state.name}"` : "",
})}
/>
<button
@ -89,7 +88,11 @@ export const StateDelete = observer(function StateDelete(props: TStateDelete) {
>
<Tooltip
tooltipContent={
state.default ? "Cannot delete the default state." : totalStates === 1 ? `Cannot have an empty group.` : ``
state.default
? t("project_settings.states.delete.disabled_default")
: totalStates === 1
? t("project_settings.states.delete.disabled_last")
: ""
}
isMobile={isMobile}
disabled={!isDeleteDisabled}

View File

@ -8,6 +8,7 @@ import { useState } from "react";
import { observer } from "mobx-react";
import { useParams } from "next/navigation";
// Plane imports
import { useTranslation } from "@plane/i18n";
import { TOAST_TYPE, setToast } from "@plane/propel/toast";
import type { IState } from "@plane/types";
// ui
@ -23,6 +24,7 @@ type TStateDeleteModal = {
export const StateDeleteModal = observer(function StateDeleteModal(props: TStateDeleteModal) {
const { isOpen, onClose, data } = props;
const { t } = useTranslation();
// states
const [isDeleteLoading, setIsDeleteLoading] = useState(false);
// router
@ -47,15 +49,14 @@ export const StateDeleteModal = observer(function StateDeleteModal(props: TState
if (err.status === 400)
setToast({
type: TOAST_TYPE.ERROR,
title: "Error!",
message:
"This state contains some work items within it, please move them to some other state to delete this state.",
title: t("common.error.label"),
message: t("project_settings.states.delete.blocked"),
});
else
setToast({
type: TOAST_TYPE.ERROR,
title: "Error!",
message: "State could not be deleted. Please try again.",
title: t("common.error.label"),
message: t("project_settings.states.delete.error"),
});
})
.finally(() => {
@ -69,13 +70,11 @@ export const StateDeleteModal = observer(function StateDeleteModal(props: TState
handleSubmit={handleDeletion}
isSubmitting={isDeleteLoading}
isOpen={isOpen}
title="Delete State"
content={
<>
Are you sure you want to delete state- <span className="font-medium text-primary">{data?.name}</span>? All of
the data related to the state will be permanently removed. This action cannot be undone.
</>
}
title={t("entity.delete.label", { entity: t("common.state") })}
content={t("entity.delete.confirmation", {
entity: t("common.state").toLowerCase(),
identifier: data?.name ? `"${data.name}"` : "",
})}
/>
);
});

View File

@ -7,6 +7,7 @@
import { useState } from "react";
import { useParams } from "next/navigation";
// ui
import { useTranslation } from "@plane/i18n";
import { TOAST_TYPE, setToast } from "@plane/propel/toast";
import { AlertModalCore } from "@plane/ui";
// hooks
@ -20,6 +21,7 @@ interface IDeleteWebhook {
export function DeleteWebhookModal(props: IDeleteWebhook) {
const { isOpen, onClose } = props;
const { t } = useTranslation();
// states
const [isDeleting, setIsDeleting] = useState(false);
// router
@ -41,14 +43,14 @@ export function DeleteWebhookModal(props: IDeleteWebhook) {
router.replace(`/${workspaceSlug}/settings/webhooks/`);
setToast({
type: TOAST_TYPE.SUCCESS,
title: "Success!",
message: "Webhook deleted successfully.",
title: t("workspace_settings.settings.webhooks.toasts.removed.title"),
message: t("workspace_settings.settings.webhooks.toasts.removed.message"),
});
} catch (_error) {
setToast({
type: TOAST_TYPE.ERROR,
title: "Error!",
message: "Webhook could not be deleted. Please try again.",
title: t("workspace_settings.settings.webhooks.toasts.not_removed.title"),
message: t("workspace_settings.settings.webhooks.toasts.not_removed.message"),
});
}
setIsDeleting(false);
@ -60,13 +62,8 @@ export function DeleteWebhookModal(props: IDeleteWebhook) {
handleSubmit={handleDelete}
isSubmitting={isDeleting}
isOpen={isOpen}
title="Delete webhook"
content={
<>
Are you sure you want to delete this webhook? Future events will not be delivered to this webhook. This action
cannot be undone.
</>
}
title={t("workspace_settings.settings.webhooks.delete.title")}
content={t("workspace_settings.settings.webhooks.delete.description")}
/>
);
}

View File

@ -6,6 +6,7 @@
import { Disclosure, Transition } from "@headlessui/react";
import { WORKSPACE_SETTINGS_TRACKER_ELEMENTS } from "@plane/constants";
import { useTranslation } from "@plane/i18n";
import { Button } from "@plane/propel/button";
import { ChevronDownIcon, ChevronUpIcon } from "@plane/propel/icons";
@ -15,13 +16,14 @@ type Props = {
export function WebhookDeleteSection(props: Props) {
const { openDeleteModal } = props;
const { t } = useTranslation();
return (
<Disclosure as="div" className="border-t border-subtle">
{({ open }) => (
<div className="w-full">
<Disclosure.Button as="button" type="button" className="flex w-full items-center justify-between py-4">
<span className="text-16 tracking-tight">Danger zone</span>
<span className="text-16 tracking-tight">{t("workspace_settings.settings.webhooks.delete.title")}</span>
{open ? <ChevronUpIcon className="h-5 w-5" /> : <ChevronDownIcon className="h-5 w-5" />}
</Disclosure.Button>
@ -36,18 +38,16 @@ export function WebhookDeleteSection(props: Props) {
>
<Disclosure.Panel>
<div className="flex flex-col gap-8">
<span className="text-13 tracking-tight">
Once a webhook is deleted, it cannot be restored. Future events will no longer be delivered to this
webhook.
</span>
<span className="text-13 tracking-tight">{t("workspace_settings.settings.webhooks.delete.description")}</span>
<div>
<Button
variant="error-fill"
size="lg"
onClick={openDeleteModal}
className="nodedc-modal-danger-button"
data-ph-element={WORKSPACE_SETTINGS_TRACKER_ELEMENTS.WEBHOOK_DELETE_BUTTON}
>
Delete webhook
{t("workspace_settings.settings.webhooks.delete.title")}
</Button>
</div>
</div>

View File

@ -218,9 +218,16 @@
}
.nodedc-glass-modal {
background:
linear-gradient(180deg, rgba(255, 255, 255, 0.036) 0%, rgba(255, 255, 255, 0.012) 100%),
rgba(6, 6, 8, 0.9) !important;
border: 0 !important;
outline: none !important;
-webkit-backdrop-filter: blur(42px);
backdrop-filter: blur(42px);
box-shadow:
0 20px 56px rgba(0, 0, 0, 0.34),
0 4px 16px rgba(0, 0, 0, 0.18);
0 24px 64px rgba(0, 0, 0, 0.42),
0 8px 22px rgba(0, 0, 0, 0.24);
}
.nodedc-glass-surface {
@ -268,11 +275,24 @@
}
.nodedc-glass-modal button:focus-visible,
.nodedc-glass-modal [role="button"]:focus-visible {
.nodedc-glass-modal [role="button"]:focus-visible,
.nodedc-glass-modal :focus-visible {
outline: none !important;
box-shadow: none !important;
}
.nodedc-modal-alert-icon {
border: 0 !important;
outline: none !important;
box-shadow: none !important;
}
.nodedc-modal-alert-icon-danger,
.nodedc-modal-alert-icon-primary {
background: color-mix(in srgb, rgb(var(--nodedc-accent-rgb)) 18%, transparent) !important;
color: rgb(var(--nodedc-accent-rgb)) !important;
}
.nodedc-modal-field {
background:
linear-gradient(180deg, rgba(255, 255, 255, 0.03) 0%, rgba(255, 255, 255, 0.012) 100%),
@ -485,17 +505,19 @@
outline: none !important;
box-shadow: none !important;
border-radius: 1.25rem !important;
background: rgb(var(--nodedc-card-active-rgb)) !important;
background: rgb(var(--nodedc-accent-rgb)) !important;
color: #0b1117 !important;
padding-inline: 1.25rem !important;
}
.nodedc-modal-primary-button:hover {
background: color-mix(in srgb, rgb(var(--nodedc-card-active-rgb)) 82%, white) !important;
background: color-mix(in srgb, rgb(var(--nodedc-accent-rgb)) 82%, white) !important;
}
.nodedc-modal-primary-button,
.nodedc-modal-primary-button *,
.nodedc-modal-danger-button,
.nodedc-modal-danger-button *,
.nodedc-settings-primary-button,
.nodedc-settings-primary-button *,
.nodedc-settings-save-button,
@ -509,9 +531,34 @@
outline: none !important;
box-shadow: none !important;
border-radius: 1.25rem !important;
background: rgb(var(--nodedc-accent-rgb)) !important;
color: #0b1117 !important;
padding-inline: 1.25rem !important;
}
.nodedc-modal-danger-button:hover {
background: color-mix(in srgb, rgb(var(--nodedc-accent-rgb)) 82%, white) !important;
}
.nodedc-glass-modal button.bg-danger-primary,
.nodedc-glass-modal button.border-danger-strong {
border: 0 !important;
outline: none !important;
box-shadow: none !important;
background: rgb(var(--nodedc-accent-rgb)) !important;
color: #0b1117 !important;
}
.nodedc-glass-modal button.bg-danger-primary:hover,
.nodedc-glass-modal button.border-danger-strong:hover {
background: color-mix(in srgb, rgb(var(--nodedc-accent-rgb)) 82%, white) !important;
}
.nodedc-glass-modal button.bg-danger-primary *,
.nodedc-glass-modal button.border-danger-strong * {
color: #0b1117 !important;
}
.nodedc-modal-chip {
min-height: 2.5rem;
border: 0 !important;

View File

@ -1138,9 +1138,14 @@ export default {
file_size_limit: "File must be of {size}MB or less in size.",
drag_and_drop: "Drag and drop anywhere to upload",
delete: "Delete attachment",
delete_confirmation:
"Are you sure you want to delete attachment {value}? This attachment will be permanently removed and this action cannot be undone.",
},
label: {
select: "Add labels",
delete_confirmation:
'Are you sure you want to delete label "{value}"? It will be removed from all work items and from any views where this label is used.',
delete_error: "Label could not be deleted. Please try again.",
create: {
success: "Label created successfully",
failed: "Label creation failed",
@ -1205,7 +1210,7 @@ export default {
modals: {
decline: {
title: "Decline work item",
content: "Are you sure you want to decline work item {value}?",
content: "Are you sure you want to decline work item {value}? This action cannot be undone.",
},
delete: {
title: "Delete work item",
@ -1691,6 +1696,11 @@ export default {
description: "Automate notifications to external services when project events occur.",
title: "Webhooks",
add_webhook: "Add webhook",
delete: {
title: "Delete webhook",
description:
"Are you sure you want to delete this webhook? Future events will no longer be delivered to it. This action cannot be undone.",
},
modal: {
title: "Create webhook",
details: "Webhook details",
@ -1886,6 +1896,12 @@ export default {
heading: "States",
description: "Define and customize workflow states to track the progress of your work items.",
describe_this_state_for_your_members: "Describe this state for your members.",
delete: {
blocked: "This state still contains work items. Move them to another state before deleting it.",
error: "State could not be deleted. Please try again.",
disabled_default: "Cannot delete the default state.",
disabled_last: "Cannot leave the group without states.",
},
empty_state: {
title: "No states available for the {groupKey} group",
description: "Please create a new state",
@ -1914,6 +1930,17 @@ export default {
label: "Estimates",
title: "Enable estimates for my project",
enable_description: "They help you in communicating complexity and workload of the team.",
delete_modal: {
title: "Delete estimate system",
description:
'Deleting the estimate system "{value}" will permanently remove it from every work item in this project. If you enable estimates again later, you will need to configure work items again.',
submit: "Delete estimate system",
loading: "Deleting",
success_title: "Estimate system deleted",
success_message: "The estimate system has been removed from the project.",
error_title: "Estimate system delete failed",
error_message: "We were unable to delete the estimate system. Please try again.",
},
list_heading: "Estimate list",
archived_heading: "Archived estimates",
archived_description: "These are estimates from earlier project versions that are not currently in use. Read more",
@ -2304,6 +2331,14 @@ export default {
},
},
project_page: {
delete_modal: {
title: "Delete page",
content: 'Are you sure you want to delete page "{value}"? The page will be permanently removed and this action cannot be undone.',
success_title: "Page deleted",
success_message: "Page deleted successfully.",
error_title: "Page delete failed",
error_message: "Page could not be deleted. Please try again.",
},
empty_state: {
general: {
title:

View File

@ -1294,9 +1294,14 @@ export default {
file_size_limit: "Максимальный размер файла - {size} МБ",
drag_and_drop: "Перетащите файл для загрузки",
delete: "Удалить вложение",
delete_confirmation:
"Вы уверены, что хотите удалить вложение {value}? Вложение будет удалено без возможности восстановления.",
},
label: {
select: "Выбрать метку",
delete_confirmation:
'Вы уверены, что хотите удалить метку "{value}"? Она будет удалена из всех рабочих элементов и представлений, где используется.',
delete_error: "Не удалось удалить метку. Попробуйте снова.",
create: {
success: "Метка создана",
failed: "Ошибка создания метки",
@ -1361,7 +1366,7 @@ export default {
modals: {
decline: {
title: "Отклонить рабочий элемент",
content: "Вы уверены, что хотите отклонить рабочий элемент {value}?",
content: "Вы уверены, что хотите отклонить рабочий элемент {value}? Это действие нельзя отменить.",
},
delete: {
title: "Удалить рабочий элемент",
@ -1853,6 +1858,11 @@ export default {
description: "Автоматизируйте уведомления во внешние сервисы при событиях проекта.",
title: "Вебхуки",
add_webhook: "Добавить вебхук",
delete: {
title: "Удалить вебхук",
description:
"Вы уверены, что хотите удалить этот вебхук? События больше не будут отправляться в этот вебхук. Это действие нельзя отменить.",
},
modal: {
title: "Создать вебхук",
details: "Детали вебхука",
@ -2045,6 +2055,12 @@ export default {
heading: "Статусы",
description: "Определяйте и настраивайте статусы рабочего процесса для отслеживания прогресса рабочих элементов.",
describe_this_state_for_your_members: "Опишите этот статус для участников",
delete: {
blocked: "В этом статусе есть рабочие элементы. Переместите их в другой статус, чтобы удалить текущий.",
error: "Не удалось удалить статус. Попробуйте снова.",
disabled_default: "Нельзя удалить статус по умолчанию.",
disabled_last: "Нельзя оставить группу без статусов.",
},
empty_state: {
title: "Нет статусов для группы {groupKey}",
description: "Создайте новый статус",
@ -2073,6 +2089,17 @@ export default {
label: "Оценки",
title: "Включить оценки для моего проекта",
enable_description: "Они помогают вам в общении о сложности и рабочей нагрузке команды.",
delete_modal: {
title: "Удалить систему оценок",
description:
'Удаление системы оценок "{value}" безвозвратно уберет её из всех рабочих элементов проекта. Если вы включите оценки снова, рабочие элементы придется настраивать заново.',
submit: "Удалить систему оценок",
loading: "Удаление",
success_title: "Система оценок удалена",
success_message: "Система оценок удалена из проекта.",
error_title: "Не удалось удалить систему оценок",
error_message: "Мы не смогли удалить систему оценок. Попробуйте снова.",
},
list_heading: "Список оценок",
archived_heading: "Архивные оценки",
archived_description: "Это оценки из предыдущих версий проекта, которые сейчас не используются. Подробнее о них",
@ -2461,6 +2488,14 @@ export default {
},
},
project_page: {
delete_modal: {
title: "Удалить страницу",
content: 'Вы уверены, что хотите удалить страницу "{value}"? Страница будет удалена без возможности восстановления.',
success_title: "Страница удалена",
success_message: "Страница успешно удалена.",
error_title: "Не удалось удалить страницу",
error_message: "Не удалось удалить страницу. Попробуйте снова.",
},
empty_state: {
general: {
title: "Создавайте заметки, документы или базу знаний. Используйте Galileo, ИИ-помощник NODE.DC.",

View File

@ -41,9 +41,8 @@ export interface DialogTitleProps extends React.ComponentProps<typeof BaseDialog
}
// Constants
const OVERLAY_CLASSNAME = cn("fixed inset-0 z-90 bg-backdrop/70 backdrop-blur-sm");
const BASE_CLASSNAME =
"nodedc-glass-modal relative w-full rounded-[28px] border border-subtle/70 bg-surface-1/78 text-left shadow-[0_16px_48px_rgba(0,0,0,0.34)] backdrop-blur-2xl z-100";
const OVERLAY_CLASSNAME = cn("fixed inset-0 z-90 bg-black/60 backdrop-blur-md");
const BASE_CLASSNAME = "nodedc-glass-modal relative z-100 w-full rounded-[28px] text-left";
// Utility functions
const getPositionClassNames = (position: DialogPosition) =>

View File

@ -84,7 +84,7 @@ export function ModalPortal({
const positionClass = fullScreen ? "" : PORTAL_POSITION_CLASSES[position];
return cn(
"nodedc-glass-modal absolute top-0 h-full rounded-[28px] border border-subtle/70 bg-surface-1/78 shadow-[0_16px_48px_rgba(0,0,0,0.34)] backdrop-blur-2xl transition-transform duration-300 ease-out",
"nodedc-glass-modal absolute top-0 h-full rounded-[28px] transition-transform duration-300 ease-out",
widthClass,
positionClass,
contentClassName
@ -101,7 +101,7 @@ export function ModalPortal({
>
{showOverlay && (
<div
className={cn("absolute inset-0 bg-black/50 backdrop-blur-sm transition-colors duration-300", overlayClassName)}
className={cn("absolute inset-0 bg-black/60 backdrop-blur-md transition-colors duration-300", overlayClassName)}
onClick={handleOverlayClick}
aria-hidden="true"
/>

View File

@ -34,6 +34,7 @@
"@headlessui/react": "^1.7.3",
"@plane/constants": "workspace:*",
"@plane/hooks": "workspace:*",
"@plane/i18n": "workspace:*",
"@plane/propel": "workspace:*",
"@plane/types": "workspace:*",
"@plane/utils": "workspace:*",

View File

@ -8,6 +8,7 @@ import type { LucideIcon } from "lucide-react";
import { AlertTriangle, Info } from "lucide-react";
import React from "react";
// components
import { useTranslation } from "@plane/i18n";
import type { TButtonVariant } from "@plane/propel/button";
import { Button } from "@plane/propel/button";
import { cn } from "../utils";
@ -43,16 +44,17 @@ const VARIANT_ICONS: Record<TModalVariant, LucideIcon> = {
};
const BUTTON_VARIANTS: Record<TModalVariant, TButtonVariant> = {
danger: "error-fill",
danger: "primary",
primary: "primary",
};
const VARIANT_CLASSES: Record<TModalVariant, string> = {
danger: "bg-danger-subtle text-danger-primary",
primary: "bg-accent-primary/20 text-accent-primary",
danger: "nodedc-modal-alert-icon nodedc-modal-alert-icon-danger",
primary: "nodedc-modal-alert-icon nodedc-modal-alert-icon-primary",
};
export function AlertModalCore(props: Props) {
const { t } = useTranslation();
const {
content,
handleClose,
@ -62,10 +64,10 @@ export function AlertModalCore(props: Props) {
isOpen,
position = EModalPosition.CENTER,
primaryButtonText = {
loading: "Deleting",
default: "Delete",
loading: t("deleting"),
default: t("delete"),
},
secondaryButtonText = "Cancel",
secondaryButtonText = t("cancel"),
title,
variant = "danger",
width = EModalWidth.XL,
@ -94,11 +96,11 @@ export function AlertModalCore(props: Props) {
</span>
)}
<div className="text-center sm:text-left">
<h3 className="text-18 font-medium text-secondary">{title}</h3>
<h3 className="text-18 font-medium text-primary">{title}</h3>
<p className="mt-1 text-13 text-secondary">{content}</p>
</div>
</div>
<div className="flex flex-col-reverse gap-3 border-t border-subtle/70 px-6 py-4 sm:flex-row sm:justify-end">
<div className="flex flex-col-reverse gap-3 px-6 py-4 sm:flex-row sm:justify-end">
<Button variant="secondary" onClick={handleClose} className="nodedc-modal-secondary-button min-w-[8.25rem]">
{secondaryButtonText}
</Button>
@ -108,8 +110,8 @@ export function AlertModalCore(props: Props) {
onClick={handleSubmit}
loading={isSubmitting}
className={cn("min-w-[8.25rem]", {
"nodedc-modal-danger-button": variant === "danger",
"nodedc-modal-primary-button": variant === "primary",
"nodedc-modal-danger-button": variant === "danger",
})}
>
{isSubmitting ? primaryButtonText.loading : primaryButtonText.default}

View File

@ -41,7 +41,7 @@ export function ModalCore(props: Props) {
leaveFrom="opacity-100"
leaveTo="opacity-0"
>
<div className="fixed inset-0 bg-backdrop/70 backdrop-blur-sm transition-opacity" />
<div className="fixed inset-0 bg-black/60 backdrop-blur-md transition-opacity" />
</Transition.Child>
<div className="fixed inset-0 z-30 overflow-y-auto">
@ -57,7 +57,7 @@ export function ModalCore(props: Props) {
>
<Dialog.Panel
className={cn(
"nodedc-glass-modal relative w-full transform rounded-[28px] border border-subtle/70 bg-surface-1/78 text-left shadow-[0_16px_48px_rgba(0,0,0,0.34)] backdrop-blur-2xl transition-all",
"nodedc-glass-modal relative w-full transform rounded-[28px] text-left transition-all",
width,
className
)}

View File

@ -1317,6 +1317,9 @@ importers:
'@plane/hooks':
specifier: workspace:*
version: link:../hooks
'@plane/i18n':
specifier: workspace:*
version: link:../i18n
'@plane/propel':
specifier: workspace:*
version: link:../propel