-
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathcpedit.coffee
827 lines (801 loc) · 26.5 KB
/
cpedit.coffee
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
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
margin = 0.5
defaultPage = ->
xMin: 0
yMin: 0
xMax: 4
yMax: 4
FOLD = require 'fold'
foldAngleToOpacity = (foldAngle, assignment) ->
if assignment in ['M', 'V']
Math.max 0.1, (Math.abs foldAngle ? 180) / 180
else
1
class Editor
constructor: (@svg) ->
@undoStack = []
@redoStack = []
@updateUndoStack()
@fold =
file_spec: 1.2
file_creator: 'Crease Pattern Editor'
file_classes: ['singleModel']
frame_classes: ['creasePattern']
vertices_coords: []
edges_vertices: []
edges_assignment: []
edges_foldAngle: []
"cpedit:page": defaultPage()
@gridGroup = @svg.group()
.addClass 'grid'
@creaseGroup = @svg.group()
.addClass 'crease'
@creaseLine = {}
@vertexGroup = @svg.group()
.addClass 'vertex'
@vertexCircle = {}
@dragGroup = @svg.group()
.addClass 'drag'
@updateGrid()
updateGrid: ->
# Call whenever page dimensions change
page = @fold["cpedit:page"]
document?.getElementById('width')?.innerHTML = page.xMax
document?.getElementById('height')?.innerHTML = page.yMax
@gridGroup.clear()
for x in [0..page.xMax]
@gridGroup.line x, page.yMin, x, page.yMax
for y in [0..page.yMax]
@gridGroup.line page.xMin, y, page.xMax, y
@svg.viewbox page.xMin - margin, page.yMin - margin, page.xMax - page.xMin + 2*margin, page.yMax - page.yMin + 2*margin
nearestFeature: (pt) ->
p = [pt.x, pt.y]
page = @fold["cpedit:page"]
closest =
[
Math.max page.xMin, Math.min page.xMax, Math.round pt.x
Math.max page.yMin, Math.min page.yMax, Math.round pt.y
]
v = FOLD.geom.closestIndex p, @fold.vertices_coords
if v?
vertex = @fold.vertices_coords[v]
if FOLD.geom.dist(vertex, p) < FOLD.geom.dist(closest, p)
closest = vertex
x: closest[0]
y: closest[1]
setTitle: (title) ->
@fold['file_title'] = title
setMode: (mode) ->
@mode?.exit @
@mode = mode
@mode.enter @
setLineType: (@lineType) ->
setAbsFoldAngle: (@absFoldAngle) ->
getFoldAngle: ->
if @lineType == 'V'
@absFoldAngle
else if @lineType == 'M'
-@absFoldAngle
else
0
escape: ->
@mode?.escape? @
addVertex: (v) ->
[i, changedEdges] =
FOLD.filter.addVertexAndSubdivide @fold, [v.x, v.y], FOLD.geom.EPS
@drawVertex i if i == @fold.vertices_coords.length - 1 # new vertex
@drawEdge e for e in changedEdges
i
addCrease: (p1, p2, assignment, foldAngle) ->
p1 = @addVertex p1
p2 = @addVertex p2
newVertices = @fold.vertices_coords.length
changedEdges = FOLD.filter.addEdgeAndSubdivide @fold, p1, p2, FOLD.geom.EPS
for e in changedEdges[0]
@fold.edges_assignment[e] = assignment
@fold.edges_foldAngle[e] = foldAngle
@drawEdge e for e in changedEdges[i] for i in [0, 1]
@drawVertex v for v in [newVertices ... @fold.vertices_coords.length]
#console.log @fold
#@loadFold @fold
subdivide: ->
FOLD.filter.collapseNearbyVertices @fold, FOLD.geom.EPS
FOLD.filter.subdivideCrossingEdges_vertices @fold, FOLD.geom.EPS
@loadFold @fold
saveForUndo: ->
@undoStack.push FOLD.convert.deepCopy @fold
@redoStack = []
@updateUndoStack()
undo: ->
return unless @undoStack.length
@redoStack.push @fold
@fold = @undoStack.pop()
@loadFold @fold
@updateUndoStack()
redo: ->
return unless @redoStack.length
@undoStack.push @fold
@fold = @redoStack.pop()
@loadFold @fold
@updateUndoStack()
updateUndoStack: ->
document?.getElementById('undo')?.disabled = (@undoStack.length == 0)
document?.getElementById('redo')?.disabled = (@redoStack.length == 0)
transform: (matrix, integerize = true) ->
###
Main transforms we care about (reflection and 90-degree rotation) should
preserve integrality of coordinates. Force this when integerize is true.
###
@saveForUndo()
integers = (Number.isInteger(x) for x in coords \
for coords in @fold.vertices_coords) if integerize
FOLD.filter.transform @fold, matrix
if integerize
for ints, v in integers
for int, i in ints when int
@fold.vertices_coords[v][i] = Math.round @fold.vertices_coords[v][i]
@loadFold @fold
reflectX: ->
{xMin, xMax} = @fold['cpedit:page']
@transform FOLD.geom.matrixReflectAxis 0, 2, (xMin + xMax) / 2
reflectY: ->
{yMin, yMax} = @fold['cpedit:page']
@transform FOLD.geom.matrixReflectAxis 1, 2, (yMin + yMax) / 2
rotate90: (cw) ->
{xMin, xMax, yMin, yMax} = @fold['cpedit:page']
if cw
angle = Math.PI/2
else
angle = -Math.PI/2
@transform FOLD.geom.matrixRotate2D angle,
[(xMin + xMax) / 2, (yMin + yMax) / 2]
rotateCW: -> @rotate90 true
rotateCCW: -> @rotate90 false
translate: (dx, dy) ->
@transform FOLD.geom.matrixTranslate [dx, dy]
shiftL: -> @translate -1, 0
shiftR: -> @translate +1, 0
shiftU: -> @translate 0, -1
shiftD: -> @translate 0, +1
loadFold: (@fold) ->
@fold.version = 1.2
@mode?.exit @
@drawVertices()
@fold.edges_foldAngle ?=
for assignment in @fold.edges_assignment
switch assignment
when 'V'
180 # "The fold angle is positive for valley folds,"
when 'M'
-180 # "negative for mountain folds, and"
else
0 # "zero for flat, unassigned, and border folds"
@drawEdges()
@fold["cpedit:page"] ?=
if @fold.vertices_coords?.length
xMin: Math.min ...(v[0] for v in @fold.vertices_coords)
yMin: Math.min ...(v[1] for v in @fold.vertices_coords)
xMax: Math.max ...(v[0] for v in @fold.vertices_coords)
yMax: Math.max ...(v[1] for v in @fold.vertices_coords)
else
defaultPage()
@updateGrid()
document?.getElementById('title').value = @fold.file_title ? ''
@mode?.enter @
drawVertices: ->
@vertexGroup.clear()
@drawVertex v for v in [[email protected]_coords.length]
drawEdges: ->
@creaseGroup.clear()
@drawEdge e for e in [[email protected]_vertices.length]
drawVertex: (v) ->
@vertexCircle[v]?.remove()
@vertexCircle[v] = @vertexGroup.circle 0.2
.center ...(@fold.vertices_coords[v])
.attr 'data-index', v
drawEdge: (e) ->
@creaseLine[e]?.remove()
coords = (@fold.vertices_coords[v] for v in @fold.edges_vertices[e])
@creaseLine[e] =
l = @creaseGroup.line coords[0][0], coords[0][1], coords[1][0], coords[1][1]
.addClass @fold.edges_assignment[e]
.attr 'stroke-opacity',
foldAngleToOpacity @fold.edges_foldAngle[e], @fold.edges_assignment[e]
.attr 'data-index', e
cleanup: ->
# Check for vertices of degree 0, or of degree 2
# where the two incident edges are parallel.
# Consider vertices in decreasing order so that indices don't change.
FOLD.convert.edges_vertices_to_vertices_edges_unsorted @fold
for v in [@fold.vertices_coords.length-1 .. 0]
if @fold.vertices_edges[v].length == 0
FOLD.filter.removeVertex @fold, v
else if @fold.vertices_edges[v].length == 2
edges = @fold.vertices_edges[v]
vectors =
for edge in edges
vertices = @fold.edges_vertices[edge]
coords =
for vertex in vertices
@fold.vertices_coords[vertex]
FOLD.geom.mul (FOLD.geom.unit FOLD.geom.sub coords[0], coords[1]),
if vertices[0] == v then 1 else -1
if (FOLD.geom.dot vectors[0], vectors[1]) <= -1 + FOLD.geom.EPS
for otherV in @fold.edges_vertices[edges[1]]
break unless v == otherV
vertices = @fold.edges_vertices[edges[0]]
for i in [0...2]
if vertices[i] == v
vertices[i] = otherV
FOLD.filter.removeEdge @fold, edges[1]
FOLD.filter.removeVertex @fold, v
FOLD.convert.edges_vertices_to_vertices_edges_unsorted @fold
delete @fold.vertices_edges
@drawVertices()
@drawEdges()
convertToFold: (splitCuts, json = true) ->
## Add face structure to @fold
fold = FOLD.convert.deepCopy @fold
FOLD.convert.edges_vertices_to_vertices_edges_sorted fold
fold.frame_classes = (c for c in fold.frame_classes ? [] \
when c not in ['cuts', 'noCuts'])
unless FOLD.filter.cutEdges(fold).length
fold.frame_classes.push 'noCuts'
else if splitCuts
fold.frame_classes.push 'noCuts'
FOLD.filter.splitCuts fold
#console.log 'cut', fold
else
fold.frame_classes.push 'cuts'
FOLD.convert.vertices_edges_to_faces_vertices_edges fold
#console.log fold
fold = FOLD.convert.toJSON fold if json
fold
downloadFold: ->
#json = FOLD.convert.toJSON @fold # minimal content
@download @convertToFold(false), 'application/json', '.fold'
downloadSplitFold: ->
@download @convertToFold(true), 'application/json', '-split.fold'
convertToSVG: (options) ->
svg = @svg.clone()
svg.find('.C').front()
svg.find('.B').front()
svg.find('.B').stroke color: '#000000'
if options?.nice
## Cuts look the same as boundary, and are very thick (0.2).
svg.find('.B, .C').stroke width: 0.2
svg.find('.C').stroke color: '#000000'
## Nice blue/red, even in grayscale
svg.find('.M').stroke color: '#ff6060'
svg.find('.V').stroke color: '#385dcf'
## Instead of opacity, use thickness for bigger folds.
## 90 degrees has thickness 0.1, while 180 degrees has thickness 0.15.
svg.find('.M, .V, .B, .C').each ->
t = @attr 'stroke-opacity'
@stroke width: (1-t) * 0.05 + t * 0.15
@attr 'stroke-opacity', 1
else
svg.find('.M, .V, .B, .C').stroke width: 0.1
svg.find('.C').stroke color: '#00ff00'
svg.find('.M').stroke color: '#ff0000'
svg.find('.V').stroke color: '#0000ff'
unless options?.noUnfold
svg.find('.U').stroke color: '#ffff00', width: 0.1
svg.find('.vertex, .drag').remove()
if options?.grid
svg.find('.grid').stroke color: '#dddddd', width: 0.05
else
svg.find('.grid').remove()
svg.attr 'width', "#{@svg.viewbox().width}cm"
svg.attr 'height', "#{@svg.viewbox().height}cm"
svg.element('style').words '''
line { stroke-linecap: round; }
'''
svg.svg()
.replace /[ ]id="[^"]+"/g, ''
downloadSVG: ->
@download @convertToSVG(), 'image/svg+xml', '.svg'
download: (content, type, extension) ->
a = document.getElementById 'download'
a.href = url = URL.createObjectURL new Blob [content], {type}
a.download = (@fold.file_title or 'creasepattern') + extension
a.click()
a.href = ''
URL.revokeObjectURL url
class Mode
class LineDrawMode extends Mode
enter: (editor) ->
svg = editor.svg
@which = 0 ## 0 = first point, 1 = second point
@points = {}
@circles = []
@crease = @line = null
@dragging = false
svg.mousemove move = (e) =>
point = editor.nearestFeature svg.point e.clientX, e.clientY
## Wait for distance threshold in drag before triggering drag
if e.buttons
if @down?
unless point.x == @down.x and
point.y == @down.y
@dragging = true
@which = 1
else if @down == null
@down = point
@points[@which] = point
unless @which < @circles.length
@circles.push editor.dragGroup.circle 0.3
@circles[@which].center @points[@which].x, @points[@which].y
if @which == 1
@line ?= editor.dragGroup.line().addClass 'drag'
@crease ?= editor.dragGroup.line().addClass editor.lineType
.attr 'stroke-opacity',
foldAngleToOpacity editor.getFoldAngle(), editor.lineType
@line.plot @points[0].x, @points[0].y, @points[1].x, @points[1].y
@crease.plot @points[0].x, @points[0].y, @points[1].x, @points[1].y
svg.mousedown (e) =>
@down = null # special value meaning 'set'
move e
svg.mouseup (e) =>
move e
## Click, click style line drawing: advance to second point if not
## currently in drag mode, and didn't just @escape (no "down" point).
if @which == 0 and not @dragging and @down != undefined
@which = 1
else
## Commit new crease, unless it's zero length.
unless @which == 0 or (
@points[0].x == @points[1].x and @points[0].y == @points[1].y
)
editor.saveForUndo()
editor.addCrease @points[0], @points[1],
editor.lineType, editor.getFoldAngle()
@escape editor
move e
svg.mouseenter (e) =>
## Cancel crease if user exits, lets go of button, and re-enters
@escape editor if @dragging and e.buttons == 0
move e
svg.mouseleave (e) =>
if @circles.length == @which + 1
@circles.pop().remove()
escape: (editor) ->
@circles.pop().remove() while @circles.length
@crease?.remove()
@line?.remove()
@crease = @line = null
@which = 0
@dragging = false
@down = undefined
exit: (editor) ->
@escape editor
editor.svg
.mousemove null
.mousedown null
.mouseup null
.mouseenter null
.mouseleave null
class LinePaintMode extends Mode
enter: (editor) ->
svg = editor.svg
svg.mousedown change = (e) =>
return unless e.buttons
return unless e.target.tagName == 'line'
edge = parseInt e.target.getAttribute 'data-index'
return if isNaN edge
@paint editor, edge
svg.mouseover change # painting
exit: (editor) ->
editor.svg
.mousedown null
.mouseover null
class LineAssignMode extends LinePaintMode
paint: (editor, edge) ->
unless editor.fold.edges_assignment[edge] == editor.lineType and
editor.fold.edges_foldAngle[edge] == editor.getFoldAngle()
editor.saveForUndo()
editor.fold.edges_assignment[edge] = editor.lineType
editor.fold.edges_foldAngle[edge] = editor.getFoldAngle()
editor.drawEdge edge
class LineEraseMode extends LinePaintMode
paint: (editor, edge) ->
editor.saveForUndo()
vertices = editor.fold.edges_vertices[edge]
FOLD.filter.removeEdge editor.fold, edge
editor.drawEdges()
# Remove any now-isolated vertices
incident = {}
for edgeVertices in editor.fold.edges_vertices
for vertex in edgeVertices
incident[vertex] = true
# Remove vertices in decreasing order so that indices don't change
if vertices[0] < vertices[1]
vertices = [vertices[1], vertices[0]]
for vertex in vertices
unless incident[vertex]
FOLD.filter.removeVertex editor.fold, vertex
editor.drawVertices() # might get called twice
class VertexMoveMode extends Mode
enter: (editor) ->
svg = editor.svg
svg.mousemove move = (e) =>
@point = editor.nearestFeature svg.point e.clientX, e.clientY
if @vertex?
@drag editor
svg.mousedown (e) =>
@vertex = parseInt e.target.getAttribute 'data-index'
if e.target.tagName == 'circle' and @vertex?
@circle = e.target.instance
.addClass 'drag'
@down = null # special value meaning 'set'
move e
else
@circle = @vertex = null
svg.mouseup (e) =>
move e
if @vertex?
## Commit new location
unless @point.x == editor.fold.vertices_coords[@vertex][0] and
@point.y == editor.fold.vertices_coords[@vertex][1]
editor.saveForUndo()
editor.fold.vertices_coords[@vertex][0] = @point.x
editor.fold.vertices_coords[@vertex][1] = @point.y
@vertex = null
editor.subdivide()
#editor.drawVertex @vertex
#for vertices, edge in editor.fold.edges_vertices
# editor.drawEdge edge if @vertex in vertices
@escape editor
svg.mouseover (e) =>
return if @vertex?
return unless e.target.tagName == 'circle' and index = e.target.getAttribute 'data-index'
e.target.instance.addClass 'drag'
svg.mouseout (e) =>
return unless e.target.tagName == 'circle' and e.target.getAttribute 'data-index'
return if @vertex == parseInt e.target.getAttribute 'data-index'
e.target.instance.removeClass 'drag'
#svg.mouseenter (e) =>
# ## Cancel crease if user exits, lets go of button, and re-enters
# @escape editor if @dragging and e.buttons == 0
# move e
#svg.mouseleave (e) =>
# if @circles.length == @which + 1
# @circles.pop().remove()
escape: (editor) ->
if @vertex?
@circle.removeClass 'drag'
@point =
x: editor.fold.vertices_coords[@vertex][0]
y: editor.fold.vertices_coords[@vertex][1]
@drag editor
@circle = @vertex = null
exit: (editor) ->
@escape editor
editor.svg
.find '.vertex circle.drag'
.removeClass 'drag'
editor.svg
.mousemove null
.mousedown null
.mouseup null
.mouseenter null
.mouseleave null
drag: (editor) ->
@circle.center @point.x, @point.y
vertex = @vertex
point = @point
editor.svg.find '.crease line'
.each ->
edge = @attr 'data-index'
i = editor.fold.edges_vertices[edge].indexOf vertex
if i >= 0
@attr "x#{i+1}", point.x
@attr "y#{i+1}", point.y
modes =
drawLine: new LineDrawMode
assignLine: new LineAssignMode
eraseLine: new LineEraseMode
moveVertex: new VertexMoveMode
window?.onload = ->
svg = SVG().addTo '#interface'
editor = new Editor svg
for input in document.getElementsByTagName 'input'
do (input) ->
switch input.getAttribute 'name'
when 'mode'
if input.checked
editor.setMode modes[input.id]
input.addEventListener 'change', (e) ->
return unless input.checked
if input.id of modes
editor.setMode modes[input.id]
else
console.warn "Unrecognized mode #{input.id}"
when 'line'
if input.checked
editor.setLineType input.value
input.addEventListener 'change', (e) ->
return unless input.checked
editor.setLineType input.value
input.parentElement.addEventListener 'click', (e) ->
unless e.target == input or e.target.tagName in ['LABEL', 'INPUT', 'A']
input.click()
window.addEventListener 'keyup', (e) =>
switch e.key
when 'd', 'D'
document.getElementById('drawLine').click()
when 'a', 'A'
document.getElementById('assignLine').click()
when 'e', 'E'
document.getElementById('eraseLine').click()
when 'm'
document.getElementById('moveVertex').click()
when 'b', 'B'
document.getElementById('boundary').click()
when 'M'
document.getElementById('mountain').click()
when 'V'
document.getElementById('valley').click()
when 'u', 'U'
document.getElementById('unfolded').click()
when 'c', 'C'
document.getElementById('cut').click()
when 'Escape'
editor.escape()
when 'z'
editor.undo()
when 'y', 'Z'
editor.redo()
for id in ['cleanup', 'undo', 'redo', 'reflectX', 'reflectY', 'rotateCCW', 'rotateCW', 'shiftL', 'shiftD', 'shiftU', 'shiftR']
do (id) ->
document.getElementById(id).addEventListener 'click', (e) ->
e.stopPropagation()
editor[id]()
document.getElementById('loadFold').addEventListener 'click', (e) ->
e.stopPropagation()
document.getElementById('fileFold').click()
document.getElementById('fileFold').addEventListener 'input', (e) ->
e.stopPropagation()
return unless e.target.files.length
file = e.target.files[0]
reader = new FileReader
reader.onload = ->
editor.loadFold JSON.parse reader.result
reader.readAsText file
document.getElementById('downloadFold').addEventListener 'click', (e) ->
e.stopPropagation()
editor.downloadFold()
document.getElementById('downloadSplitFold').addEventListener 'click', (e) ->
e.stopPropagation()
editor.downloadSplitFold()
document.getElementById('downloadSVG').addEventListener 'click', (e) ->
e.preventDefault()
e.stopPropagation()
editor.downloadSVG()
for [size, dim] in [['width', 'x'], ['height', 'y']]
for [delta, op] in [[-1, 'Dec'], [+1, 'Inc']]
do (size, dim, delta, op) ->
document.getElementById(size+op).addEventListener 'click', (e) ->
e.stopPropagation()
editor.saveForUndo()
editor.fold["cpedit:page"][dim + 'Max'] += delta
editor.updateGrid()
document.getElementById('title').addEventListener 'input', (e) ->
editor.setTitle document.getElementById('title').value
## Fold angle
angleInput = document.getElementById 'angle'
angle = null
setAngle = (value) ->
return unless typeof value == 'number'
return if isNaN value
angle = value
angle = Math.max angle, 0
angle = Math.min angle, 180
angleInput.value = angle
editor.setAbsFoldAngle angle
setAngle parseFloat angleInput.value # initial value
angleInput.addEventListener 'change', (e) ->
setAngle eval angleInput.value # allow formulas via eval
for [sign, op] in [[+1, 'Add'], [-1, 'Sub']]
for amt in [1, 90]
document.getElementById("angle#{op}#{amt}").addEventListener 'click',
do (sign, amt) -> (e) ->
setAngle angle + sign * amt
## Origami Simulator
simulator = null
ready = false
onReady = null
checkReady = ->
if ready
onReady?()
onReady = null
window.addEventListener 'message', (e) ->
if e.data and e.data.from == 'OrigamiSimulator' and e.data.status == 'ready'
ready = true
checkReady()
document.getElementById('simulate').addEventListener 'click', (e) ->
if simulator? and not simulator.closed
simulator.focus()
else
ready = false
#simulator = window.open 'OrigamiSimulator/?model=', 'simulator'
simulator = window.open 'https://origamisimulator.org/?model=', 'simulator'
fold = editor.convertToFold true, false # split cuts, no JSON
## Origami Simulator wants 'F' for unfolded (facet) creases;
## it uses 'U' for undriven creases. :-/
fold.edges_assignment =
for assignment in fold.edges_assignment
if assignment == 'U'
'F'
else
assignment
onReady = -> simulator.postMessage
op: 'importFold'
fold: fold
, '*'
checkReady()
## CLI
## VDOM simulation of used subset of svg.js interface
class VSVG
constructor: (@tag, @parent) ->
@classes = new Set
@attrs = new Map
@children = []
svg: ->
s = ''
if @tag == 'svg'
s += '''
<?xml version="1.0" encoding="utf-8"?>
'''
@attrs.set 'xmlns', 'http://www.w3.org/2000/svg'
if @classes.size
@attrs.set 'class', (c for c from @classes).join ' '
else
@attrs.delete 'class'
s += "<#{@tag}"
for [key, value] from @attrs
s += " #{key}=\"#{value}\""
if @innerHTML
s + ">\n" + @innerHTML + "\n</#{@tag}>"
else if @children.length
s + ">\n" + (
for child in @children when not child.removed
child.svg()
).join("\n") +
"\n</#{@tag}>"
else
s + "/>"
remove: ->
@removed = true
@
clear: ->
child.parent = undefined for child in @children
@children = []
@
attr: (key, value) ->
if value? # setter
@attrs.set key, value
@
else # getter
@attrs.get key
viewbox: (x, y, width, height) ->
if x? # setter
@attr 'viewBox', "#{x} #{y} #{width} #{height}"
else # getter
coords = @attr 'viewBox'
.split /\s+/
.map parseFloat
x: coords[0]
y: coords[1]
width: coords[2]
height: coords[3]
addClass: (c) ->
@classes.add c
@
group: ->
@children.push child = new VSVG 'g', @
child
line: (x1, y1, x2, y2) ->
@children.push child = new VSVG 'line', @
child
.attr 'x1', x1
.attr 'y1', y1
.attr 'x2', x2
.attr 'y2', y2
stroke: ({color, width}) ->
@attr 'stroke', color if color?
@attr 'stroke-width', width if width?
@
circle: (diameter) ->
@children.push child = new VSVG 'circle', @
child.attr 'r', diameter / 2
center: (x, y) ->
console.assert @tag == 'circle'
@attr 'cx', x
.attr 'cy', y
front: ->
i = @parent.children.indexOf @
console.assert i >= 0
@parent.children.splice i, 1
@parent.children.push @
@
element: (tag) ->
@children.push child = new VSVG tag, @
child
words: (child) ->
@innerHTML = child
@
clone: ->
# Ignore clone operation because we're not rendering to DOM
@
find: (pattern) ->
classes =
for part in pattern.split /\s*,\s*/
match = part.match /^\.([^.]+)$/
throw new Error "Bad select pattern '#{part}'" unless match?
match[1]
results = []
results.each = (f) -> f.call node for node in @
for shortcut in ['stroke', 'remove', 'front']
do (shortcut) ->
results[shortcut] = (...args) ->
for node in results
node[shortcut](...args)
recurse = (node) ->
match = false
for class_ in classes
if node.classes.has class_
match = true
break
results.push node if match
for child in node.children when not child.removed
recurse child
return
recurse @
results
cli = (args = process.argv[2..]) ->
fs = require 'fs'
unless args.length
console.log """
Usage: coffee cpedit.coffee [formats/options] file1.fold file2.fold ...
Formats:
-s/--svg .svg
-f/--fold .fold
Options:
-c/--cleanup Remove unnecessary degree-0 and -2 vertices
-g/--grid Keep grid lines
-u/--no-unfold Don't color unfolded creases yellow
-n/--nice Nice colors instead of pure RGB for Origami Simulator
"""
formats = []
cpFiles = []
cleanup = false
options = {}
for arg in args
switch arg
when '-c', '--clean', '--cleanup'
cleanup = true
when '-s', '--svg'
formats.push 'SVG'
when '-f', '--fold'
formats.push 'Fold'
when '-u', '--no-unfold'
options.noUnfold = true
when '-g', '--grid'
options.grid = true
when '-n', '--nice'
options.nice = true
else
if arg.startsWith '-'
console.log "Unknown option: #{arg}"
continue
cpFiles.push arg
for cpFile in cpFiles
editor = new Editor new VSVG 'svg'
cpData = JSON.parse fs.readFileSync cpFile, encoding: 'utf8'
editor.loadFold cpData
editor.cleanup() if cleanup
for format in formats
output = editor["convertTo#{format}"] options
outputPath = cpFile.replace /(\.(fold|cp))?$/, ".#{format.toLowerCase()}"
fs.writeFileSync outputPath, output, encoding: 'utf8'
cli() if module? and require?.main == module