@taskctrl/canvas-timeline 0.1.0 → 0.1.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,35 @@
1
+ import type { Item } from '../types';
2
+ export type InteractionMode = 'move' | 'resize-left' | 'resize-right';
3
+ export interface InteractionState {
4
+ item: Item;
5
+ mode: InteractionMode;
6
+ startX: number;
7
+ startY: number;
8
+ currentX: number;
9
+ currentY: number;
10
+ deltaX: number;
11
+ originalGroup: string | number;
12
+ currentGroup: string | number;
13
+ }
14
+ export declare class InteractionHandler {
15
+ private state;
16
+ private dragSnap;
17
+ private activated;
18
+ constructor(dragSnap: number);
19
+ startInteraction(item: Item, mode: InteractionMode, x: number, y: number): void;
20
+ update(x: number, y: number): void;
21
+ setCurrentGroup(groupId: string | number): void;
22
+ endMove(pixelsPerMs: number): {
23
+ newStartTime: number;
24
+ newGroupId: string | number;
25
+ } | null;
26
+ endResize(pixelsPerMs: number): {
27
+ newTime: number;
28
+ edge: 'left' | 'right';
29
+ } | null;
30
+ cancel(): void;
31
+ getState(): InteractionState | null;
32
+ getMode(): InteractionMode | null;
33
+ isActive(): boolean;
34
+ isPending(): boolean;
35
+ }
@@ -0,0 +1,9 @@
1
+ export declare class ScrollHandler {
2
+ private onScrollVertical;
3
+ private onScrollHorizontal;
4
+ constructor(onScrollVertical: (deltaY: number) => void, onScrollHorizontal: (deltaX: number) => void);
5
+ handleWheel(event: WheelEvent): void;
6
+ handleVerticalScroll(deltaY: number): void;
7
+ handleHorizontalScroll(deltaX: number): void;
8
+ attach(element: HTMLElement): () => void;
9
+ }
@@ -0,0 +1,10 @@
1
+ export declare class ZoomHandler {
2
+ private onZoom;
3
+ private visibleTimeStart;
4
+ private visibleTimeEnd;
5
+ private minZoom;
6
+ private maxZoom;
7
+ constructor(onZoom: (newStart: number, newEnd: number) => void, visibleTimeStart: number, visibleTimeEnd: number, minZoom: number, maxZoom: number);
8
+ updateBounds(start: number, end: number): void;
9
+ handleWheelZoom(e: WheelEvent, cursorRatio: number): void;
10
+ }
@@ -0,0 +1,149 @@
1
+ import type React from 'react';
2
+ export interface Group {
3
+ id: number | string;
4
+ title: string;
5
+ type?: string;
6
+ hierarchy?: string;
7
+ root?: boolean;
8
+ [key: string]: unknown;
9
+ }
10
+ export interface Item {
11
+ id: number;
12
+ group: number | string;
13
+ start_time: number;
14
+ end_time: number;
15
+ type?: string;
16
+ [key: string]: unknown;
17
+ }
18
+ export interface ItemBounds {
19
+ x: number;
20
+ y: number;
21
+ width: number;
22
+ height: number;
23
+ }
24
+ export interface ItemState {
25
+ selected: boolean;
26
+ hovered: boolean;
27
+ dragging: boolean;
28
+ filtered: boolean;
29
+ }
30
+ export interface DrawHelpers {
31
+ roundRect(x: number, y: number, w: number, h: number, radius?: number): void;
32
+ fillText(text: string, x: number, y: number, maxWidth?: number): void;
33
+ gradient(x: number, w: number, color1: string, color2: string): CanvasGradient;
34
+ leftBar(color: string, width?: number): void;
35
+ icon(type: 'check' | 'danger-red' | 'danger-yellow', x: number, y: number, size?: number): void;
36
+ badge(text: string, x: number, y: number, bgColor: string): void;
37
+ }
38
+ export type CanvasItemRenderer = (ctx: CanvasRenderingContext2D, item: Item, bounds: ItemBounds, state: ItemState, helpers: DrawHelpers) => void;
39
+ export type CanvasGroupItemRenderer = CanvasItemRenderer;
40
+ export interface DayStyle {
41
+ backgroundColor?: string;
42
+ borderColor?: string;
43
+ opacity?: number;
44
+ }
45
+ export interface RowStyle {
46
+ backgroundColor?: string;
47
+ borderBottomColor?: string;
48
+ }
49
+ export interface Dependency {
50
+ fromItemId: number;
51
+ toItemId: number;
52
+ type?: 'finish-to-start' | 'start-to-start' | 'finish-to-finish';
53
+ color?: string;
54
+ }
55
+ export interface TimelineTheme {
56
+ primary: string;
57
+ trainColors: Record<string, string>;
58
+ status: {
59
+ red: string;
60
+ yellow: string;
61
+ green: string;
62
+ };
63
+ grid: {
64
+ line: string;
65
+ rowAlt: string;
66
+ weekend: string;
67
+ };
68
+ item: {
69
+ radius: number;
70
+ text: string;
71
+ selectedRing: string;
72
+ };
73
+ marker: {
74
+ today: string;
75
+ milestone: string;
76
+ cursor: string;
77
+ };
78
+ sidebar: {
79
+ bg: string;
80
+ border: string;
81
+ text: string;
82
+ };
83
+ header: {
84
+ bg: string;
85
+ border: string;
86
+ text: string;
87
+ };
88
+ }
89
+ export declare const DEFAULT_THEME: TimelineTheme;
90
+ export interface MarkerConfig {
91
+ date: number;
92
+ color: string;
93
+ width: number;
94
+ label?: string;
95
+ }
96
+ export interface CanvasTimelineProps {
97
+ groups: Group[];
98
+ items: Item[];
99
+ defaultTimeStart: number;
100
+ defaultTimeEnd: number;
101
+ visibleTimeStart?: number;
102
+ visibleTimeEnd?: number;
103
+ sidebarWidth: number;
104
+ lineHeight: number;
105
+ itemHeightRatio: number;
106
+ stackItems: boolean;
107
+ buffer?: number;
108
+ canMove: boolean;
109
+ canResize: false | 'left' | 'right' | 'both';
110
+ canChangeGroup: boolean;
111
+ dragSnap: number;
112
+ minZoom: number;
113
+ maxZoom: number;
114
+ theme?: Partial<TimelineTheme>;
115
+ dayStyle?: (date: Date) => DayStyle | null;
116
+ rowStyle?: (group: Group) => RowStyle | null;
117
+ showCursorLine?: boolean;
118
+ itemRenderer: CanvasItemRenderer;
119
+ groupRenderer?: CanvasGroupItemRenderer;
120
+ sidebarGroupRenderer: (group: Group) => React.ReactNode;
121
+ rightSidebarWidth?: number;
122
+ rightSidebarGroupRenderer?: (group: Group) => React.ReactNode;
123
+ dependencies?: Dependency[];
124
+ onItemClick?: (itemId: number, e: PointerEvent) => void;
125
+ onItemDoubleClick?: (itemId: number, e: PointerEvent) => void;
126
+ onItemContextMenu?: (itemId: number, e: PointerEvent) => void;
127
+ onItemMove?: (itemId: number, newStartTime: number, newGroupId: string | number) => void;
128
+ onItemResize?: (itemId: number, newTime: number, edge: 'left' | 'right') => void;
129
+ /** Validate and optionally constrain move/resize. Return the (possibly modified) time. */
130
+ moveResizeValidator?: (action: 'move' | 'resize', itemId: number, time: number, edge?: 'left' | 'right') => number;
131
+ onItemHover?: (itemId: number | null, e: PointerEvent) => void;
132
+ onCanvasDoubleClick?: (groupId: number, time: number) => void;
133
+ onCanvasContextMenu?: (groupId: number, time: number, e: PointerEvent) => void;
134
+ onTimeChange?: (start: number, end: number) => void;
135
+ onZoom?: (start: number, end: number) => void;
136
+ selected?: number[];
137
+ /** Called when the timeline is ready, providing the capture API */
138
+ onReady?: (api: CanvasTimelineRef) => void;
139
+ children?: React.ReactNode;
140
+ }
141
+ export interface CaptureOptions {
142
+ timeStart: number;
143
+ timeEnd: number;
144
+ scale: number;
145
+ sidebarWidth: number;
146
+ }
147
+ export interface CanvasTimelineRef {
148
+ captureToCanvas(options: CaptureOptions): HTMLCanvasElement;
149
+ }
package/package.json CHANGED
@@ -1,13 +1,18 @@
1
1
  {
2
2
  "name": "@taskctrl/canvas-timeline",
3
- "version": "0.1.0",
3
+ "version": "0.1.2",
4
4
  "description": "High-performance canvas-based timeline component for React",
5
5
  "scripts": {
6
6
  "build": "rimraf ./dist && tsc && vite build",
7
7
  "test": "vitest run",
8
8
  "test:watch": "vitest",
9
9
  "test:coverage": "vitest run --coverage",
10
- "lint": "eslint --ext .ts --ext .tsx ./src"
10
+ "lint": "eslint --ext .ts --ext .tsx ./src",
11
+ "release": "yarn release:patch",
12
+ "release:patch": "npm version patch -m \"chore: release v%s\" && git push origin HEAD --follow-tags",
13
+ "release:minor": "npm version minor -m \"chore: release v%s\" && git push origin HEAD --follow-tags",
14
+ "release:major": "npm version major -m \"chore: release v%s\" && git push origin HEAD --follow-tags",
15
+ "release:beta": "npm version prerelease --preid=beta -m \"chore: release v%s\" && git push origin HEAD --follow-tags"
11
16
  },
12
17
  "main": "./dist/canvas-timeline.cjs.js",
13
18
  "module": "./dist/canvas-timeline.es.js",