573 lines
16 KiB
Python
Raw Normal View History

#!/usr/bin/env python3
# Copyright 2016 Luke Phillips (lukerazor@hotmail.com)
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
#along with this program; if not, write to the Free Software
#Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
# Extension dirs
# linux:~/.config/inkscape/extensions
# windows: [Drive]:\Program Files\Inkscape\share\extensions
from lxml import etree
import inkex
import copy
CUTOUT_TOP = 1
CUTOUT_BOTTOM = 2
CUTOUT_LEFT = 4
CUTOUT_RIGHT = 8
inkex.NSS[u'cs'] = u'http://www.razorfoss.org/tuckboxextension/'
class EffectDimensionProvider():
def __init__(self, effect, x = 0, y = 0):
self.Effect = effect
self.Layer = effect.svg.get_current_layer()
self.Width = effect.options.DeckWidth
self.Height = effect.options.DeckHeight
self.Depth = effect.options.DeckDepth
self.Allowance = effect.options.DeckAllowance
self.X = x
self.Y = y
def MMtoUU(self, mmval):
if hasattr(self.Effect.svg, "unittouu"):
return str(self.Effect.svg.unittouu("{0}mm".format(mmval)))
else:
MM_TO_PIXELS = 3.5433071
return str(MM_TO_PIXELS * mmval)
def MaximiseHeight(self):
if self.Height < self.Width: # always choose the smallest to be the "width"
temp = self.Width
self.Width = self.Height
self.Height = temp
class BoxBase():
def __init__(self, dimensionProvider):
#create a group
self.DimProvider = dimensionProvider
self.Group = etree.SubElement(dimensionProvider.Layer, inkex.addNS('g','svg'), {} )
self.Width = dimensionProvider.Width + dimensionProvider.Allowance
self.Height = dimensionProvider.Height + dimensionProvider.Allowance
self.Depth = dimensionProvider.Depth + dimensionProvider.Allowance
self.X = dimensionProvider.X
self.Y = dimensionProvider.Y
self.MinY = 0
self.MinX = 0
### init some common sizes ###
self.ThumbSize = 20
# tuck flap size
self.FlapOffset = 1.5
self.FlapHeight = self.Depth
if self.Depth < 7 or self.Depth > 25:
self.FlapHeight = 20
# main flap size
self.MainFlapHeight = (4 * self.Depth)/3
if self.MainFlapHeight < self.ThumbSize:
self.MainFlapHeight = 24
### colour ###
self.Fill = '#ffffff'
self.StrokeWidth = self.DimProvider.MMtoUU(0.5)
def _CreateRectangleInMillimetres(self, height, width, x, y):
style = {'stroke': '#000000', 'stroke-width': self.StrokeWidth, 'fill' : self.Fill}
attribs = {'style': str(inkex.Style(style)), 'height': self.DimProvider.MMtoUU(height), 'width': self.DimProvider.MMtoUU(width), 'x': self.DimProvider.MMtoUU(x), 'y': self.DimProvider.MMtoUU(y)}
etree.SubElement(self.Group, inkex.addNS('rect','svg'), attribs )
def _CreateRectangleInMillimetresWithCutouts(self, height, width, x, y, cutoutPositions):
cmds = []
t = self.ThumbSize
cr = (3*t)/4 # curve ratio
# start position
cmds.append(["m", x, y])
# Top side
if cutoutPositions & CUTOUT_TOP == CUTOUT_TOP:
cmds.append(["h", (self.Width - t)/2])
cmds.append(["c", 0, cr,
t, cr,
t, 0])
cmds.append(["h", (self.Width - t)/2])
else:
cmds.append(["h", width])
# Right Side
if cutoutPositions & CUTOUT_RIGHT == CUTOUT_RIGHT:
cmds.append(["v", (self.Height - t)/2])
cmds.append(["c", -cr, 0,
-cr, t,
0, t])
cmds.append(["v", (self.Height - t)/2])
else:
cmds.append(["v", height])
# Bottom Side
if cutoutPositions & CUTOUT_BOTTOM == CUTOUT_BOTTOM:
cmds.append(["h", -(self.Width - t)/2])
cmds.append(["c", 0, -cr,
-t, -cr,
-t, 0])
cmds.append(["h", -(self.Width - t)/2])
else:
cmds.append(["h", -width])
# Left Side
if cutoutPositions & CUTOUT_LEFT == CUTOUT_LEFT:
cmds.append(["v", -(self.Height - t)/2])
cmds.append(["c", cr, 0,
cr, -t,
0, -t])
cmds.append(["v", -(self.Height - t)/2])
else:
cmds.append(["v", -height])
self._CreatePathinMillimetres(cmds)
def _CreatePathinMillimetres(self, cmds):
pathStr = ""
for cmd in cmds:
pathStr += cmd[0] + " "
for coord in cmd[1:]:
pathStr += self.DimProvider.MMtoUU(coord) + " "
pathStr += "z"
#raise Exception(pathStr)
style = {'stroke': '#000000', 'stroke-width': self.StrokeWidth, 'fill' : self.Fill}
attribs = {'style': str(inkex.Style(style)), 'd': pathStr}
etree.SubElement(self.Group, inkex.addNS('path','svg'), attribs )
class SingleFlappedTuckBox(BoxBase):
def __init__(self, dimensionProvider):
BoxBase.__init__(self, dimensionProvider)
def Create(self):
self.FlapOffset = 1.5
self.FlapHeight = min(20, self.Depth)
# Figure out some row and column values,
# note rows and cols work left to right, top to bottom, but both calculated in reverse
col5 = self.X - self.Depth
col4 = col5 - self.Width
col3 = col4 - self.Depth
col2 = col3 - self.Width
col1 = col2 - self.Depth
self.MinX = col1
row4 = self.Y - self.Depth
row3 = row4 - self.Height
row2 = row3 - self.Depth
row1 = row2 - self.Depth
self.MinY = row1
### COLUMN 1 ###
#create left glue panel
self._CreateRectangleInMillimetres(self.Height, self.Depth, col1, row3)
### COLUMN 2 ###
#create box back print panel
self._CreateRectangleInMillimetresWithCutouts(self.Height, self.Width, col2, row3, CUTOUT_TOP)
#create box bottom glue panel
self._CreateRectangleInMillimetres(self.Depth, self.Width, col2, row4)
### COLUMN 3 ###
#create left flap
self._CreatePathinMillimetres(
[
["m", col3, row3],
["h", self.Depth],
["l", -self.FlapOffset, -self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
#create left print panel
self._CreateRectangleInMillimetres(self.Height, self.Depth, col3, row3)
#create bottom left glue panel
self._CreateRectangleInMillimetres(self.Depth, self.Depth, col3, row4)
### COLUMN 4 ###
#create main flap
self._CreatePathinMillimetres(
[
["m", col4, row2],
["c", 0, -self.MainFlapHeight, self.Width, -self.MainFlapHeight, self.Width, 0]
])
#create box top print panel
self._CreateRectangleInMillimetres(self.Depth, self.Width, col4, row2)
#create box front print panel
self._CreateRectangleInMillimetres(self.Height, self.Width, col4, row3)
#create box bottom print panel
self._CreateRectangleInMillimetres(self.Depth, self.Width, col4, row4)
### COLUMN 5 ###
#create right flap
self._CreatePathinMillimetres(
[
["m", col5, row3],
["h", self.Depth],
["l", -self.FlapOffset, -self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
#create right print panel
self._CreateRectangleInMillimetres(self.Height, self.Depth, col5, row3)
#create bottom right glue panel
self._CreateRectangleInMillimetres(self.Depth, self.Depth, col5, row4)
class DoubleFlappedTuckBox(BoxBase):
def __init__(self, dimensionProvider):
BoxBase.__init__(self, dimensionProvider)
def Create(self):
# Figure out some row and column values,
# note rows and cols work left to right, top to bottom
col5 = self.X - self.Depth
col4 = col5 - self.Width
col3 = col4 - self.Depth
col2 = col3 - self.Width
col1 = col2 - self.Depth
self.MinX = col1
row5 = self.Y - self.Depth
row4 = row5 - self.Depth
row3 = row4 - self.Height
row2 = row3 - self.Depth
row1 = row2 - self.Depth
self.MinY = row1
### COLUMN 1 ###
#create left glue panel
self._CreateRectangleInMillimetres(self.Height, self.Depth, col1, row3)
### COLUMN 2 ###
#create box back print panel
self._CreateRectangleInMillimetresWithCutouts(self.Height, self.Width, col2, row3, CUTOUT_TOP | CUTOUT_BOTTOM)
### COLUMN 3 ###
#create top left flap
self._CreatePathinMillimetres(
[
["m", col3, row3],
["h", self.Depth],
["l", -self.FlapOffset, -self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
#create left print panel
self._CreateRectangleInMillimetres(self.Height, self.Depth, col3, row3)
#create bottom left flap
self._CreatePathinMillimetres(
[
["m", col3, row4],
["h", self.Depth],
["l", -self.FlapOffset, self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
### COLUMN 4 ###
#create top main flap
self._CreatePathinMillimetres(
[
["m", col4, row2],
["c", 0, -self.MainFlapHeight, self.Width, -self.MainFlapHeight, self.Width, 0]
])
#create box top print panel
self._CreateRectangleInMillimetres(self.Depth, self.Width, col4, row2)
#create box front print panel
self._CreateRectangleInMillimetres(self.Height, self.Width, col4, row3)
#create box bottom print panel
self._CreateRectangleInMillimetres(self.Depth, self.Width, col4, row4)
#create bottom main flap
self._CreatePathinMillimetres(
[
["m", col4, row5],
["c", 0, self.MainFlapHeight, self.Width, self.MainFlapHeight, self.Width, 0]
])
### COLUMN 5 ###
#create top right flap
self._CreatePathinMillimetres(
[
["m", col5, row3],
["h", self.Depth],
["l", -self.FlapOffset, -self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
#create right print panel
self._CreateRectangleInMillimetres(self.Height, self.Depth, col5, row3)
#create bottom right flap
self._CreatePathinMillimetres(
[
["m", col5, row4],
["h", self.Depth],
["l", -self.FlapOffset, self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
class SlipcaseTuckBox(BoxBase):
def __init__(self, dimensionProvider):
BoxBase.__init__(self, dimensionProvider)
def Create(self):
self.FlapOffset = 1.5
# Figure out some row and column values,
# note rows and cols work left to right, top to bottom
col5 = self.X - self.Depth
col4 = col5 - self.Width
col3 = col4 - self.Depth
col2 = col3 - self.Width
col1 = col2 - self.Depth
row1 = self.Y - self.Height
self.MinY = row1
### COLUMN 1 ###
#create left glue flap
self._CreatePathinMillimetres(
[
["m", col2, row1],
["v", self.Height],
["l", -(self.Depth - self.FlapOffset), -self.FlapOffset],
["v", -(self.Height - (2*self.FlapOffset))],
])
### COLUMN 2 ###
#create box back print panel
self._CreateRectangleInMillimetres(self.Height, self.Width, col2, row1)
### COLUMN 3 ###
#create left print panel
self._CreateRectangleInMillimetres(self.Height, self.Depth, col3, row1)
### COLUMN 4 ###
#create box front print panel
self._CreateRectangleInMillimetres(self.Height, self.Width, col4, row1)
### COLUMN 5 ###
#create right print panel
self._CreateRectangleInMillimetres(self.Height, self.Depth, col5, row1)
class Matchbox(BoxBase):
def __init__(self, dimensionProvider, numFlaps):
BoxBase.__init__(self, dimensionProvider)
self.DimProvider.MaximiseHeight()
self.NumFlaps = numFlaps
def Create(self):
if self.NumFlaps == 2:
tuckbox = DoubleFlappedTuckBox(self.DimProvider)
else:
tuckbox = SingleFlappedTuckBox(self.DimProvider)
tuckbox.Create()
#################################
# Create Drawer for inside the box
dimProvider = copy.copy(self.DimProvider)
dimProvider.Width -= 2
dimProvider.Height -= 2
dimProvider.Depth -= 2
dimProvider.Y = tuckbox.MinY - 20
drawer = MatcboxDrawer(dimProvider)
drawer.Create()
class TelescopingBox(BoxBase):
def __init__(self, dimensionProvider):
BoxBase.__init__(self, dimensionProvider)
self.DimProvider.MaximiseHeight()
def Create(self):
#################################
# Create box top
top = MatcboxDrawer(self.DimProvider, includeFingerCutouts=True)
top.Create()
#################################
# Create box bottom
dimProvider = copy.copy(self.DimProvider)
dimProvider.Width -= 1
dimProvider.Height -= 1
dimProvider.Depth -= 1
dimProvider.Y = top.MinY - 20
drawer = MatcboxDrawer(dimProvider)
drawer.Create()
class MatcboxDrawer(BoxBase):
def __init__(self, dimensionProvider, includeFingerCutouts=False):
BoxBase.__init__(self, dimensionProvider)
self.IncludeFingerCutouts = includeFingerCutouts
def Create(self):
fudgeDepth = self.Depth - 2 # overlap panels should be a little smaller to avoid touching box base
# Figure out some row and column co-ord values,
# note rows and cols work left to right, top to bottom, values start at 0 and go negative
col5 = self.X - fudgeDepth
col4 = col5 - self.Depth
col3 = col4 - self.Width
col2 = col3 - self.Depth
col1 = col2 - fudgeDepth
row5 = self.Y - fudgeDepth
row4 = row5 - self.Depth
row3 = row4 - self.Height
row2 = row3 - self.Depth
row1 = row2 - fudgeDepth
self.MinY = row1
### COLUMN 1 ###
#create left overlap panel
if self.IncludeFingerCutouts:
self._CreateRectangleInMillimetresWithCutouts(self.Height, fudgeDepth, col1, row3, CUTOUT_RIGHT)
else:
self._CreateRectangleInMillimetres(self.Height, fudgeDepth, col1, row3)
### COLUMN 2 ###
#create top left flap
self._CreatePathinMillimetres(
[
["m", col2, row3],
["h", self.Depth],
["l", -self.FlapOffset, -self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
#create box left side print panel
if self.IncludeFingerCutouts:
self._CreateRectangleInMillimetresWithCutouts(self.Height, self.Depth, col2, row3, CUTOUT_LEFT)
else:
self._CreateRectangleInMillimetres(self.Height, self.Depth, col2, row3)
#create bottom left flap
self._CreatePathinMillimetres(
[
["m", col2, row4],
["h", self.Depth],
["l", -self.FlapOffset, self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
### COLUMN 3 ###
#create top side overlap panel
self._CreateRectangleInMillimetres(fudgeDepth, self.Width, col3, row1)
#create top box side panel
self._CreateRectangleInMillimetres(self.Depth, self.Width, col3, row2)
#create box bottom
self._CreateRectangleInMillimetres(self.Height, self.Width, col3, row3)
#create bottom box side panel
self._CreateRectangleInMillimetres(self.Depth, self.Width, col3, row4)
#create bottom side overlap panel
self._CreateRectangleInMillimetres(fudgeDepth, self.Width, col3, row5)
### COLUMN 4 ###
#create top right flap
self._CreatePathinMillimetres(
[
["m", col4, row3],
["h", self.Depth],
["l", -self.FlapOffset, -self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
#create box right side print panel
if self.IncludeFingerCutouts:
self._CreateRectangleInMillimetresWithCutouts(self.Height, self.Depth, col4, row3, CUTOUT_RIGHT)
else:
self._CreateRectangleInMillimetres(self.Height, self.Depth, col4, row3)
#create bottom right flap
self._CreatePathinMillimetres(
[
["m", col4, row4],
["h", self.Depth],
["l", -self.FlapOffset, self.FlapHeight],
["h", -(self.Depth - (2*self.FlapOffset))],
])
### COLUMN 5 ###
#create right overlap panel
if self.IncludeFingerCutouts:
self._CreateRectangleInMillimetresWithCutouts(self.Height, fudgeDepth, col5, row3, CUTOUT_LEFT)
else:
self._CreateRectangleInMillimetres(self.Height, fudgeDepth, col5, row3)
class Tuckbox(inkex.EffectExtension):
def __init__(self):
inkex.Effect.__init__(self)
self.arg_parser.add_argument('-t', '--type', type = str, dest = 'BoxType')
self.arg_parser.add_argument('-n', '--num_flaps', type = int, dest = 'NumFlaps')
self.arg_parser.add_argument('-w', '--deck_width', type = float, dest = 'DeckWidth')
self.arg_parser.add_argument('-r', '--deck_height', type = float, dest = 'DeckHeight')
self.arg_parser.add_argument('-d', '--deck_depth', type = float, dest = 'DeckDepth')
self.arg_parser.add_argument('-a', '--box_allowance', type = float, dest = 'DeckAllowance')
def GetPaths(self):
paths = []
def effect(self):
dimProvider = EffectDimensionProvider(self)
if self.options.BoxType == "TUCKBOX":
if self.options.NumFlaps == 2:
box = DoubleFlappedTuckBox(dimProvider)
else:
box = SingleFlappedTuckBox(dimProvider)
elif self.options.BoxType == "SLIPCASE":
box = SlipcaseTuckBox(dimProvider)
elif self.options.BoxType == "MATCHBOX":
box = Matchbox(dimProvider, self.options.NumFlaps)
elif self.options.BoxType == "TELESCOPE":
box = TelescopingBox(dimProvider)
elif self.options.BoxType == "DISH":
box = MatcboxDrawer(dimProvider)
else:
raise Exception("Box type '{0}' is undefined".format(self.options.BoxType))
box.Create()
if __name__ == '__main__':
Tuckbox().run()