Compare commits
	
		
			5 Commits
		
	
	
		
			release/0.
			...
			master
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 06f0a9b4ac | |||
| 0b43e4ecd7 | |||
|   | e279321367 | ||
|   | fee71b651c | ||
|   | d05d22afe7 | 
| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "LiteRyzJS/Project", |   "name": "LiteRyzJS/Project", | ||||||
|   "version": "0.2.0.508", |   "version": "0.2.1.114", | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "css-loader": "^7.1.2", |     "css-loader": "^7.1.2", | ||||||
|     "sass": "^1.77.8", |     "sass": "^1.77.8", | ||||||
|  | |||||||
							
								
								
									
										204
									
								
								src/project/background-canvas.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										204
									
								
								src/project/background-canvas.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,204 @@ | |||||||
|  | import Canvas from '../references/canvas.js'; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class BackgroundCanvas extends Canvas { | ||||||
|  |   Options = null; | ||||||
|  |   Debug = false; | ||||||
|  | 
 | ||||||
|  |   StartDate = new Date(); | ||||||
|  |   Duration = 30; | ||||||
|  |   StartOfWeek = 1; | ||||||
|  |   RowCount = 8; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   constructor(el) { | ||||||
|  |     super(el); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   get HeaderRow1Rectangle() { | ||||||
|  |     return { | ||||||
|  |       X: 0, | ||||||
|  |       Y: 0, | ||||||
|  |       W: this.Width, | ||||||
|  |       H: (this.Options.HeaderRow.Height[0] - this.Options.BorderWidth) | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get HeaderRow2Rectangle() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     return { | ||||||
|  |       X: 0, | ||||||
|  |       Y: this.Options.HeaderRow.Height[0], | ||||||
|  |       W: this.Width, | ||||||
|  |       H: (this.Options.HeaderRow.Height[1] - this.Options.BorderWidth) | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get HeaderHeight() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     return a.Options.HeaderRow.Height[0] + a.Options.HeaderRow.Height[1]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   Load(startDate, duration, startOfWeek, rowCount) { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     a.StartDate = new Date(startDate); | ||||||
|  |     a.Duration = duration; | ||||||
|  |     a.StartOfWeek = startOfWeek;    // 1 = Monday
 | ||||||
|  |     a.RowCount = rowCount; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   Invalidate() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     if (a.Options == null) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (a.StartDate == null) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     a.#drawChartHeader(); | ||||||
|  |     a.#drawColumnLayout(); | ||||||
|  |     a.#drawRowLayout(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   #drawChartHeader() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     const displayDays = a.Duration + 2; | ||||||
|  | 
 | ||||||
|  |     if (a.Debug) { | ||||||
|  |       a.DrawRectangle(a.HeaderRow1Rectangle, "red", {}); | ||||||
|  |       a.DrawRectangle(a.HeaderRow2Rectangle, "orange", {}); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let startDate = new Date(a.StartDate); | ||||||
|  |     startDate.addDays(-1); | ||||||
|  | 
 | ||||||
|  |     // Draw horizontal line under months
 | ||||||
|  |     a.#drawHorizontalLine(a.Options.HeaderRow.Height[0]); | ||||||
|  | 
 | ||||||
|  |     // Draw vertical lines for dates
 | ||||||
|  |     for (let i=1; i<displayDays; i++) { | ||||||
|  |       a.DrawVerticalLine((a.Options.DayWidth * i), (a.Options.HeaderRow.Height[0] + a.Options.BorderWidth), (a.Options.HeaderRow.Height[1] - a.Options.BorderWidth), a.Options.BorderColour, {}); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Draw horizontal line under dates
 | ||||||
|  |     a.#drawHorizontalLine(a.HeaderHeight); | ||||||
|  | 
 | ||||||
|  |     // Write dates
 | ||||||
|  |     for (let i=0; i<displayDays; i++) { | ||||||
|  |       const date = Date.addDays(startDate, i); | ||||||
|  |       const x = (a.Options.DayWidth * i); | ||||||
|  | 
 | ||||||
|  |       // Draw month
 | ||||||
|  |       if (date.getDate() == 1) { | ||||||
|  |         const size = a.MeasureText(a.Options.DateFont, "#"); | ||||||
|  |         const monthPoint = { | ||||||
|  |           X: (x + 2), | ||||||
|  |           Y: Math.half(a.Options.HeaderRow.Height[0] - size.H) | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         a.DrawText(monthPoint.X, monthPoint.Y, date.toCString("MMMM"), a.Options.DateFont, a.Options.DateForeColour); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       // Draw day
 | ||||||
|  |       const dateRectangle = { | ||||||
|  |         X: x, | ||||||
|  |         Y: a.Options.HeaderRow.Height[0], | ||||||
|  |         W: a.Options.DayWidth - a.Options.BorderWidth, | ||||||
|  |         H: (a.Options.HeaderRow.Height[1] - a.Options.BorderWidth) | ||||||
|  |       }; | ||||||
|  | 
 | ||||||
|  |       a.FillText(dateRectangle, date.getDate(), a.Options.DateFont, a.Options.DateForeColour); | ||||||
|  | 
 | ||||||
|  |       if (a.Debug) { | ||||||
|  |         a.DrawRectangle(dateRectangle, "red", {}); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   #drawColumnLayout() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     const height = a.Height; | ||||||
|  |     const displayDays = a.Duration + 2; | ||||||
|  | 
 | ||||||
|  |     let startDate = new Date(a.StartDate); | ||||||
|  |     startDate.addDays(-1); | ||||||
|  | 
 | ||||||
|  |     // Write dates
 | ||||||
|  |     for (let i=0; i<displayDays; i++) { | ||||||
|  |       const date = Date.addDays(startDate, i); | ||||||
|  |       const x = (a.Options.DayWidth * i); | ||||||
|  | 
 | ||||||
|  |       const dateRectangle = { | ||||||
|  |         X: x, | ||||||
|  |         Y: a.HeaderHeight, | ||||||
|  |         W: a.Options.DayWidth - a.Options.BorderWidth, | ||||||
|  |         H: height - a.Options.BorderWidth | ||||||
|  |       }; | ||||||
|  | 
 | ||||||
|  |       // Fill background for Saturday and Sunday
 | ||||||
|  |       if (date.getDay() == 6) { | ||||||
|  |         a.DrawRectangle(dateRectangle, a.Options.Column.SatColour, { FillColour: a.Options.Column.SatColour }); | ||||||
|  |       } else if (date.getDay() == 0) { | ||||||
|  |         a.DrawRectangle(dateRectangle, a.Options.Column.SunColour, { FillColour: a.Options.Column.SunColour }); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       // Draw vertical date lines
 | ||||||
|  |       if (a.StartOfWeek == date.getDay()) { | ||||||
|  |         if (a.Options.ShowStartDayOfWeekLine) { | ||||||
|  |           a.DrawVerticalLine(dateRectangle.X, a.HeaderHeight, dateRectangle.H, a.Options.BorderColour, {}); | ||||||
|  |         } | ||||||
|  |       } else { | ||||||
|  |         if (a.Options.ShowDateLines) { | ||||||
|  |           a.DrawVerticalLine(dateRectangle.X, a.HeaderHeight, dateRectangle.H, a.Options.BorderColour, { LineDash: a.Options.BorderDashPattern }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   #drawRowLayout() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     for (let i=0; i<a.RowCount; i++) { | ||||||
|  |       const rowRectangle = { | ||||||
|  |         X: 0, | ||||||
|  |         Y: (a.HeaderHeight + (a.Options.Row.Height * i)), | ||||||
|  |         W: a.Width, | ||||||
|  |         H: (a.Options.Row.Height - a.Options.BorderWidth) | ||||||
|  |       }; | ||||||
|  | 
 | ||||||
|  |       if (a.Options.ShowRowStripes) { | ||||||
|  |         if ((i % 2) == 1) { | ||||||
|  |           a.DrawRectangle(rowRectangle, a.Options.Row.BackColour, { FillColour: a.Options.Row.BackColour }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       if (a.Options.ShowRowLines) { | ||||||
|  |         a.DrawHorizontalLine(0, (rowRectangle.Y + a.Options.Row.Height), rowRectangle.W, a.Options.BorderColour, { LineDash: a.Options.BorderDashPattern }); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   #drawHorizontalLine(y) { | ||||||
|  |     const a = this; | ||||||
|  |     const width = a.Width; | ||||||
|  | 
 | ||||||
|  |     a.DrawHorizontalLine(0, y, width, this.Options.BorderColour); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | export default BackgroundCanvas; | ||||||
							
								
								
									
										80
									
								
								src/project/flourish-canvas.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								src/project/flourish-canvas.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,80 @@ | |||||||
|  | import Canvas from '../references/canvas.js'; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class FlourishCanvas extends Canvas { | ||||||
|  |   Options = null; | ||||||
|  |   Debug = false; | ||||||
|  | 
 | ||||||
|  |   XPos = -1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   constructor(el) { | ||||||
|  |     super(el); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   get HeaderHeight() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     return a.Options.HeaderRow.Height[0] + a.Options.HeaderRow.Height[1]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   Load() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     a.el.addEventListener('mousemove', function (e) { | ||||||
|  |       // Hottracking
 | ||||||
|  |       if (a.Options.EnableHotTracking) { | ||||||
|  |         const point = { X: e.offsetX, Y: e.offsetY }; | ||||||
|  | 
 | ||||||
|  |         if (a.Debug) { | ||||||
|  |           console.log(point); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         a.XPos = point.X; | ||||||
|  | 
 | ||||||
|  |         a.Invalidate(); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     a.el.addEventListener('mouseout', function (e) { | ||||||
|  |       a.XPos = -1; | ||||||
|  | 
 | ||||||
|  |       a.Invalidate(); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   Invalidate() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     if (a.Options == null) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     a.Clear(); | ||||||
|  | 
 | ||||||
|  |     if (a.XPos < 0) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     a.#drawVerticalLine(a.XPos); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   #drawVerticalLine(x) { | ||||||
|  |     const a = this; | ||||||
|  |     const h = this.Height - a.HeaderHeight; | ||||||
|  | 
 | ||||||
|  |     if (a.Options.HotTrackLine.Width > 1) { | ||||||
|  |       x += Math.half(a.Options.HotTrackLine.Width); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     a.DrawVerticalLine(x, a.HeaderHeight, h, a.Options.HotTrackLine.Colour, { LineWidth: a.Options.HotTrackLine.Width }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | export default FlourishCanvas; | ||||||
							
								
								
									
										157
									
								
								src/project/foreground-canvas.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								src/project/foreground-canvas.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,157 @@ | |||||||
|  | import Canvas from '../references/canvas.js'; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class ForegroundCanvas extends Canvas { | ||||||
|  |   Options = null; | ||||||
|  |   Debug = false; | ||||||
|  | 
 | ||||||
|  |   StartDate = new Date(); | ||||||
|  |   Tasks = []; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   constructor(el) { | ||||||
|  |     super(el); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   get HeaderHeight() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     return a.Options.HeaderRow.Height[0] + a.Options.HeaderRow.Height[1]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   Load(startDate, tasks) { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     a.StartDate = new Date(startDate); | ||||||
|  |     a.Tasks = tasks; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   Invalidate() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     if (a.Options == null) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (a.StartDate == null) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     a.#drawTasks(); | ||||||
|  |     a.#drawConnectorLines(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   #drawTasks() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     for (let i=0; i<a.Tasks.length; i++) { | ||||||
|  |       let style = a.Options.Row.Task; | ||||||
|  | 
 | ||||||
|  |       if (a.Tasks[i].IsCollated == true) { | ||||||
|  |         style = a.Options.Row.CollatedTask; | ||||||
|  |       } else if (a.Tasks[i].CollatedTaskID != null) { | ||||||
|  |         style = a.Options.Row.ChildTask; | ||||||
|  |       } else if (a.Tasks[i].PredecessorTaskID == null) { | ||||||
|  |         style = a.Options.Row.OrphanTask; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       const rectangle = a.#getTaskRectangle(a.Tasks[i], style); | ||||||
|  | 
 | ||||||
|  |       if (a.Tasks[i].Duration <= 0) { | ||||||
|  |         a.DrawDiamond(rectangle, style.BorderColour, { FillColour: style.FillColour }); | ||||||
|  |       } else { | ||||||
|  |         a.DrawRectangle(rectangle, style.BorderColour, { FillColour: style.FillColour }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   #drawConnectorLines() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     for (let i=0; i<a.Tasks.length; i++) { | ||||||
|  |       if (a.Tasks[i].PredecessorTaskID == null) { | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       if (a.Tasks[i].PredecessorTaskID == null) { | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       const style = ((a.Tasks[i].IsCollated == true) ? a.Options.Row.CollatedTask : a.Options.Row.Task); | ||||||
|  |       const rectangle = a.#getTaskRectangle(a.Tasks[i], style); | ||||||
|  |       const paddingX = (a.Options.Row.CollatedTask.Height + a.Options.BorderWidth); | ||||||
|  |       const offsetX = (rectangle.X + paddingX); | ||||||
|  | 
 | ||||||
|  |       const predecessorTask = a.Tasks.first("ID", a.Tasks[i].PredecessorTaskID); | ||||||
|  |       if (predecessorTask == null) { | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       const predecessorStyle = ((predecessorTask.IsCollated == true) ? a.Options.Row.CollatedTask : a.Options.Row.Task); | ||||||
|  |       const predecessorRectangle = a.#getTaskRectangle(predecessorTask, predecessorStyle); | ||||||
|  | 
 | ||||||
|  |       let points = []; | ||||||
|  |       points.push({ X: (predecessorRectangle.X + predecessorRectangle.W), Y: (predecessorRectangle.Y + Math.half(predecessorRectangle.H)) }); | ||||||
|  |       points.push({ X: offsetX, Y: (predecessorRectangle.Y + Math.half(predecessorRectangle.H)) }); | ||||||
|  | 
 | ||||||
|  |       points.push({ X: offsetX, Y: (rectangle.Y - paddingX) }); | ||||||
|  |       points.push({ X: offsetX, Y: rectangle.Y }); | ||||||
|  | 
 | ||||||
|  |       a.DrawLines(points, a.Options.Line.Colour, { LineWidth: a.Options.Line.Width }); | ||||||
|  | 
 | ||||||
|  |       const arrowRectangle = { | ||||||
|  |         X: (offsetX - Math.half(a.Options.Line.ArrowSize)), | ||||||
|  |         Y: (rectangle.Y - a.Options.Line.ArrowSize), | ||||||
|  |         W: a.Options.Line.ArrowSize, | ||||||
|  |         H: a.Options.Line.ArrowSize | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       a.DrawArrowS(arrowRectangle, a.Options.Line.Colour, { FillColour: a.Options.Line.Colour }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   #getTaskRectangle(task, style) { | ||||||
|  |     const a = this; | ||||||
|  |     const i = (task.Order - 1); | ||||||
|  | 
 | ||||||
|  |     const rowRectangle = { | ||||||
|  |       X: 0, | ||||||
|  |       Y: (a.HeaderHeight + (a.Options.Row.Height * i)), | ||||||
|  |       W: a.Width, | ||||||
|  |       H: (a.Options.Row.Height - a.Options.BorderWidth) | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     if (a.Debug) { | ||||||
|  |       a.DrawRectangle(rowRectangle, "red", {}); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (task.Duration <= 0) { | ||||||
|  |       let result = { | ||||||
|  |         X: (a.Options.DayWidth * (Date.diffDays(a.StartDate, task.StartDate) + 1)), | ||||||
|  |         Y: (a.HeaderHeight + (a.Options.Row.Height * i) + Math.half(rowRectangle.H - style.Height)), | ||||||
|  |         W: style.Height, | ||||||
|  |         H: style.Height | ||||||
|  |       }; | ||||||
|  | 
 | ||||||
|  |       result.X += Math.half(a.Options.DayWidth - (style.Height + a.Options.BorderWidth)); | ||||||
|  | 
 | ||||||
|  |       return result; | ||||||
|  |     } else { | ||||||
|  |       return { | ||||||
|  |         X: (a.Options.DayWidth * (Date.diffDays(a.StartDate, task.StartDate) + 1)), | ||||||
|  |         Y: (a.HeaderHeight + (a.Options.Row.Height * i) + Math.half(rowRectangle.H - style.Height)), | ||||||
|  |         W: (a.Options.DayWidth * task.Duration), | ||||||
|  |         H: style.Height | ||||||
|  |       }; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | export default ForegroundCanvas; | ||||||
							
								
								
									
										58
									
								
								src/project/gantt-chart-options.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/project/gantt-chart-options.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | |||||||
|  | class GanttChartOptions { | ||||||
|  |   DayWidth = 24; | ||||||
|  |   HeaderRow = { | ||||||
|  |     Height: [ 20, 20 ] | ||||||
|  |   }; | ||||||
|  |   Row = { | ||||||
|  |     Height: 28, | ||||||
|  |     BackColour: "rgba(235, 235, 235, 0.3)", | ||||||
|  |     Task: { | ||||||
|  |       Height: 14, | ||||||
|  |       BorderColour: "#555555", | ||||||
|  |       FillColour: "#D8EEDB" | ||||||
|  |     }, | ||||||
|  |     OrphanTask: { | ||||||
|  |       Height: 14, | ||||||
|  |       BorderColour: "#555555", | ||||||
|  |       FillColour: "#9CC2E6" | ||||||
|  |     }, | ||||||
|  |     ChildTask: { | ||||||
|  |       Height: 14, | ||||||
|  |       BorderColour: "#555555", | ||||||
|  |       FillColour: "#FFF5C1" | ||||||
|  |     }, | ||||||
|  |     CollatedTask: { | ||||||
|  |       Height: 6, | ||||||
|  |       BorderColour: "#555555", | ||||||
|  |       FillColour: "#555555" | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |   Column = { | ||||||
|  |     SatColour: "rgba(233, 237, 239, 0.8)", | ||||||
|  |     SunColour: "rgba(233, 237, 239, 0.9)" | ||||||
|  |   }; | ||||||
|  |   Line = { | ||||||
|  |     Colour: "#555555", | ||||||
|  |     Width: 1, | ||||||
|  |     ArrowSize: 5 | ||||||
|  |   }; | ||||||
|  |   HotTrackLine = { | ||||||
|  |     Colour: "#D04437", | ||||||
|  |     Width: 1 | ||||||
|  |   }; | ||||||
|  |   DateFont = "7pt sans-serif"; | ||||||
|  |   DateForeColour = "#636363"; | ||||||
|  |   BorderWidth = 1; | ||||||
|  |   BorderColour = "#B8B8B8"; | ||||||
|  |   BorderDashPattern = [1, 1]; | ||||||
|  |   EnableHotTracking = true; | ||||||
|  |   MinimumRowCount = 6; | ||||||
|  |   ShowDateLines = true; | ||||||
|  |   ShowStartDayOfWeekLine = true; | ||||||
|  |   ShowRowLines = true; | ||||||
|  |   ShowRowStripes = true; | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | export default GanttChartOptions; | ||||||
| @ -1,20 +1,29 @@ | |||||||
| import Canvas from '../references/canvas.js'; | import GanttChartOptions from './gantt-chart-options.js'; | ||||||
|  | 
 | ||||||
|  | import CanvasContainer from '../references/canvas-container.js'; | ||||||
|  | import BackgroundCanvas from './background-canvas.js'; | ||||||
|  | import ForegroundCanvas from './foreground-canvas.js'; | ||||||
|  | import FlourishCanvas from './flourish-canvas.js'; | ||||||
| 
 | 
 | ||||||
| import './project.scss'; | import './project.scss'; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class GanttChart { | class GanttChart { | ||||||
|  |   Options = null; | ||||||
|  |   CanvasContainer = null; | ||||||
|  | 
 | ||||||
|  |   #debug = false; | ||||||
|  |   Tasks = []; | ||||||
|  |   StartDate = null; | ||||||
|  |   Duration = 30; | ||||||
|  |   StartOfWeek = 1; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|   constructor(el, options) { |   constructor(el, options) { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     a.Canvas = new Canvas(el); |     a.Options = Object.assign(new GanttChartOptions(), options); | ||||||
|     a.Options = Object.assign(a.DefaultOptions, options); |     a.CanvasContainer = new CanvasContainer(el); | ||||||
| 
 |  | ||||||
|     a.Debug = false; |  | ||||||
|     a.Tasks = []; |  | ||||||
|     a.StartDate = null; |  | ||||||
|     a.Duration = 30; |  | ||||||
|     a.StartOfWeek = 1; |  | ||||||
| 
 | 
 | ||||||
|     a.#initialiseComponents(); |     a.#initialiseComponents(); | ||||||
|   } |   } | ||||||
| @ -22,92 +31,37 @@ class GanttChart { | |||||||
|   #initialiseComponents() { |   #initialiseComponents() { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     if (!a.Canvas.el.classList.contains("literyzjs-project")) { |     // Add background canvas layer
 | ||||||
|       a.Canvas.el.classList.add("literyzjs-project"); |     const layer1 = a.CanvasContainer.AddLayer(); | ||||||
|  | 
 | ||||||
|  |     const backCanvasLayer = new BackgroundCanvas(layer1); | ||||||
|  |     backCanvasLayer.Options = a.Options; | ||||||
|  | 
 | ||||||
|  |     a.CanvasContainer.Layer.push(backCanvasLayer); | ||||||
|  | 
 | ||||||
|  |     // Add foreground canvas layer
 | ||||||
|  |     const layer2 = a.CanvasContainer.AddLayer(); | ||||||
|  | 
 | ||||||
|  |     const foreCanvasLayer = new ForegroundCanvas(layer2); | ||||||
|  |     foreCanvasLayer.Options = a.Options; | ||||||
|  | 
 | ||||||
|  |     a.CanvasContainer.Layer.push(foreCanvasLayer); | ||||||
|  | 
 | ||||||
|  |     // Add flourish canvas layer
 | ||||||
|  |     const layer3 = a.CanvasContainer.AddLayer(); | ||||||
|  | 
 | ||||||
|  |     const flourishCanvasLayer = new FlourishCanvas(layer3); | ||||||
|  |     flourishCanvasLayer.Options = a.Options; | ||||||
|  | 
 | ||||||
|  |     a.CanvasContainer.Layer.push(flourishCanvasLayer); | ||||||
|  | 
 | ||||||
|  |     // Invalidate every canvas
 | ||||||
|  |     a.CanvasContainer.Invalidate(); | ||||||
|  | 
 | ||||||
|  |     Document.removeClass(a.CanvasContainer.FlowContainer, "border"); | ||||||
|  |     Document.addClass(a.CanvasContainer.FlowContainer, "gantt-chart"); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|     if (a.Canvas.FlowContainer.classList.contains("border")) { |  | ||||||
|       a.Canvas.FlowContainer.classList.remove("border"); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (!a.Canvas.FlowContainer.classList.contains("gantt-chart")) { |  | ||||||
|       a.Canvas.FlowContainer.classList.add("gantt-chart"); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|   get DefaultOptions() { |  | ||||||
|     return { |  | ||||||
|       DayWidth: 24, |  | ||||||
|       HeaderRow: { |  | ||||||
|         Height: [ 20, 21 ] |  | ||||||
|       }, |  | ||||||
|       Row: { |  | ||||||
|         Height: 28, |  | ||||||
|         BackColour: "rgba(235, 235, 235, 0.3)", |  | ||||||
|         Task: { |  | ||||||
|           Height: 14, |  | ||||||
|           BorderColour: "#555555", |  | ||||||
|           FillColour: "#D8EEDB" |  | ||||||
|         }, |  | ||||||
|         OrphanTask: { |  | ||||||
|           Height: 14, |  | ||||||
|           BorderColour: "#555555", |  | ||||||
|           FillColour: "#9CC2E6" |  | ||||||
|         }, |  | ||||||
|         ChildTask: { |  | ||||||
|           Height: 14, |  | ||||||
|           BorderColour: "#555555", |  | ||||||
|           FillColour: "#FFF5C1" |  | ||||||
|         }, |  | ||||||
|         CollatedTask: { |  | ||||||
|           Height: 6, |  | ||||||
|           BorderColour: "#555555", |  | ||||||
|           FillColour: "#555555" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       Column: { |  | ||||||
|         SatColour: "rgba(233, 237, 239, 0.8)", |  | ||||||
|         SunColour: "rgba(233, 237, 239, 0.9)", |  | ||||||
|       }, |  | ||||||
|       Line: { |  | ||||||
|         Colour: "#555555", |  | ||||||
|         Width: 1, |  | ||||||
|         ArrowSize: 5 |  | ||||||
|       }, |  | ||||||
|       DateFont: "7pt sans-serif", |  | ||||||
|       DateForeColour: "#636363", |  | ||||||
|       BorderWidth: 1, |  | ||||||
|       BorderColour: "#B8B8B8", |  | ||||||
|       BorderDashPattern: [1, 1], |  | ||||||
|       MinimumRowCount: 6, |  | ||||||
|       ShowDateLines: true, |  | ||||||
|       ShowStartDayOfWeekLine: true, |  | ||||||
|       ShowRowLines: true, |  | ||||||
|       ShowRowStripes: true |  | ||||||
|     }; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get HeaderRow1Rectangle() { |  | ||||||
|     return { |  | ||||||
|       X: 0, |  | ||||||
|       Y: 0, |  | ||||||
|       W: this.Canvas.Width, |  | ||||||
|       H: (this.Options.HeaderRow.Height[0] - this.Options.BorderWidth) |  | ||||||
|     }; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get HeaderRow2Rectangle() { |  | ||||||
|     const a = this; |  | ||||||
| 
 |  | ||||||
|     return { |  | ||||||
|       X: 0, |  | ||||||
|       Y: this.Options.HeaderRow.Height[0], |  | ||||||
|       W: this.Canvas.Width, |  | ||||||
|       H: (this.Options.HeaderRow.Height[1] - this.Options.BorderWidth) |  | ||||||
|     }; |  | ||||||
|   } |  | ||||||
| 
 | 
 | ||||||
|   get HeaderHeight() { |   get HeaderHeight() { | ||||||
|     const a = this; |     const a = this; | ||||||
| @ -115,43 +69,64 @@ class GanttChart { | |||||||
|     return a.Options.HeaderRow.Height[0] + a.Options.HeaderRow.Height[1]; |     return a.Options.HeaderRow.Height[0] + a.Options.HeaderRow.Height[1]; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   get Debug() { | ||||||
|  |     return this.#debug; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   set Debug(value) { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     a.#debug = value; | ||||||
|  | 
 | ||||||
|  |     a.CanvasContainer.Layer.forEach(e => { | ||||||
|  |       if (typeof(e.Debug) == "undefined") { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       e.Debug = value; | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
|   Clear() { |   Clear() { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     a.Canvas.Clear(); |     a.CanvasContainer.Clear(); | ||||||
| 
 | 
 | ||||||
|     a.StartDate = new Date(); |     a.StartDate = new Date(); | ||||||
|     a.Duration = 30; |     a.Duration = 30; | ||||||
|     a.StartOfWeek = 1;    // 1 = Monday
 |     a.StartOfWeek = 1;    // 1 = Monday
 | ||||||
|     a.Tasks = []; |     a.Tasks = []; | ||||||
| 
 | 
 | ||||||
|  |     // Invalidate canvas (background, foreground)
 | ||||||
|  |     a.CanvasContainer.Layer[0].Load(new Date(), a.Duration, a.StartOfWeek, a.Tasks.length); | ||||||
|  |     a.CanvasContainer.Layer[1].Load(new Date(), a.Tasks); | ||||||
|  |     a.CanvasContainer.Layer[2].Load(); | ||||||
|  | 
 | ||||||
|     a.Invalidate(); |     a.Invalidate(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   Invalidate() { |   Invalidate() { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     a.Canvas.Clear(); |     a.CanvasContainer.Clear(); | ||||||
| 
 | 
 | ||||||
|     const width = ((a.Duration + 2) * a.Options.DayWidth); |     const width = ((a.Duration + 2) * a.Options.DayWidth); | ||||||
|     const height = (Math.max(a.Tasks.length, a.Options.MinimumRowCount) * a.Options.Row.Height) + a.HeaderHeight; |     const height = (Math.max(a.Tasks.length, a.Options.MinimumRowCount) * a.Options.Row.Height) + a.HeaderHeight; | ||||||
| 
 | 
 | ||||||
|     a.Canvas.Size = { W: width, H: height }; |     a.CanvasContainer.Size = { | ||||||
|  |       W: width, | ||||||
|  |       H: height | ||||||
|  |     }; | ||||||
| 
 | 
 | ||||||
|     a.Canvas.Invalidate(); |     a.CanvasContainer.Invalidate(); | ||||||
| 
 |  | ||||||
|     a.#drawChartHeader(); |  | ||||||
|     a.#drawColumnLayout(); |  | ||||||
|     a.#drawRowLayout(); |  | ||||||
|     a.#drawTasks(); |  | ||||||
|     a.#drawConnectorLines(); |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   Load(project) { |   Load(project) { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     a.Canvas.Clear(); |     a.CanvasContainer.Clear(); | ||||||
| 
 | 
 | ||||||
|     if (project == null) { |     if (project == null) { | ||||||
|       return; |       return; | ||||||
| @ -162,244 +137,14 @@ class GanttChart { | |||||||
|     a.StartOfWeek = project.Project.StartOfWeek; |     a.StartOfWeek = project.Project.StartOfWeek; | ||||||
|     a.Tasks = project.ExportTasks(); |     a.Tasks = project.ExportTasks(); | ||||||
| 
 | 
 | ||||||
|  |     // Invalidate canvas (background, foreground)
 | ||||||
|  |     a.CanvasContainer.Layer[0].Load(new Date(project.StartDate), project.Duration, project.Project.StartOfWeek, a.Tasks.length); | ||||||
|  |     a.CanvasContainer.Layer[1].Load(new Date(project.StartDate), a.Tasks); | ||||||
|  |     a.CanvasContainer.Layer[2].Load(); | ||||||
|  | 
 | ||||||
|     a.Invalidate(); |     a.Invalidate(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   #drawChartHeader() { |  | ||||||
|     const a = this; |  | ||||||
| 
 |  | ||||||
|     const displayDays = a.Duration + 2; |  | ||||||
| 
 |  | ||||||
|     if (a.Debug) { |  | ||||||
|       a.Canvas.DrawRectangle(a.HeaderRow1Rectangle, "red", {}); |  | ||||||
|       a.Canvas.DrawRectangle(a.HeaderRow2Rectangle, "orange", {}); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     let startDate = new Date(a.StartDate); |  | ||||||
|     startDate.addDays(-1); |  | ||||||
| 
 |  | ||||||
|     // Draw horizontal line under months
 |  | ||||||
|     a.#drawHorizontalLine(a.Options.HeaderRow.Height[0]); |  | ||||||
| 
 |  | ||||||
|     // Draw vertical lines for dates
 |  | ||||||
|     for (let i=1; i<displayDays; i++) { |  | ||||||
|       a.Canvas.DrawVerticalLine((a.Options.DayWidth * i), (a.Options.HeaderRow.Height[0] + a.Options.BorderWidth), (a.Options.HeaderRow.Height[1] - a.Options.BorderWidth), a.Options.BorderColour, {}); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // Draw horizontal line under dates
 |  | ||||||
|     a.#drawHorizontalLine(a.HeaderHeight); |  | ||||||
| 
 |  | ||||||
|     // Write dates
 |  | ||||||
|     for (let i=0; i<displayDays; i++) { |  | ||||||
|       const date = Date.addDays(startDate, i); |  | ||||||
|       const x = (a.Options.DayWidth * i); |  | ||||||
| 
 |  | ||||||
|       // Draw month
 |  | ||||||
|       if (date.getDate() == 1) { |  | ||||||
|         const size = a.Canvas.MeasureText(a.Options.DateFont, "#"); |  | ||||||
|         const monthPoint = { |  | ||||||
|           X: (x + 2), |  | ||||||
|           Y: Math.half(a.Options.HeaderRow.Height[0] - size.H) |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         a.Canvas.DrawText(monthPoint.X, monthPoint.Y, date.toCString("MMMM"), a.Options.DateFont, a.Options.DateForeColour); |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       // Draw day
 |  | ||||||
|       const dateRectangle = { |  | ||||||
|         X: x, |  | ||||||
|         Y: a.Options.HeaderRow.Height[0], |  | ||||||
|         W: a.Options.DayWidth - a.Options.BorderWidth, |  | ||||||
|         H: (a.Options.HeaderRow.Height[1] - a.Options.BorderWidth) |  | ||||||
|       }; |  | ||||||
| 
 |  | ||||||
|       a.Canvas.FillText(dateRectangle, date.getDate(), a.Options.DateFont, a.Options.DateForeColour); |  | ||||||
| 
 |  | ||||||
|       if (a.Debug) { |  | ||||||
|         a.Canvas.DrawRectangle(dateRectangle, "red", {}); |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   #drawColumnLayout() { |  | ||||||
|     const a = this; |  | ||||||
| 
 |  | ||||||
|     const height = a.Canvas.Height; |  | ||||||
|     const displayDays = a.Duration + 2; |  | ||||||
| 
 |  | ||||||
|     let startDate = new Date(a.StartDate); |  | ||||||
|     startDate.addDays(-1); |  | ||||||
| 
 |  | ||||||
|     // Write dates
 |  | ||||||
|     for (let i=0; i<displayDays; i++) { |  | ||||||
|       const date = Date.addDays(startDate, i); |  | ||||||
|       const x = (a.Options.DayWidth * i); |  | ||||||
| 
 |  | ||||||
|       const dateRectangle = { |  | ||||||
|         X: x, |  | ||||||
|         Y: a.HeaderHeight, |  | ||||||
|         W: a.Options.DayWidth - a.Options.BorderWidth, |  | ||||||
|         H: height - a.Options.BorderWidth |  | ||||||
|       }; |  | ||||||
| 
 |  | ||||||
|       // Fill background for Saturday and Sunday
 |  | ||||||
|       if (date.getDay() == 6) { |  | ||||||
|         a.Canvas.DrawRectangle(dateRectangle, a.Options.Column.SatColour, { FillColour: a.Options.Column.SatColour }); |  | ||||||
|       } else if (date.getDay() == 0) { |  | ||||||
|         a.Canvas.DrawRectangle(dateRectangle, a.Options.Column.SunColour, { FillColour: a.Options.Column.SunColour }); |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       // Draw vertical date lines
 |  | ||||||
|       if (a.StartOfWeek == date.getDay()) { |  | ||||||
|         if (a.Options.ShowStartDayOfWeekLine) { |  | ||||||
|           a.Canvas.DrawVerticalLine(dateRectangle.X, a.HeaderHeight, dateRectangle.H, a.Options.BorderColour, {}); |  | ||||||
|         } |  | ||||||
|       } else { |  | ||||||
|         if (a.Options.ShowDateLines) { |  | ||||||
|           a.Canvas.DrawVerticalLine(dateRectangle.X, a.HeaderHeight, dateRectangle.H, a.Options.BorderColour, { LineDash: a.Options.BorderDashPattern }); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   #drawRowLayout() { |  | ||||||
|     const a = this; |  | ||||||
| 
 |  | ||||||
|     for (let i=0; i<a.Tasks.length; i++) { |  | ||||||
|       const rowRectangle = { |  | ||||||
|         X: 0, |  | ||||||
|         Y: (a.HeaderHeight + (a.Options.Row.Height * i)), |  | ||||||
|         W: a.Canvas.Width, |  | ||||||
|         H: (a.Options.Row.Height - a.Options.BorderWidth) |  | ||||||
|       }; |  | ||||||
| 
 |  | ||||||
|       if (a.Options.ShowRowStripes) { |  | ||||||
|         if ((i % 2) == 1) { |  | ||||||
|           a.Canvas.DrawRectangle(rowRectangle, a.Options.Row.BackColour, { FillColour: a.Options.Row.BackColour }); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       if (a.Options.ShowRowLines) { |  | ||||||
|         a.Canvas.DrawHorizontalLine(0, (rowRectangle.Y + a.Options.Row.Height), rowRectangle.W, a.Options.BorderColour, { LineDash: a.Options.BorderDashPattern }); |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   #drawTasks() { |  | ||||||
|     const a = this; |  | ||||||
| 
 |  | ||||||
|     for (let i=0; i<a.Tasks.length; i++) { |  | ||||||
|       let style = a.Options.Row.Task; |  | ||||||
| 
 |  | ||||||
|       if (a.Tasks[i].IsCollated == true) { |  | ||||||
|         style = a.Options.Row.CollatedTask; |  | ||||||
|       } else if (a.Tasks[i].CollatedTaskID != null) { |  | ||||||
|         style = a.Options.Row.ChildTask; |  | ||||||
|       } else if (a.Tasks[i].PredecessorTaskID == null) { |  | ||||||
|         style = a.Options.Row.OrphanTask; |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       const rectangle = a.#getTaskRectangle(a.Tasks[i], style); |  | ||||||
| 
 |  | ||||||
|       if (a.Tasks[i].Duration <= 0) { |  | ||||||
|         a.Canvas.DrawDiamond(rectangle, style.BorderColour, { FillColour: style.FillColour }); |  | ||||||
|       } else { |  | ||||||
|         a.Canvas.DrawRectangle(rectangle, style.BorderColour, { FillColour: style.FillColour }); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   #drawConnectorLines() { |  | ||||||
|     const a = this; |  | ||||||
| 
 |  | ||||||
|     for (let i=0; i<a.Tasks.length; i++) { |  | ||||||
|       if (a.Tasks[i].PredecessorTaskID == null) { |  | ||||||
|         continue; |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       if (a.Tasks[i].PredecessorTaskID == null) { |  | ||||||
|         continue; |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       const style = ((a.Tasks[i].IsCollated == true) ? a.Options.Row.CollatedTask : a.Options.Row.Task); |  | ||||||
|       const rectangle = a.#getTaskRectangle(a.Tasks[i], style); |  | ||||||
|       const paddingX = (a.Options.Row.CollatedTask.Height + a.Options.BorderWidth); |  | ||||||
|       const offsetX = (rectangle.X + paddingX); |  | ||||||
| 
 |  | ||||||
|       const predecessorTask = a.Tasks.first("ID", a.Tasks[i].PredecessorTaskID); |  | ||||||
|       if (predecessorTask == null) { |  | ||||||
|         continue; |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       const predecessorStyle = ((predecessorTask.IsCollated == true) ? a.Options.Row.CollatedTask : a.Options.Row.Task); |  | ||||||
|       const predecessorRectangle = a.#getTaskRectangle(predecessorTask, predecessorStyle); |  | ||||||
| 
 |  | ||||||
|       let points = []; |  | ||||||
|       points.push({ X: (predecessorRectangle.X + predecessorRectangle.W), Y: (predecessorRectangle.Y + Math.half(predecessorRectangle.H)) }); |  | ||||||
|       points.push({ X: offsetX, Y: (predecessorRectangle.Y + Math.half(predecessorRectangle.H)) }); |  | ||||||
| 
 |  | ||||||
|       points.push({ X: offsetX, Y: (rectangle.Y - paddingX) }); |  | ||||||
|       points.push({ X: offsetX, Y: rectangle.Y }); |  | ||||||
| 
 |  | ||||||
|       a.Canvas.DrawLines(points, a.Options.Line.Colour, { LineWidth: a.Options.Line.Width }); |  | ||||||
| 
 |  | ||||||
|       const arrowRectangle = { |  | ||||||
|         X: (offsetX - Math.half(a.Options.Line.ArrowSize)), |  | ||||||
|         Y: (rectangle.Y - a.Options.Line.ArrowSize), |  | ||||||
|         W: a.Options.Line.ArrowSize, |  | ||||||
|         H: a.Options.Line.ArrowSize |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       a.Canvas.DrawArrowS(arrowRectangle, a.Options.Line.Colour, { FillColour: a.Options.Line.Colour }); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   #drawHorizontalLine(y) { |  | ||||||
|     const a = this; |  | ||||||
|     const width = a.Canvas.Width; |  | ||||||
| 
 |  | ||||||
|     a.Canvas.DrawHorizontalLine(0, y, width, this.Options.BorderColour); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   #getTaskRectangle(task, style) { |  | ||||||
|     const a = this; |  | ||||||
|     const i = (task.Order - 1); |  | ||||||
| 
 |  | ||||||
|     const rowRectangle = { |  | ||||||
|       X: 0, |  | ||||||
|       Y: (a.HeaderHeight + (a.Options.Row.Height * i)), |  | ||||||
|       W: a.Canvas.Width, |  | ||||||
|       H: (a.Options.Row.Height - a.Options.BorderWidth) |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     if (a.Debug) { |  | ||||||
|       a.Canvas.DrawRectangle(rowRectangle, "red", {}); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (task.Duration <= 0) { |  | ||||||
|       let result = { |  | ||||||
|         X: (a.Options.DayWidth * (Date.diffDays(a.StartDate, task.StartDate) + 1)), |  | ||||||
|         Y: (a.HeaderHeight + (a.Options.Row.Height * i) + Math.half(rowRectangle.H - style.Height)), |  | ||||||
|         W: style.Height, |  | ||||||
|         H: style.Height |  | ||||||
|       }; |  | ||||||
| 
 |  | ||||||
|       result.X += Math.half(a.Options.DayWidth - (style.Height + a.Options.BorderWidth)); |  | ||||||
| 
 |  | ||||||
|       return result; |  | ||||||
|     } else { |  | ||||||
|       return { |  | ||||||
|         X: (a.Options.DayWidth * (Date.diffDays(a.StartDate, task.StartDate) + 1)), |  | ||||||
|         Y: (a.HeaderHeight + (a.Options.Row.Height * i) + Math.half(rowRectangle.H - style.Height)), |  | ||||||
|         W: (a.Options.DayWidth * task.Duration), |  | ||||||
|         H: style.Height |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -9,7 +9,7 @@ class Project { | |||||||
|     const a = this; |     const a = this; | ||||||
|     const _options = Object.assign(a.DefaultOptions, options); |     const _options = Object.assign(a.DefaultOptions, options); | ||||||
| 
 | 
 | ||||||
|     a.Debug = true; |     a.Debug = false; | ||||||
|     a.Project = _options; |     a.Project = _options; | ||||||
|     a.Tasks = []; |     a.Tasks = []; | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -117,17 +117,25 @@ | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|   .gantt-chart { | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | .gantt-chart { | ||||||
|   border-color: #B8B8B8; |   border-color: #B8B8B8; | ||||||
|   border-style: solid; |   border-style: solid; | ||||||
|   border-width: 1px; |   border-width: 1px; | ||||||
|   overflow: hidden; |   overflow: hidden; | ||||||
|   scrollbar-width: thin; |   scrollbar-width: thin; | ||||||
|   scrollbar-color: #CDCDCD #F0F0F0; |   scrollbar-color: #CDCDCD #F0F0F0; | ||||||
|   } |  | ||||||
| 
 | 
 | ||||||
|   .gantt-chart.scroll-x { | 
 | ||||||
|  |   &.scroll-x { | ||||||
|     overflow-x: scroll !important; |     overflow-x: scroll !important; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   canvas { | ||||||
|  |     display: block; | ||||||
|  |     position: relative; | ||||||
|  |   } | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | |||||||
							
								
								
									
										229
									
								
								src/references/canvas-container.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										229
									
								
								src/references/canvas-container.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,229 @@ | |||||||
|  | class CanvasContainer { | ||||||
|  | 
 | ||||||
|  |   constructor(el) { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     a.el = el; | ||||||
|  | 
 | ||||||
|  |     a.Layer = []; | ||||||
|  |     a.BorderWidth = 2; | ||||||
|  |     a.ScrollbarWidth = 11; | ||||||
|  |     a.Padding = { | ||||||
|  |       Top: 0, | ||||||
|  |       Right: 0, | ||||||
|  |       Bottom: 0, | ||||||
|  |       Left: 0 | ||||||
|  |     }; | ||||||
|  |     a.EnableDock = true; | ||||||
|  | 
 | ||||||
|  |     a.#initialiseComponents(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   #initialiseComponents() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     a.ClearLayers(); | ||||||
|  | 
 | ||||||
|  |     // a.Invalidate();
 | ||||||
|  |     // a.Clear();
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   get FlowContainer() { | ||||||
|  |     return this.el.getElementsByTagName("div")[0]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get Height() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     if (a.Layer.length <= 0) { | ||||||
|  |       return 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return Document.getHeight(a.Layer[0].el); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   set Height(value) { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     // if (a.el != null) {
 | ||||||
|  |     //   a.el.style.height = value + "px";
 | ||||||
|  |     // }
 | ||||||
|  | 
 | ||||||
|  |     if (a.FlowContainer != null) { | ||||||
|  |       a.FlowContainer.style.height = value + "px"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // if (a.CanvasContainer != null) {
 | ||||||
|  |     //   a.CanvasContainer.style.height = value + "px";
 | ||||||
|  |     // }
 | ||||||
|  | 
 | ||||||
|  |     // if (a.CanvasContext != null) {
 | ||||||
|  |     //   a.CanvasContext.canvas.height = value;
 | ||||||
|  |     // }
 | ||||||
|  | 
 | ||||||
|  |     a.Layer.forEach((e, i) => { | ||||||
|  |       e.Height = value; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get Width() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     if (a.Layer.length <= 0) { | ||||||
|  |       return 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return Document.getWidth(a.Layer[0].el); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   set Width(value) { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     // if (a.el != null) {
 | ||||||
|  |     //   a.el.style.width = value + "px";
 | ||||||
|  |     // }
 | ||||||
|  | 
 | ||||||
|  |     if (a.FlowContainer != null) { | ||||||
|  |       a.FlowContainer.style.width = value + "px"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // if (a.CanvasContainer != null) {
 | ||||||
|  |     //   a.CanvasContainer.style.width = value + "px";
 | ||||||
|  |     // }
 | ||||||
|  | 
 | ||||||
|  |     // if (a.CanvasContext != null) {
 | ||||||
|  |     //   a.CanvasContext.canvas.width = value;
 | ||||||
|  |     // }
 | ||||||
|  | 
 | ||||||
|  |     a.Layer.forEach((e, i) => { | ||||||
|  |       e.Width = value; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get Size() { | ||||||
|  |     return { | ||||||
|  |       W: this.Width, | ||||||
|  |       H: this.Height | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   set Size(value) { | ||||||
|  |     this.Width = value.W; | ||||||
|  |     this.Height = value.H; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get ClientRectangle() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     return { | ||||||
|  |       X: a.Padding.Left, | ||||||
|  |       Y: a.Padding.Top, | ||||||
|  |       W: (a.Width - (a.Padding.Left + a.Padding.Right)), | ||||||
|  |       H: (a.Height - (a.Padding.Top + a.Padding.Bottom)) | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get Rectangle() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     return { | ||||||
|  |       X: 0, | ||||||
|  |       Y: 0, | ||||||
|  |       W: a.Width, | ||||||
|  |       H: a.Height | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   Clear() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     // if (a.CanvasContext == null) {
 | ||||||
|  |     //   return;
 | ||||||
|  |     // }
 | ||||||
|  | 
 | ||||||
|  |     // a.CanvasContext.clearRect(0, 0, a.CanvasContext.canvas.width, a.CanvasContext.canvas.height);
 | ||||||
|  | 
 | ||||||
|  |     a.Layer.forEach((e, i) => { | ||||||
|  |       e.Clear(); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   ClearLayers() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     if (a.el != null) { | ||||||
|  |       a.el.innerHTML = "<div class='border'></div>"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     a.Layer = []; | ||||||
|  | 
 | ||||||
|  |     a.Invalidate(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   AddLayer() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     if (a.el == null) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const divList = a.el.getElementsByTagName("div"); | ||||||
|  |     if (divList.length <= 0) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Add another canvas element
 | ||||||
|  |     Document.appendHtml(divList[0], "canvas"); | ||||||
|  | 
 | ||||||
|  |     const canvasList = a.el.getElementsByTagName("canvas"); | ||||||
|  |     const result = canvasList[(canvasList.length - 1)]; | ||||||
|  | 
 | ||||||
|  |     return result; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   Invalidate() { | ||||||
|  |     const a = this; | ||||||
|  | 
 | ||||||
|  |     if (a.EnableDock) { | ||||||
|  |       a.FlowContainer.style.width = "100%"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const w = (a.Layer.length <= 0 ? 0 : Document.getWidth(a.Layer[0].el)); | ||||||
|  |     const h = (a.Layer.length <= 0 ? 0 : Document.getHeight(a.Layer[0].el)); | ||||||
|  | 
 | ||||||
|  |     if (w > (Document.getWidth(a.FlowContainer) + (a.BorderWidth * 2))) { | ||||||
|  |       a.FlowContainer.classList.add("scroll-x"); | ||||||
|  |       a.FlowContainer.style.height = (h + a.ScrollbarWidth) + "px"; | ||||||
|  |     } else { | ||||||
|  |       a.FlowContainer.classList.remove("scroll-x"); | ||||||
|  |       a.FlowContainer.style.height = h + "px"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     a.Layer.forEach(e => { | ||||||
|  |       if (typeof(e.Invalidate) != "undefined") { | ||||||
|  |         e.Invalidate(); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     let offsetTop = 0; | ||||||
|  | 
 | ||||||
|  |     a.Layer.forEach((e, i) => { | ||||||
|  |       if (i <= 0) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       // Correct positon because of position-relative to keep container overflow working
 | ||||||
|  |       const h = Document.getHeight(a.Layer[(i - 1)].el); | ||||||
|  | 
 | ||||||
|  |       offsetTop += h; | ||||||
|  | 
 | ||||||
|  |       a.Layer[i].el.style.top = "-" + offsetTop + "px"; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | export default CanvasContainer; | ||||||
| @ -1,19 +1,10 @@ | |||||||
| class Canvas { | class Canvas { | ||||||
| 
 |  | ||||||
|   constructor(el) { |   constructor(el) { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     a.el = el; |     a.el = el; | ||||||
| 
 | 
 | ||||||
|     a.BorderWidth = 2; |     a.BorderWidth = 2; | ||||||
|     a.ScrollbarWidth = 11; |  | ||||||
|     a.Padding = { |  | ||||||
|       Top: 0, |  | ||||||
|       Right: 0, |  | ||||||
|       Bottom: 0, |  | ||||||
|       Left: 0 |  | ||||||
|     }; |  | ||||||
|     a.EnableDock = true; |  | ||||||
| 
 | 
 | ||||||
|     a.#initialiseComponents(); |     a.#initialiseComponents(); | ||||||
|   } |   } | ||||||
| @ -25,48 +16,30 @@ class Canvas { | |||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     a.el.innerHTML = "<div class='border'><canvas></canvas></div>"; |  | ||||||
| 
 |  | ||||||
|     a.Invalidate(); |  | ||||||
|     a.Clear(); |     a.Clear(); | ||||||
|  |     a.Invalidate(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|   get FlowContainer() { |  | ||||||
|     return this.el.getElementsByTagName("div")[0]; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get CanvasContainer() { |  | ||||||
|     return this.FlowContainer.getElementsByTagName("canvas")[0]; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get CanvasContext() { |   get CanvasContext() { | ||||||
|     return this.CanvasContainer.getContext("2d"); |     return this.el.getContext("2d"); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   get Height() { |   get Height() { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     if (a.CanvasContainer == null) { |     if (a.el == null) { | ||||||
|       return 0; |       return 0; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return a.#getHeight(a.CanvasContainer); |     return Document.getHeight(a.el); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   set Height(value) { |   set Height(value) { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     // if (a.el != null) {
 |     if (a.el != null) { | ||||||
|     //   a.el.style.height = value + "px";
 |       a.el.style.height = value + "px"; | ||||||
|     // }
 |  | ||||||
| 
 |  | ||||||
|     if (a.FlowContainer != null) { |  | ||||||
|       a.FlowContainer.style.height = value + "px"; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (a.CanvasContainer != null) { |  | ||||||
|       a.CanvasContainer.style.height = value + "px"; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (a.CanvasContext != null) { |     if (a.CanvasContext != null) { | ||||||
| @ -77,26 +50,18 @@ class Canvas { | |||||||
|   get Width() { |   get Width() { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     if (a.CanvasContainer == null) { |     if (a.el == null) { | ||||||
|       return 0; |       return 0; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return a.#getWidth(a.CanvasContainer); |     return Document.getWidth(a.el); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   set Width(value) { |   set Width(value) { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
|     // if (a.el != null) {
 |     if (a.el != null) { | ||||||
|     //   a.el.style.width = value + "px";
 |       a.el.style.width = value + "px"; | ||||||
|     // }
 |  | ||||||
| 
 |  | ||||||
|     if (a.FlowContainer != null) { |  | ||||||
|       a.FlowContainer.style.width = value + "px"; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (a.CanvasContainer != null) { |  | ||||||
|       a.CanvasContainer.style.width = value + "px"; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (a.CanvasContext != null) { |     if (a.CanvasContext != null) { | ||||||
| @ -116,17 +81,6 @@ class Canvas { | |||||||
|     this.Height = value.H; |     this.Height = value.H; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   get ClientRectangle() { |  | ||||||
|     const a = this; |  | ||||||
| 
 |  | ||||||
|     return { |  | ||||||
|       X: a.Padding.Left, |  | ||||||
|       Y: a.Padding.Top, |  | ||||||
|       W: (a.Width - (a.Padding.Left + a.Padding.Right)), |  | ||||||
|       H: (a.Height - (a.Padding.Top + a.Padding.Bottom)) |  | ||||||
|     }; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get Rectangle() { |   get Rectangle() { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
| @ -466,23 +420,6 @@ class Canvas { | |||||||
|     a.CanvasContext.fillText(text, x, y); |     a.CanvasContext.fillText(text, x, y); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   Invalidate() { |  | ||||||
|     const a = this; |  | ||||||
| 
 |  | ||||||
|     if (a.EnableDock) { |  | ||||||
|       a.FlowContainer.style.width = "100%"; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (a.#getWidth(a.CanvasContainer) > (a.#getWidth(a.FlowContainer) + (a.BorderWidth * 2))) { |  | ||||||
|       a.FlowContainer.classList.add("scroll-x"); |  | ||||||
|       a.FlowContainer.style.height = (a.#getHeight(a.CanvasContainer) + a.ScrollbarWidth) + "px"; |  | ||||||
|     } else { |  | ||||||
|       a.FlowContainer.classList.remove("scroll-x"); |  | ||||||
|       a.FlowContainer.style.height = a.#getHeight(a.CanvasContainer) + "px"; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   MeasureText(font, value) { |   MeasureText(font, value) { | ||||||
|     const a = this; |     const a = this; | ||||||
| 
 | 
 | ||||||
| @ -500,30 +437,6 @@ class Canvas { | |||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   #getWidth(el) { |  | ||||||
|     if (el == null) { |  | ||||||
|       return 0; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (typeof(el) == "undefined") { |  | ||||||
|       return 0; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return (el.offsetWidth || el.innerWidth || el.clientWidth); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   #getHeight(el) { |  | ||||||
|     if (el == null) { |  | ||||||
|       return 0; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (typeof(el) == "undefined") { |  | ||||||
|       return 0; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return (el.offsetHeight || el.innerHeight || el.clientHeight); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
		Loading…
	
		Reference in New Issue
	
	Block a user