1
+ <!DOCTYPE html>
2
+ < html >
3
+ < head >
4
+ < title > </ title >
5
+ < script src ="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/3.9.1/chart.min.js "> </ script >
6
+ < script src ="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.4/moment.min.js " crossorigin ="anonymous " referrerpolicy ="no-referrer "> </ script >
7
+ < script src ="https://cdnjs.cloudflare.com/ajax/libs/chartjs-adapter-moment/1.0.0/chartjs-adapter-moment.min.js " crossorigin ="anonymous " referrerpolicy ="no-referrer "> </ script >
8
+ < script >
9
+ function decimal ( x , places )
10
+ {
11
+ const pow = Math . pow ( 10 , places ) ;
12
+ return Math . round ( x * pow ) / pow ;
13
+ }
14
+ function millenium_fmt ( x )
15
+ {
16
+ const list = [ [ 1e9 , "G" , "Giga" ] , [ 1e6 , "M" , "Mega" ] , [ 1e3 , "k" , "Kilo" ] , [ 1 , "" , "" ] ,
17
+ [ 1e-3 , "m" , "Milli" ] , [ 1e-6 , "\xb5" , "Micro" ] , [ 1e-9 , "n" , "Nano" ] ] ;
18
+
19
+ for ( const [ threshold , word , longword ] of list )
20
+ {
21
+ if ( x > threshold )
22
+ {
23
+ return { value : x / threshold , word : word , longword : longword } ;
24
+ }
25
+ }
26
+ // Default to the last one
27
+ const [ threshold , word , longword ] = list [ list . length - 1 ] ;
28
+ return { value : x / threshold , word : word , longword : longword } ;
29
+ }
30
+ const avogadrosConstant = 6.02214076e23 ;
31
+ const elementaryCharge = 1.602176634e-19 ;
32
+ const factor = avogadrosConstant * elementaryCharge ;
33
+ function coulomb_to_mol ( x )
34
+ {
35
+ return x / factor ;
36
+ }
37
+ </ script >
38
+ < script >
39
+ const file = "%FILECONTENT%" ;
40
+ const decimals = 2 ;
41
+
42
+ function dateISOFmt ( date )
43
+ {
44
+ return moment ( date ) . format ( "YYYY-MM-DD HH:mm:ss" ) ;
45
+ }
46
+
47
+ function loaded ( )
48
+ {
49
+ const split = file . split ( '|' ) . filter ( x => x . trim ( ) . length > 0 ) ;
50
+ let data = [ ] ;
51
+ for ( const row of split )
52
+ {
53
+ const rowparts = row . split ( ';' ) ;
54
+ data . push ( { x : dateISOFmt ( Number ( rowparts [ 0 ] ) * 1000 ) , current : Math . abs ( Number ( rowparts [ 1 ] ) ) , charge : rowparts [ 2 ] , mol : coulomb_to_mol ( Number ( rowparts [ 2 ] ) ) } ) ;
55
+ }
56
+ data . sort ( function ( a , b ) { return a . x - b . x ; } ) ;
57
+
58
+ const ctx = document . getElementById ( 'chart' ) ;
59
+ const myChart = new Chart ( ctx , {
60
+ type : 'line' ,
61
+ data : {
62
+ datasets : [ {
63
+ label : 'Current' ,
64
+ backgroundColor : 'rgb(255, 99, 132)' ,
65
+ borderColor : 'rgb(255, 99, 153)' ,
66
+ data : data ,
67
+ yAxisID : 'currentAxis' ,
68
+ parsing : {
69
+ yAxisKey : 'current'
70
+ }
71
+ } , {
72
+ label : 'Charge' ,
73
+ backgroundColor : 'rgb(153, 51, 255)' ,
74
+ borderColor : 'rgb(153, 9, 255)' ,
75
+ data : data ,
76
+ yAxisID : 'chargeAxis' ,
77
+ parsing : {
78
+ yAxisKey : 'charge'
79
+ }
80
+ } , {
81
+ label : 'Electrons' ,
82
+ backgroundColor : 'rgb(26, 234, 11)' ,
83
+ borderColor : 'rgb(42, 222, 5)' ,
84
+ data : data ,
85
+ yAxisID : 'molAxis' ,
86
+ parsing : {
87
+ yAxisKey : 'mol'
88
+ }
89
+ } ] ,
90
+ indexAxis : 'x'
91
+ } ,
92
+ options : {
93
+ normalized : true ,
94
+ animation : false ,
95
+ interaction : {
96
+ mode : 'nearest' ,
97
+ axis : 'x' ,
98
+ intersect : false
99
+ } ,
100
+ scales : {
101
+ x : {
102
+ type : 'time'
103
+ } ,
104
+ currentAxis : {
105
+ type : 'linear' ,
106
+ position : 'left' ,
107
+ grid : { display : false } ,
108
+ ticks : {
109
+ callback : function ( value , index , ticks ) {
110
+ x = millenium_fmt ( value ) ;
111
+ return x . value + " " + x . word + "A" ;
112
+ }
113
+ }
114
+ } ,
115
+ chargeAxis : {
116
+ type : 'linear' ,
117
+ position : 'right' ,
118
+ grid : { display : false } ,
119
+ ticks : {
120
+ callback : function ( value , index , ticks ) {
121
+ x = millenium_fmt ( value ) ;
122
+ return x . value + " " + x . word + "C" ;
123
+ }
124
+ }
125
+ } ,
126
+ molAxis : {
127
+ type : 'linear' ,
128
+ position : 'right' ,
129
+ grid : { display : false } ,
130
+ ticks : { display : false }
131
+ }
132
+ } ,
133
+ plugins : {
134
+ tooltip : {
135
+ callbacks : {
136
+ label : function ( context ) {
137
+ let label = context . dataset . label || '' ;
138
+
139
+ if ( label ) {
140
+ label += ': ' ;
141
+ }
142
+ if ( context . parsed . y !== null ) {
143
+ mfmt = millenium_fmt ( context . parsed . y ) ;
144
+ label += decimal ( mfmt . value , decimals ) ;
145
+ label += ' ' ;
146
+ // Suffix based on dataset
147
+ switch ( context . dataset . parsing . yAxisKey )
148
+ {
149
+ case "current" :
150
+ label += mfmt . word ;
151
+ label += 'A' ;
152
+ break ;
153
+ case "charge" :
154
+ label += mfmt . word ;
155
+ label += 'C' ;
156
+ break ;
157
+ case "mol" :
158
+ label += mfmt . word ;
159
+ label += 'mol' ;
160
+ break ;
161
+ }
162
+ }
163
+ return label ;
164
+ } ,
165
+ title : function ( context ) {
166
+ context = context [ 0 ] ;
167
+ let label = context . label || '' ;
168
+ label = new Date ( label ) . toLocaleString ( ) ;
169
+ return label ;
170
+ }
171
+ }
172
+ }
173
+ }
174
+ }
175
+ } ) ;
176
+ }
177
+
178
+ window . addEventListener ( 'load' , loaded , true ) ;
179
+ </ script >
180
+ </ head >
181
+ < body >
182
+ < form action ="/ ">
183
+ < input type ="submit " value ="Back ">
184
+ </ form >
185
+ < canvas id ="chart " width ="100 " height ="40 "> </ canvas >
186
+ </ body >
187
+ </ html >
0 commit comments