function sketch(
geojson,
{
title,
title_size = 32,
title_fill = "#333333",
paper = true,
width = 700,
margin = 10,
projection = d3.geoIdentity().reflectY(true),
k = 0.1,
fill = "#346E9C",
fillOpacity = 1,
fillWeight = 0.5,
fillStyle = "zigzag",
hachureGap = 2.5,
roughness = 8,
bowing = 0,
overflow = true,
stroke = "#000",
strokeWidth1 = 1,
strokeWidth2 = 0.5,
baseFrequency1 = 0.03,
baseFrequency2 = 0.06,
scale1 = 5,
scale2 = 7
} = {}
) {
let land = geo.simplify(geojson, { k: k, merge: true });
let delta = title ? 55 : 0;
let adjust = paper ? [width / 12, 10, 20] : [0, 0, 0];
const [[x0, y0], [x1, y1]] = d3
.geoPath(projection.fitWidth(width - adjust[0] - margin * 2, land))
.bounds(land);
let trans = projection.translate();
projection.translate([
trans[0] + adjust[0] + margin,
trans[1] + adjust[1] + margin + delta
]);
let height = Math.ceil(y1 - y0) + adjust[2] + margin * 2 + delta;
// start svg
const svg = d3
.create("svg")
.attr("viewBox", [0, 0, width, height])
.style("width", width)
.style("height", height)
.style("background", "white");
if (paper) {
svg
.append("g")
.append("image")
.attr("xlink:href", background)
.attr("width", width)
.attr("x", 0)
.attr("y", 0);
}
// rough
const rc = rough.svg(svg.node());
// svg filters (stroke)
let defs = svg.append("defs");
const pencil1 = defs.append("filter").attr("id", "pencil1");
pencil1.append("feTurbulence").attr("baseFrequency", baseFrequency1);
pencil1
.append("feDisplacementMap")
.attr("in", "SourceGraphic")
.attr("scale", scale1);
const pencil2 = defs.append("filter").attr("id", "pencil2");
pencil2.append("feTurbulence").attr("baseFrequency", baseFrequency2);
pencil2
.append("feDisplacementMap")
.attr("in", "SourceGraphic")
.attr("scale", scale2);
// contour smoothing
const path = smooth(d3.curveBasisClosed, projection);
// land clip (for clipping)
svg
.append("clipPath")
.attr("id", `myclip`)
.append("path")
.datum(land)
.attr("d", path);
// background pattern (fill)
svg
.append("g")
.attr("opacity", fillOpacity)
.attr("clip-path", overflow == true ? `none` : `url(#myclip)`)
.node()
.append(
rc.path(path(land.features[0]), {
stroke: "none",
bowing: bowing,
fill: fill,
fillWeight: fillWeight,
hachureGap: hachureGap,
roughness: roughness,
fillStyle: fillStyle
})
);
// contour patern (stroke)
svg
.append("path")
.datum(land)
.attr("d", path)
.attr("fill", "none")
.attr("stroke", stroke)
.attr("stroke-width", strokeWidth1)
.attr("filter", "url(#pencil1)");
svg
.append("path")
.datum(land)
.attr("d", path)
.attr("fill", "none")
.attr("stroke", stroke)
.attr("stroke-width", strokeWidth2)
.attr("filter", "url(#pencil2)");
// Title
svg
.append("text")
.attr("x", width / 2)
.attr("y", 45)
.attr("text-anchor", "middle")
.style("font-family", "Permanent Marker")
.attr("font-size", title_size)
.attr("fill", title_fill)
.text(title);
return svg.node();
}