add rotation fuction
This commit is contained in:
parent
9febca7da6
commit
690df56bb7
17 changed files with 12374 additions and 19 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
|
@ -3,3 +3,4 @@
|
|||
*.pyc
|
||||
__pycache__
|
||||
pcb_tools_extension.egg-info
|
||||
test/outputs
|
||||
|
|
|
|||
201
LICENSE
Normal file
201
LICENSE
Normal file
|
|
@ -0,0 +1,201 @@
|
|||
Apache License
|
||||
Version 2.0, January 2004
|
||||
http://www.apache.org/licenses/
|
||||
|
||||
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||
|
||||
1. Definitions.
|
||||
|
||||
"License" shall mean the terms and conditions for use, reproduction,
|
||||
and distribution as defined by Sections 1 through 9 of this document.
|
||||
|
||||
"Licensor" shall mean the copyright owner or entity authorized by
|
||||
the copyright owner that is granting the License.
|
||||
|
||||
"Legal Entity" shall mean the union of the acting entity and all
|
||||
other entities that control, are controlled by, or are under common
|
||||
control with that entity. For the purposes of this definition,
|
||||
"control" means (i) the power, direct or indirect, to cause the
|
||||
direction or management of such entity, whether by contract or
|
||||
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||
|
||||
"You" (or "Your") shall mean an individual or Legal Entity
|
||||
exercising permissions granted by this License.
|
||||
|
||||
"Source" form shall mean the preferred form for making modifications,
|
||||
including but not limited to software source code, documentation
|
||||
source, and configuration files.
|
||||
|
||||
"Object" form shall mean any form resulting from mechanical
|
||||
transformation or translation of a Source form, including but
|
||||
not limited to compiled object code, generated documentation,
|
||||
and conversions to other media types.
|
||||
|
||||
"Work" shall mean the work of authorship, whether in Source or
|
||||
Object form, made available under the License, as indicated by a
|
||||
copyright notice that is included in or attached to the work
|
||||
(an example is provided in the Appendix below).
|
||||
|
||||
"Derivative Works" shall mean any work, whether in Source or Object
|
||||
form, that is based on (or derived from) the Work and for which the
|
||||
editorial revisions, annotations, elaborations, or other modifications
|
||||
represent, as a whole, an original work of authorship. For the purposes
|
||||
of this License, Derivative Works shall not include works that remain
|
||||
separable from, or merely link (or bind by name) to the interfaces of,
|
||||
the Work and Derivative Works thereof.
|
||||
|
||||
"Contribution" shall mean any work of authorship, including
|
||||
the original version of the Work and any modifications or additions
|
||||
to that Work or Derivative Works thereof, that is intentionally
|
||||
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||
or by an individual or Legal Entity authorized to submit on behalf of
|
||||
the copyright owner. For the purposes of this definition, "submitted"
|
||||
means any form of electronic, verbal, or written communication sent
|
||||
to the Licensor or its representatives, including but not limited to
|
||||
communication on electronic mailing lists, source code control systems,
|
||||
and issue tracking systems that are managed by, or on behalf of, the
|
||||
Licensor for the purpose of discussing and improving the Work, but
|
||||
excluding communication that is conspicuously marked or otherwise
|
||||
designated in writing by the copyright owner as "Not a Contribution."
|
||||
|
||||
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||
on behalf of whom a Contribution has been received by Licensor and
|
||||
subsequently incorporated within the Work.
|
||||
|
||||
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||
this License, each Contributor hereby grants to You a perpetual,
|
||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||
copyright license to reproduce, prepare Derivative Works of,
|
||||
publicly display, publicly perform, sublicense, and distribute the
|
||||
Work and such Derivative Works in Source or Object form.
|
||||
|
||||
3. Grant of Patent License. Subject to the terms and conditions of
|
||||
this License, each Contributor hereby grants to You a perpetual,
|
||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||
(except as stated in this section) patent license to make, have made,
|
||||
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||
where such license applies only to those patent claims licensable
|
||||
by such Contributor that are necessarily infringed by their
|
||||
Contribution(s) alone or by combination of their Contribution(s)
|
||||
with the Work to which such Contribution(s) was submitted. If You
|
||||
institute patent litigation against any entity (including a
|
||||
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||
or a Contribution incorporated within the Work constitutes direct
|
||||
or contributory patent infringement, then any patent licenses
|
||||
granted to You under this License for that Work shall terminate
|
||||
as of the date such litigation is filed.
|
||||
|
||||
4. Redistribution. You may reproduce and distribute copies of the
|
||||
Work or Derivative Works thereof in any medium, with or without
|
||||
modifications, and in Source or Object form, provided that You
|
||||
meet the following conditions:
|
||||
|
||||
(a) You must give any other recipients of the Work or
|
||||
Derivative Works a copy of this License; and
|
||||
|
||||
(b) You must cause any modified files to carry prominent notices
|
||||
stating that You changed the files; and
|
||||
|
||||
(c) You must retain, in the Source form of any Derivative Works
|
||||
that You distribute, all copyright, patent, trademark, and
|
||||
attribution notices from the Source form of the Work,
|
||||
excluding those notices that do not pertain to any part of
|
||||
the Derivative Works; and
|
||||
|
||||
(d) If the Work includes a "NOTICE" text file as part of its
|
||||
distribution, then any Derivative Works that You distribute must
|
||||
include a readable copy of the attribution notices contained
|
||||
within such NOTICE file, excluding those notices that do not
|
||||
pertain to any part of the Derivative Works, in at least one
|
||||
of the following places: within a NOTICE text file distributed
|
||||
as part of the Derivative Works; within the Source form or
|
||||
documentation, if provided along with the Derivative Works; or,
|
||||
within a display generated by the Derivative Works, if and
|
||||
wherever such third-party notices normally appear. The contents
|
||||
of the NOTICE file are for informational purposes only and
|
||||
do not modify the License. You may add Your own attribution
|
||||
notices within Derivative Works that You distribute, alongside
|
||||
or as an addendum to the NOTICE text from the Work, provided
|
||||
that such additional attribution notices cannot be construed
|
||||
as modifying the License.
|
||||
|
||||
You may add Your own copyright statement to Your modifications and
|
||||
may provide additional or different license terms and conditions
|
||||
for use, reproduction, or distribution of Your modifications, or
|
||||
for any such Derivative Works as a whole, provided Your use,
|
||||
reproduction, and distribution of the Work otherwise complies with
|
||||
the conditions stated in this License.
|
||||
|
||||
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||
any Contribution intentionally submitted for inclusion in the Work
|
||||
by You to the Licensor shall be under the terms and conditions of
|
||||
this License, without any additional terms or conditions.
|
||||
Notwithstanding the above, nothing herein shall supersede or modify
|
||||
the terms of any separate license agreement you may have executed
|
||||
with Licensor regarding such Contributions.
|
||||
|
||||
6. Trademarks. This License does not grant permission to use the trade
|
||||
names, trademarks, service marks, or product names of the Licensor,
|
||||
except as required for reasonable and customary use in describing the
|
||||
origin of the Work and reproducing the content of the NOTICE file.
|
||||
|
||||
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||
agreed to in writing, Licensor provides the Work (and each
|
||||
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||
implied, including, without limitation, any warranties or conditions
|
||||
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||
appropriateness of using or redistributing the Work and assume any
|
||||
risks associated with Your exercise of permissions under this License.
|
||||
|
||||
8. Limitation of Liability. In no event and under no legal theory,
|
||||
whether in tort (including negligence), contract, or otherwise,
|
||||
unless required by applicable law (such as deliberate and grossly
|
||||
negligent acts) or agreed to in writing, shall any Contributor be
|
||||
liable to You for damages, including any direct, indirect, special,
|
||||
incidental, or consequential damages of any character arising as a
|
||||
result of this License or out of the use or inability to use the
|
||||
Work (including but not limited to damages for loss of goodwill,
|
||||
work stoppage, computer failure or malfunction, or any and all
|
||||
other commercial damages or losses), even if such Contributor
|
||||
has been advised of the possibility of such damages.
|
||||
|
||||
9. Accepting Warranty or Additional Liability. While redistributing
|
||||
the Work or Derivative Works thereof, You may choose to offer,
|
||||
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||
or other liability obligations and/or rights consistent with this
|
||||
License. However, in accepting such obligations, You may act only
|
||||
on Your own behalf and on Your sole responsibility, not on behalf
|
||||
of any other Contributor, and only if You agree to indemnify,
|
||||
defend, and hold each Contributor harmless for any liability
|
||||
incurred by, or claims asserted against, such Contributor by reason
|
||||
of your accepting any such warranty or additional liability.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
APPENDIX: How to apply the Apache License to your work.
|
||||
|
||||
To apply the Apache License to your work, attach the following
|
||||
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||
replaced with your own identifying information. (Don't include
|
||||
the brackets!) The text should be enclosed in the appropriate
|
||||
comment syntax for the file format. We also recommend that a
|
||||
file or class name and description of purpose be included on the
|
||||
same "printed page" as the copyright notice for easier
|
||||
identification within third-party archives.
|
||||
|
||||
Copyright [yyyy] [name of copyright owner]
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
|
|
@ -5,8 +5,8 @@ This library is designed based on [PCB tools](https://github.com/curtacircuitos/
|
|||
|
||||
PCB tools extension adds following function to PCB tools.
|
||||
|
||||
- Rotate PCB data (imprementation is not completed)
|
||||
- Save loding PCB data
|
||||
- Rotate PCB data
|
||||
- Write back loded PCB data (original PCB tools does not work completely)
|
||||
- Merge multiple PCB data
|
||||
- Translate DXF file to gerber data
|
||||
|
||||
|
|
@ -25,12 +25,16 @@ ctx.merge(metal1)
|
|||
|
||||
metal2 = gerberex.read('board2.gtl')
|
||||
metal2.to_metric()
|
||||
metal2.rotate(-20)
|
||||
metal2.offset(30, 0)
|
||||
ctx.merge(metal2)
|
||||
|
||||
ctx.dump('panelized-board.gtl')
|
||||
```
|
||||
|
||||
```rotate()``` method can be used to rotate PCB data counterclockwise. you have to specify angle in degree<br>
|
||||
```offset()``` method can be used to move PCB data. Specified offset values are interpreted according to unit setting of PCB data. In case of the above code, ```board2.gtl``` move to 30mm left since ```to_metric()``` is called.
|
||||
|
||||
In case of Excellon drill data, you have to use ```DrillCompositon``` instead of ```GerberComposition```.
|
||||
|
||||
```python
|
||||
|
|
@ -43,6 +47,7 @@ ctx.merge(drill1)
|
|||
|
||||
drill2 = gerberex.read('board2.txt')
|
||||
drill2.to_metric()
|
||||
drill2.rotate(-20)
|
||||
drill2.offset(30, 0)
|
||||
ctx.merge(drill2)
|
||||
|
||||
|
|
|
|||
|
|
@ -7,13 +7,21 @@ from gerber.utils import *
|
|||
from gerber.am_statements import *
|
||||
from gerber.am_eval import OpCode
|
||||
|
||||
from gerberex.am_expression import eval_macro
|
||||
from gerberex.am_expression import eval_macro, AMConstantExpression, AMOperatorExpression
|
||||
|
||||
class AMPrimitiveDef(AMPrimitive):
|
||||
def __init__(self, code, exposure=None, rotation=0):
|
||||
def __init__(self, code, exposure=None, rotation=None):
|
||||
super(AMPrimitiveDef, self).__init__(code, exposure)
|
||||
if not rotation:
|
||||
rotation = AMConstantExpression(0)
|
||||
self.rotation = rotation
|
||||
|
||||
def rotate(self, angle, center=None):
|
||||
self.rotation = AMOperatorExpression(AMOperatorExpression.ADD,
|
||||
self.rotation,
|
||||
AMConstantExpression(float(angle)))
|
||||
self.rotation = self.rotation.optimize()
|
||||
|
||||
def to_inch(self):
|
||||
pass
|
||||
|
||||
|
|
@ -44,12 +52,12 @@ class AMCommentPrimitiveDef(AMPrimitiveDef):
|
|||
class AMCirclePrimitiveDef(AMPrimitiveDef):
|
||||
@classmethod
|
||||
def from_modifiers(cls, code, modifiers):
|
||||
exposure = 'on' if modifiers[0] == 1 else 'off',
|
||||
diameter = modifiers[1],
|
||||
center_x = modifiers[2],
|
||||
center_y = modifiers[3],
|
||||
exposure = 'on' if modifiers[0].value == 1 else 'off'
|
||||
diameter = modifiers[1]
|
||||
center_x = modifiers[2]
|
||||
center_y = modifiers[3]
|
||||
rotation = modifiers[4]
|
||||
return cls(code, expressions, center_x, center_y, rotation)
|
||||
return cls(code, exposure, diameter, center_x, center_y, rotation)
|
||||
|
||||
def __init__(self, code, exposure, diameter, center_x, center_y, rotation):
|
||||
super(AMCirclePrimitiveDef, self).__init__(code, exposure, rotation)
|
||||
|
|
@ -87,7 +95,7 @@ class AMVectorLinePrimitiveDef(AMPrimitiveDef):
|
|||
@classmethod
|
||||
def from_modifiers(cls, code, modifiers):
|
||||
code = code
|
||||
exposure = 'on' if modifiers[0] == 1 else 'off'
|
||||
exposure = 'on' if modifiers[0].value == 1 else 'off'
|
||||
width = modifiers[1]
|
||||
start_x = modifiers[2]
|
||||
start_y = modifiers[3]
|
||||
|
|
@ -141,7 +149,7 @@ class AMCenterLinePrimitiveDef(AMPrimitiveDef):
|
|||
@classmethod
|
||||
def from_modifiers(cls, code, modifiers):
|
||||
code = code
|
||||
exposure = 'on' if modifiers[0] == 1 else 'off'
|
||||
exposure = 'on' if modifiers[0].value == 1 else 'off'
|
||||
width = modifiers[1]
|
||||
height = modifiers[2]
|
||||
x = modifiers[3]
|
||||
|
|
@ -191,7 +199,7 @@ class AMOutlinePrimitiveDef(AMPrimitiveDef):
|
|||
def from_modifiers(cls, code, modifiers):
|
||||
num_points = modifiers[1] + 1
|
||||
code = code
|
||||
exposure = 'on' if modifiers[0] == 1 else 'off'
|
||||
exposure = 'on' if modifiers[0].value == 1 else 'off'
|
||||
addrs = modifiers[2:num_points * 2]
|
||||
rotation = modifiers[3 + num_points * 2]
|
||||
return cls(code, exposure, addrs, rotation)
|
||||
|
|
@ -231,7 +239,7 @@ class AMPolygonPrimitiveDef(AMPrimitiveDef):
|
|||
@classmethod
|
||||
def from_modifiers(cls, code, modifiers):
|
||||
code = code
|
||||
exposure = 'on' if modifiers[0] == 1 else 'off'
|
||||
exposure = 'on' if modifiers[0].value == 1 else 'off'
|
||||
vertices = modifiers[1]
|
||||
x = modifiers[2]
|
||||
y = modifiers[3]
|
||||
|
|
@ -417,6 +425,9 @@ class AMVariableDef(object):
|
|||
yield i
|
||||
yield (OpCode.STORE, self.number)
|
||||
|
||||
def rotate(self, angle, center=None):
|
||||
pass
|
||||
|
||||
def to_primitive_defs(instructions):
|
||||
classes = {
|
||||
0: AMCommentPrimitiveDef,
|
||||
|
|
@ -434,4 +445,4 @@ def to_primitive_defs(instructions):
|
|||
yield AMVariableDef(-code, modifiers[0])
|
||||
else:
|
||||
primitive = classes[code]
|
||||
yield primitive.from_modifiers(code, modifiers)
|
||||
yield primitive.from_modifiers(code, modifiers)
|
||||
|
|
|
|||
|
|
@ -101,7 +101,7 @@ class GerberComposition(Composition):
|
|||
|
||||
if not self.settings:
|
||||
self.settings = file.settings
|
||||
self.param_statements = file.header
|
||||
self.param_statements = [file.header]
|
||||
|
||||
|
||||
def _register_aperture_macro(self, statement):
|
||||
|
|
|
|||
|
|
@ -111,7 +111,7 @@ class DxfArcStatement(DxfStatement):
|
|||
settings.zero_suppression),
|
||||
write_gerber_value(begin_y, settings.format,
|
||||
settings.zero_suppression),
|
||||
'03' if deg0 > deg1 else '02',
|
||||
'03',
|
||||
write_gerber_value(end_x, settings.format,
|
||||
settings.zero_suppression),
|
||||
write_gerber_value(end_y, settings.format,
|
||||
|
|
|
|||
|
|
@ -6,6 +6,7 @@
|
|||
from gerber.excellon import (ExcellonParser, detect_excellon_format, ExcellonFile)
|
||||
from gerber.excellon_statements import UnitStmt
|
||||
from gerber.cam import FileSettings
|
||||
from gerberex.utility import rotate
|
||||
|
||||
def loads(data, filename=None, settings=None, tools=None, format=None):
|
||||
if not settings:
|
||||
|
|
@ -27,6 +28,12 @@ class ExcellonFileEx(ExcellonFile):
|
|||
def __init__(self, statements, tools, hits, settings, filename=None):
|
||||
super(ExcellonFileEx, self).__init__(statements, tools, hits, settings, filename)
|
||||
|
||||
def rotate(self, angle, center=(0,0)):
|
||||
if angle % 360 == 0:
|
||||
return
|
||||
for hit in self.hits:
|
||||
hit.position = rotate(hit.position[0], hit.position[1], angle, center)
|
||||
|
||||
class UnitStmtEx(UnitStmt):
|
||||
@classmethod
|
||||
def from_statement(cls, stmt):
|
||||
|
|
|
|||
|
|
@ -4,7 +4,9 @@
|
|||
# Copyright 2019 Hiroshi Murayama <opiopan@gmail.com>
|
||||
|
||||
import gerber.rs274x
|
||||
from gerberex.statements import (AMParamStmt, AMParamStmtEx)
|
||||
from gerber.gerber_statements import ADParamStmt, CoordStmt
|
||||
from gerberex.statements import AMParamStmt, AMParamStmtEx
|
||||
from gerberex.utility import rotate
|
||||
|
||||
class GerberFile(gerber.rs274x.GerberFile):
|
||||
@classmethod
|
||||
|
|
@ -23,3 +25,64 @@ class GerberFile(gerber.rs274x.GerberFile):
|
|||
|
||||
def __init__(self, statements, settings, primitives, apertures, filename=None):
|
||||
super(GerberFile, self).__init__(statements, settings, primitives, apertures, filename)
|
||||
|
||||
def rotate(self, angle, center=(0,0)):
|
||||
if angle % 360 == 0:
|
||||
return
|
||||
self._generalize_aperture()
|
||||
for statement in self.statements:
|
||||
if isinstance(statement, AMParamStmtEx):
|
||||
statement.rotate(angle, center)
|
||||
elif isinstance(statement, CoordStmt) and statement.x != None and statement.y != None:
|
||||
statement.x, statement.y = rotate(statement.x, statement.y, angle, center)
|
||||
|
||||
def _generalize_aperture(self):
|
||||
RECTANGLE = 0
|
||||
LANDSCAPE_OBROUND = 1
|
||||
PORTRATE_OBROUND = 2
|
||||
POLYGON = 3
|
||||
macro_defs = [
|
||||
('MACR', AMParamStmtEx.rectangle),
|
||||
('MACLO', AMParamStmtEx.landscape_obround),
|
||||
('MACPO', AMParamStmtEx.portrate_obround),
|
||||
('MACP', AMParamStmtEx.polygon)
|
||||
]
|
||||
|
||||
need_to_change = False
|
||||
insert_point = 0
|
||||
last_aperture = 0
|
||||
macros = {}
|
||||
for idx in range(0, len(self.statements)):
|
||||
statement = self.statements[idx]
|
||||
if isinstance(statement, AMParamStmtEx):
|
||||
macros[statement.name] = statement
|
||||
if not need_to_change:
|
||||
insert_point = idx + 1
|
||||
if isinstance(statement, ADParamStmt) and statement.shape in ['R', 'O', 'P']:
|
||||
need_to_change = True
|
||||
last_aperture = idx
|
||||
|
||||
if need_to_change:
|
||||
for idx in range(0, len(macro_defs)):
|
||||
macro_def = macro_defs[idx]
|
||||
name = macro_def[0]
|
||||
num = 1
|
||||
while name in macros:
|
||||
name = '%s_%d' % (macro_def[0], num)
|
||||
num += 1
|
||||
self.statements.insert(insert_point, macro_def[1](name))
|
||||
macro_defs[idx] = (name, macro_def[1])
|
||||
for idx in range(insert_point, last_aperture + len(macro_defs) + 1):
|
||||
statement = self.statements[idx]
|
||||
if isinstance(statement, ADParamStmt):
|
||||
if statement.shape == 'R':
|
||||
statement.shape = macro_defs[RECTANGLE][0]
|
||||
elif statement.shape == 'O':
|
||||
x = statement.modifiers[0] \
|
||||
if len(statement.modifiers) > 0 else 0
|
||||
y = statement.modifiers[1] \
|
||||
if len(statement.modifiers) > 1 else 0
|
||||
statement.shape = macro_defs[LANDSCAPE_OBROUND][0] \
|
||||
if x > y else macro_defs[PORTRATE_OBROUND][0]
|
||||
elif statement.shape == 'P':
|
||||
statement.shape = macro_defs[POLYGON][0]
|
||||
|
|
|
|||
|
|
@ -11,9 +11,41 @@ class AMParamStmtEx(AMParamStmt):
|
|||
def from_stmt(cls, stmt):
|
||||
return cls(stmt.param, stmt.name, stmt.macro)
|
||||
|
||||
@classmethod
|
||||
def circle(cls, name):
|
||||
return cls('AM', name, '1,1,$1,0,0,0*1,0,$2,0,0,0')
|
||||
|
||||
@classmethod
|
||||
def rectangle(cls, name):
|
||||
return cls('AM', name, '21,1,$1,$2,0,0,0*1,0,$3,0,0,0')
|
||||
|
||||
@classmethod
|
||||
def landscape_obround(cls, name):
|
||||
return cls(
|
||||
'AM', name,
|
||||
'$4=$1-$2*'
|
||||
'21,1,$1-$4,$2,0,0,0*'
|
||||
'1,1,$4,$4/2,0,0*'
|
||||
'1,1,$4,-$4/2,0,0*'
|
||||
'1,0,$3,0,0,0')
|
||||
|
||||
@classmethod
|
||||
def portrate_obround(cls, name):
|
||||
return cls(
|
||||
'AM', name,
|
||||
'$4=$2-$1*'
|
||||
'21,1,$1,$2-$4,0,0,0*'
|
||||
'1,1,$4,0,$4/2,0*'
|
||||
'1,1,$4,0,-$4/2,0*'
|
||||
'1,0,$3,0,0,0')
|
||||
|
||||
@classmethod
|
||||
def polygon(cls, name):
|
||||
return cls('AM', name, '5,1,$2,0,0,$1,$3*1,0,$4,0,0,0')
|
||||
|
||||
def __init__(self, param, name, macro):
|
||||
super(AMParamStmtEx, self).__init__(param, name, macro)
|
||||
self.primitive_defs = to_primitive_defs(self.instructions)
|
||||
self.primitive_defs = list(to_primitive_defs(self.instructions))
|
||||
|
||||
def to_inch(self):
|
||||
if self.units == 'metric':
|
||||
|
|
@ -32,3 +64,7 @@ class AMParamStmtEx(AMParamStmt):
|
|||
for p in self.primitive_defs:
|
||||
yield p.to_gerber(settings)
|
||||
return "%%AM%s*\n%s%%" % (self.name, '\n'.join(plist()))
|
||||
|
||||
def rotate(self, angle, center=None):
|
||||
for primitive_def in self.primitive_defs:
|
||||
primitive_def.rotate(angle, center)
|
||||
|
|
|
|||
13
gerberex/utility.py
Normal file
13
gerberex/utility.py
Normal file
|
|
@ -0,0 +1,13 @@
|
|||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
# Copyright 2019 Hiroshi Murayama <opiopan@gmail.com>
|
||||
|
||||
from math import cos, sin, pi
|
||||
|
||||
def rotate(x, y, angle, center):
|
||||
x0 = x - center[0]
|
||||
y0 = y - center[1]
|
||||
angle = angle * pi / 180.0
|
||||
return (cos(angle) * x0 - sin(angle) * y0 + center[0],
|
||||
sin(angle) * x0 + cos(angle) * y0 + center[1])
|
||||
2792
test/data/fill.dxf
Normal file
2792
test/data/fill.dxf
Normal file
File diff suppressed because it is too large
Load diff
51
test/data/merge.py
Executable file
51
test/data/merge.py
Executable file
|
|
@ -0,0 +1,51 @@
|
|||
#!/usr/bin/env python
|
||||
import os
|
||||
import gerberex
|
||||
from gerberex.dxf import DxfFile
|
||||
|
||||
exts = ['GTL', 'GTO', 'GTP', 'GTS', 'GBL', 'GBO', 'GBP', 'GBS', 'TXT']
|
||||
boards=[
|
||||
('../../sonopi-digi/pcb/CAMOutputs/sonopi-digi.', 0, 0),
|
||||
('../../sonopi-digi/pcb/CAMOutputs/sonopi-digi.', 0, 22.5),
|
||||
('../../rcstick-f/pcb/small/CAMOutputs/rcstick-f-small.', 0, 60),
|
||||
('../../rcstick-f/pcb/small/CAMOutputs/rcstick-f-small.', 20, 60),
|
||||
('../../rcstick-f/pcb/small/CAMOutputs/rcstick-f-small.', 40, 60),
|
||||
('../../rcstick-f/pcb/large/CAMOutputs/rcstick-f.', 72.4, 0),
|
||||
('../../rcstick-f/pcb/jig/CAMOutputs/rcstick-jig.', 0, 44),
|
||||
('../../stm32breakout/pcb/CAMOutputs/stm32breakout.', 78.0, 59.36),
|
||||
('../../stm32breakout/pcb/CAMOutputs/stm32breakout.', 100.0, 59.36),
|
||||
]
|
||||
outline = 'outline.dxf'
|
||||
fill = 'fill.dxf'
|
||||
|
||||
outputs = 'outputs/elecrow-panelized'
|
||||
|
||||
os.chdir(os.path.dirname(__file__))
|
||||
|
||||
for ext in exts:
|
||||
print('merging %s: ' % ext ,end='', flush=True)
|
||||
if ext == 'TXT':
|
||||
ctx = gerberex.DrillComposition()
|
||||
else:
|
||||
ctx = gerberex.GerberComposition()
|
||||
for board in boards:
|
||||
file = gerberex.read(board[0] + ext)
|
||||
file.to_metric()
|
||||
file.offset(board[1], board[2])
|
||||
ctx.merge(file)
|
||||
print('.', end='', flush=True)
|
||||
if ext != 'TXT':
|
||||
file = gerberex.read(outline)
|
||||
ctx.merge(file)
|
||||
ctx.dump(outputs + '.' + ext)
|
||||
print(' end', flush=True)
|
||||
|
||||
print('generating GML: ', end='', flush=True)
|
||||
file = gerberex.read(outline)
|
||||
file.write(outputs + '.GML')
|
||||
print('.', end='', flush=True)
|
||||
file = gerberex.read(fill)
|
||||
file.to_metric()
|
||||
file.draw_mode = DxfFile.DM_FILL
|
||||
file.write(outputs + '-fill.GML')
|
||||
print('. end', flush=True)
|
||||
2602
test/data/outline.dxf
Normal file
2602
test/data/outline.dxf
Normal file
File diff suppressed because it is too large
Load diff
6518
test/data/test.GTL
Normal file
6518
test/data/test.GTL
Normal file
File diff suppressed because it is too large
Load diff
45
test/data/test.TXT
Normal file
45
test/data/test.TXT
Normal file
|
|
@ -0,0 +1,45 @@
|
|||
M48
|
||||
;GenerationSoftware,Autodesk,EAGLE,9.3.0*%
|
||||
;CreationDate,2019-03-17T14:26:03Z*%
|
||||
FMAT,2
|
||||
ICI,OFF
|
||||
METRIC,TZ,000.000
|
||||
T5C0.350
|
||||
T4C0.508
|
||||
T3C0.800
|
||||
T2C0.930
|
||||
T1C1.200
|
||||
%
|
||||
G90
|
||||
M71
|
||||
T1
|
||||
X5450Y2685
|
||||
X9950Y2685
|
||||
T2
|
||||
X11200Y4785
|
||||
X8700Y4785
|
||||
X6700Y4785
|
||||
X4200Y4785
|
||||
T3
|
||||
X5588Y25527
|
||||
T4
|
||||
X2062Y6253
|
||||
X2062Y7503
|
||||
X2062Y8753
|
||||
X2062Y10003
|
||||
T5
|
||||
X4191Y15113
|
||||
X11557Y15494
|
||||
X7747Y17018
|
||||
X7747Y9017
|
||||
X4064Y27178
|
||||
X7112Y27178
|
||||
X8890Y26035
|
||||
X1651Y24892
|
||||
X6096Y22479
|
||||
X12700Y18796
|
||||
X13335Y18161
|
||||
X7747Y22225
|
||||
X7112Y23907
|
||||
X14450Y34250
|
||||
M30
|
||||
0
test/panelimage.py
Normal file → Executable file
0
test/panelimage.py
Normal file → Executable file
12
test/test.py
Normal file → Executable file
12
test/test.py
Normal file → Executable file
|
|
@ -1,3 +1,4 @@
|
|||
import os
|
||||
import gerberex
|
||||
from gerberex.dxf import DxfFile
|
||||
import gerber
|
||||
|
|
@ -44,9 +45,18 @@ def merge2():
|
|||
ctx.dump('test-merged.TXT')
|
||||
|
||||
|
||||
os.chdir(os.path.dirname(__file__))
|
||||
|
||||
#merge2()
|
||||
|
||||
file = gerberex.read('outline.dxf')
|
||||
file = gerberex.read('data/test.GTL')
|
||||
file.rotate(45)
|
||||
file.write('outputs/test_changed.GTL')
|
||||
file = gerberex.read('data/test.TXT')
|
||||
file.rotate(45)
|
||||
file.write('outputs/test_changed.TXT')
|
||||
|
||||
file = gerberex.read('data/outline.dxf')
|
||||
file.to_metric()
|
||||
w = file.width
|
||||
file.draw_mode = DxfFile.DM_FILL
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue