forked from legalese/legalese-google-app
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdependencies-to-d3.js
executable file
·609 lines (527 loc) · 24.4 KB
/
dependencies-to-d3.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
// read the hand-coded dependencies sheet and output a simple JSON list of nodes and links suitable for consumption by d3.layout.force().
// in future, we can take a list of nodes and links produced by the main dependencies.js and either output it to the spreadsheet view as an intermediate step
// or produce a node/links layout directly.
function readDepSheet(){
var sheet = SpreadsheetApp.getActiveSheet();
var mydepSheet = new depSheet();
mydepSheet.read(sheet);
return mydepSheet;
}
function depWriteForceLayout() {
var mydepSheet = readDepSheet();
mydepSheet.layout_stages("stages.json");
// mydepSheet.layout_d3("dag.json");
// mydepSheet.layout_htmlFileIndex();
dumpMyLogStats();
}
function depSheet() {
// we internally maintain a list of vertices and edges.
this.vertex = []; // { type=BLANK|SUBSECTION|PDF|PARTY, title=String, URL=String, templatename=String, state=(done|waived|optional|pending|unready) }
this.edge = []; // { source=vertex_id, target=vertex_id } // for a Target to be considered done, all its Sources must be done, waived, or optional
// we keep track of parsed spreadsheet cells
this.cell = []; // cell[y][x] = VertexIndex
// type VertexIndex = Int
// later, we export to a list of nodes and links suitable for consumption by d3.
this.d3 = { nodes: [], links: [] };
this.stages = [ ]; // [ {id:, nodes:, links:} ]
this.v2n_all = [];
this.v2n_current = [];
}
depSheet.prototype.read = function(sheet) {
this.sheet = sheet;
this.sheetName = sheet.getSheetName();
// sheet nodes are given by rows which begin with the word DEPENDENCIES
// the name is the second word to appear in the row, no matter what cell it's on.
//
// PDF nodes are given by a pattern match: [BLANK] [BLANK]
// [BLANK] [PDF ]
//
// Party nodes are given by a pattern match: [PDF ] [BLANK]
// [BLANK] [PARTY]
//
// or an alternative pattern match [BLANK] [PARTY]
// [BLANK] [PARTY]
//
// so we basically parse the entire sheet cell by cell building up the lists of .vertices and .edges as we go.
// it is possible to parse left-to-right, top-to-bottom.
//
var rows = sheet.getDataRange();
var numRows = rows.getNumRows(); this.numRows = numRows;
var values = rows.getValues();
this.formulas = rows.getFormulas();
var formats = rows.getNumberFormats();
var display = rows.getDisplayValues();
var section = "";
var currentsubsection = "";
var currentsubsectionvertex = undefined;
for (var i = 0; i <= numRows - 1; i++) {
var row = values[i];
this.cell[i] = this.cell[i] || [];
// process header rows
if (row.filter(function(c){return c.length > 0}).length == 0) { deLog(["row %s is blank, skipping", i],9); continue; }
else if (row[0]) deLog(["row %s: processing row %s",i+1,row.slice(0,1)],9);
// we have a new sheet subsection
if (row[0] == "DEPENDENCIES") {
section=row[0];
var words = row.slice(1).filter(function(e){return (e != undefined && e.length)});
currentsubsection = words.shift();
deLog(["matched DEPENDENCIES subsection: %s", currentsubsection],8);
var vertex = {type:"subsection", subsection:currentsubsection, title:currentsubsection, cell:[i,0], endrow:numRows};
if (words.length && words[0] == "requires") {
words.shift();
deLog([" requires %s", words],8);
vertex.requiresSubsections = words;
}
this.cell[i][0] = this.vertex.length;
if (currentsubsectionvertex) { currentsubsectionvertex.endrow = i-1 }
this.vertex.push(vertex);
currentsubsectionvertex = vertex;
continue;
}
else if (row[0] == "DEP NODES") {
section=row[0];
if (currentsubsectionvertex) { currentsubsectionvertex.endrow = i-1 }
}
else if (row[0] == "OUTPUT FOLDER") { // this really should be a CONFIG section
section=row[0];
deLog(["matched OUTPUT FOLDER section: %s", this.formulas[i]],8);
var words = this.formulas[i].slice(1).filter(function(e){return (e != undefined && e.length)});
if (! words.length) {
deLog(["matched OUTPUT FOLDER section: %s", row.slice(1)],8);
words = row.slice(1).filter(function(e){return (e != undefined && e.length)});
}
this.output_folder = formulate_folder(words.shift());
deLog(["determined output folder is %s", this.output_folder.name],8);
}
// main processing, section-dependent
if (section == "DEPENDENCIES") {
for (var j = 0; j < row.length; j++) {
if (j == 0) continue;
if (values[i][j]) { deLog(["looking at cell %s,%s: %s", i, j, values[i][j]],9); }
var vertex = undefined;
// match PDF cell
if (values[i][j] && ! values[i-1][j-1] && ! values[i-1][j] && ! values[i][j-1]) {
deLog(["matched PDF cell: %s", display[i][j]], 9);
// todo: add URL, colour etc
vertex = {type:"pdf", subsection:currentsubsection, title:display[i][j], cell:[i,j]};
}
// match Party cell
else if (values[i][j] && ! values[i][j-1] &&
((this.vertex[this.cell[i-1][j-1]] && this.vertex[this.cell[i-1][j-1]].type == "pdf")
||
(this.vertex[this.cell[i-1][j]] &&this.vertex[this.cell[i-1][j]].type == "party"))) {
deLog(["matched Party cell: %s", display[i][j]], 9);
vertex = {type:"party", subsection:currentsubsection, title:display[i][j], cell:[i,j]};
}
if (vertex) {
this.cell[i][j] = this.vertex.length; // order matters here -- the index is one-off from the length
this.vertex.push(vertex);
}
}
}
}
deLog(["first pass complete; read %s subsections, %s pdfs, and %s party vertices",
this.vertex.filter(function(v){return (v.type == "subsection")}).length,
this.vertex.filter(function(v){return (v.type == "pdf") }).length,
this.vertex.filter(function(v){return (v.type == "party")}).length],8);
this.decorateVertices();
// completed first pass over the entire sheet, and built up .vertex and .cell.
// now we run a second pass over .cell, inferring edges based on position.
//
// the edge rules are:
// a subsection may require one or more other subsections.
// a given edge defines a source and a target. a source is a prerequisite to a target.
// an edge may take one of the following types:
// SOURCE -> TARGET
// subsection -> subsection
// pdf -> subsection
// pdf -> pdf
// party -> pdf
// a subsection-subsection edge is determined by the DEPENDENCIES subsection line. all the sources for a subsection are named after the "requires" keyword.
// a pdf-subsection edge is given by all the final targets for a subsection.
// a final targetis a vertex which is not a source to any other vertex.
// a pdf-pdf edge is given by a spatial relationship within the section:
// the PDF sources for a given PDF target cell are all the pdfs listed in the first column to the left that contains any pdf vertices.
// a party-pdf edge is given by a spatial relationship within the section:
// the Party targets for a given source PDF cell are all the parties listed one column to the right and contiguously down
this.findEdges_party_pdf();
this.findEdges_pdf_pdf();
this.findEdges_subsection_subsection();
this.findEdges_pdf_subsection();
// dependency status rules:
//
// for a subsection to be complete, all the subsection's final targets must be complete.
// for a vertex to be complete, all the vertex's sources must be complete.
// if a vertex has no sources, then, to be complete, it has to be coloured green -- it's probably a party source.
// how do we distinguish between unready and pending vs waiting, etc?
// this is where the green tip of coding currently awaits.
};
// extract additional information about each vertex
depSheet.prototype.decorateVertices = function() {
var self = this;
// associate the ultimate filename of the linked PDF.
// there's usually a hyperlink formula
for (var v = 0; v < this.vertex.length; v++) {
var vertex = this.vertex[v];
if (vertex.type == "pdf") {
var formula = self.formulas[vertex.cell[0]][vertex.cell[1]];
deLog(["decorating %s (formula %s)", vertex.title, formula],8);
// [16-05-05 02:08:21:775 PDT] dependencies decorating Class F Agreement - Leow Thai Chee (formula =HYPERLINK("https://drive.google.com/open?id=0BxOaYa8pqqSwS2pVZl9hZnZCdzg","Class F Agreement - Leow Thai Chee"))
// [16-05-05 02:08:21:775 PDT] dependencies decorating Directors' Resolution for Allotment (formula =HYPERLINK("https://drive.google.com/open?id=0BxOaYa8pqqSwaG1Oa2ZKQ1Q1ekk","Directors' Resolution for Allotment"))
if (formula.match(/^=HYPERLINK\("([^"]+)"/)) {
var formulated = formulate_file(formula);
vertex.url = formulated.url;
vertex.driveID = formulated.id;
vertex.name = formulated.name;
vertex.file = formulated.file;
deLog([" %s has name=%s size=%s", formulated.id, formulated.name, formulated.file],9);
}
}
}
};
function formulate_file (formula) {
var url = formula.match(/^=HYPERLINK\("([^"]+)"/) ? (formula.match(/^=HYPERLINK\("([^"]+)"/)[1]) : formula;
var id = url.match(/id=([-\w]+)/)[1];
var file = DriveApp.getFileById(id);
return { url:url, id:id, file:file, name:file.getName() };
}
function formulate_folder (formula) {
var url = formula.match(/^=HYPERLINK\("([^"]+)"/) ? (formula.match(/^=HYPERLINK\("([^"]+)"/)[1]) : formula;
var id = url.match(/(?:id=|folders\/)([-\w]+)/)[1];
var folder = DriveApp.getFolderById(id);
return { url:url, id:id, folder:folder, name:folder.getName() };
}
depSheet.prototype.findEdges_subsection_subsection = function() {
var self = this;
for (var v = 0; v < this.vertex.length; v++) {
var vertex = this.vertex[v];
if (vertex.type == "subsection") {
if (vertex.requiresSubsections) {
for (var rsi = 0; rsi < vertex.requiresSubsections.length; rsi++) {
var sourceSubsection = this.getSubsectionVertex(vertex.requiresSubsections[rsi]);
if (! sourceSubsection) {
deLog(["!!! subsection->subsection subsection %s requires %s but no subsection by that name is known!", vertex.title, vertex.requiresSubsections[rsi]],5);
}
else {
deLog([" learned subsection-subsection edge %s -> %s (%s -> %s)", self.vertex.indexOf(sourceSubsection), v, sourceSubsection.title, vertex.title], 8);
self.edge.push([self.vertex.indexOf(sourceSubsection), v]);
}
}
}
}
}
};
// a pdf-subsection edge is given by all the final targets for a subsection.
// a final target is a vertex which is not a source to any other vertex.
depSheet.prototype.findEdges_pdf_subsection = function() {
var self = this;
var subsectionVertices = this.vertex.filter(function(f){return (f.type == "subsection")});
deLog(["pdf_subsection: preparing to extract final targets of subsections %s", subsectionVertices.map(function(vertex){return vertex.title})],8);
for (var vi = 0; vi < subsectionVertices.length; vi++) {
var subsection = subsectionVertices[vi].title;
var vertexesInSubsection = this.getVertexesInSubsection(subsection);
deLog(["pdf_subsection: examining %s vertices in subsection %s", vertexesInSubsection.length, subsection], 8);
var finalTargets = vertexesInSubsection.filter(function(vertex){
return (self.targetsFor(self.vertexId(vertex)).length == 0);
});
for (var fti = 0; fti < finalTargets.length; fti++) {
var ft = finalTargets[fti];
deLog([" learned pdf-subsection edge %s -> %s (%s -> %s)", self.vertexId(ft), self.vertex.indexOf(subsectionVertices[vi]), ft.title, subsection], 8);
self.edge.push([self.vertexId(ft), self.vertex.indexOf(subsectionVertices[vi])]);
}
}
};
depSheet.prototype.sourcesFor = function(v) {
return this.edge.filter(function(e){return e[1] == v}).map(function(e){return e[0]});
}
depSheet.prototype.targetsFor = function(v) {
return this.edge.filter(function(e){return e[0] == v}).map(function(e){return e[1]});
}
depSheet.prototype.vertexId = function(vertex) {
return this.vertex.indexOf(vertex);
}
depSheet.prototype.getSubsectionVertex = function(title) {
return this.vertex.filter(function(v){ return (v.type == "subsection" && v.title == title) })[0];
};
depSheet.prototype.getVertexesInSubsection = function(subsectiontitle) {
return this.vertex.filter(function(v){ return (v.subsection == subsectiontitle && v.title != subsectiontitle && v.type != "subsection") });
};
depSheet.prototype.getSubsections = function() {
return this.vertex.filter(function(v){ return (v.type == "subsection") });
};
depSheet.prototype.topoSortedSubsections = function() {
return this.topoSort(this.getSubsections());
};
depSheet.prototype.topoSortedPDFs = function(subsection) {
var vs = this.getVertexesInSubsection(subsection);
return this.topoSort(vs);
};
depSheet.prototype.topoSort = function(input) {
var self = this;
// let's borrow Kahn's algorithm.
// First, find a list of "start nodes" which have no incoming edges and insert them into a set S; at least one such node must exist in a non-empty acyclic graph. Then:
// L ← Empty list that will contain the sorted elements
// S ← Set of all nodes with no incoming edges
// while S is non-empty do
// remove a node n from S
// add n to tail of L
// for each node m with an edge e from n to m do
// remove edge e from the graph
// if m has no other incoming edges then
// insert m into S
// if graph has edges then
// return error (graph has at least one cycle)
// else
// return L (a topologically sorted order)
//
var start = input.filter(function(vertex){
return (self.sourcesFor(self.vertexId(vertex)).length == 0);
});
deLog(["topoSort: start nodes with no incoming edges are: %s", start],8);
};
depSheet.prototype.left_neighbours = function(vertex) {
var mysubsectionvertex = this.getSubsectionVertex(vertex.subsection);
var section_start = mysubsectionvertex.cell[0];
var section_end = mysubsectionvertex.endrow;
deLog(["looking for vertex %s's left neighbours, within section %s", vertex.title, mysubsectionvertex.title],8);
var vertices = this.getVertexesInSubsection(mysubsectionvertex.title);
deLog([" considering %s candidates in %s", vertices.length, mysubsectionvertex.title],8);
var verticesInColumn = [];
for (var jj = 1; jj <= vertex.cell[1]; jj++) {
verticesInColumn = vertices.filter(function(v){ return (v.type=="pdf" && v.cell[1] == vertex.cell[1]-jj) });
if (verticesInColumn.length) { break }
}
if (verticesInColumn.length) {
deLog([" found left neighbours %s", verticesInColumn.map(function(v){return v.title})],8);
}
return verticesInColumn;
};
depSheet.prototype.findEdges_pdf_pdf = function() {
var self = this;
// locate all the pdf-pdf edges
for (var v = 0; v < this.vertex.length; v++) {
var vertex = this.vertex[v];
if (vertex.type == "pdf") {
deLog(["computing all the pdf sources for pdf %s which is at %s", vertex.title, vertex.cell],8);
// the left-neighbour column of a given pdf cell is the first column to the left which contains a pdf cell within the same dependency subsection.
this.left_neighbours(vertex).map(function(left){ // sigh, var that = this
deLog([" learned pdf-pdf edge %s -> %s (%s -> %s)", self.vertex.indexOf(left), self.vertex.indexOf(vertex), left.title, vertex.title], 8);
self.edge.push([self.vertex.indexOf(left), v]);
});
}
}
};
// locate all the party-pdf edges.
depSheet.prototype.findEdges_party_pdf = function() {
for (var v = 0; v < this.vertex.length; v++) {
var vertex = this.vertex[v];
if (vertex.type == "pdf") {
deLog(["computing all the parties for pdf %s which is at %s", vertex.title, vertex.cell],8);
// walk down the string
for (var ii = 1; ii < this.numRows; ii++) {
var i = vertex.cell[0] + ii;
var j = vertex.cell[1] + 1;
if (! this.cell[i] || ! this.cell[i][j]) { // deLog([" !!! reached end of party string at %s,%s", i,j],8);
break;
}
if (this.vertex[this.cell[i][j]].type != "party") { deLog([" !!! found non-party cell at %s,%s while walking party string: %s",
i,j,this.vertex[this.cell[i][j]]],6); }
this.edge.push([this.cell[i][j], v]);
deLog([" learned party-pdf edge %s -> %s (%s -> %s)", this.cell[i][j], v, this.vertex[this.cell[i][j]].title, vertex.title], 8);
}
}
}
};
// http://cpettitt.github.io/project/dagre-d3/latest/demo/sentence-tokenization.html
depSheet.prototype.layout_dagre = function() {
};
depSheet.prototype.vertex2node = function(vertex){
var toreturn = {title:(vertex.type == "pdf" ? vertex.subsection + " - " : "") + vertex.title,
type:vertex.type,
};
return toreturn;
}
depSheet.prototype.layout_vertex2nodes = function() {
return this.vertex.map(this.vertex2node);
};
depSheet.prototype.edge2link = function(edge){
return {source:edge[0], target:edge[1]}
};
depSheet.prototype.layout_edges2links = function() {
return this.edge.map(this.edge2link);
};
// https://vida.io/documents/fGzpzjP98Bs2ShMHW
depSheet.prototype.layout_d3 = function(outfile) {
this.idempotent_write(outfile, JSON.stringify(this.layout_for_d3({})), "text/javascript");
};
depSheet.prototype.layout_for_d3 = function(myobject) {
myobject.nodes = this.layout_vertex2nodes();
myobject.links = this.layout_edges2links();
return myobject;
};
depSheet.prototype.layout_stages = function(outfile) {
var self = this;
//
// the animation shows the growth of a tree, then the ripening of the tree.
//
// GROWTH:
// First, just one goal -- the root (type=subsection).
// Then the documents to be signed for that subsection (type=pdf).
// Then the signatories for those documents. (type=party)
// Then the other three subsections. (type=subsection)
// And their documents. (type=pdf)
// And their parties. (type=parties)
//
// RIPENING:
// basically the toposort, from source to target, early to late, leaf to root.
// Each document and party starts out a neutral gray.
// Each subsection starts out red.
// pick the earliest subsection.
// change the colour of all the nodes in that subsection to black, to indicate that subsection is now active
// Wait one beat.
// Identify the ready PDFs -- anything with no sources or whose sources are all green turns its party leaves red, and turns itself red.
// Wait one beat.
// each party that is red waits a random brief amount of time, then turns green.
// That allows the next set of documents to turn red, together with their partes.
// Eventually all the PDFs and parties in a subsection are green, and
// so we turn that subsection node green.
//
// Maybe off to one side we can maintain a running count of signature and PDFs done?
// we have the idea of a stagetime, which is ordered
// STAGES OF GROWTH:
// this.stages.nodes = [ {id:0, nodes:[rootSubsection,...], links:[]} ]
//
// STAGES OF COLOUR CHANGE:
// rootSubsection = { title:..., type:..., ripeness:[ [0,neutral], [100,black], [200,red], [300,green] ] }
//
// the first stage is the final subsection.
var finalTarget = this.vertex
.filter(function(f){return (f.type == "subsection")})
.filter(function(vertex){
deLog(["%s has targets %s",
vertex.title,
self.targetsFor(self.vertexId(vertex))
.map(function(t){return self.vertex[t] ? self.vertex[t].title : t + " not found in vertex list"})
],8);
return (self.targetsFor(self.vertexId(vertex)).length == 0);
})
;
var staged = new depSheet();
deLog(["stage 1: finalTarget = %s", finalTarget],8);
staged.vertex = finalTarget;
staged.addStage({id:1})
// the second stage: all the PDFs in the final subsection.
// we have to recalculate all the links because the links
// are numbered by the index of the vertices, and if the
// vertices keep extending, the indexes will be different
// to their number from the "all" graph.
// let's just set up a fresh graph and import the desired bits into it.
deLog(["stage 2: subsection vertices: %s",
self.getVertexesInSubsection(finalTarget[0].title)
.filter(function(f){return f.type=="pdf"})
.length
],8);
staged.vertex = staged.vertex.concat(self.getVertexesInSubsection(finalTarget[0].title)
.filter(function(f){return f.type=="pdf"}));
deLog(["stage 2: staged.vertex now has length %s", staged.vertex.length],8);
staged.findEdges_party_pdf();
staged.findEdges_pdf_pdf();
staged.findEdges_pdf_subsection();
staged.addStage({id:2});
deLog(["stage 3: subsection vertices: %s",
self.getVertexesInSubsection(finalTarget[0].title)
.filter(function(f){return f.type=="party"})
.length
],8);
staged.vertex = staged.vertex.concat(self.getVertexesInSubsection(finalTarget[0].title)
.filter(function(f){return f.type=="party"}));
deLog(["stage 3: staged.vertex now has length %s", staged.vertex.length],8);
staged.findEdges_party_pdf();
staged.findEdges_pdf_pdf();
staged.findEdges_pdf_subsection();
staged.addStage({id:3});
// output
for (var s = 0; s < staged.stages.length; s++) {
var stage = staged.stages[s];
// convert v2n_current from vertex to node
stage.nodes = stage.v2n_current.map(self.vertex2node);
delete stage.v2n_current;
}
this.idempotent_write(outfile, JSON.stringify({stages:staged.stages}), "text/plain");
};
depSheet.prototype.addStage = function(newstage){
var self = this;
var previous = this.stages[this.stages.length-1];
// add the nodes which aren't already nodes from a previous stage.
deLog(["i have %s vertices available to consider.", this.vertex.length],8);
newstage.v2n_current = [];
newstage.links = this.layout_edges2links(); // need make run the same dedup as for nodes
for (var v = 0; v < self.vertex.length; v++) {
var vertex = self.vertex[v];
if (self.alreadyStaged(vertex)) {
deLog(["addStage %s: vertex %s previously staged.", newstage.id, vertex.title],8);
continue }
else {
deLog(["addStage %s: staging new vertex %s", newstage.id, vertex.title],8);
self.v2n_all.push(vertex);
newstage.v2n_current.push(vertex);
}
}
this.stages.push(newstage);
};
depSheet.prototype.alreadyStaged = function(test) { // lol, check out the O(n*m) profligacy
var self = this;
if (test.title == "Series Seed") {
deLog(["looking inside v2n_all for %s", test],8);
}
deLog(["search result: %s", self.v2n_all.indexOf(test)],8);
return (self.v2n_all.indexOf(test) >= 0);
}
// this Google Drive equivalent of cat > filename; if the file already exists, overwrite; if not, create.
depSheet.prototype.idempotent_write = function(filename, content, mimetype) {
var outfile;
var iterator = this.output_folder.folder.getFilesByName(filename);
if (iterator.hasNext()) {
outfile = iterator.next();
outfile.setContent(content);
}
else {
outfile = this.output_folder.folder.createFile(filename, content, mimetype);
}
return outfile;
}
// output an HTML index listing all the PDFs, organized by section.
depSheet.prototype.layout_htmlFileIndex = function() {
var self = this;
var root = XmlService.createElement("html");
var head = XmlService.createElement("head"); root.addContent(head);
var body = XmlService.createElement("body"); root.addContent(body);
var subsections = self.getSubsections();
var existingFiles = {};
var iterator = this.output_folder.folder.getFiles();
while (iterator.hasNext()) { var file = iterator.next(); existingFiles[file] = file }
for (var vs = 0; vs < subsections.length; vs++) {
body.addContent(XmlService.createElement("h2").setText(subsections[vs].title));
var pdflist = XmlService.createElement("ol"); body.addContent(pdflist);
var pdfs = this.getVertexesInSubsection(subsections[vs].title).filter(function(e){return e.type=="pdf"});
for (var vpdf = 0; vpdf < pdfs.length; vpdf++) {
pdflist.addContent(XmlService.createElement("li")
.addContent((pdfs[vpdf].name
? (XmlService.createElement("a").setAttribute("href",encodeURI(pdfs[vpdf].name)))
: (XmlService.createElement("b"))
).setText(pdfs[vpdf].title))
);
// if the PDF isn't in the output folder, add it.
if (pdfs[vpdf].file) {
if (existingFiles[pdfs[vpdf].file]) {
deLog(["html output: %s already exists, no need to add", pdfs[vpdf].name],9);
}
else {
deLog(["html output: adding %s to output folder %s", pdfs[vpdf].name, this.output_folder.name],8);
this.output_folder.folder.addFile(pdfs[vpdf].file);
}
}
}
}
this.idempotent_write("index.html", XmlService.getPrettyFormat().format(XmlService.createDocument(root)), "text/html");
};