File: test_flatten.rb

package info (click to toggle)
klayout 0.30.0-1
  • links: PTS, VCS
  • area: main
  • in suites: sid, trixie
  • size: 292,204 kB
  • sloc: cpp: 2,068,428; ruby: 47,823; xml: 26,924; python: 14,404; sh: 1,812; tcl: 212; perl: 170; makefile: 112; ansic: 42
file content (222 lines) | stat: -rw-r--r-- 5,470 bytes parent folder | download | duplicates (3)
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

class MenuAction < RBA::Action
  def initialize( title, shortcut, &action ) 
    self.title = title
    self.shortcut = shortcut
    @action = action
  end
  def triggered 
    @action.call( self ) 
  end
private
  @action
end

# HINT: this simple implementation does not account 
# for variant building - the shapes and instances are simply taken
# out from the original shapes and therefore might disappear somewhere else.

$f7_handler = MenuAction.new( "Flatten", "F7" ) do 

  app = RBA::Application.instance
  mw = app.main_window

  lv = mw.current_view
  if lv == nil
    raise "Flatten: No view selected"
  end

  # start transaction for "undo"
  lv.transaction( "Flatten" )

  begin

    # because objects might be referenced multiple times (thru different hierarchy paths)
    # we must first copy and then delete them

    # copy & transform them
    lv.each_object_selected do |sel|

      cv = lv.cellview( sel.cv_index ) 
      target = cv.cell 

      # only if not flat already ..
      if target.cell_index != sel.cell_index

        source = cv.layout.cell( sel.cell_index )

        if sel.is_cell_inst? 

          # copy and transform
          new_inst = target.insert( sel.inst )
          target.transform( new_inst, sel.source_trans )

        else

          # copy and transform
          target_shapes = target.shapes( sel.layer )
          new_shape = target_shapes.insert( sel.shape )
          target_shapes.transform( new_shape, sel.source_trans )
          
        end

      end

    end

    # delete the objects
    # HINT: since it is possible that a certain object is used multiple times, we need to test
    # each reference, if it is still valid (i.e. the object has not been deleted yet).
    lv.each_object_selected do |sel|

      cv = lv.cellview( sel.cv_index ) 
      target = cv.cell 

      # only if not flat already ..
      if target.cell_index != sel.cell_index

        source = cv.layout.cell( sel.cell_index )

        if sel.is_cell_inst? 
          if source.is_valid?( sel.inst )
            source.erase( sel.inst )
          end
        else
          if source.shapes( sel.layer ).is_valid?( sel.shape )
            source.shapes( sel.layer ).erase( sel.shape )
          end
        end

      end

    end

  ensure

    # always execute that code:

    # commit transaction
    lv.commit

    # clear selection and cancel all other edit operations, so 
    # nothing refers to shapes that might have been deleted.
    lv.cancel

  end

end

# HINT: this simple implementation does not account 
# for variant building - the shapes and instances are simply taken
# out from the original shapes and therefore might disappear somewhere else.

$f8_handler = MenuAction.new( "Propagate", "F8" ) do 

  app = RBA::Application.instance
  mw = app.main_window

  lv = mw.current_view
  if lv == nil
    raise "Propagate: No view selected"
  end

  # start transaction for "undo"
  lv.transaction( "Propagate" )

  begin

    # because objects might be referenced multiple times (thru different hierarchy paths)
    # we must first copy and then delete them

    # copy & transform them
    lv.each_object_selected do |sel|

      cv = lv.cellview( sel.cv_index ) 

      # only if not flat already ..
      if cv.cell_index != sel.cell_index

        source = cv.layout.cell( sel.cell_index )

        if sel.is_cell_inst? 

          # copy and transform
          if sel.path_length <= 2
            target = cv.cell 
          else
            target = cv.layout.cell( sel.path_nth( sel.path_length - 3 ).cell_inst.cell_index )
          end
          new_inst = target.insert( sel.inst )
          target.transform( new_inst, sel.path_nth( sel.path_length - 2 ).specific_cplx_trans )

        else

          # copy and transform
          if sel.path_length <= 1
            target = cv.cell 
          else
            target = cv.layout.cell( sel.path_nth( sel.path_length - 2 ).cell_inst.cell_index )
          end
          target_shapes = target.shapes( sel.layer )
          new_shape = target_shapes.insert( sel.shape )
          target_shapes.transform( new_shape, sel.path_nth( sel.path_length - 1 ).specific_cplx_trans )
          
        end

      end

    end

    # delete the objects
    # HINT: since it is possible that a certain object is used multiple times, we need to test
    # each reference, if it is still valid (i.e. the object has not been deleted yet).
    lv.each_object_selected do |sel|

      cv = lv.cellview( sel.cv_index ) 
      target = cv.cell 

      # only if not flat already ..
      if target.cell_index != sel.cell_index

        source = cv.layout.cell( sel.cell_index )

        if sel.is_cell_inst? 
          if source.is_valid?( sel.inst )
            source.erase( sel.inst )
          end
        else
          if source.shapes( sel.layer ).is_valid?( sel.shape )
            source.shapes( sel.layer ).erase( sel.shape )
          end
        end

      end

    end

  ensure

    # always execute that code:

    # commit transaction
    lv.commit

    # clear selection and cancel all other edit operations, so 
    # nothing refers to shapes that might have been deleted.
    lv.cancel

  end

end

app = RBA::Application.instance
mw = app.main_window

menu = mw.menu
menu.insert_separator("@toolbar.end", "name")
menu.insert_item("@toolbar.end", "rba_flatten", $f7_handler)
menu.insert_item("@toolbar.end", "rba_propagate", $f8_handler)

app.exec