+
+
+
+
Workspace
+
{workspaceDisplayName}
+
);
diff --git a/plane-src/apps/web/core/components/gantt-chart/preview/timeline-preview.tsx b/plane-src/apps/web/core/components/gantt-chart/preview/timeline-preview.tsx
index 9053fd5..f2ce50d 100644
--- a/plane-src/apps/web/core/components/gantt-chart/preview/timeline-preview.tsx
+++ b/plane-src/apps/web/core/components/gantt-chart/preview/timeline-preview.tsx
@@ -5,7 +5,7 @@
*/
import { useEffect, useMemo, useRef, useState } from "react";
-import { CalendarDays, Check, Filter, SlidersHorizontal } from "lucide-react";
+import { Check, Filter, SlidersHorizontal, X } from "lucide-react";
import type { ChartDataType, IGanttBlock } from "@plane/types";
import { cn } from "@plane/utils";
import { getItemPositionWidth } from "@/components/gantt-chart/views/helpers";
@@ -249,11 +249,12 @@ const sortPreviewItems = (items: TGanttTimelinePreviewItem[], sortMode: TGanttPr
});
export function GanttTimelinePreview(props: TGanttTimelinePreviewProps) {
- const { emptyMessage, isLoading = false, items, locale, subtitle, title } = props;
+ const { emptyMessage, isLoading = false, items, locale } = props;
const [activeRange, setActiveRange] = useState
("Live");
const [activePanel, setActivePanel] = useState<"filters" | "view" | null>(null);
const [activeDateFilters, setActiveDateFilters] = useState([]);
const [activeStatusFilters, setActiveStatusFilters] = useState([]);
+ const [selectedPreviewItemId, setSelectedPreviewItemId] = useState(null);
const [showFullTaskName, setShowFullTaskName] = useState(false);
const [sortMode, setSortMode] = useState("target_date_asc");
const scrollContainerRef = useRef(null);
@@ -329,6 +330,15 @@ export function GanttTimelinePreview(props: TGanttTimelinePreviewProps) {
const hiddenItemsCount = Math.max(items.length - timeline.blocks.length, 0);
const activeFilterCount =
activeDateFilters.length + activeStatusFilters.length + (sortMode === "target_date_asc" ? 0 : 1);
+ const selectedPreviewItem =
+ timeline.blocks.find((item) => item.id === selectedPreviewItemId) ?? timeline.blocks.find((item) => item.id === items[0]?.id);
+ const selectedPreviewItemDate = selectedPreviewItem?.target_date
+ ? getDateFromValue(selectedPreviewItem.target_date)
+ : undefined;
+ const selectedPreviewItemStartDate = selectedPreviewItem?.start_date
+ ? getDateFromValue(selectedPreviewItem.start_date)
+ : undefined;
+ const formatPreviewDate = (date?: Date) => (date ? getShortDateLabel(date, locale) : "Нет");
const toggleStatusFilter = (filterKey: TGanttPreviewStatusFilter) =>
setActiveStatusFilters((currentFilters) =>
@@ -350,6 +360,13 @@ export function GanttTimelinePreview(props: TGanttTimelinePreviewProps) {
setSortMode("target_date_asc");
};
+ useEffect(() => {
+ if (!selectedPreviewItemId) return;
+ if (timeline.blocks.some((item) => item.id === selectedPreviewItemId)) return;
+
+ setSelectedPreviewItemId(null);
+ }, [selectedPreviewItemId, timeline.blocks]);
+
useEffect(() => {
const scrollElement = scrollContainerRef.current;
if (!scrollElement || isLoading) return;
@@ -366,28 +383,24 @@ export function GanttTimelinePreview(props: TGanttTimelinePreviewProps) {
return (
-
-
-
-
-
-
{title}
- {subtitle &&
{subtitle}
}
-
-
+
-
- {GANTT_RANGES.map((item) => (
-
- ))}
+
+
+ {GANTT_RANGES.map((item) => (
+
+ ))}
+
{activePanel === "view" && (
@@ -551,6 +566,38 @@ export function GanttTimelinePreview(props: TGanttTimelinePreviewProps) {
+ {selectedPreviewItemId && selectedPreviewItem && (
+
+
+
+
+ {selectedPreviewItem.identifier}
+
+
{selectedPreviewItem.name}
+
+
+
Начало
+
{formatPreviewDate(selectedPreviewItemStartDate)}
+
+
+
Срок
+
{formatPreviewDate(selectedPreviewItemDate)}
+
+
+
+ Клик по строке или полосе Ганта выбирает задачу без перехода в полную карточку.
+
+
+
+ )}
+
{isLoading ? (
Array.from({ length: 4 }, (_, index) => (
@@ -560,7 +607,19 @@ export function GanttTimelinePreview(props: TGanttTimelinePreviewProps) {
timeline.blocks.map((item) => (
setSelectedPreviewItemId(item.id)}
+ onKeyDown={(event) => {
+ if (event.key !== "Enter" && event.key !== " ") return;
+
+ event.preventDefault();
+ setSelectedPreviewItemId(item.id);
+ }}
style={{
gridTemplateColumns: `${GANTT_LABEL_COLUMN_WIDTH}px ${timeline.timelineWidth}px`,
}}
@@ -579,6 +638,13 @@ export function GanttTimelinePreview(props: TGanttTimelinePreviewProps) {
{
+ event.stopPropagation();
+ setSelectedPreviewItemId(item.id);
+ }}
style={{
left: `${item.left}px`,
width: `${item.width}px`,
diff --git a/plane-src/apps/web/core/components/home/home-dashboard-widgets.tsx b/plane-src/apps/web/core/components/home/home-dashboard-widgets.tsx
index da6f479..bb06ebe 100644
--- a/plane-src/apps/web/core/components/home/home-dashboard-widgets.tsx
+++ b/plane-src/apps/web/core/components/home/home-dashboard-widgets.tsx
@@ -15,6 +15,7 @@ import { cn } from "@plane/utils";
// hooks
import { useHome } from "@/hooks/store/use-home";
import { useProject } from "@/hooks/store/use-project";
+import { useWorkspace } from "@/hooks/store/use-workspace";
// plane web components
import { HomePageHeader } from "@/plane-web/components/home/header";
import { ProjectService } from "@/services/project";
@@ -23,7 +24,7 @@ import { WorkspaceService } from "@/services/workspace.service";
import { HomeCardShell } from "./home-card-shell";
import { HomeGanttPreview } from "./home-gantt-preview";
import { HomeRecentIssueDecks } from "./home-recent-issue-decks";
-import { HomeActivityTrendCard, HomeOperationsOverview } from "./home-project-insights";
+import { HomeActivityTrendCard, HomeOperationsCard, HomeRhythmRecentOverview } from "./home-project-insights";
import { HomeProjectStack } from "./home-project-stack";
import { aggregateProjectAnalytics, type THomeProjectData } from "./home.utils";
import { StickiesWidget } from "../stickies/widget";
@@ -81,6 +82,7 @@ export const DashboardWidgets = observer(function DashboardWidgets(props: Dashbo
const pathname = usePathname();
// store hooks
const { toggleWidgetSettings, widgetsMap, showWidgetSettings, loading } = useHome();
+ const { currentWorkspace } = useWorkspace();
const { loader, joinedProjectIds, getPartialProjectById, fetchProjectAnalyticsCount, getProjectAnalyticsCountById } =
useProject();
// plane hooks
@@ -190,21 +192,15 @@ export const DashboardWidgets = observer(function DashboardWidgets(props: Dashbo
].filter(Boolean);
return (
-
-
+
toggleWidgetSettings(false)}
/>
-
-
+
+
-
+
+
-
-
+
+
+
+
diff --git a/plane-src/apps/web/core/components/home/home-project-insights.tsx b/plane-src/apps/web/core/components/home/home-project-insights.tsx
index 95811b8..28b4484 100644
--- a/plane-src/apps/web/core/components/home/home-project-insights.tsx
+++ b/plane-src/apps/web/core/components/home/home-project-insights.tsx
@@ -293,111 +293,125 @@ export function HomeActivityTrendCard(props: HomeProjectInsightsProps) {
);
}
-export function HomeOperationsOverview(props: HomeProjectInsightsProps) {
- const { recentActivitySlot } = props;
+export function HomeRhythmCard(props: HomeProjectInsightsProps) {
const {
completedIssues,
- completionRate,
metricCards,
openIssues,
- progressRows,
project,
recentTouchpoints,
totalIssues,
} = useHomeProjectInsightData(props);
return (
-
-
-
-
-
+
+
+
+
Ритм исполнения
+
Закрытый объём и открытый остаток по фокусу.
+
+
+
+
+
+
+
+ {metricCards.map((metric) => (
+
+
{metric.label}
+
{metric.value}
-
-
Операционный срез
-
Команда, циклы и модули относительно текущего workspace.
+ ))}
+
+
+
+
+
+ Закрытые задачи
+ {completedIssues}
+
+
+
0 ? (completedIssues / totalIssues) * 100 : 0}%` }}
+ />
-
- {progressRows.map((row) => {
- const percent = row.max > 0 ? Math.max((row.value / row.max) * 100, row.value > 0 ? 10 : 0) : 0;
+
+
+ Открытый остаток
+ {openIssues}
+
+
+
0 ? (openIssues / totalIssues) * 100 : 0}%`,
+ height: "100%",
+ }}
+ />
+
+
+
- return (
-
-
- {row.label}
- {row.value}
-
-
+
+ {project ? project.identifier : "Workspace"}
+ держит
+ {totalIssues}
+ задач и
+ {recentTouchpoints}
+ недавних касаний.
+
+
+ );
+}
+
+export function HomeOperationsCard(props: HomeProjectInsightsProps) {
+ const {
+ progressRows,
+ } = useHomeProjectInsightData(props);
+
+ return (
+
+
+
+
+
+
+
Операционный срез
+
Команда, циклы и модули относительно текущего workspace.
+
+
+
+
+ {progressRows.map((row) => {
+ const percent = row.max > 0 ? Math.max((row.value / row.max) * 100, row.value > 0 ? 10 : 0) : 0;
+
+ return (
+
+
+ {row.label}
+ {row.value}
- );
- })}
-
+
+
+ );
+ })}
+
+ );
+}
-
-
-
-
Ритм исполнения
-
Закрытый объём и открытый остаток по фокусу.
-
-
-
-
-
+export function HomeRhythmRecentOverview(props: HomeProjectInsightsProps) {
+ const { recentActivitySlot } = props;
+ const { completionRate } = useHomeProjectInsightData(props);
-
- {metricCards.map((metric) => (
-
-
{metric.label}
-
{metric.value}
-
- ))}
-
-
-
-
-
- Закрытые задачи
- {completedIssues}
-
-
-
0 ? (completedIssues / totalIssues) * 100 : 0}%` }}
- />
-
-
-
-
-
- Открытый остаток
- {openIssues}
-
-
-
0 ? (openIssues / totalIssues) * 100 : 0}%`,
- height: "100%",
- }}
- />
-
-
-
-
-
- {project ? project.identifier : "Workspace"}
- держит
- {totalIssues}
- задач и
- {recentTouchpoints}
- недавних касаний.
-
-
+ return (
+
+
{recentActivitySlot ? (
@@ -423,6 +437,15 @@ export function HomeOperationsOverview(props: HomeProjectInsightsProps) {
);
}
+export function HomeOperationsOverview(props: HomeProjectInsightsProps) {
+ return (
+
+
+
+
+ );
+}
+
export function HomeProjectInsights(props: HomeProjectInsightsProps) {
return (
diff --git a/plane-src/apps/web/core/components/home/home-project-stack.tsx b/plane-src/apps/web/core/components/home/home-project-stack.tsx
index 9fd5024..602ec47 100644
--- a/plane-src/apps/web/core/components/home/home-project-stack.tsx
+++ b/plane-src/apps/web/core/components/home/home-project-stack.tsx
@@ -6,6 +6,7 @@
import { FolderOpenDot, Layers3, Search, UsersRound } from "lucide-react";
import Link from "next/link";
+import { useRouter } from "next/navigation";
import type { TActivityEntityData, TProjectAnalyticsCount } from "@plane/types";
import { Logo } from "@plane/propel/emoji-icon-picker";
import { cn } from "@plane/utils";
@@ -29,6 +30,7 @@ const ACTIVE_CARD_HEIGHT = 248;
const STACK_OFFSET = 88;
export function HomeProjectStack(props: HomeProjectStackProps) {
+ const router = useRouter();
const {
className,
projects,
@@ -201,7 +203,7 @@ export function HomeProjectStack(props: HomeProjectStackProps) {
)}
-
+
Быстрый выбор
@@ -213,45 +215,64 @@ export function HomeProjectStack(props: HomeProjectStackProps) {
-
+
{orderedProjects.map((project: THomeProjectData) => {
const analytics = analyticsMap[project.id];
+ const isActive = project.id === selectedProject?.id;
+
return (
);
})}
{selectedProject && (
-
-
+
+
Фокус
-
{selectedProject.identifier}
+
+ {selectedProject.identifier}
+
-
+
Команда
-
+
{analyticsMap[selectedProject.id]?.total_members ?? 0}
-
+
Контур
-
+
{activityCountByProject[selectedProject.id] ?? 0} касаний
diff --git a/plane-src/apps/web/core/components/home/root.tsx b/plane-src/apps/web/core/components/home/root.tsx
index b1aeb2f..5ea74b3 100644
--- a/plane-src/apps/web/core/components/home/root.tsx
+++ b/plane-src/apps/web/core/components/home/root.tsx
@@ -57,8 +57,8 @@ export const WorkspaceHomeView = observer(function WorkspaceHomeView() {
)}
<>
-
-
+
+
diff --git a/plane-src/apps/web/core/components/voice-tasker/global-control.tsx b/plane-src/apps/web/core/components/voice-tasker/global-control.tsx
index 1d02132..33cde1c 100644
--- a/plane-src/apps/web/core/components/voice-tasker/global-control.tsx
+++ b/plane-src/apps/web/core/components/voice-tasker/global-control.tsx
@@ -389,15 +389,15 @@ export function VoiceTaskerGlobalControl({ workspaceSlug }: Props) {
diff --git a/plane-src/apps/web/styles/globals.css b/plane-src/apps/web/styles/globals.css
index c6a0c3d..fa68bd0 100644
--- a/plane-src/apps/web/styles/globals.css
+++ b/plane-src/apps/web/styles/globals.css
@@ -1790,6 +1790,60 @@
box-shadow 160ms ease;
}
+ .nodedc-home-route-surface {
+ min-height: 100vh;
+ background: #333333 !important;
+ }
+
+ main:has(.nodedc-home-route-surface) {
+ background: #333333 !important;
+ }
+
+ .nodedc-home-page-shell {
+ max-width: min(1840px, calc(100vw - 5rem));
+ }
+
+ .nodedc-home-top-toolbar > .nodedc-glass-modal {
+ max-width: min(1840px, calc(100vw - 5rem));
+ margin-inline: auto;
+ border: 0 !important;
+ background: transparent !important;
+ padding-inline: 0 !important;
+ box-shadow: none !important;
+ -webkit-backdrop-filter: none !important;
+ backdrop-filter: none !important;
+ }
+
+ .nodedc-home-top-toolbar {
+ padding-inline: 0 !important;
+ }
+
+ .nodedc-home-dashboard-shell {
+ gap: 0.75rem;
+ }
+
+ .nodedc-home-dashboard-grid {
+ align-items: stretch;
+ gap: 0.75rem;
+ }
+
+ .nodedc-home-lower-grid {
+ gap: 0.75rem;
+ align-items: stretch;
+ }
+
+ .nodedc-home-project-panel {
+ margin-top: 1.75rem;
+ height: calc(100% - 1.75rem) !important;
+ min-height: 0;
+ }
+
+ .nodedc-home-main-column {
+ display: flex;
+ flex-direction: column;
+ gap: 0.75rem;
+ }
+
.nodedc-home-hero {
position: relative;
overflow: hidden;
@@ -1797,17 +1851,16 @@
border: 0 !important;
outline: none !important;
border-radius: 1.9rem !important;
- background:
- linear-gradient(180deg, rgba(255, 255, 255, 0.028) 0%, rgba(255, 255, 255, 0.008) 100%), rgba(10, 10, 12, 0.74) !important;
- padding: 1rem;
- box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.022) !important;
- -webkit-backdrop-filter: blur(28px);
- backdrop-filter: blur(28px);
+ background: transparent !important;
+ padding: 0;
+ box-shadow: none !important;
+ -webkit-backdrop-filter: none;
+ backdrop-filter: none;
}
@media (min-width: 768px) {
.nodedc-home-hero {
- padding: 1.2rem;
+ padding: 0;
}
}
@@ -1817,42 +1870,58 @@
justify-content: flex-end;
gap: 0.8rem;
min-height: 1.75rem;
- padding: 0.15rem 0.35rem 0.85rem;
+ padding: 0.1rem 0.35rem 0.45rem;
text-align: right;
}
.nodedc-home-hero-grid {
+ --nodedc-home-title-width: 13.25rem;
display: grid;
min-width: 0;
- gap: 1rem;
+ position: relative;
+ min-height: 8.55rem;
}
@media (min-width: 1280px) {
.nodedc-home-hero-grid {
- grid-template-columns: minmax(320px, 360px) minmax(0, 1fr);
+ grid-template-columns: minmax(0, 1fr);
align-items: stretch;
}
}
.nodedc-home-hero-title-cell {
+ position: absolute;
+ inset: 0;
+ z-index: 1;
display: flex;
- min-height: 8rem;
+ min-height: 8.55rem;
flex-direction: column;
- justify-content: flex-end;
+ justify-content: center;
border-radius: 1.7rem;
- background: linear-gradient(180deg, rgba(255, 255, 255, 0.022), rgba(255, 255, 255, 0.006)), rgba(0, 0, 0, 0.18);
+ background: #474747 !important;
padding: 1.25rem;
+ padding-right: max(1.25rem, calc(100% - var(--nodedc-home-title-width)));
+ box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.035) !important;
}
.nodedc-home-hero-title-cell h1 {
- max-width: 16rem;
+ max-width: 18rem;
color: var(--text-color-primary);
- font-size: clamp(2rem, 3vw, 3.35rem);
+ font-size: clamp(1.05rem, 1.25vw, 1.45rem);
font-weight: 700;
- line-height: 0.92;
+ line-height: 1.02;
letter-spacing: 0;
}
+ .nodedc-home-hero-title-label {
+ margin-bottom: 0.45rem;
+ color: var(--text-color-secondary);
+ font-size: 0.7rem;
+ font-weight: 650;
+ letter-spacing: 0.16em;
+ text-transform: uppercase;
+ }
+
.nodedc-home-hero-title-cell p {
margin-top: 0.75rem;
color: var(--text-color-secondary);
@@ -1886,21 +1955,58 @@
}
.nodedc-home-market-band {
+ position: absolute;
+ top: 0;
+ right: 0;
+ left: var(--nodedc-home-title-width);
+ z-index: 2;
display: flex;
+ width: auto;
min-width: 0;
- min-height: 8rem;
+ min-height: 8.55rem;
align-items: flex-end;
- gap: 1.5rem;
- border-radius: 1.7rem;
+ gap: 1rem;
+ border-radius: 1.7rem !important;
background: rgb(var(--nodedc-card-active-rgb)) !important;
- padding: 1.25rem;
+ padding: 1rem;
color: rgb(var(--nodedc-on-card-active-rgb));
box-shadow:
inset 0 1px 0 rgba(255, 255, 255, 0.42),
0 18px 42px rgba(0, 0, 0, 0.18) !important;
}
+ .nodedc-home-market-focus-title {
+ line-height: 1.22 !important;
+ }
+
+ .nodedc-home-market-progress {
+ height: 1.05rem;
+ margin-top: 0.5rem;
+ overflow: hidden;
+ border-radius: 999px;
+ background: rgba(0, 0, 0, 0.18);
+ }
+
@media (max-width: 767px) {
+ .nodedc-home-hero-grid {
+ display: flex;
+ min-height: auto;
+ flex-direction: column;
+ gap: 0.75rem;
+ }
+
+ .nodedc-home-hero-title-cell,
+ .nodedc-home-market-band {
+ position: relative;
+ inset: auto;
+ width: 100%;
+ min-height: 6rem;
+ }
+
+ .nodedc-home-hero-title-cell {
+ padding: 1rem;
+ }
+
.nodedc-home-market-band {
flex-direction: column;
align-items: stretch;
@@ -1913,11 +2019,10 @@
isolation: isolate;
min-height: 30rem;
border-radius: 2rem !important;
- background:
- linear-gradient(180deg, rgba(255, 255, 255, 0.026) 0%, rgba(255, 255, 255, 0.008) 100%), rgba(8, 8, 10, 0.78) !important;
- box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.02) !important;
- -webkit-backdrop-filter: blur(28px);
- backdrop-filter: blur(28px);
+ background: #050506 !important;
+ box-shadow: none !important;
+ -webkit-backdrop-filter: none;
+ backdrop-filter: none;
}
.nodedc-home-gantt-toolbar {
@@ -1926,11 +2031,67 @@
display: flex;
flex-wrap: wrap;
align-items: center;
- justify-content: space-between;
+ justify-content: flex-start;
gap: 1rem;
padding: 1.25rem;
}
+ .nodedc-home-gantt-toolbar-spacer {
+ display: none;
+ }
+
+ .nodedc-home-gantt-controls {
+ display: flex;
+ flex-wrap: wrap;
+ align-items: center;
+ justify-content: flex-start;
+ gap: 0.65rem;
+ }
+
+ .nodedc-home-gantt-range-group {
+ display: inline-flex;
+ align-items: center;
+ gap: 0.15rem;
+ border-radius: 999px;
+ background: #171718;
+ padding: 0.2rem;
+ box-shadow:
+ inset 0 1px 0 rgba(255, 255, 255, 0.045),
+ 0 14px 28px rgba(0, 0, 0, 0.24);
+ }
+
+ .nodedc-home-gantt-range-button {
+ display: inline-flex;
+ height: 2.15rem;
+ min-width: 2.15rem;
+ align-items: center;
+ justify-content: center;
+ border: 0 !important;
+ outline: none !important;
+ border-radius: 999px !important;
+ background: transparent !important;
+ padding-inline: 0.82rem;
+ color: var(--text-color-secondary);
+ font-size: 0.72rem;
+ font-weight: 800;
+ transition:
+ background 160ms ease,
+ color 160ms ease,
+ transform 160ms ease;
+ }
+
+ .nodedc-home-gantt-range-button:hover {
+ color: var(--text-color-primary);
+ }
+
+ .nodedc-home-gantt-range-button-active {
+ background: rgb(var(--nodedc-card-active-rgb)) !important;
+ color: rgb(var(--nodedc-on-card-active-rgb)) !important;
+ box-shadow:
+ inset 0 1px 0 rgba(255, 255, 255, 0.42),
+ 0 10px 22px rgba(var(--nodedc-card-active-rgb), 0.18);
+ }
+
.nodedc-home-gantt-chip {
display: inline-flex;
height: 2.35rem;
@@ -1961,10 +2122,34 @@
border: 0 !important;
outline: none !important;
border-radius: 999px !important;
- background: rgba(0, 0, 0, 0.42) !important;
+ background: #151516 !important;
color: var(--text-color-primary);
}
+ .nodedc-home-gantt-filter-button-has-count {
+ display: inline-flex !important;
+ width: auto;
+ min-width: 3.85rem;
+ gap: 0.45rem;
+ padding-inline: 0.72rem;
+ background: rgba(0, 0, 0, 0.56) !important;
+ color: var(--text-color-primary) !important;
+ }
+
+ .nodedc-home-gantt-filter-count {
+ display: grid;
+ width: 1.25rem;
+ min-width: 1.25rem;
+ height: 1.25rem;
+ place-items: center;
+ border-radius: 999px;
+ background: rgba(255, 255, 255, 0.96);
+ color: #08080a;
+ font-size: 0.68rem;
+ font-weight: 900;
+ line-height: 1;
+ }
+
.nodedc-home-gantt-round-button:hover {
background: rgba(0, 0, 0, 0.58) !important;
}
@@ -1979,6 +2164,12 @@
display: inline-flex;
align-items: center;
gap: 0.5rem;
+ border-radius: 999px;
+ background: #0f0f10;
+ padding: 0.2rem;
+ box-shadow:
+ inset 0 1px 0 rgba(255, 255, 255, 0.03),
+ 0 14px 28px rgba(0, 0, 0, 0.22);
}
.nodedc-home-gantt-popover {
@@ -2092,7 +2283,7 @@
margin: 0 1.25rem 1.25rem;
overflow: hidden;
border-radius: 1.75rem;
- background: rgba(0, 0, 0, 0.28);
+ background: #050506;
}
.nodedc-home-gantt-scroll {
@@ -2121,7 +2312,9 @@
position: relative;
min-height: 23.5rem;
padding: 1rem;
- background: linear-gradient(180deg, rgba(255, 255, 255, 0.035) 1px, transparent 1px) 0 0 / 100% 4.2rem;
+ background:
+ linear-gradient(180deg, rgba(255, 255, 255, 0.035) 1px, transparent 1px) 0 0 / 100% 4.2rem,
+ #050506;
}
.nodedc-home-gantt-floating {
@@ -2141,6 +2334,48 @@
backdrop-filter: blur(24px);
}
+ .nodedc-home-gantt-inspector {
+ position: absolute;
+ top: 5.35rem;
+ left: calc(12rem + 2.25rem);
+ z-index: 6;
+ width: min(31rem, calc(100% - 16rem));
+ min-height: 11.5rem;
+ border: 1px solid rgba(255, 255, 255, 0.065);
+ border-radius: 1.65rem;
+ background:
+ linear-gradient(135deg, rgba(255, 255, 255, 0.13), rgba(255, 255, 255, 0.025) 44%, rgba(255, 255, 255, 0.08)),
+ rgba(34, 34, 38, 0.58);
+ padding: 1rem 1.05rem 1.05rem 3.85rem;
+ box-shadow:
+ 0 22px 60px rgba(0, 0, 0, 0.42),
+ inset 0 1px 0 rgba(255, 255, 255, 0.16);
+ -webkit-backdrop-filter: blur(28px) saturate(1.22);
+ backdrop-filter: blur(28px) saturate(1.22);
+ }
+
+ .nodedc-home-gantt-inspector-close {
+ position: absolute;
+ top: 0.72rem;
+ left: 0.72rem;
+ display: grid;
+ width: 2.35rem;
+ min-width: 2.35rem;
+ height: 2.35rem;
+ place-items: center;
+ border: 0 !important;
+ outline: none !important;
+ border-radius: 1.05rem !important;
+ background: rgba(0, 0, 0, 0.44) !important;
+ color: rgba(255, 255, 255, 0.82);
+ box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.06);
+ }
+
+ .nodedc-home-gantt-inspector-close:hover {
+ background: rgba(0, 0, 0, 0.62) !important;
+ color: rgb(var(--nodedc-card-active-rgb));
+ }
+
.nodedc-home-gantt-grid {
position: absolute;
top: 1rem;
@@ -2239,6 +2474,20 @@
align-items: center;
gap: 1rem;
min-height: 3.45rem;
+ cursor: pointer;
+ border-radius: 1.05rem;
+ transition:
+ background 140ms ease,
+ color 140ms ease;
+ }
+
+ .nodedc-home-gantt-row:hover,
+ .nodedc-home-gantt-row-selected {
+ background: rgba(255, 255, 255, 0.035);
+ }
+
+ .nodedc-home-gantt-row-selected .nodedc-home-gantt-row-label {
+ color: rgb(var(--nodedc-card-active-rgb));
}
.nodedc-home-gantt-row-compact {
@@ -2257,7 +2506,7 @@
left: 0;
z-index: 2;
border-radius: 1rem;
- background: linear-gradient(90deg, rgba(7, 7, 9, 0.94) 0%, rgba(7, 7, 9, 0.72) 76%, transparent 100%);
+ background: linear-gradient(90deg, #050506 0%, rgba(5, 5, 6, 0.78) 76%, transparent 100%);
padding: 0.45rem 0.75rem 0.45rem 0;
}
@@ -2275,6 +2524,7 @@
height: 1.4rem;
min-width: 2.5rem;
border-radius: 999px;
+ cursor: pointer;
}
.nodedc-home-gantt-bar-accent {
@@ -2757,6 +3007,131 @@
opacity: 0.8;
}
+ .nodedc-home-project-quick-list {
+ display: flex;
+ width: calc(100% + 2rem);
+ margin-inline: -1rem;
+ flex-direction: column;
+ gap: 0;
+ overflow: hidden;
+ border-radius: 1.25rem;
+ background: rgba(8, 8, 10, 0.62);
+ }
+
+ .nodedc-home-project-quick-button {
+ display: flex;
+ width: 100%;
+ min-height: 3.22rem;
+ align-items: center;
+ justify-content: space-between;
+ gap: 0.8rem;
+ border: 0 !important;
+ outline: none !important;
+ border-radius: 0 !important;
+ background: rgba(255, 255, 255, 0.028) !important;
+ padding: 0.7rem 0.9rem;
+ color: var(--text-color-primary);
+ font-size: 0.95rem;
+ font-weight: 520;
+ text-align: left;
+ transition:
+ background 160ms ease,
+ color 160ms ease,
+ transform 160ms ease,
+ box-shadow 160ms ease;
+ }
+
+ .nodedc-home-project-quick-button + .nodedc-home-project-quick-button {
+ box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.045);
+ }
+
+ .nodedc-home-project-quick-button:first-child {
+ border-top-left-radius: 1.25rem !important;
+ border-top-right-radius: 1.25rem !important;
+ }
+
+ .nodedc-home-project-quick-button:last-child {
+ border-bottom-right-radius: 1.25rem !important;
+ border-bottom-left-radius: 1.25rem !important;
+ }
+
+ .nodedc-home-project-quick-button:hover {
+ background: rgba(255, 255, 255, 0.08) !important;
+ color: var(--text-color-primary);
+ transform: translateY(-1px);
+ }
+
+ .nodedc-home-project-quick-button[data-active="true"] {
+ background: rgba(255, 255, 255, 0.13) !important;
+ color: var(--text-color-primary);
+ box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.09);
+ }
+
+ .nodedc-home-project-quick-main {
+ display: flex;
+ min-width: 0;
+ align-items: center;
+ gap: 0.62rem;
+ }
+
+ .nodedc-home-project-quick-logo {
+ display: grid;
+ width: 1.7rem;
+ min-width: 1.7rem;
+ height: 1.7rem;
+ place-items: center;
+ border-radius: 999px;
+ background: rgba(255, 255, 255, 0.08);
+ box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.05);
+ }
+
+ .nodedc-home-project-quick-button[data-active="true"] .nodedc-home-project-quick-logo {
+ background: rgba(var(--nodedc-card-active-rgb), 0.92);
+ color: #0a0a0b;
+ }
+
+ .nodedc-home-project-quick-metric {
+ display: inline-flex;
+ min-width: max-content;
+ align-items: center;
+ gap: 0.38rem;
+ font-size: 0.72rem;
+ font-weight: 650;
+ opacity: 0.78;
+ }
+
+ .nodedc-home-project-quick-dot {
+ display: block;
+ width: 0.48rem;
+ height: 0.48rem;
+ border-radius: 999px;
+ background: rgba(255, 255, 255, 0.58);
+ }
+
+ .nodedc-home-project-quick-button[data-active="true"] .nodedc-home-project-quick-dot {
+ background: rgb(var(--nodedc-card-active-rgb));
+ }
+
+ .nodedc-home-project-focus-grid {
+ grid-template-columns: repeat(3, minmax(0, 1fr));
+ background: transparent !important;
+ padding: 0 !important;
+ }
+
+ .nodedc-home-project-focus-item {
+ min-width: 0;
+ overflow: hidden;
+ border-radius: 0 !important;
+ background: transparent !important;
+ padding-inline: 0.75rem !important;
+ }
+
+ .nodedc-home-project-focus-value {
+ min-width: 0;
+ line-height: 1.2;
+ overflow-wrap: anywhere;
+ }
+
.nodedc-home-user-card {
position: relative;
overflow: hidden;
@@ -3020,6 +3395,14 @@
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.01) !important;
}
+ .nodedc-home-operations-card {
+ background:
+ linear-gradient(180deg, rgba(255, 255, 255, 0.022) 0%, rgba(255, 255, 255, 0.006) 100%),
+ rgba(10, 10, 12, 0.68) !important;
+ -webkit-backdrop-filter: blur(28px);
+ backdrop-filter: blur(28px);
+ }
+
.nodedc-home-soft-badge {
background: rgba(0, 0, 0, 0.34) !important;
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.01) !important;