boxespy/boxes/generators/cardbox.py

255 lines
9.9 KiB
Python

#!/usr/bin/env python3
# Copyright (C) 2013-2014 Florian Festi
# Copyright (C) 2018 jens persson <jens@persson.cx>
# Copyright (C) 2023 Manuel Lohoff
#
# 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 3 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, see <http://www.gnu.org/licenses/>.
from boxes import edges, Boxes, BoolArg
class InsetEdgeSettings(edges.Settings):
"""Settings for InsetEdge"""
absolute_params = {
"thickness": 0,
}
class InsetEdge(edges.BaseEdge):
"""An edge with space to slide in a lid"""
def __call__(self, length, **kw):
t = self.settings.thickness
self.corner(90)
self.edge(t, tabs=2)
self.corner(-90)
self.edge(length, tabs=2)
self.corner(-90)
self.edge(t, tabs=2)
self.corner(90)
class FingerHoleEdgeSettings(edges.Settings):
"""Settings for FingerHoleEdge"""
absolute_params = {
"wallheight": 0,
"fingerholedepth": 0,
}
class FingerHoleEdge(edges.BaseEdge):
"""An edge with room to get your fingers around cards"""
def __call__(self, length, **kw):
depth = self.settings.fingerholedepth-10
self.edge(length/2-10, tabs=2)
self.corner(90)
self.edge(depth, tabs=2)
self.corner(-180, 10)
self.edge(depth, tabs=2)
self.corner(90)
self.edge(length/2-10, tabs=2)
class CardBox(Boxes):
"""Box for storage of playing cards, with versatile options"""
ui_group = "Box"
description = """
### Description
Versatile Box for Storage of playing cards. Multiple different styles of storage are supportet, e.g. a flat storage or a trading card deck box style storage. See images for ideas.
#### Building instructions
Place inner walls on floor first (if any). Then add the outer walls. Glue the two walls without finger joins to the inside of the side walls. Make sure there is no squeeze out on top, as this is going to form the rail for the lid.
Add the top of the rails to the sides (front open) or to the back and front (right side open) and the grip rail to the lid.
Details of the lid and rails
![Details](static/samples/CardBox-detail.jpg)
Whole box (early version still missing grip rail on the lid):
"""
def __init__(self) -> None:
Boxes.__init__(self)
self.addSettingsArgs(edges.FingerJointSettings)
self.buildArgParser(y=68, h=92, outside=False, sx="65*4")
self.argparser.add_argument(
"--openingdirection", action="store", type=str, default="front",
choices=['front', 'right'],
help="Direction in which the lid slides open. Lid length > Lid width recommended.")
self.argparser.add_argument(
"--fingerhole", action="store", type=str, default="regular",
choices=['regular', 'deep', 'custom'],
help="Depth of cutout to grab the cards")
self.argparser.add_argument(
"--fingerhole_depth", action="store", type=float, default=20,
help="Depth of cutout if fingerhole is set to 'custom'. Disabled otherwise.")
self.argparser.add_argument(
"--add_lidtopper", action="store", type=BoolArg(), default=False,
help="Add an additional lid topper for optical reasons and customisation"
)
@property
def fingerholedepth(self):
if self.fingerhole == 'custom':
return self.fingerhole_depth
elif self.fingerhole == 'regular':
a = self.h/4
if a < 35:
return a
else:
return 35
elif self.fingerhole == 'deep':
return self.h-self.thickness-10
#inner dimensions of surrounding box (disregarding inlays)
@property
def boxhight(self):
if self.outside:
return self.h - 3 * self.thickness
else:
return self.h
@property
def boxwidth(self):
return (len(self.sx) + 1) * self.thickness + sum(self.sx)
@property
def boxdepth(self):
if not self.outside:
if self.openingdirection == 'right':
return self.y + 2 * self.thickness
else:
return self.y
else:
return self.y - 2 * self.thickness
def divider_bottom(self):
t = self.thickness
sx = self.sx
y = self.boxdepth
pos = 0.5 * t
for i in sx[:-1]:
pos += i + t
self.fingerHolesAt(pos, 0, y, 90)
def divider_back_and_front(self):
t = self.thickness
sx = self.sx
y = self.boxhight
pos = 0.5 * t
for i in sx[:-1]:
pos += i + t
self.fingerHolesAt(pos, 0, y, 90)
def render(self):
t = self.thickness
h = self.boxhight
x = self.boxwidth
y = self.boxdepth
sx = self.sx
s = InsetEdgeSettings(thickness=t)
p = InsetEdge(self, s)
p.char = "a"
self.addPart(p)
s = FingerHoleEdgeSettings(thickness=t, wallheight=h, fingerholedepth=self.fingerholedepth)
p = FingerHoleEdge(self, s)
p.char = "A"
self.addPart(p)
if self.openingdirection == 'right':
with self.saved_context():
self.rectangularWall(x, y-t*.2, "eFee", move="right", label="Lid")
self.rectangularWall(x, y, "ffff", callback=[self.divider_bottom],
move="right", label="Bottom")
self.rectangularWall(x, y, "eEEE", move="up only")
self.rectangularWall(x, t, "feee", move="up", label="Lip Front")
self.rectangularWall(x, t, "eefe", move="up", label="Lip Back")
with self.saved_context():
self.rectangularWall(x, h+t, "FfFf",
callback=[self.divider_back_and_front],
move="right",
label="Back")
self.rectangularWall(x, h+t, "FfFf",
callback=[self.divider_back_and_front],
move="right",
label="Front")
self.rectangularWall(x, h+t, "EEEE", move="up only")
with self.saved_context():
self.rectangularWall(y, h+t, "FFEF", move="right", label="Outer Side Left")
self.rectangularWall(y, h+t, "FFaF", move="right", label="Outer Side Right")
self.rectangularWall(y, h+t, "fFfF", move="up only")
with self.saved_context():
self.rectangularWall(y, h, "Aeee", move="right", label="Inner Side Left")
self.rectangularWall(y, h, "Aeee", move="right", label="Inner Side Right")
self.rectangularWall(y, h, "eAee", move="up only")
with self.saved_context():
self.rectangularWall(y-t*.2, t, "fEeE", move="right", label="Lid Lip")
self.rectangularWall(y, t*2, "efee", move="up only")
for i in range(len(sx) - 1):
self.rectangularWall(h, y, "fAff", move="right", label="Divider")
for c in sx:
self.rectangularWall(c, h, "eeee", move="right", label="Front inlay")
self.rectangularWall(c, h, "eeee", move="right", label="Back inlay")
if self.add_lidtopper:
self.rectangularWall(x, y - 2*t, "eeee", move="right", label="Lid topper")
elif self.openingdirection == 'front':
with self.saved_context():
self.rectangularWall(x - t * .2, y, "eeFe", move="right", label="Lid")
self.rectangularWall(x, y, "ffff", callback=[self.divider_bottom],
move="right", label="Bottom")
self.rectangularWall(x, y, "eEEE", move="up only")
self.rectangularWall(x - t * .2, t, "fEeE", move="up", label="Lid Lip")
with self.saved_context():
self.rectangularWall(x, h + t, "FFEF",
callback=[self.divider_back_and_front],
move="right",
label="Back")
self.rectangularWall(x, h + t, "FFaF",
callback=[self.divider_back_and_front],
move="right",
label="Front")
self.rectangularWall(x, h + t, "EEEE", move="up only")
with self.saved_context():
self.rectangularWall(y, h + t, "FfFf", move="right", label="Outer Side Left")
self.rectangularWall(y, h + t, "FfFf", move="right", label="Outer Side Right")
self.rectangularWall(y, h + t, "fFfF", move="up only")
with self.saved_context():
self.rectangularWall(y, h, "Aeee", move="right", label="Inner Side Left")
self.rectangularWall(y, h, "Aeee", move="right", label="Inner Side Right")
self.rectangularWall(y, h, "eAee", move="up only")
with self.saved_context():
self.rectangularWall(y, t, "eefe", move="right", label="Lip Left")
self.rectangularWall(y, t, "feee", move="right", label="Lip Right")
self.rectangularWall(y, t * 2, "efee", move="up only")
for i in range(len(sx) - 1):
self.rectangularWall(h, y, "fAff", move="right", label="Divider")
if self.add_lidtopper:
self.rectangularWall(x, y - 2 * t, "eeee", move="right", label="Lid topper (optional)")