{f"{hole_dia:.2f}" if hole_dia is not None else ""}
'
f'
{area_count(patterns)}
'
f'
{"Yes" if symmetric else "No"}
'
f'
{"Yes" if has_ground_plane(patterns) else "No"}
'
f'
{"Yes" if has_manhattan_area(patterns) else "No"}
'
f'
{"Yes" if has_square_pads(patterns) else "No"}
'
f'
{"Yes" if has_large_holes(patterns) else "No"}
'
f'
{format_pitches(tht_pitches(patterns))}
'
f'
{format_pitches(smd_pitches(patterns))}
'
'
')
for path, (w, h, hole_dia, symmetric, patterns) in index.items()
]
table_content = '\n'.join(table_rows)
length_sort = lambda length: float(length.partition(' ')[0])
filter_cols = {
'Width': sorted(set(w for w, h, *rest in index.values())),
'Height': sorted(set(h for w, h, *rest in index.values())),
'Mounting Hole Diameter': sorted(set(dia for w, h, dia, *rest in index.values() if dia)) + ['None'],
'Number of Areas': sorted(set(area_count(patterns) for *_rest, patterns in index.values())),
'Symmetric Top and Bottom?': ['Yes', 'No'],
'Ground Plane?': ['Yes', 'No'],
'Manhattan Area?': ['Yes', 'No'],
'Square Pads?': ['Yes', 'No'],
'Large Holes?': ['Yes', 'No'],
'THT Pitches': sorted(set(p for *_rest, patterns in index.values() for p in tht_pitches(patterns))) + ['None'],
'SMD Pitches': sorted(set(p for *_rest, patterns in index.values() for p in smd_pitches(patterns))) + ['None'],
}
filter_headers = '\n'.join(f'
{key}
' for key in filter_cols)
key_id = lambda key: key.lower().replace("?", "").replace(" ", "_")
val_id = lambda value: str(value).replace(".", "_")
def format_value(value):
if isinstance(value, str):
return value
elif isinstance(value, int):
return str(value)
elif isinstance(value, bool):
return value and 'Yes' or 'No'
else:
return format_length(value)
filter_cols = {
key: '\n'.join(f'' for value in values)
for key, values in filter_cols.items() }
filter_cols = [f'
{values}
' for key, values in filter_cols.items()]
filter_content = '\n'.join(filter_cols)
filter_js = textwrap.dedent('''
function get_filters(){
let filters = {};
table = document.querySelector('#filter');
for (let filter of table.querySelectorAll('td')) {
selected = [];
for (let checkbox of filter.querySelectorAll('input')) {
if (checkbox.checked) {
selected.push(checkbox.nextElementSibling.textContent.replace(/ mm$/, ''));
}
}
filters[filter.id.replace(/^filter-/, '')] = selected;
}
return filters;
}
filter_indices = {
};
for (const [i, header] of document.querySelectorAll("#listing th").entries()) {
if (header.hasAttribute('data-filter-key')) {
filter_indices[header.attributes['data-filter-key'].value] = i;
}
}
function filter_row(filters, row) {
cols = row.querySelectorAll('td');
for (const [filter_id, values] of Object.entries(filters)) {
if (values.length == 0) {
continue;
}
const row_value = cols[filter_indices[filter_id]].textContent;
if (values.includes("None") && !row_value) {
continue;
}
if (values.includes(row_value)) {
continue;
}
return false;
}
return true;
}
let timeout = undefined;
function apply_filters() {
if (timeout) {
clearTimeout(timeout);
timeout = undefined;
}
const filters = get_filters();
for (let row of document.querySelectorAll("#listing tbody tr")) {
if (filter_row(filters, row)) {
row.style.display = '';
} else {
row.style.display = 'none';
}
}
}
function refresh_filters() {
if (timeout) {
clearTimeout(timeout);
}
timeout = setTimeout(apply_filters, 2000);
}
function reset_filters() {
for (let checkbox of document.querySelectorAll("#filter input")) {
checkbox.checked = false;
}
refresh_filters();
}
document.querySelector("#apply").onclick = apply_filters;
document.querySelector("#reset-filters").onclick = reset_filters;
for (let checkbox of document.querySelectorAll("#filter input")) {
checkbox.onchange = refresh_filters;
}
apply_filters();
'''.strip())
style = textwrap.dedent('''
:root {
--gray1: #d0d0d0;
--gray2: #eeeeee;
font-family: sans-serif;
}
table {
border-collapse: collapse;
box-shadow: 0 0 3px gray;
width: 100%;
}
td {
border: 1px solid var(--gray1);
padding: .1em .5em;
}
th {
border: 1px solid var(--gray1);
padding: .5em;
background: linear-gradient(0deg, #e0e0e0, #eeeeee);
}
#listing tr:hover {
background-color: #ffff80;
}
#listing tr td {
text-align: center;
}
#listing tr td:nth-child(4), #listing tr td:nth-child(5) {
text-align: right;
}
#filter {
margin-top: 2em;
}
button {
margin: 2em 0.2em;
padding: .5em 1em;
}
body {
max-width: 80em;
margin: 3em auto;
}
body > div {
width: 100%;
}
'''.strip())
html = textwrap.dedent(f'''
Gerbolyze Protoboard Index
Gerbolyze Protoboard Index
This page contains gerbers for many different types of prototype circuit boards. Everything from different pitches
of THT hole patterns to SMD pad patterns is included in many different sizes and with several mounting hole options.
All downloads on this page are licensed under the Unlicense. This means you can
download what you like and do with it whatever you want. Just note that everything here is provided without any
warranty, so if you send files you find here to a pcb board house and what you get back from them is all wrong,
that's your problem.
All files on this page have been generated automatically from a number of templates using
gerbolyze
(github mirror). If you have any suggestions for additional layouts
or layout options, please feel free to file an issue on
Gerbolyze's issue tracker on github.
{filter_headers}
{filter_content}
Download
Preview
Source SVG
Width [mm]
Height [mm]
Has Mounting Holes?
Mounting Hole Diameter [mm]
Number of Areas
Symmetric Top and Bottom?
Ground Plane?
Manhattan Area?
Square Pads?
Large Holes?
THT Pitches [mm]
SMD Pitches [mm]
{table_content}
'''.strip())
(outdir / 'index.html').write_text(html)
def generate(outdir, fun, sizes=sizes_large, name=None, generate_svg=True):
name = name or fun.__name__
outdir = outdir / f'{name}'
plain_dir = outdir / 'no_mounting_holes'
plain_dir.mkdir(parents=True, exist_ok=True)
for w, h in sizes:
outfile = plain_dir / f'{name}_{w}x{h}.svg'
board = fun((w, h))
yield outfile, (float(w), float(h), None, board.symmetric_sides, board.used_patterns)
if generate_svg:
outfile.write_text(board.generate(w, h))
for dia in (2, 2.5, 3, 4):
hole_dir = outdir / f'mounting_holes_M{dia:.1f}'
hole_dir.mkdir(exist_ok=True)
for w, h in sizes:
if w < 25 or h < 25:
continue
outfile = hole_dir / f'{name}_{w}x{h}_holes_M{dia:.1f}.svg'
try:
# Add 0.2 mm tolerance to mounting holes for easier insertion of screw
board = fun((w, h), (dia+0.2, dia+2))
yield outfile, (float(w), float(h), float(dia), board.symmetric_sides, board.used_patterns)
if generate_svg:
outfile.write_text(board.generate(w, h))
except ValueError: # mounting hole keepout too large for small board, ignore.
pass
@click.command()
@click.argument('outdir', type=click.Path(file_okay=False, dir_okay=True, path_type=pathlib.Path))
@click.option('--generate-svg/--no-generate-svg')
def generate_all(outdir, generate_svg):
index_d = {}
def index(sizes=sizes_large, name=None):
def deco(fun):
nonlocal index_d
index_d.update(generate(outdir / 'svg', fun, sizes=sizes, name=name, generate_svg=generate_svg))
return fun
return deco
@index()
def tht_normal_pitch100mil(size, mounting_holes=None):
return ProtoBoard(common_defs, 'tht', mounting_holes, border=2)
@index()
def tht_normal_pitch100mil_large_holes(size, mounting_holes=None):
return ProtoBoard(common_defs, 'thtl', mounting_holes, border=2)
@index()
def tht_normal_pitch100mil_xl_holes(size, mounting_holes=None):
return ProtoBoard(common_defs, 'thtl', mounting_holes, border=2)
@index()
def tht_normal_pitch100mil_square_pads(size, mounting_holes=None):
return ProtoBoard(common_defs, 'thtl', mounting_holes, border=2)
@index()
def tht_pitch_50mil(size, mounting_holes=None):
return ProtoBoard(common_defs, 'tht50', mounting_holes, border=2)
@index()
def tht_pitch_50mil_square_pads(size, mounting_holes=None):
return ProtoBoard(common_defs, 'tht50', mounting_holes, border=2)
@index()
def tht_mixed_pitch(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
return ProtoBoard(common_defs, f'tht50@{f}mm / tht', mounting_holes, border=2, tight_layout=True)
@index()
def tht_mixed_pitch_square_pads(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
return ProtoBoard(common_defs, f'tht50@{f}mm / tht', mounting_holes, border=2, tight_layout=True)
for pattern, name in connector_pitches.items():
@index(name=f'tht_and_connector_area_{name}')
def tht_and_connector_area(size, mounting_holes=None):
w, h = size
f = max(3.96*2.1, min(15, h*0.1))
return ProtoBoard(common_defs, f'{pattern}@{f}mm / tht', border=2, tight_layout=True)
@index()
def tht_and_connector_areas(size, mounting_holes=None):
w, h = size
fh = max(3.96*2.1, min(15, h*0.1))
fw = max(3.96*2.1, min(15, w*0.1))
return ProtoBoard(common_defs, f'conn396@{fw}mm | ((tht50 | conn200)@{fh}mm / tht / (conn125|conn250)@{fh}mm) | conn350@{fw}mm', border=2, tight_layout=True)
for pattern, name in smd_basic.items():
pattern_sizes = sizes_small if pattern not in ['manhattan'] else sizes_medium
# Default to ground plane on back for manhattan proto boards
pattern_back = pattern if pattern not in ['manhattan'] else 'ground'
@index(sizes=pattern_sizes, name=f'{name}_ground_plane')
def gen(size, mounting_holes=None):
return ProtoBoard(common_defs, f'{pattern} + ground', mounting_holes, border=1)
@index(sizes=pattern_sizes, name=f'{name}_single_side')
def gen(size, mounting_holes=None):
return ProtoBoard(common_defs, f'{pattern} + empty', mounting_holes, border=1)
@index(sizes=pattern_sizes, name=f'{name}_double_side')
def gen(size, mounting_holes=None):
return ProtoBoard(common_defs, f'{pattern} + {pattern}', mounting_holes, border=1)
@index(sizes=pattern_sizes, name=f'tht_and_{name}_large_holes')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
return ProtoBoard(common_defs, f'({pattern} + {pattern_back})@{f}mm / thtl', mounting_holes, border=1, tight_layout=True)
@index(sizes=pattern_sizes, name=f'{name}_and_tht_large_holes')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
return ProtoBoard(common_defs, f'({pattern} + {pattern_back}) / thtl@{f}mm', mounting_holes, border=1, tight_layout=True)
@index(sizes=pattern_sizes, name=f'tht_and_{name}')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
return ProtoBoard(common_defs, f'({pattern} + {pattern_back})@{f}mm / tht', mounting_holes, border=1, tight_layout=True)
@index(sizes=pattern_sizes, name=f'{name}_and_tht')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
return ProtoBoard(common_defs, f'({pattern} + {pattern_back}) / tht@{f}mm', mounting_holes, border=1, tight_layout=True)
@index(sizes=min_dim(pattern_sizes, 20), name=f'{name}_and_connector_areas')
def gen(size, mounting_holes=None):
w, h = size
fh = max(3.96*2.1, min(15, h*0.1))
fw = max(3.96*2.1, min(15, w*0.1))
return ProtoBoard(common_defs, f'conn396@{fw}mm | ((tht50 | conn200)@{fh}mm / ({pattern} + {pattern_back}) / (conn125|conn250)@{fh}mm) | conn350@{fw}mm', border=2, tight_layout=True)
*_, suffix = name.split('_')
if suffix not in ('100mil', '950um'):
@index(sizes=sizes_medium, name=f'tht_and_three_smd_100mil_950um_{suffix}')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(50, h*0.3))
f2 = max(1.27*5, min(30, w*0.2))
pattern_rot = f'{pattern}r' if pattern not in ['manhattan'] else pattern
pattern_back_rot = f'{pattern_back}r' if pattern not in ['manhattan'] else 'ground'
return ProtoBoard(common_defs, f'((smd100 + smd100) | (smd950 + smd950) | ({pattern_rot} + {pattern_back_rot})@{f2}mm)@{f}mm / tht', mounting_holes, border=1, tight_layout=True)
for (pattern1, name1), (pattern2, name2) in itertools.combinations(smd_basic.items(), 2):
*_, name1 = name1.split('_')
*_, name2 = name2.split('_')
@index(sizes=sizes_small, name=f'tht_and_two_smd_{name1}_{name2}')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
return ProtoBoard(common_defs, f'(({pattern1} + {pattern1}) | ({pattern2} + {pattern2}))@{f}mm / tht', mounting_holes, border=1, tight_layout=True)
@index(sizes=sizes_small, name=f'tht_and_two_sided_smd_{name1}_{name2}')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
return ProtoBoard(common_defs, f'({pattern1} + {pattern2})@{f}mm / tht', mounting_holes, border=1, tight_layout=True)
@index(sizes=sizes_small, name=f'two_sided_smd_{name1}_{name2}')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
return ProtoBoard(common_defs, f'{pattern1} + {pattern2}', mounting_holes, border=1)
@index(sizes_medium, name=f'tht_and_50mil_and_two_smd_100mil_950um')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(50, h*0.3))
f2 = max(1.27*5, min(30, w*0.2))
return ProtoBoard(common_defs, f'((smd100 + smd100) | (smd950 + smd950) | tht50@{f2}mm)@{f}mm / tht', mounting_holes, border=1, tight_layout=True)
@index(sizes=min_dim(sizes_medium, 60), name=f'all_tht_and_smd')
def gen(size, mounting_holes=None):
w, h = size
f = max(1.27*5, min(30, h*0.3))
f2 = max(1.27*5, min(25, w*0.1))
return ProtoBoard(common_defs, f'tht50@10mm | tht | ((smd100r + smd100r) / (smd950r + smd950r) / (smd800 + smd800)@{f2}mm / (smd650 + smd650)@{f2}mm / (smd500 + smd500)@{f2}mm)@{f}mm', mounting_holes, border=1, tight_layout=True)
write_index(index_d, outdir)
if __name__ == '__main__':
generate_all()