Skip to main content

Opencascade bottle

If you have dabbled in opencascade you have read the bottle tutorial. So what does it look like with replicad?

We can have a look at the rendered shape.

And here is what the code looks like.

const defaultParams = {
width: 50,
height: 70,
thickness: 30,
};

/** @typedef { typeof import("replicad") } replicadLib */
/** @type {function(replicadLib, typeof defaultParams): any} */
const main = (
{ Sketcher, FaceSketcher, makeCylinder, makeOffset, FaceFinder },
{ width: myWidth, height: myHeight, thickness: myThickness }
) => {
let shape = new Sketcher()
.movePointerTo([-myWidth / 2, 0])
.vLine(-myThickness / 4)
.threePointsArc(myWidth, 0, myWidth / 2, -myThickness / 4)
.vLine(myThickness / 4)
.closeWithMirror()
.extrude(myHeight)
.fillet(myThickness / 12);

const myNeckRadius = myThickness / 4;
const myNeckHeight = myHeight / 10;
const neck = makeCylinder(
myNeckRadius,
myNeckHeight,
[0, 0, myHeight],
[0, 0, 1]
);

shape = shape.fuse(neck);

shape = shape.shell(myThickness / 50, (f) =>
f.inPlane("XY", [0, 0, myHeight + myNeckHeight])
);

const neckFace = new FaceFinder()
.containsPoint([0, myNeckRadius, myHeight])
.ofSurfaceType("CYLINDRE")
.find(shape.clone(), { unique: true });

const bottomThreadFace = makeOffset(neckFace, -0.01 * myNeckRadius).faces[0];
const baseThreadSketch = new FaceSketcher(bottomThreadFace)
.movePointerTo([0.75, 0.25])
.halfEllipse(2, 0.5, 0.1)
.close();

const topThreadFace = makeOffset(neckFace, 0.05 * myNeckRadius).faces[0];
const topThreadSketch = new FaceSketcher(topThreadFace)
.movePointerTo([0.75, 0.25])
.halfEllipse(2, 0.5, 0.05)
.close();

const thread = baseThreadSketch.loftWith(topThreadSketch);

return shape.fuse(thread);
};