feat: integrate task creation functionality in HomePageClient and Kanban components
- Added task creation modal in HomePageClient with state management for visibility. - Implemented `handleCreateTask` function to handle task submissions. - Updated Kanban components to accept `onCreateTask` prop for task creation, ensuring consistent task management across the application. - Removed unused task creation UI elements from Kanban components to streamline the interface.
This commit is contained in:
@@ -5,7 +5,10 @@ import { KanbanBoardContainer } from '@/components/kanban/BoardContainer';
|
|||||||
import { Header } from '@/components/ui/Header';
|
import { Header } from '@/components/ui/Header';
|
||||||
import { TasksProvider, useTasksContext } from '@/contexts/TasksContext';
|
import { TasksProvider, useTasksContext } from '@/contexts/TasksContext';
|
||||||
import { Task, Tag, TaskStats } from '@/lib/types';
|
import { Task, Tag, TaskStats } from '@/lib/types';
|
||||||
|
import { CreateTaskData } from '@/clients/tasks-client';
|
||||||
import { userPreferencesService } from '@/services/user-preferences';
|
import { userPreferencesService } from '@/services/user-preferences';
|
||||||
|
import { CreateTaskForm } from '@/components/forms/CreateTaskForm';
|
||||||
|
import { Button } from '@/components/ui/Button';
|
||||||
|
|
||||||
interface HomePageClientProps {
|
interface HomePageClientProps {
|
||||||
initialTasks: Task[];
|
initialTasks: Task[];
|
||||||
@@ -14,9 +17,10 @@ interface HomePageClientProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function HomePageContent() {
|
function HomePageContent() {
|
||||||
const { stats, syncing } = useTasksContext();
|
const { stats, syncing, createTask } = useTasksContext();
|
||||||
const [showFilters, setShowFilters] = useState(true);
|
const [showFilters, setShowFilters] = useState(true);
|
||||||
const [showObjectives, setShowObjectives] = useState(true);
|
const [showObjectives, setShowObjectives] = useState(true);
|
||||||
|
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
||||||
|
|
||||||
// Charger les préférences depuis le service
|
// Charger les préférences depuis le service
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -38,6 +42,12 @@ function HomePageContent() {
|
|||||||
userPreferencesService.updateViewPreferences({ showObjectives: newValue });
|
userPreferencesService.updateViewPreferences({ showObjectives: newValue });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Handler pour la création de tâche depuis la barre de contrôles
|
||||||
|
const handleCreateTask = async (data: CreateTaskData) => {
|
||||||
|
await createTask(data);
|
||||||
|
setIsCreateModalOpen(false);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="min-h-screen bg-[var(--background)]">
|
<div className="min-h-screen bg-[var(--background)]">
|
||||||
<Header
|
<Header
|
||||||
@@ -50,40 +60,51 @@ function HomePageContent() {
|
|||||||
{/* Barre de contrôles de visibilité */}
|
{/* Barre de contrôles de visibilité */}
|
||||||
<div className="bg-[var(--card)]/30 border-b border-[var(--border)]/30">
|
<div className="bg-[var(--card)]/30 border-b border-[var(--border)]/30">
|
||||||
<div className="container mx-auto px-6 py-2">
|
<div className="container mx-auto px-6 py-2">
|
||||||
<div className="flex items-center gap-4">
|
<div className="flex items-center justify-between w-full">
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-4">
|
||||||
<button
|
<div className="flex items-center gap-2">
|
||||||
onClick={handleToggleFilters}
|
<button
|
||||||
className={`flex items-center gap-2 px-3 py-1.5 rounded-md text-sm font-mono transition-all ${
|
onClick={handleToggleFilters}
|
||||||
showFilters
|
className={`flex items-center gap-2 px-3 py-1.5 rounded-md text-sm font-mono transition-all ${
|
||||||
? 'bg-[var(--primary)]/20 text-[var(--primary)] border border-[var(--primary)]/30'
|
showFilters
|
||||||
: 'bg-[var(--card)] text-[var(--muted-foreground)] border border-[var(--border)] hover:border-[var(--primary)]/50'
|
? 'bg-[var(--primary)]/20 text-[var(--primary)] border border-[var(--primary)]/30'
|
||||||
}`}
|
: 'bg-[var(--card)] text-[var(--muted-foreground)] border border-[var(--border)] hover:border-[var(--primary)]/50'
|
||||||
>
|
}`}
|
||||||
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
>
|
||||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M12 6V4m0 2a2 2 0 100 4m0-4a2 2 0 110 4m-6 8a2 2 0 100-4m0 4a2 2 0 100 4m0-4v2m0-6V4m6 6v10m6-2a2 2 0 100-4m0 4a2 2 0 100 4m0-4v2m0-6V4" />
|
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
</svg>
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M12 6V4m0 2a2 2 0 100 4m0-4a2 2 0 110 4m-6 8a2 2 0 100-4m0 4a2 2 0 100 4m0-4v2m0-6V4m6 6v10m6-2a2 2 0 100-4m0 4a2 2 0 100 4m0-4v2m0-6V4" />
|
||||||
Filtres
|
</svg>
|
||||||
</button>
|
Filtres
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
onClick={handleToggleObjectives}
|
||||||
|
className={`flex items-center gap-2 px-3 py-1.5 rounded-md text-sm font-mono transition-all ${
|
||||||
|
showObjectives
|
||||||
|
? 'bg-[var(--accent)]/20 text-[var(--accent)] border border-[var(--accent)]/30'
|
||||||
|
: 'bg-[var(--card)] text-[var(--muted-foreground)] border border-[var(--border)] hover:border-[var(--accent)]/50'
|
||||||
|
}`}
|
||||||
|
>
|
||||||
|
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 12l2 2 4-4M7.835 4.697a3.42 3.42 0 001.946-.806 3.42 3.42 0 014.438 0 3.42 3.42 0 001.946.806 3.42 3.42 0 013.138 3.138 3.42 3.42 0 00.806 1.946 3.42 3.42 0 010 4.438 3.42 3.42 0 00-.806 1.946 3.42 3.42 0 01-3.138 3.138 3.42 3.42 0 00-1.946.806 3.42 3.42 0 01-4.438 0 3.42 3.42 0 00-1.946-.806 3.42 3.42 0 01-3.138-3.138 3.42 3.42 0 00-.806-1.946 3.42 3.42 0 010-4.438 3.42 3.42 0 00.806-1.946 3.42 3.42 0 013.138-3.138z" />
|
||||||
|
</svg>
|
||||||
|
Objectifs
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
<button
|
|
||||||
onClick={handleToggleObjectives}
|
|
||||||
className={`flex items-center gap-2 px-3 py-1.5 rounded-md text-sm font-mono transition-all ${
|
|
||||||
showObjectives
|
|
||||||
? 'bg-[var(--accent)]/20 text-[var(--accent)] border border-[var(--accent)]/30'
|
|
||||||
: 'bg-[var(--card)] text-[var(--muted-foreground)] border border-[var(--border)] hover:border-[var(--accent)]/50'
|
|
||||||
}`}
|
|
||||||
>
|
|
||||||
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
||||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 12l2 2 4-4M7.835 4.697a3.42 3.42 0 001.946-.806 3.42 3.42 0 014.438 0 3.42 3.42 0 001.946.806 3.42 3.42 0 013.138 3.138 3.42 3.42 0 00.806 1.946 3.42 3.42 0 010 4.438 3.42 3.42 0 00-.806 1.946 3.42 3.42 0 01-3.138 3.138 3.42 3.42 0 00-1.946.806 3.42 3.42 0 01-4.438 0 3.42 3.42 0 00-1.946-.806 3.42 3.42 0 01-3.138-3.138 3.42 3.42 0 00-.806-1.946 3.42 3.42 0 010-4.438 3.42 3.42 0 00.806-1.946 3.42 3.42 0 013.138-3.138z" />
|
|
||||||
</svg>
|
|
||||||
Objectifs
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="text-xs text-[var(--muted-foreground)] font-mono">
|
|
||||||
Affichage des composants
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* Bouton d'ajout de tâche */}
|
||||||
|
<Button
|
||||||
|
variant="primary"
|
||||||
|
onClick={() => setIsCreateModalOpen(true)}
|
||||||
|
className="flex items-center gap-2"
|
||||||
|
>
|
||||||
|
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M12 4v16m8-8H4" />
|
||||||
|
</svg>
|
||||||
|
Nouvelle tâche
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -94,6 +115,14 @@ function HomePageContent() {
|
|||||||
showObjectives={showObjectives}
|
showObjectives={showObjectives}
|
||||||
/>
|
/>
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
|
{/* Modal de création de tâche */}
|
||||||
|
<CreateTaskForm
|
||||||
|
isOpen={isCreateModalOpen}
|
||||||
|
onClose={() => setIsCreateModalOpen(false)}
|
||||||
|
onSubmit={handleCreateTask}
|
||||||
|
loading={false}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,8 +2,6 @@
|
|||||||
|
|
||||||
import { Task, TaskStatus } from '@/lib/types';
|
import { Task, TaskStatus } from '@/lib/types';
|
||||||
import { KanbanColumn } from './Column';
|
import { KanbanColumn } from './Column';
|
||||||
import { Button } from '@/components/ui/Button';
|
|
||||||
import { CreateTaskForm } from '@/components/forms/CreateTaskForm';
|
|
||||||
import { CreateTaskData } from '@/clients/tasks-client';
|
import { CreateTaskData } from '@/clients/tasks-client';
|
||||||
import { useMemo, useState } from 'react';
|
import { useMemo, useState } from 'react';
|
||||||
import { useColumnVisibility } from '@/hooks/useColumnVisibility';
|
import { useColumnVisibility } from '@/hooks/useColumnVisibility';
|
||||||
@@ -21,18 +19,16 @@ import { TaskCard } from './TaskCard';
|
|||||||
|
|
||||||
interface KanbanBoardProps {
|
interface KanbanBoardProps {
|
||||||
tasks: Task[];
|
tasks: Task[];
|
||||||
onCreateTask?: (data: CreateTaskData) => Promise<Task | null>;
|
onCreateTask?: (data: CreateTaskData) => Promise<void>;
|
||||||
onDeleteTask?: (taskId: string) => Promise<void>;
|
onDeleteTask?: (taskId: string) => Promise<void>;
|
||||||
onEditTask?: (task: Task) => void;
|
onEditTask?: (task: Task) => void;
|
||||||
onUpdateTitle?: (taskId: string, newTitle: string) => Promise<void>;
|
onUpdateTitle?: (taskId: string, newTitle: string) => Promise<void>;
|
||||||
onUpdateStatus?: (taskId: string, newStatus: TaskStatus) => Promise<void>;
|
onUpdateStatus?: (taskId: string, newStatus: TaskStatus) => Promise<void>;
|
||||||
loading?: boolean;
|
|
||||||
compactView?: boolean;
|
compactView?: boolean;
|
||||||
visibleStatuses?: TaskStatus[];
|
visibleStatuses?: TaskStatus[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export function KanbanBoard({ tasks, onCreateTask, onDeleteTask, onEditTask, onUpdateTitle, onUpdateStatus, loading = false, compactView = false, visibleStatuses }: KanbanBoardProps) {
|
export function KanbanBoard({ tasks, onCreateTask, onDeleteTask, onEditTask, onUpdateTitle, onUpdateStatus, compactView = false, visibleStatuses }: KanbanBoardProps) {
|
||||||
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
|
||||||
const [activeTask, setActiveTask] = useState<Task | null>(null);
|
const [activeTask, setActiveTask] = useState<Task | null>(null);
|
||||||
|
|
||||||
// Gestion de la visibilité des colonnes (utilise les props si disponibles)
|
// Gestion de la visibilité des colonnes (utilise les props si disponibles)
|
||||||
@@ -72,11 +68,6 @@ export function KanbanBoard({ tasks, onCreateTask, onDeleteTask, onEditTask, onU
|
|||||||
allColumns.filter(column => visibleStatuses.includes(column.id)) :
|
allColumns.filter(column => visibleStatuses.includes(column.id)) :
|
||||||
getVisibleStatuses(allColumns);
|
getVisibleStatuses(allColumns);
|
||||||
|
|
||||||
const handleCreateTask = async (data: CreateTaskData) => {
|
|
||||||
if (onCreateTask) {
|
|
||||||
await onCreateTask(data);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Gestion du début du drag
|
// Gestion du début du drag
|
||||||
const handleDragStart = (event: DragStartEvent) => {
|
const handleDragStart = (event: DragStartEvent) => {
|
||||||
@@ -111,25 +102,8 @@ export function KanbanBoard({ tasks, onCreateTask, onDeleteTask, onEditTask, onU
|
|||||||
onDragEnd={handleDragEnd}
|
onDragEnd={handleDragEnd}
|
||||||
>
|
>
|
||||||
<div className="h-full flex flex-col bg-[var(--background)]">
|
<div className="h-full flex flex-col bg-[var(--background)]">
|
||||||
{/* Header avec bouton nouvelle tâche */}
|
{/* Espacement supérieur */}
|
||||||
<div className="flex justify-between items-center p-6 pb-0">
|
<div className="pt-4"></div>
|
||||||
<div className="flex items-center gap-3">
|
|
||||||
<div className="w-2 h-2 bg-[var(--primary)] rounded-full animate-pulse"></div>
|
|
||||||
<h2 className="text-lg font-mono font-bold text-[var(--foreground)] uppercase tracking-wider">
|
|
||||||
Kanban Board
|
|
||||||
</h2>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{onCreateTask && (
|
|
||||||
<Button
|
|
||||||
variant="primary"
|
|
||||||
onClick={() => setIsCreateModalOpen(true)}
|
|
||||||
disabled={loading}
|
|
||||||
>
|
|
||||||
+ Nouvelle tâche
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
{/* Board tech dark */}
|
{/* Board tech dark */}
|
||||||
@@ -139,7 +113,7 @@ export function KanbanBoard({ tasks, onCreateTask, onDeleteTask, onEditTask, onU
|
|||||||
key={column.id}
|
key={column.id}
|
||||||
id={column.id}
|
id={column.id}
|
||||||
tasks={column.tasks}
|
tasks={column.tasks}
|
||||||
onCreateTask={onCreateTask ? handleCreateTask : undefined}
|
onCreateTask={onCreateTask}
|
||||||
onDeleteTask={onDeleteTask}
|
onDeleteTask={onDeleteTask}
|
||||||
onEditTask={onEditTask}
|
onEditTask={onEditTask}
|
||||||
onUpdateTitle={onUpdateTitle}
|
onUpdateTitle={onUpdateTitle}
|
||||||
@@ -148,15 +122,6 @@ export function KanbanBoard({ tasks, onCreateTask, onDeleteTask, onEditTask, onU
|
|||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Modal de création */}
|
|
||||||
{onCreateTask && (
|
|
||||||
<CreateTaskForm
|
|
||||||
isOpen={isCreateModalOpen}
|
|
||||||
onClose={() => setIsCreateModalOpen(false)}
|
|
||||||
onSubmit={handleCreateTask}
|
|
||||||
loading={loading}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Overlay pour le drag & drop */}
|
{/* Overlay pour le drag & drop */}
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import { KanbanFilters } from './KanbanFilters';
|
|||||||
import { EditTaskForm } from '@/components/forms/EditTaskForm';
|
import { EditTaskForm } from '@/components/forms/EditTaskForm';
|
||||||
import { useTasksContext } from '@/contexts/TasksContext';
|
import { useTasksContext } from '@/contexts/TasksContext';
|
||||||
import { Task, TaskStatus } from '@/lib/types';
|
import { Task, TaskStatus } from '@/lib/types';
|
||||||
import { UpdateTaskData } from '@/clients/tasks-client';
|
import { UpdateTaskData, CreateTaskData } from '@/clients/tasks-client';
|
||||||
import { useColumnVisibility } from '@/hooks/useColumnVisibility';
|
import { useColumnVisibility } from '@/hooks/useColumnVisibility';
|
||||||
import { getAllStatuses } from '@/lib/status-config';
|
import { getAllStatuses } from '@/lib/status-config';
|
||||||
|
|
||||||
@@ -67,6 +67,11 @@ export function KanbanBoardContainer({
|
|||||||
// Obtenir le nom du tag épinglé pour l'affichage
|
// Obtenir le nom du tag épinglé pour l'affichage
|
||||||
const pinnedTagName = tags.find(tag => tag.isPinned)?.name;
|
const pinnedTagName = tags.find(tag => tag.isPinned)?.name;
|
||||||
|
|
||||||
|
// Wrapper pour adapter le type de createTask
|
||||||
|
const handleCreateTask = async (data: CreateTaskData): Promise<void> => {
|
||||||
|
await createTask(data);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{/* Barre de filtres - conditionnelle */}
|
{/* Barre de filtres - conditionnelle */}
|
||||||
@@ -95,7 +100,7 @@ export function KanbanBoardContainer({
|
|||||||
kanbanFilters.swimlanesMode === 'priority' ? (
|
kanbanFilters.swimlanesMode === 'priority' ? (
|
||||||
<PrioritySwimlanesBoard
|
<PrioritySwimlanesBoard
|
||||||
tasks={filteredTasks}
|
tasks={filteredTasks}
|
||||||
onCreateTask={createTask}
|
onCreateTask={handleCreateTask}
|
||||||
onDeleteTask={deleteTask}
|
onDeleteTask={deleteTask}
|
||||||
onEditTask={handleEditTask}
|
onEditTask={handleEditTask}
|
||||||
onUpdateTitle={handleUpdateTitle}
|
onUpdateTitle={handleUpdateTitle}
|
||||||
@@ -107,7 +112,7 @@ export function KanbanBoardContainer({
|
|||||||
) : (
|
) : (
|
||||||
<SwimlanesBoard
|
<SwimlanesBoard
|
||||||
tasks={filteredTasks}
|
tasks={filteredTasks}
|
||||||
onCreateTask={createTask}
|
onCreateTask={handleCreateTask}
|
||||||
onDeleteTask={deleteTask}
|
onDeleteTask={deleteTask}
|
||||||
onEditTask={handleEditTask}
|
onEditTask={handleEditTask}
|
||||||
onUpdateTitle={handleUpdateTitle}
|
onUpdateTitle={handleUpdateTitle}
|
||||||
@@ -120,12 +125,11 @@ export function KanbanBoardContainer({
|
|||||||
) : (
|
) : (
|
||||||
<KanbanBoard
|
<KanbanBoard
|
||||||
tasks={filteredTasks}
|
tasks={filteredTasks}
|
||||||
onCreateTask={createTask}
|
onCreateTask={handleCreateTask}
|
||||||
onDeleteTask={deleteTask}
|
onDeleteTask={deleteTask}
|
||||||
onEditTask={handleEditTask}
|
onEditTask={handleEditTask}
|
||||||
onUpdateTitle={handleUpdateTitle}
|
onUpdateTitle={handleUpdateTitle}
|
||||||
onUpdateStatus={handleUpdateStatus}
|
onUpdateStatus={handleUpdateStatus}
|
||||||
loading={loading}
|
|
||||||
compactView={kanbanFilters.compactView}
|
compactView={kanbanFilters.compactView}
|
||||||
visibleStatuses={visibleStatuses}
|
visibleStatuses={visibleStatuses}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import { SwimlanesBase, SwimlaneData } from './SwimlanesBase';
|
|||||||
|
|
||||||
interface PrioritySwimlanesoardProps {
|
interface PrioritySwimlanesoardProps {
|
||||||
tasks: Task[];
|
tasks: Task[];
|
||||||
onCreateTask?: (data: CreateTaskData) => Promise<Task | null>;
|
onCreateTask?: (data: CreateTaskData) => Promise<void>;
|
||||||
onDeleteTask?: (taskId: string) => Promise<void>;
|
onDeleteTask?: (taskId: string) => Promise<void>;
|
||||||
onEditTask?: (task: Task) => void;
|
onEditTask?: (task: Task) => void;
|
||||||
onUpdateTitle?: (taskId: string, newTitle: string) => Promise<void>;
|
onUpdateTitle?: (taskId: string, newTitle: string) => Promise<void>;
|
||||||
@@ -64,7 +64,6 @@ export function PrioritySwimlanesBoard({
|
|||||||
return (
|
return (
|
||||||
<SwimlanesBase
|
<SwimlanesBase
|
||||||
tasks={tasks}
|
tasks={tasks}
|
||||||
title="Swimlanes par Priorité"
|
|
||||||
swimlanes={swimlanesData}
|
swimlanes={swimlanesData}
|
||||||
onCreateTask={onCreateTask}
|
onCreateTask={onCreateTask}
|
||||||
onDeleteTask={onDeleteTask}
|
onDeleteTask={onDeleteTask}
|
||||||
@@ -73,7 +72,6 @@ export function PrioritySwimlanesBoard({
|
|||||||
onUpdateStatus={onUpdateStatus}
|
onUpdateStatus={onUpdateStatus}
|
||||||
compactView={compactView}
|
compactView={compactView}
|
||||||
visibleStatuses={visibleStatuses}
|
visibleStatuses={visibleStatuses}
|
||||||
loading={loading}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -3,8 +3,6 @@
|
|||||||
import { Task, TaskStatus } from '@/lib/types';
|
import { Task, TaskStatus } from '@/lib/types';
|
||||||
import { TaskCard } from './TaskCard';
|
import { TaskCard } from './TaskCard';
|
||||||
import { QuickAddTask } from './QuickAddTask';
|
import { QuickAddTask } from './QuickAddTask';
|
||||||
import { CreateTaskForm } from '@/components/forms/CreateTaskForm';
|
|
||||||
import { Button } from '@/components/ui/Button';
|
|
||||||
import { CreateTaskData } from '@/clients/tasks-client';
|
import { CreateTaskData } from '@/clients/tasks-client';
|
||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
import { useColumnVisibility } from '@/hooks/useColumnVisibility';
|
import { useColumnVisibility } from '@/hooks/useColumnVisibility';
|
||||||
@@ -115,21 +113,18 @@ export interface SwimlaneData {
|
|||||||
|
|
||||||
interface SwimlanesBaseProps {
|
interface SwimlanesBaseProps {
|
||||||
tasks: Task[];
|
tasks: Task[];
|
||||||
title: string;
|
|
||||||
swimlanes: SwimlaneData[];
|
swimlanes: SwimlaneData[];
|
||||||
onCreateTask?: (data: CreateTaskData) => Promise<Task | null>;
|
onCreateTask?: (data: CreateTaskData) => Promise<void>;
|
||||||
onDeleteTask?: (taskId: string) => Promise<void>;
|
onDeleteTask?: (taskId: string) => Promise<void>;
|
||||||
onEditTask?: (task: Task) => void;
|
onEditTask?: (task: Task) => void;
|
||||||
onUpdateTitle?: (taskId: string, newTitle: string) => Promise<void>;
|
onUpdateTitle?: (taskId: string, newTitle: string) => Promise<void>;
|
||||||
onUpdateStatus?: (taskId: string, newStatus: TaskStatus) => Promise<void>;
|
onUpdateStatus?: (taskId: string, newStatus: TaskStatus) => Promise<void>;
|
||||||
compactView?: boolean;
|
compactView?: boolean;
|
||||||
visibleStatuses?: TaskStatus[];
|
visibleStatuses?: TaskStatus[];
|
||||||
loading?: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export function SwimlanesBase({
|
export function SwimlanesBase({
|
||||||
tasks,
|
tasks,
|
||||||
title,
|
|
||||||
swimlanes,
|
swimlanes,
|
||||||
onCreateTask,
|
onCreateTask,
|
||||||
onDeleteTask,
|
onDeleteTask,
|
||||||
@@ -137,12 +132,10 @@ export function SwimlanesBase({
|
|||||||
onUpdateTitle,
|
onUpdateTitle,
|
||||||
onUpdateStatus,
|
onUpdateStatus,
|
||||||
compactView = false,
|
compactView = false,
|
||||||
visibleStatuses,
|
visibleStatuses
|
||||||
loading = false
|
|
||||||
}: SwimlanesBaseProps) {
|
}: SwimlanesBaseProps) {
|
||||||
const [activeTask, setActiveTask] = useState<Task | null>(null);
|
const [activeTask, setActiveTask] = useState<Task | null>(null);
|
||||||
const [collapsedSwimlanes, setCollapsedSwimlanes] = useState<Set<string>>(new Set());
|
const [collapsedSwimlanes, setCollapsedSwimlanes] = useState<Set<string>>(new Set());
|
||||||
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
|
||||||
const [showQuickAdd, setShowQuickAdd] = useState<{ [key: string]: boolean }>({});
|
const [showQuickAdd, setShowQuickAdd] = useState<{ [key: string]: boolean }>({});
|
||||||
|
|
||||||
// Gestion de la visibilité des colonnes
|
// Gestion de la visibilité des colonnes
|
||||||
@@ -194,12 +187,6 @@ export function SwimlanesBase({
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Handlers pour la création de tâches
|
// Handlers pour la création de tâches
|
||||||
const handleCreateTask = async (data: CreateTaskData) => {
|
|
||||||
if (onCreateTask) {
|
|
||||||
await onCreateTask(data);
|
|
||||||
setIsCreateModalOpen(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleQuickAdd = async (data: CreateTaskData, columnId: string) => {
|
const handleQuickAdd = async (data: CreateTaskData, columnId: string) => {
|
||||||
if (onCreateTask) {
|
if (onCreateTask) {
|
||||||
@@ -220,24 +207,8 @@ export function SwimlanesBase({
|
|||||||
onDragEnd={handleDragEnd}
|
onDragEnd={handleDragEnd}
|
||||||
>
|
>
|
||||||
<div className="flex flex-col h-full bg-[var(--background)]">
|
<div className="flex flex-col h-full bg-[var(--background)]">
|
||||||
{/* Header */}
|
{/* Espacement supérieur */}
|
||||||
<div className="flex-shrink-0 px-6 py-4 border-b border-[var(--border)]/50">
|
<div className="flex-shrink-0 py-2"></div>
|
||||||
<div className="flex items-center justify-between">
|
|
||||||
<h2 className="text-lg font-mono font-bold text-[var(--foreground)] uppercase tracking-wider">
|
|
||||||
{title}
|
|
||||||
</h2>
|
|
||||||
|
|
||||||
{onCreateTask && (
|
|
||||||
<Button
|
|
||||||
variant="primary"
|
|
||||||
onClick={() => setIsCreateModalOpen(true)}
|
|
||||||
disabled={loading}
|
|
||||||
>
|
|
||||||
+ Nouvelle tâche
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
{/* Headers des colonnes visibles */}
|
{/* Headers des colonnes visibles */}
|
||||||
@@ -341,15 +312,6 @@ export function SwimlanesBase({
|
|||||||
)}
|
)}
|
||||||
</DragOverlay>
|
</DragOverlay>
|
||||||
|
|
||||||
{/* Modal de création complète */}
|
|
||||||
{onCreateTask && (
|
|
||||||
<CreateTaskForm
|
|
||||||
isOpen={isCreateModalOpen}
|
|
||||||
onClose={() => setIsCreateModalOpen(false)}
|
|
||||||
onSubmit={handleCreateTask}
|
|
||||||
loading={loading}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</DndContext>
|
</DndContext>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import { SwimlanesBase, SwimlaneData } from './SwimlanesBase';
|
|||||||
|
|
||||||
interface SwimlanesboardProps {
|
interface SwimlanesboardProps {
|
||||||
tasks: Task[];
|
tasks: Task[];
|
||||||
onCreateTask?: (data: CreateTaskData) => Promise<Task | null>;
|
onCreateTask?: (data: CreateTaskData) => Promise<void>;
|
||||||
onDeleteTask?: (taskId: string) => Promise<void>;
|
onDeleteTask?: (taskId: string) => Promise<void>;
|
||||||
onEditTask?: (task: Task) => void;
|
onEditTask?: (task: Task) => void;
|
||||||
onUpdateTitle?: (taskId: string, newTitle: string) => Promise<void>;
|
onUpdateTitle?: (taskId: string, newTitle: string) => Promise<void>;
|
||||||
@@ -84,7 +84,6 @@ export function SwimlanesBoard({
|
|||||||
return (
|
return (
|
||||||
<SwimlanesBase
|
<SwimlanesBase
|
||||||
tasks={tasks}
|
tasks={tasks}
|
||||||
title="Swimlanes par Tag"
|
|
||||||
swimlanes={swimlanesData}
|
swimlanes={swimlanesData}
|
||||||
onCreateTask={onCreateTask}
|
onCreateTask={onCreateTask}
|
||||||
onDeleteTask={onDeleteTask}
|
onDeleteTask={onDeleteTask}
|
||||||
@@ -93,7 +92,6 @@ export function SwimlanesBoard({
|
|||||||
onUpdateStatus={onUpdateStatus}
|
onUpdateStatus={onUpdateStatus}
|
||||||
compactView={compactView}
|
compactView={compactView}
|
||||||
visibleStatuses={visibleStatuses}
|
visibleStatuses={visibleStatuses}
|
||||||
loading={loading}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user