-
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
-- Copyright 2026 Shota FUJI
--
-- This Source Code Form is subject to the terms of the Mozilla Public
-- License, v. 2.0. If a copy of the MPL was not distributed with this
-- file, You can obtain one at https://mozilla.org/MPL/2.0/.
--
-- SPDX-License-Identifier: MPL-2.0
module Template exposing (template)
import Length exposing (Length, toMM)
import Parameters exposing (Parameters, canvasSizeDimension)
import String
import Svg exposing (..)
import Svg.Attributes exposing (..)
import Svg.Path exposing (LargeArcFlag(..), PathCommand(..), PointMode(..), SweepFlag(..))
type Anchor
= TopLeft ( Float, Float )
| TopRight ( Float, Float )
| BottomLeft ( Float, Float )
| BottomRight ( Float, Float )
type alias Size =
{ width : Float, height : Float }
topLeftFor : Anchor -> Size -> ( Float, Float )
topLeftFor anchor { width, height } =
case anchor of
TopLeft ( x, y ) ->
( x, y )
TopRight ( x, y ) ->
( x - width, y )
BottomLeft ( x, y ) ->
( x, y - height )
BottomRight ( x, y ) ->
( x - width, y - height )
template : Parameters -> List (Svg.Attribute msg) -> Svg.Svg msg
template params attrs =
let
( canvasWidth, canvasHeight ) =
canvasSizeDimension params.rendering.size
|> Tuple.mapBoth toMM toMM
margin =
toMM params.rendering.margin
lugWidth =
toMM params.lugWidth
gap =
toMM params.rendering.gap
in
svg
(viewBox (String.join " " [ "0", "0", String.fromInt (ceiling canvasWidth), String.fromInt (ceiling canvasHeight) ])
:: class "print"
:: attrs
)
[ scaleChecker (BottomLeft ( margin, canvasHeight - margin ))
, legends params (BottomLeft ( margin + scaleCheckerSize.width + 10, canvasHeight - margin ))
, defs
[]
[ pattern
[ id "SkivingPattern"
, x "0"
, y "0"
, width "3"
, height "3"
, patternUnits "userSpaceOnUse"
]
[ Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( -1, -1 )
, LineTo Absolute ( 4, 4 )
]
, fill "none"
, stroke "currentColor"
, strokeWidth "0.1"
, strokeDasharray "1"
]
[]
]
]
, longPiece params (TopLeft ( margin, margin ))
, shortPiece params (TopLeft ( margin + lugWidth + gap, margin ))
, linings params (TopRight ( canvasWidth - margin, margin ))
]
scaleCheckerSize : Size
scaleCheckerSize =
Size 10 10
scaleChecker : Anchor -> Svg msg
scaleChecker anchor =
let
( ox, oy ) =
topLeftFor anchor scaleCheckerSize
px : Float -> String
px n =
String.fromFloat (ox + n)
py : Float -> String
py n =
String.fromFloat (oy + n)
in
g
[]
[ g
[ stroke "currentColor", strokeWidth "0.2" ]
[ line
[ x1 (px 0), y1 (py 8), x2 (px 0), y2 (py 10) ]
[]
, line
[ x1 (px 0), y1 (py 9), x2 (px 10), y2 (py 9) ]
[]
, line
[ x1 (px 10), y1 (py 8), x2 (px 10), y2 (py 10) ]
[]
]
, text_
[ x (px 5)
, y (py 7)
, fontSize "3"
, fontWeight "100"
, textAnchor "middle"
, fill "currentColor"
]
[ text "10mm" ]
]
legendsSize : Size
legendsSize =
Size 34 23
legends : Parameters -> Anchor -> Svg msg
legends params anchor =
let
( ox, oy ) =
topLeftFor anchor legendsSize
px : Float -> String
px n =
String.fromFloat (ox + n)
py : Float -> String
py n =
String.fromFloat (oy + n)
in
g
[]
[ Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy + 5 )
, HorizontalLineTo Relative 8
]
, stroke "currentColor"
, strokeWidth (params.rendering.lineWidth |> toMM |> String.fromFloat)
]
[]
, text_
[ x (px 10)
, y (py 6)
, fontSize "4"
, fontWeight "100"
, textAnchor "left"
, fill "currentColor"
]
[ text "Cut line" ]
, rect
[ x (px 0)
, y (py 10)
, stroke "currentColor"
, width "8"
, height "5"
, strokeWidth "0.2"
, fill "url(#SkivingPattern)"
]
[]
, text_
[ x (px 10)
, y (py 14)
, fontSize "4"
, fontWeight "100"
, textAnchor "left"
, fill "currentColor"
]
[ text "Skiving area" ]
, g
[ fill "none"
, stroke "currentColor"
, strokeWidth "0.1"
]
[ Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox + 1.5, oy + 20.5 )
, HorizontalLineTo Relative 5
]
]
[]
, Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox + 4, oy + 18 )
, VerticalLineTo Relative 5
]
]
[]
]
, text_
[ x (px 10)
, y (py 22)
, fontSize "4"
, fontWeight "100"
, textAnchor "left"
, fill "currentColor"
]
[ text "Hole center" ]
]
skivingSeamStroke : Parameters -> List (Svg.Attribute msg)
skivingSeamStroke params =
[ fill "none"
, stroke "currentColor"
, strokeWidth (toMM params.rendering.lineWidth |> String.fromFloat)
, strokeDasharray "1 0.5"
]
longPieceSize : Parameters -> Size
longPieceSize params =
Size (toMM params.lugWidth) (toMM params.longPiece.length + toMM params.longPiece.flap)
longPiece : Parameters -> Anchor -> Svg msg
longPiece params anchor =
let
lugWidth =
toMM params.lugWidth
length =
toMM params.longPiece.length
flap =
toMM params.longPiece.flap
( ox, oy ) =
topLeftFor anchor (longPieceSize params)
in
g
[]
(Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy )
, VerticalLineTo Relative (length + flap - lugWidth / 2)
, EllipticalArcCurve Relative
{ rx = lugWidth / 2
, ry = lugWidth / 2
, angle = 0.0
, largeArcFlag = LargeArc
, sweepFlag = Counterclockwise
, x = lugWidth
, y = 0
}
[]
, VerticalLineTo Relative -(length + flap - lugWidth / 2)
, ClosePath
]
, fill "none"
, stroke "currentColor"
, strokeWidth (toMM params.rendering.lineWidth |> String.fromFloat)
]
[]
:: Svg.path
(Svg.Path.d
[ MoveTo Absolute ( ox, oy + flap )
, HorizontalLineTo Relative lugWidth
]
:: skivingSeamStroke params
)
[]
:: Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy )
, HorizontalLineTo Relative lugWidth
, VerticalLineTo Relative flap
, HorizontalLineTo Relative -lugWidth
, ClosePath
]
, fill "url(#SkivingPattern)"
]
[]
:: (List.range 0 (params.longPiece.buckleHole.count - 1)
|> List.map
(\index ->
let
{ buckleHole } =
params.longPiece
offset =
toMM buckleHole.offset
interval =
toMM buckleHole.interval
in
hole
params
buckleHole.diameter
( ox + lugWidth / 2
, oy + offset + interval * toFloat index
)
)
)
)
hole : Parameters -> Length -> ( Float, Float ) -> Svg msg
hole params diameter ( cx, cy ) =
let
radius =
toMM diameter / 2
in
g
[]
[ circle
[ Svg.Attributes.cx (String.fromFloat cx)
, Svg.Attributes.cy (String.fromFloat cy)
, r (String.fromFloat radius)
, fill "none"
, stroke "currentColor"
, strokeWidth (toMM params.rendering.lineWidth |> String.fromFloat)
]
[]
, g
[ fill "none"
, stroke "currentColor"
, strokeWidth "0.1"
]
[ Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( cx - radius / 2, cy )
, HorizontalLineTo Relative radius
]
]
[]
, Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( cx, cy - radius / 2 )
, VerticalLineTo Relative radius
]
]
[]
]
]
shortPieceSize : Parameters -> Size
shortPieceSize params =
Size (toMM params.lugWidth) (toMM params.shortPiece.length + toMM params.shortPiece.caseSideFlap)
shortPiece : Parameters -> Anchor -> Svg msg
shortPiece params anchor =
let
lugWidth =
toMM params.lugWidth
length =
toMM params.shortPiece.length
caseSideFlap =
toMM params.shortPiece.caseSideFlap
claspSideFlap =
toMM params.shortPiece.claspSideFlap
( ox, oy ) =
topLeftFor anchor (shortPieceSize params)
in
g
[]
[ Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy )
, VerticalLineTo Relative (length + caseSideFlap + claspSideFlap)
, HorizontalLineTo Relative lugWidth
, VerticalLineTo Relative -(length + caseSideFlap + claspSideFlap)
, ClosePath
]
, fill "none"
, stroke "currentColor"
, strokeWidth (toMM params.rendering.lineWidth |> String.fromFloat)
]
[]
, g
(skivingSeamStroke params)
[ Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy + caseSideFlap )
, HorizontalLineTo Relative lugWidth
]
]
[]
, Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy + caseSideFlap + length )
, HorizontalLineTo Relative lugWidth
]
]
[]
]
, g
[ fill "url(#SkivingPattern)" ]
[ Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy )
, HorizontalLineTo Relative lugWidth
, VerticalLineTo Relative caseSideFlap
, HorizontalLineTo Relative -lugWidth
, ClosePath
]
]
[]
, Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy + caseSideFlap + length )
, HorizontalLineTo Relative lugWidth
, VerticalLineTo Relative claspSideFlap
, HorizontalLineTo Relative -lugWidth
, ClosePath
]
]
[]
]
]
linings : Parameters -> Anchor -> Svg msg
linings params anchor =
if toMM params.lining == 0 then
g [] []
else
let
gap =
toMM params.rendering.gap
lugWidth =
toMM params.lugWidth
sectionWidth =
lugWidth * 2 + gap
headerHeight =
5.0
size =
Size
sectionWidth
(headerHeight + gap / 2 + Basics.max (toMM params.longPiece.length) (toMM params.shortPiece.length))
( ox, oy ) =
topLeftFor anchor size
in
g
[]
[ longLining params (TopLeft ( ox, oy + headerHeight + gap / 2 ))
, shortLining params (TopLeft ( ox + lugWidth + gap, oy + headerHeight + gap / 2 ))
, text_
[ x (ox + sectionWidth / 2 |> String.fromFloat)
, y (oy + 4 |> String.fromFloat)
, fontSize "5"
, fontWeight "100"
, textAnchor "middle"
, fill "currentColor"
]
[ text ("Linings / t = " ++ (toMM params.lining |> String.fromFloat) ++ "mm") ]
]
longLiningSize : Parameters -> Size
longLiningSize params =
Size (toMM params.lugWidth) (toMM params.longPiece.length)
longLining : Parameters -> Anchor -> Svg msg
longLining params anchor =
let
lugWidth =
toMM params.lugWidth
length =
toMM params.longPiece.length
( ox, oy ) =
topLeftFor anchor (longLiningSize params)
in
g
[]
[ Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy )
, VerticalLineTo Relative (length - lugWidth / 2)
, EllipticalArcCurve Relative
{ rx = lugWidth / 2
, ry = lugWidth / 2
, angle = 0.0
, largeArcFlag = LargeArc
, sweepFlag = Counterclockwise
, x = lugWidth
, y = 0
}
[]
, VerticalLineTo Relative -(length - lugWidth / 2)
, ClosePath
]
, fill "none"
, stroke "currentColor"
, strokeWidth (toMM params.rendering.lineWidth |> String.fromFloat)
]
[]
]
shortLiningSize : Parameters -> Size
shortLiningSize params =
Size (toMM params.lugWidth) (toMM params.shortPiece.length)
shortLining : Parameters -> Anchor -> Svg msg
shortLining params anchor =
let
lugWidth =
toMM params.lugWidth
length =
toMM params.shortPiece.length
( ox, oy ) =
topLeftFor anchor (shortLiningSize params)
in
g
[]
[ Svg.path
[ Svg.Path.d
[ MoveTo Absolute ( ox, oy )
, VerticalLineTo Relative length
, HorizontalLineTo Relative lugWidth
, VerticalLineTo Relative -length
, ClosePath
]
, fill "none"
, stroke "currentColor"
, strokeWidth (toMM params.rendering.lineWidth |> String.fromFloat)
]
[]
]