| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  | import {VariableUiElement} from "../Base/VariableUIElement"; | 
					
						
							|  |  |  | import {UIEventSource} from "../../Logic/UIEventSource"; | 
					
						
							|  |  |  | import Table from "../Base/Table"; | 
					
						
							|  |  |  | import Combine from "../Base/Combine"; | 
					
						
							|  |  |  | import {FixedUiElement} from "../Base/FixedUiElement"; | 
					
						
							|  |  |  | import {Utils} from "../../Utils"; | 
					
						
							|  |  |  | import BaseUIElement from "../BaseUIElement"; | 
					
						
							| 
									
										
										
										
											2022-01-16 02:00:55 +01:00
										 |  |  | import Svg from "../../Svg"; | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | export default class Histogram<T> extends VariableUiElement { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private static defaultPalette = [ | 
					
						
							|  |  |  |         "#ff5858", | 
					
						
							|  |  |  |         "#ffad48", | 
					
						
							|  |  |  |         "#ffff59", | 
					
						
							|  |  |  |         "#56bd56", | 
					
						
							|  |  |  |         "#63a9ff", | 
					
						
							| 
									
										
										
										
											2022-01-16 02:00:55 +01:00
										 |  |  |         "#9d62d9", | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |         "#fa61fa" | 
					
						
							|  |  |  |     ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     constructor(values: UIEventSource<string[]>, | 
					
						
							|  |  |  |                 title: string | BaseUIElement, | 
					
						
							|  |  |  |                 countTitle: string | BaseUIElement, | 
					
						
							| 
									
										
										
										
											2022-01-26 21:40:38 +01:00
										 |  |  |                 options?: { | 
					
						
							|  |  |  |                     assignColor?: (t0: string) => string, | 
					
						
							| 
									
										
										
										
											2022-01-22 02:56:35 +01:00
										 |  |  |                     sortMode?: "name" | "name-rev" | "count" | "count-rev" | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |     ) { | 
					
						
							| 
									
										
										
										
											2022-01-26 21:40:38 +01:00
										 |  |  |         const sortMode = new UIEventSource<"name" | "name-rev" | "count" | "count-rev">(options?.sortMode ?? "name") | 
					
						
							| 
									
										
										
										
											2022-01-16 02:00:55 +01:00
										 |  |  |         const sortName = new VariableUiElement(sortMode.map(m => { | 
					
						
							|  |  |  |             switch (m) { | 
					
						
							|  |  |  |                 case "name": | 
					
						
							|  |  |  |                     return Svg.up_svg() | 
					
						
							|  |  |  |                 case "name-rev": | 
					
						
							|  |  |  |                     return Svg.up_svg().SetStyle("transform: rotate(180deg)") | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     return Svg.circle_svg() | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         })) | 
					
						
							|  |  |  |         const titleHeader = new Combine([sortName.SetClass("w-4 mr-2"), title]).SetClass("flex") | 
					
						
							|  |  |  |             .onClick(() => { | 
					
						
							|  |  |  |                 if (sortMode.data === "name") { | 
					
						
							|  |  |  |                     sortMode.setData("name-rev") | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     sortMode.setData("name") | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const sortCount = new VariableUiElement(sortMode.map(m => { | 
					
						
							|  |  |  |             switch (m) { | 
					
						
							|  |  |  |                 case "count": | 
					
						
							|  |  |  |                     return Svg.up_svg() | 
					
						
							|  |  |  |                 case "count-rev": | 
					
						
							|  |  |  |                     return Svg.up_svg().SetStyle("transform: rotate(180deg)") | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     return Svg.circle_svg() | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         })) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const countHeader = new Combine([sortCount.SetClass("w-4 mr-2"), countTitle]).SetClass("flex").onClick(() => { | 
					
						
							|  |  |  |             if (sortMode.data === "count-rev") { | 
					
						
							|  |  |  |                 sortMode.setData("count") | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 sortMode.setData("count-rev") | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const header = [ | 
					
						
							|  |  |  |             titleHeader, | 
					
						
							|  |  |  |             countHeader | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |         super(values.map(values => { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (values === undefined) { | 
					
						
							|  |  |  |                 return undefined; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             values = Utils.NoNull(values) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             const counts = new Map<string, number>() | 
					
						
							|  |  |  |             for (const value of values) { | 
					
						
							|  |  |  |                 const c = counts.get(value) ?? 0; | 
					
						
							|  |  |  |                 counts.set(value, c + 1); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             const keys = Array.from(counts.keys()); | 
					
						
							| 
									
										
										
										
											2022-01-16 02:00:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             switch (sortMode.data) { | 
					
						
							|  |  |  |                 case "name": | 
					
						
							|  |  |  |                     keys.sort() | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case "name-rev": | 
					
						
							|  |  |  |                     keys.sort().reverse() | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case "count": | 
					
						
							|  |  |  |                     keys.sort((k0, k1) => counts.get(k0) - counts.get(k1)) | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case "count-rev": | 
					
						
							|  |  |  |                     keys.sort((k0, k1) => counts.get(k1) - counts.get(k0)) | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-09-09 00:05:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |             const max = Math.max(...Array.from(counts.values())) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             const fallbackColor = (keyValue: string) => { | 
					
						
							|  |  |  |                 const index = keys.indexOf(keyValue) | 
					
						
							|  |  |  |                 return Histogram.defaultPalette[index % Histogram.defaultPalette.length] | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |             let actualAssignColor = undefined; | 
					
						
							| 
									
										
										
										
											2022-01-22 02:56:35 +01:00
										 |  |  |             if (options?.assignColor === undefined) { | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |                 actualAssignColor = fallbackColor; | 
					
						
							| 
									
										
										
										
											2021-09-09 00:05:51 +02:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |                 actualAssignColor = (keyValue: string) => { | 
					
						
							| 
									
										
										
										
											2022-01-22 02:56:35 +01:00
										 |  |  |                     return options.assignColor(keyValue) ?? fallbackColor(keyValue) | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return new Table( | 
					
						
							| 
									
										
										
										
											2022-01-16 02:00:55 +01:00
										 |  |  |                 header, | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |                 keys.map(key => [ | 
					
						
							|  |  |  |                     key, | 
					
						
							|  |  |  |                     new Combine([ | 
					
						
							| 
									
										
										
										
											2021-09-09 00:05:51 +02:00
										 |  |  |                         new Combine([new FixedUiElement("" + counts.get(key)).SetClass("font-bold rounded-full block")]) | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |                             .SetClass("flex justify-center rounded border border-black") | 
					
						
							|  |  |  |                             .SetStyle(`background: ${actualAssignColor(key)}; width: ${100 * counts.get(key) / max}%`) | 
					
						
							|  |  |  |                     ]).SetClass("block w-full") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 ]), | 
					
						
							|  |  |  |                 keys.map(_ => ["width: 20%"]) | 
					
						
							| 
									
										
										
										
											2022-01-19 20:34:04 +01:00
										 |  |  |             ).SetClass("w-full zebra-table"); | 
					
						
							| 
									
										
										
										
											2022-01-16 02:00:55 +01:00
										 |  |  |         }, [sortMode])); | 
					
						
							| 
									
										
										
										
											2021-06-21 03:12:43 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } |