forked from MakeMagazinDE/GRBLize
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathimport_files.pas
994 lines (891 loc) · 32.7 KB
/
import_files.pas
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
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
unit import_files;
interface
uses
SysUtils, StrUtils, Windows, Classes, Forms, Controls, Menus, MMSystem,
Math, StdCtrls, Graphics, FileCtrl, Dialogs, Clipper;
const
c_hpgl_scale = 40;
c_numOfFiles = 8 -1;
c_numOfPens = 32 -1;
type
Thpgl_cmd = (cmd_none, cmd_pa, cmd_pu, cmd_pd, cmd_sp, cmd_in,
cmd_number, cmd_drill, cmd_exit, cmd_nextline);
Taction = (none, lift, seek, mill, drill);
Tshape = (contour, inside, outside, pocket, drillhole);
Trotate = (deg0, deg90, deg180, deg270);
T_parseReturnType = (p_none, p_endofline, p_letters, p_number);
TFloatPoint = record
X: Double;
Y: Double;
end;
Tbounds = record
min: TIntPoint;
max: TIntPoint;
mid: TIntPoint;
end;
TfloatBounds = record
min: TFloatPoint;
max: TFloatPoint;
mid: TFloatPoint;
end;
Tfile_param = record
valid: boolean;
enable: boolean;
penoverride: Integer;
rotate: Trotate;
mirror: Boolean;
scale: Double;
isdrillfile:Boolean;
bounds: Tbounds;
offset: TintPoint;
user1: Double;
end;
Tpen_record = record
used: boolean;
enable: boolean;
shape: Tshape;
color: TColor;
diameter: Double;
tipdia: Double;
offset: TIntPoint; // in HPGL Plotter units 40/mm
scale: Double;
speed: Integer;
z_end: Double;
z_inc: Double;
atc: Integer;
tooltip: Integer;
force_closed: Boolean;
end;
Tatc_record = record
// enable: boolean; // ... und Pen ist eingeschaltet
used: boolean; // ... und Pen ist eingeschaltet
isInSpindle: boolean; // Flag wird von Wechsler-Routine aktualisiert
pen: Integer;
TREFok: boolean;
TLCok: boolean;
TLCref: Double;
TLCdelta: Double;
end;
Tblock_record = record
enable: boolean;
pen: Integer;
fileID: Integer; // von welchem File?
closed: boolean; // closed Polygon (TRUE) oder offener Linienpfad (FALSE)
isChild: boolean; // hat einen Parent
parentID: Integer; // -1 wenn kein Parent gefunden, sonst Block-#
isParent: boolean; // hat eine ChildList
childList: Array of Integer;
bounds: Tbounds;
outline_raw: Tpath; // outline path original (Integer Points)
outline: Tpath; // skaliert und mit Offsets
end;
Tfinal = record
enable: Boolean;
out_of_work: Boolean;
pen: Integer;
shape: Tshape;
closed: Boolean;
fileID: Integer; // von welchem File?
was_closed: Boolean;
bounds: Tbounds;
outlines: Tpaths;
millings: Tpaths;
milling_enables: array of Boolean; // äußere [0] und Child-Pfade
end;
Tjob = record
fileDelimStrings: Array[0..c_numOfFiles] of String[255];
pens: Array[0..31] of Tpen_record;
partsize_x: Double;
partsize_y: Double;
partsize_z: Double;
z_feed: Integer;
z_penlift: Double;
z_penup: Double;
z_gauge: Double;
cam_x: Double;
cam_y: Double;
cam_z_abs: Double;
park_x: Double;
park_y: Double;
park_z: Double;
toolchange_x: Double;
toolchange_y: Double;
toolchange_z: Double;
probe_x: Double;
probe_y: Double;
probe_z: Double;
probe_z_gauge: Double;
invert_z: Boolean;
parkposition_on_end: Boolean;
toolchange_pause: Boolean;
use_excellon_dia: Boolean;
optimize_drills: Boolean;
use_fixed_probe: Boolean;
use_part_probe: Boolean;
spindle_wait: Integer; // Hochlaufzeit
atc_enabled: Boolean;
atc_zero_x: Double;
atc_zero_y: Double;
atc_pickup_z: Double;
atc_delta_x: Double;
atc_delta_y: Double;
table_x: Double;
table_y: Double;
table_z: Double;
fix1_x: Double;
fix1_y: Double;
fix1_z: Double;
fix2_x: Double;
fix2_y: Double;
fix2_z: Double;
end;
const
ActionArray: Array [0..4] of String[7] =
('none', 'lift', 'seek', 'drill', 'mill');
ShapeArray: Array [0..4] of String[15] =
('Contour', 'Inside', 'Outside', 'Pocket', 'Drill');
ShapeColorArray: Array [0..4] of Tcolor =
(clBlack, clBlue, clRed, clFuchsia, clgreen);
ToolTipArray: Array [0..7] of String[15] =
('Flat Tip', 'Cone 30°', 'Cone 45°', 'Cone 60°', 'Cone 90°','Ball Tip','Drill', 'Dummy');
zeroPoint: TIntPoint = (
X: 0;
Y: 0;
);
PenInit: Tpen_record = (
used: false;
enable: false;
shape: contour;
color: clblack;
diameter: 2.0;
tipdia: 2.0;
scale: 100;
speed: 400;
z_end: 2.0;
z_inc: 1.0;
);
BlockZeroEntry: Tblock_record =
( enable: false;
pen: -1;
fileID: 0;
closed: false;
isChild: false;
parentID: -1;
isParent: false;
);
var
jp_old: TIntPoint;
job: Tjob;
use_inches_in_drillfile: Boolean;
FileParamArray: Array[0..c_numOfFiles] of Tfile_param;
blockArrays: Array [0..c_numOfFiles] of Array of Tblock_record;
final_array: Array of Tfinal;
final_bounds: Tbounds; // Abmessungen gesamt inkl. Offsets in HPGL-Units, wird in ListBlocks gesetzt
final_bounds_mm: TfloatBounds; // Abmessungen gesamt inkl. Offsets in mm, wird in ListBlocks gesetzt
ATCarray: Array[0..31] of Tatc_record;
CurrentPen, PendingPen: Integer;
LastAction, PendingAction: Taction;
CurrentBlockID,
CurrentfileID: Integer;
Last_z: Double;
LastPoint: TIntPoint;
FileIndex: Integer;
procedure init_blockArrays;
function point_in_bounds(my_point: TIntPoint; my_bounds: Tbounds): Boolean;
function bounds_in_bounds(my_bounds1, my_bounds2: Tbounds): Boolean;
procedure hpgl_fileload(my_name:String; fileID, penOverride: Integer);
procedure svg_fileload(my_name:String; fileID, penOverride: Integer);
procedure drill_fileload(my_name:String; fileID, penOverride: Integer; useDrillDia: Boolean);
procedure gcode_fileload(my_name:String; fileID, penOverride: Integer);
procedure apply_pen_change;
procedure item_change(arr_idx: Integer);
function RoundToDigits(zahl: double; n: integer): double;
function FloatToStrDot(my_val: Double):String;
function StrDotToFloat(my_str: String): Double;
// alle Pfad-Enables des übergebenen Blocks auf enable_status setzen
procedure enable_all_millings(var my_entry: Tfinal; enable_status: Boolean);
// ein bestimmtes Pfad-Enable des übergebenen Blocks auf enable_status setzen
procedure enable_single_milling(var my_entry: Tfinal; path_idx: Integer; enable_status: Boolean);
function is_any_milling_enabled(var my_entry: Tfinal):Boolean;
// sucht in Array my_path nach Punkt mit geringstem Abstand zu last_xy
function find_nearest_point(var search_path: Tpath; last_x, last_y: Integer): Integer;
// Zerlegt String nach Zahlen und Buchtstaben(ketten),
// beginnt my_line an Position my_pos nach Buchstaben oder Zahlen anbzusuchen.
// Wurde eine Zahl gefunden, ist Result = p_number, ansonsten p_letter.
// Wurde nichts (mehr) gefunden, ist Result = p_endofline.
// POSITION zeigt zum Schluss auf das Zeichen NACH dem letzten gültigen Wert.
// T_parseReturnType = (p_none, p_endofline, p_letters, p_number);
function ParseLine(var position: Integer; const linetoparse: string;
var value: Double; var letters: String): T_parseReturnType;
// Dekodiert einen einzelnes Befehlsbuchstaben/Wert-Paar, beginnend an Position
// Liefert Buchstaben in "letter" und folgenden Wert in "value" zurück
// Ergebnis ist TRUE, wenn Befehlsbuchstaben/Wert-Paar gefunden wurde
// POSITION zeigt zum Schluss auf das Zeichen NACH dem letzten gültigen Wert.
function ParseCommand(var position: Integer; var linetoparse: string;
var value: Double; var letter: char): boolean;
procedure ExecuteFile(const AFilename: String;
AParameter, ACurrentDir: String; AWait, AHide: Boolean);
implementation
uses grbl_player_main;
procedure enable_all_millings(var my_entry: Tfinal; enable_status: Boolean);
// alle Pfad-Enables des übergebenen Blocks auf enable_status setzen
var i: Integer;
begin
if length(my_entry.millings) > 0 then
for i := 0 to high(my_entry.millings) do
my_entry.milling_enables[i]:= enable_status;
if length(my_entry.millings) = 1 then
my_entry.enable:= enable_status;
end;
function is_any_milling_enabled(var my_entry: Tfinal):Boolean;
var i: Integer;
begin
result:= false;
if length(my_entry.millings) > 0 then
for i := 0 to high(my_entry.millings) do
if my_entry.milling_enables[i] then begin
result:= true;
break;
end;
end;
procedure enable_single_milling(var my_entry: Tfinal; path_idx: Integer; enable_status: Boolean);
// ein bestimmtes Pfad-Enable des übergebenen Blocks auf enable_status setzen
begin
if (path_idx < length(my_entry.millings)) and (length(my_entry.millings) > 0) then
my_entry.milling_enables[path_idx]:= enable_status;
if length(my_entry.millings) = 1 then
my_entry.enable:= enable_status;
end;
function RoundToDigits(zahl: double; n: integer): double;
// Runde Zahl auf n Nachkommastellen
var multi: double;
begin
multi:=IntPower(10, n);
zahl:=round(zahl*multi);
result:=zahl/multi;
end;
procedure ExecuteFile(const AFilename: String;
AParameter, ACurrentDir: String; AWait, AHide: Boolean);
var
si: TStartupInfo;
pi: TProcessInformation;
begin
if Length(ACurrentDir) = 0 then
ACurrentDir := ExtractFilePath(AFilename)
else if AnsiLastChar(ACurrentDir) = '\' then
Delete(ACurrentDir, Length(ACurrentDir), 1);
FillChar(si, SizeOf(si), 0);
with si do begin
cb := SizeOf(si);
dwFlags := STARTF_USESHOWWINDOW;
if AHide then
wShowWindow := SW_HIDE
else
wShowWindow := SW_NORMAL;
end;
FillChar(pi, SizeOf(pi), 0);
AParameter := Format('"%s" %s', [AFilename, TrimRight(AParameter)]);
if CreateProcess(Nil, PChar(AParameter), Nil, Nil, False,
CREATE_DEFAULT_ERROR_MODE or CREATE_NEW_CONSOLE or
NORMAL_PRIORITY_CLASS, Nil, PChar(ACurrentDir), si, pi) then
try
if AWait then
while WaitForSingleObject(pi.hProcess, 50) <> Wait_Object_0 do begin
end;
TerminateProcess(pi.hProcess, Cardinal(-1));
finally
CloseHandle(pi.hProcess);
CloseHandle(pi.hThread);
end;
end;
function FloatToStrDot(my_val: Double):String;
var
my_Settings: TFormatSettings;
begin
my_Settings.Create;
my_Settings.DecimalSeparator := '.';
FloatToStrDot:= FormatFloat('0.00',my_val,my_Settings);
end;
function StrDotToFloat(my_str: String): Double;
var
my_Settings: TFormatSettings;
begin
my_Settings.Create;
my_Settings.DecimalSeparator := '.';
StrDotToFloat:= StrToFloatDef(my_str,0,my_Settings);
end;
function ParseLine (var position: Integer; const linetoparse: string;
var value: Double; var letters: String): T_parseReturnType;
// Zerlegt String nach Zahlen und Buchtstaben(ketten),
// beginnt my_line an Position my_pos nach Buchstaben oder Zahlen anbzusuchen.
// Wurde eine Zahl gefunden, ist Result = p_number, ansonsten p_letter.
// Wurde nichts (mehr) gefunden, ist Result = p_endofline.
// POSITION zeigt zum Schluss auf das Zeichen NACH dem letzten gültigen Wert.
// T_parseReturnType = (p_none, p_endofline, p_letters, p_number);
var
my_str: String;
my_char: char;
my_end, i: integer;
begin
result:= p_endofline;
my_end:= length(linetoparse);
value:= 0;
letters:= '';
result:= p_none;
if (position > my_end) then
exit;
// Leer- und Steuerzeichen überspringen
repeat
my_char := linetoparse[position]; // erstes Zeichen
inc(position);
until (my_char in ['0'..'9', '.', '+', '-', 'A'..'z']) or (position > my_end);
dec(position); // Zeigt auf erstes relevantes Zeichen oder Ende
if position > my_end then
exit;
my_char := linetoparse[position]; // erstes relevantes Zeichen
if my_char = 'T' then begin
value:= 0;
end;
my_str:='';
if my_char in ['A'..'z'] then begin
result:= p_letters;
for i:= position to my_end do begin
if not (linetoparse[i] in ['A'..'z']) then
break;
my_str:= my_str+ linetoparse[i];
end;
position:= i;
letters:= my_str;
end else if my_char in ['0'..'9', '.', '+', '-', 'e', 'E'] then begin
result:= p_number;
for i:= position to my_end do begin
if not (linetoparse[i] in ['0'..'9', '.', '+', '-', 'e', 'E']) then
break;
my_str:= my_str+ linetoparse[i];
end;
position:= i;
value:= StrDotToFloat(my_str);
end;
end;
function ParseCommand(var position: Integer; var linetoparse: string;
var value: Double; var letter: char): boolean;
// Dekodiert einen einzelnes Befehlsbuchstaben/Wert-Paar, beginnend an Position
// Liefert Buchstaben in "letter" und folgenden Wert in "value" zurück
// Ergebnis ist TRUE, wenn Befehlsbuchstaben/Wert-Paar gefunden wurde
var
my_str: String;
begin
letter:= #13;
result:= false;
if position < length(linetoparse) then begin
letter:= linetoparse[position];
inc(position);
if (letter >= 'A') and (letter <= 'z') then begin
ParseLine(position, linetoparse, value, my_str);
result:= true;
end;
end;
end;
// #############################################################################
// #############################################################################
procedure init_blockArrays;
var i: Integer;
begin
CurrentBlockID:= 0;
for i := 0 to c_numOfFiles do begin
SetLength(blockArrays[i], 0);
end;
end;
function new_block(fileID: Integer): Integer;
// hängt leeren Block an
var my_len: Integer;
begin
my_len:= length(blockArrays[fileID]);
SetLength(blockArrays[fileID], my_len+1);
blockArrays[fileID,my_len]:= BlockZeroEntry;
SetLength(blockArrays[fileID,my_len].outline_raw, 0);
SetLength(blockArrays[fileID,my_len].outline, 0);
new_block:= my_len;
end;
procedure append_point(fileID, blockID: Integer; new_pt: TintPoint);
// für File-Import:
// hängt übergebenen Punkt an Block-Pfad an und setzt File-Bounds
var my_len: Integer;
begin
my_len:= length(blockArrays[fileID, blockID].outline_raw);
// // Skalierung und Offsets des Files
// new_pt.X:= FileParamArray[fileID].offset.x
// + (round(new_pt.X * 10 * FileParamArray[fileID].scale) div 1000);
// new_pt.Y:= FileParamArray[fileID].offset.y
// + (round(new_pt.Y * 10 * FileParamArray[fileID].scale) div 1000);
// Bounds des Files neu setzen
if new_pt.X < FileParamArray[fileID].bounds.min.x then
FileParamArray[fileID].bounds.min.x:= new_pt.X;
if new_pt.X > FileParamArray[fileID].bounds.max.x then
FileParamArray[fileID].bounds.max.x:= new_pt.X;
if new_pt.Y < FileParamArray[fileID].bounds.min.y then
FileParamArray[fileID].bounds.min.y:= new_pt.Y;
if new_pt.Y > FileParamArray[fileID].bounds.max.y then
FileParamArray[fileID].bounds.max.y:= new_pt.Y;
// Bounds des Blocks neu setzen
if new_pt.X < blockArrays[fileID, blockID].bounds.min.x then
blockArrays[fileID, blockID].bounds.min.x:= new_pt.X;
if new_pt.X > blockArrays[fileID, blockID].bounds.max.x then
blockArrays[fileID, blockID].bounds.max.x:= new_pt.X;
if new_pt.Y < blockArrays[fileID, blockID].bounds.min.y then
blockArrays[fileID, blockID].bounds.min.y:= new_pt.Y;
if new_pt.Y > FileParamArray[fileID].bounds.max.y then
blockArrays[fileID, blockID].bounds.max.y:= new_pt.Y;
// Raw-Points bereits mit File-Offset/Scale
SetLength(blockArrays[fileID, blockID].outline_raw, my_len+1);
blockArrays[fileID, blockID].outline_raw[my_len]:= new_pt;
SetLength(blockArrays[fileID, blockID].outline, my_len+1);
blockArrays[fileID, blockID].outline[my_len]:= new_pt;
end;
// #############################################################################
procedure file_rotate_mirror(fileID: Integer; auto_close_polygons: boolean);
// Jeden Block prüfen, ob geschlossener Pfad; danach outline_raw-Pfade
// rotieren und spiegeln
// Verwendet beim Import gesetzte File-Bounds
// muss gleich nach Import geschehen
var b,p, my_pathlen, my_blocklen: Integer;
nx, ny: Integer;
my_file_entry: Tfile_param;
my_pt, my_first_pt, my_last_pt: TIntPoint;
my_offset: TintPoint;
my_scale: Double;
begin
my_file_entry:= FileParamArray[fileID];
if not my_file_entry.valid then
exit;
my_blocklen:= length(blockArrays[fileID]);
if my_blocklen = 0 then // keine Blöcke enthalten
exit;
for b:= 0 to my_blocklen - 1 do begin
my_pathlen:= length(blockArrays[fileID, b].outline_raw);
if my_pathlen = 0 then // keine Pfade enthalten
continue;
// letzten Eintrag entfernen, falls gleich erstem Punkt, dafür "closed" setzen
my_first_pt:= blockArrays[fileID,b].outline_raw[0];
my_last_pt:= blockArrays[fileID,b].outline_raw[my_pathlen-1];
if (my_first_pt.X = my_last_pt.X) and (my_first_pt.Y = my_last_pt.Y) and
(my_pathlen > 1) and auto_close_polygons then begin
dec(my_pathlen);
blockArrays[fileID,b].closed:= true;
setlength(blockArrays[fileID,b].outline_raw, my_pathlen);
setlength(blockArrays[fileID,b].outline, my_pathlen);
end;// else
// blockArrays[fileID,b].closed:= false;
my_offset:= FileParamArray[fileID].offset;
my_scale:= FileParamArray[fileID].scale;
if FileParamArray[fileID].mirror then
my_offset.X:= -my_offset.X;
case my_file_entry.rotate of
deg90:
begin
ny:= my_offset.X;
my_offset.X:= my_offset.Y;
my_offset.Y:= -ny;
end;
deg180:
my_offset.X:= -my_offset.X;
deg270:
begin
ny:= my_offset.X;
my_offset.X:= -my_offset.Y;
my_offset.Y:= ny;
end;
end;
for p:= 0 to my_pathlen - 1 do begin
my_pt:= blockArrays[fileID, b].outline_raw[p];
// Skalierung und Offsets des Files
my_pt.X:= my_offset.X + (round(my_pt.X * 10 * my_scale) div 1000);
my_pt.Y:= my_offset.y + (round(my_pt.Y * 10 * my_scale) div 1000);
nx:= my_file_entry.bounds.min.x + my_file_entry.bounds.max.x - my_pt.X;
ny:= my_file_entry.bounds.min.y + my_file_entry.bounds.max.y - my_pt.Y;
if my_file_entry.mirror then
case my_file_entry.rotate of
deg0:
begin
my_pt.X:= nx;
end;
deg90:
begin // X:=Y, Y:=X, X und Y vertauschen
ny:= my_pt.Y;
my_pt.Y := my_pt.X;
my_pt.X := ny;
end;
deg180:
begin
my_pt.Y:= ny;
end;
deg270:
begin
my_pt.X:= ny;
my_pt.Y:= nx;
end;
end
else
case my_file_entry.rotate of
deg90:
begin
my_pt.Y:= my_pt.X;
my_pt.X:= ny;
end;
deg180:
begin
my_pt.X:= nx;
my_pt.Y:= ny;
end;
deg270:
begin
my_pt.X:= my_pt.Y;
my_pt.Y:= nx;
end;
end; // case
blockArrays[fileID, b].outline_raw[p]:= my_pt;
end; // path
end; // blocks
end;
procedure block_scale(fileID, blockID: Integer);
// Block mit Pen-Skalierung versehen
var i, my_pen, my_len: Integer;
my_bounds: Tbounds;
my_pt: TintPoint;
begin
if not FileParamArray[fileID].valid then
exit;
if length(blockArrays[fileID]) = 0 then
exit; // kein Block vorhanden
my_len:= length(blockArrays[fileID,blockID].outline_raw);
if my_len = 0 then
exit; // kein Path vorhanden
my_pen:= blockArrays[fileID, blockID].pen;
if my_pen < 0 then
exit; // kein Pen vorhanden
my_bounds.min.x:= high(Integer);
my_bounds.min.y:= high(Integer);
my_bounds.max.x:= low(Integer);
my_bounds.max.y:= low(Integer);
for i:= 0 to my_len-1 do begin
my_pt:= blockArrays[fileID, blockID].outline_raw[i];
// Skalierung des Blocks
my_pt.X:= round(my_pt.X * 10 * job.pens[my_pen].scale) div 1000;
my_pt.Y:= round(my_pt.Y * 10 * job.pens[my_pen].scale) div 1000;
if my_pt.X < my_bounds.min.x then
my_bounds.min.x:= my_pt.X;
if my_pt.X > my_bounds.max.x then
my_bounds.max.x:= my_pt.X;
if my_pt.Y < my_bounds.min.y then
my_bounds.min.y:= my_pt.Y;
if my_pt.Y > my_bounds.max.y then
my_bounds.max.y:= my_pt.Y;
blockArrays[fileID, blockID].outline[i]:= my_pt;
end;
blockArrays[fileID, blockID].enable:= FileParamArray[fileID].enable and job.pens[my_pen].enable;
my_bounds.mid.x:= (my_bounds.min.x + my_bounds.max.x) div 2;
my_bounds.mid.y:= (my_bounds.min.y + my_bounds.max.y) div 2;
blockArrays[fileID,blockID].bounds:= my_bounds;
end;
procedure block_scale_file(fileID: Integer);
// skaliert Pens/Tools über gesamtes File
var
my_blockID, my_blockcount: Integer;
begin
my_blockcount:= length(blockArrays[fileID]);
if my_blockcount = 0 then
exit;
if FileParamArray[fileID].valid then
for my_blockID:= 0 to my_blockcount - 1 do
block_scale(fileID, my_blockID);
end;
procedure block_scale_all;
// skaliert Pens/Tools über alle Files
var
i: Integer;
begin
for i:= 0 to c_numOfFiles do
block_scale_file(i);
end;
procedure block_scale_pen(penID: Integer);
// skaliert Pens/Tools über alle Files
var
my_fileID, my_blockID, my_blockcount: Integer;
begin
for my_fileID:= 0 to c_numOfFiles do begin
my_blockcount:= length(blockArrays[my_fileID]);
if FileParamArray[my_fileID].valid then
for my_blockID:= 0 to my_blockcount - 1 do begin
if blockArrays[my_fileID, my_blockID].pen = penID then
block_scale(my_fileID, my_blockID);
end;
end;
end;
function bounds_in_bounds(my_bounds1, my_bounds2: Tbounds): Boolean;
// liefert TRUE wenn my_bounds1 innerhalb oder auf my_bounds2
begin
bounds_in_bounds:= false;
if my_bounds1.min.x < my_bounds2.min.x then
exit;
if my_bounds1.min.y < my_bounds2.min.y then
exit;
if my_bounds1.max.x > my_bounds2.max.x then
exit;
if my_bounds1.max.y > my_bounds2.max.y then
exit;
bounds_in_bounds:= true;
end;
function point_in_bounds(my_point: TIntPoint; my_bounds: Tbounds): Boolean;
begin
point_in_bounds:= false;
if my_point.x < my_bounds.min.x then
exit;
if my_point.x > my_bounds.max.x then
exit;
if my_point.y < my_bounds.min.y then
exit;
if my_point.y > my_bounds.max.y then
exit;
point_in_bounds:= true;
end;
function find_nearest_point(var search_path: Tpath; last_x, last_y: Integer): Integer;
// sucht in Array my_path nach Punkt mit geringstem Abstand zu last_xy
var
p, temp_idx, lastdv, dv, dx, dy, my_len: Integer;
begin
my_len:= length(search_path);
lastdv:= high(Integer);
temp_idx:= 0;
for p:= 0 to my_len-1 do begin
dx:= search_path[p].x;
dy:= search_path[p].y;
// ist dieser Punkt ist gleich dem Ausgangspunkt?
if (dx = last_x) and (dy = last_y) then
continue; // wenn ja, überspringen
// finde nächstliegenden Punkt
dx:= abs(dx - last_x); // Abstand zum Ausgangspunkt
dy:= abs(dy - last_y);
// dv:= round(sqrt(sqr(dx) + sqr(dy)));
// schneller und ausreichend: Absolutwerte addieren
dv:= dx + dy;
if dv <= lastdv then begin
temp_idx:= p; // wird "mitgezogen"
lastdv:= dv;
end;
end;
find_nearest_point:= temp_idx;
end;
// #############################################################################
{$I hpgl_import.inc}
{$I drill_import.inc}
// #############################################################################
procedure add_outline_to_final(my_path: Tpath; my_Idx: Integer);
var my_len: Integer;
begin
if (length(my_path) > 0) and (my_Idx < length(final_array)) then begin
my_len:= length(final_array[my_Idx].outlines);
setlength(final_array[my_Idx].outlines, my_len+1);
final_array[my_Idx].outlines[my_len]:= my_path;
end;
end;
function add_block_to_final(my_block: Tblock_record): Integer;
// erzeugt neuen final_array-Eintrag, gibt Index zu neuem final zurück
var
i: Integer;
begin
i:= length(final_array);
if my_block.enable then begin
setlength(final_array, i+1);
// diese Werte können nachträglich geändert werden:
final_array[i].shape:= job.pens[my_block.pen].shape;
// diese Werte liegen seit Import fest:
final_array[i].enable:= my_block.enable;
// handle out_of_work independently from manual enable
final_array[i].out_of_work:= my_block.enable;
final_array[i].pen:= my_block.pen;
if job.pens[my_block.pen].force_closed then
final_array[i].closed:= true // ist ein Gerber-Import
else
final_array[i].closed:= my_block.closed;
final_array[i].was_closed:= my_block.closed;
final_array[i].bounds:= my_block.bounds;
// ersten Outline-Pfad übertragen
setlength(final_array[i].outlines, 1);
setlength(final_array[i].outlines[0], 1);
final_array[i].outlines[0]:= my_block.outline;
end;
add_block_to_final:= i;
end;
procedure make_final_array(fileID: Integer);
// Blocks zusammensuchen, Childs adoptieren
// Berücksichtigt nur eine Verwandschaftsebene!
// Trägt Werte aus Pen-Array ein
var i, c, p, m: Integer;
my_len: Integer;
begin
for p:= 0 to length(blockArrays[fileID])-1 do begin // Parent-Loop (p)
if blockArrays[fileID, p].parentID >= 0 then
continue; // ist bereits Parent
if not blockArrays[fileID, p].enable then
continue; // nicht aktiv
if not blockArrays[fileID, p].closed then
continue; // nicht geschlossen
if job.pens[blockArrays[fileID,0].pen].Shape in [drillhole, contour] then
continue; // ist bereits Parent
// Child-Objekte erstellen
for c:= 0 to length(blockArrays[fileID])-1 do begin // Child-Loop (c)
if p = c then
continue; // sind wir selbst
if not blockArrays[fileID, c].enable then
continue; // nicht aktiv
if blockArrays[fileID,c].pen <> blockArrays[fileID,p].pen then
continue; // nicht mein Pen
if not blockArrays[fileID,c].closed then
continue; // ist nur eine Linie
if blockArrays[fileID,c].parentID >= 0 then // ist bereits Parent
continue;
if not bounds_in_bounds(blockArrays[fileID,c].bounds, blockArrays[fileID,p].bounds) then
continue; // Child ist nicht in Parent-Bounds
// ist Outline von Block [k] in Outline von Block [i] enthalten?
// if PointInPolygon (blockArray[c].bounds.mid, blockArray[p].outline) <> 0 then begin
blockArrays[fileID,c].parentID:= p;
blockArrays[fileID,c].isChild:= true;
// Parent setzen
my_len:= length(blockArrays[fileID,p].childList);
setlength(blockArrays[fileID,p].childList, my_len+1);
blockArrays[fileID,p].childList[my_len]:= c;
blockArrays[fileID,p].isParent:= true;
// end;
end;
end;
for p:= 0 to length(blockArrays[fileID])-1 do begin
if not blockArrays[fileID,p].closed then begin
add_block_to_final(blockArrays[fileID,p]); // ist nur eine Linie
continue;
end;
if not blockArrays[fileID,p].isChild then begin // ist ersteinmal kein Child
m:= add_block_to_final(blockArrays[fileID,p]); // also hinzufügen
if blockArrays[fileID,p].isParent then // hat Block Childs?
for i:= 0 to length(blockArrays[fileID,p].childList)-1 do begin // Child-Loop (c)
c:= blockArrays[fileID,p].childList[i];
add_outline_to_final(blockArrays[fileID,c].outline, m);
end;
end;
end;
end;
// #############################################################################
// #############################################################################
// CLIPPER tools
// #############################################################################
// #############################################################################
procedure compile_milling(var my_final_entry: Tfinal);
// Wrapper für ClipperOffset für einzelnen Pen, mehrere Blocks
// erstellt milling-Paths-Array mit ggf. mehreren Pfadgruppen
// Milling-Pfade [0] sind immer outline
var i, j, rp, mp: Integer;
my_radius, my_dia: Double;
result_paths: Tpaths;
my_poly_end: TEndType;
my_cclw: boolean;
my_millingcount, my_pointcount: Integer; // millcount, pointcount
begin
// Offenbar seit Delphi XE8 funktioniert der Offset
// von innenliegenden Objekten mit Clipper nicht mehr.
// temporärer Workaround: Einzelne Objekte anlegen, keine Childs.
with TClipperOffset.Create() do
try
if (my_final_entry.shape = drillhole) or (my_final_entry.shape = contour) then begin
my_final_entry.millings:= my_final_entry.outlines;
exit; // weiter mit finally...
end;
my_radius:= job.pens[my_final_entry.pen].tipdia * (c_hpgl_scale div 2); // = mm * 40plu / 2
my_dia:= job.pens[my_final_entry.pen].tipdia/2;
if (my_final_entry.shape = inside) or (my_final_entry.shape = pocket) then begin
my_radius:= -my_radius;
end;
if length(my_final_entry.outlines)> 1 then begin
my_cclw:= Orientation(my_final_entry.outlines[0]); // ist Richtung Parent CCLW?
for i:= 1 to length(my_final_entry.outlines)-1 do begin
// wenn Parent Counterclockwise ist und auch Child, dann Child-Richtung umkehren
if Orientation(my_final_entry.outlines[i]) xor (not my_cclw) then
// Childs umkehren, damit sie ein "Loch" sind
my_final_entry.outlines[i]:= ReversePath(my_final_entry.outlines[i]);
end;
end;
Clear;
ArcTolerance:= 3;
if my_final_entry.closed then
my_poly_end:= etClosedPolygon
else
my_poly_end:= etOpenRound;
AddPaths(my_final_entry.outlines, jtRound, my_poly_end);
Execute(result_paths, my_radius);
my_final_entry.millings:= result_paths;
if my_final_entry.shape = pocket then begin
my_radius:= my_radius + 5;
if abs(my_radius) < 20 then // 0,5 mm
my_radius:= -20;
for i:= 0 to 99 do begin
Clear;
AddPaths(result_paths, jtRound, my_poly_end);
Execute(result_paths, my_radius);
if length(result_paths) = 0 then
break;
rp:= length(result_paths);
mp:= length(my_final_entry.millings);
setlength(my_final_entry.millings, mp + rp);
for j:= 0 to rp-1 do
my_final_entry.millings[mp+j]:= CleanPolygon(result_paths[j], my_dia);
end;
end;
// falls Clipper keine Pfade erzeugt hat, Kontur nehmen - sonst scheitert ListBlocks!
if length(my_final_entry.millings) = 0 then begin
my_final_entry.millings:= my_final_entry.outlines;
end else
// Außenkonturen einzelner Linien müssen geschlossen werden. Ersten als letzten Punkt anfügen
if (my_final_entry.shape = outside) then begin
my_millingcount:= length(my_final_entry.millings);
if my_millingcount > 0 then
for i:= 0 to my_millingcount - 1 do begin
my_pointcount:= length(my_final_entry.millings[i]);
setLength(my_final_entry.millings[i], my_pointcount + 1); // anfügen
my_final_entry.millings[i,my_pointcount]:= my_final_entry.millings[i,0];
end;
end;
finally
my_millingcount:= length(my_final_entry.millings);
// Enable-Flag-Array für einzelne Milling-Pfade erstellen
setLength(my_final_entry.milling_enables, my_millingcount);
if my_millingcount > 0 then
for i := 0 to my_millingcount-1 do
my_final_entry.milling_enables[i]:= my_final_entry.enable; // vorbelegen
Free;
end;
end;
// #############################################################################
// #############################################################################
procedure apply_pen_change;
// alle Änderungen, Offset etc.
var i: Integer;
begin
block_scale_all;
setlength(final_array, 0);
for i:= 0 to c_numOfFiles do
if FileParamArray[i].valid then
make_final_array(i);
// Werkzeugkorrektur-Offsets für fertiges BlockArray
for i:= 0 to high(final_array) do begin
compile_milling(final_array[i]);
// alle Pfad-Enables des übergebenen Blocks auf enable_status setzen
enable_all_millings(final_array[i], final_array[i].enable);
end;
ListBlocks;
end;
procedure item_change(arr_idx: Integer);
// Parameter-Änderungen in Final-Array anwenden
begin
if (arr_idx < length(final_array)) and (arr_idx >= 0) then
compile_milling(final_array[arr_idx]);
ListBlocks;
end;
end.