-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathoptimize_cuts_ga.py
executable file
·134 lines (107 loc) · 4.2 KB
/
optimize_cuts_ga.py
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
#!/usr/bin/env python
import sys
import striped_ga as optalg
#import segmented_ga as optalg
from optparse import OptionParser
from pyparsing import *
from itemplot import *
import itertools
def parse_input(f):
# define the grammar
ParserElement.setDefaultWhitespaceChars(" \t")
intnum = Word(nums)
floatnum = Combine(Word(nums) + "." + Word(nums) +
Optional('e'+oneOf("+ -")+Word(nums)))
number = (intnum^floatnum).setParseAction( lambda s,l,t: [ float(t[0]) ] )
#number = (intnum).setParseAction( lambda s,l,t: [ int(t[0]) ] )
comma = Literal(",").suppress()
nl = Literal("\n").suppress()
amount = Optional((comma + number).setParseAction( lambda s,l,t: [ int(t[0]) ]),default=1)
rotatable = Optional(comma + Literal("N"),default="Y").setParseAction( lambda s,l,t: [ t[0]=='Y' ] )
data = number + comma + number + Optional(amount + Optional(rotatable))
row = Group(data + restOfLine.setParseAction( lambda s,l,t: [ t[0].strip() ]))
rows = ZeroOrMore(pythonStyleComment.suppress() ^ row ^ nl.suppress())
return rows.parseFile(f)
def input_items(filename):
lines = parse_input(filename)
items = []
for line in lines:
w,h,n,r,s = line
typ = optalg.ItemType(w, h, s, r)
for i in range(n):
items.append(optalg.Item(typ))
return items
def types(items):
types = {}
for item in items:
types[item.type] = True
for typ in types.keys():
yield typ
def add_trim(items,trim):
for typ in types(items):
typ.w += trim
typ.h += trim
def remove_trim(items,trim):
for typ in types(items):
typ.w -= trim
typ.h -= trim
def verify_result(items,W,H):
# check that none of the items overlap each other
for item1,item2 in itertools.combinations(items,2):
if item1.overlaps(item2):
raise ValueError('Overlapping items')
# check that none of the items exceeds the board size
for item in items:
if item.x+item.w > W:
raise ValueError('item exceeds maximum width')
elif item.y+item.h > H:
raise ValueError('item exceeds maximum height')
if __name__=='__main__':
try:
import psyco
psyco.full()
except ImportError:
pass
parser = OptionParser()
parser.set_defaults(pairs=[])
#parser.add_option("-o","--output",dest="output",
# help="the output file name")
parser.add_option("-t","--trim",dest="trim",type="int",
help="the trim amount", default=0.)
parser.add_option("-H","--height",dest="height",type="int",
help="the plate height")
parser.add_option("-g","--generations",dest="generations",type="int",
help="the maximum number of generations",
default=200)
parser.add_option("-P","--plateau",dest="plateau",type="int",
help="the maximum number generations with no improvement",
default=10)
parser.add_option("-r","--randomize",dest="randomize",action="store_true",
default=False)
parser.add_option("--no-plot",dest="plot_result",action="store_false",
help="do not plot the resulting layout",
default=True)
parser.add_option("-p","--pop_size",dest="pop_size",type="int",
help="the size of the population",
default=100)
(options,args) = parser.parse_args()
H = options.height
trim = options.trim
generations = options.generations
randomize = options.randomize
plot_result = options.plot_result
pop_size = options.pop_size
items = input_items(args[0])
add_trim(items,trim)
W,items = optalg.optimize(items,H+trim,generations=generations,
randomize=randomize,
pop_size=pop_size,verbose=True)
# verify the result
verify_result(items,W,H+trim)
# remove the trim from the pieces
W -= trim
remove_trim(items,trim)
if plot_result:
plot_layout(items,W,H,show=True)
for item in items:
print item.w,item.h,item.x,item.y