tl;dr Show me a working example right away

Intro

The critical angle plays a pivotal role in the cutting process. To achieve total internal reflection for light rays within the gemstone, the facets must be cut at angles steeper than the critical angle. However, the challenge lies in the fact that light reflects multiple times inside the stone and can originate from any direction. Without precise tools, maximizing the potential of your design becomes impossible.

Hackagem’s optimizer focuses on minimizing the windowing effect across a wide range of tilted stone positions. You have control over the valid parameter ranges, which is vital since the optimizer may eliminate entire tiers if it helps address the windowing issue. However, this can come at the expense of reduced scintillation. As such, optimization becomes a delicate balancing act, as we will explore later.

A nice feature is that the optimizer has the ability to maintain a high diversity of solutions, allowing you to choose not only the best-performing option but also the most visually pleasing one.

Quick Overview

Before explaining how to run the optimizer yourself, let’s take a look at optimizing a Round Brilliant Cut in quartz from our previous post. During optimization there are three parts that are being shown.

Progress Graph

The progress graph tracks the iteration, the temperature (we will come to that), the “loss” at each iteration, and the lowest loss value so far.

Tilt performance graph

Showing the amount of windowing in the original design at a range of -45 to 45 degrees. The best design found so far, is shown as well. In this case the improvement is quite dramatic, at zero tilt for instance, the window decreased from 17% to about 4%.

Top 5 Solution with Diversity

The optimizer aims to maintain diversity in its solutions, so the top five options are not always the ones with the absolute lowest windowing. However, the design with the lowest windowing is guaranteed to be included in the list. In the example, we can see that the best design achieves a windowing improvement of over 40%, which is crazy and is calculated across a range of different tilt angles.

Preparing a design for optimization

There are a number of steps required to use the optimizer for your design. About 6 lines of code should be added in order to run the optimizer. First, let’s assume the code to be optimized looks as follows:

setSym(6, true)
cut("1", index(3), 42.5, centerPoint())
cut("2", index(3), 90, setSize())
crown()
cut("3", index(0), 20, meet("1,2"))

Now, lets assume we want to find the optimal angle for the first and last tier. We’ll put the cut commands inside a function and give it a name:

setSym(6, true)
function myDesign() {
  cut("1", index(3), 42.5, centerPoint())
  cut("2", index(3), 90, setSize())
  crown()
  cut("3", index(0), 20, meet("1,2"))
}

This function myDesign() will be called by the optimizer many times and it will be called with an argument containing the values for the angles that need to be optimized. We need to deconstruct that parameter to get the actual values:

setSym(6, true)
function myDesign(X) {
  const [a, b] = X
  cut("1", index(3), a, centerPoint())
  cut("2", index(3), 90, setSize())
  crown()
  cut("3", index(0), b, meet("1,2"))
}

Note that we use a and b for angles, but it can be any number for any part of the design that you like to optimize.

Just to check that everything works, we can call this function with the original values:

...
myDesign([ 42.5, 20])

The function is ready, we now need to specify initial values for a and b, together with their valid ranges. Like so:

const myVariableDefinitions = [      // Use any name you like
    { name: "a", initial: 40, min: 35, max: 50 },
    { name: "b", initial: 22, min: 15, max: 40 }
];

Note that the ordering is important, the variables should be in the same order as the reading order in myDesign(X). The actual inital values shouldn’t matter a lot, but it is good to start with a reasonable design.

The last step is to actually initiate the optimizer, the whole code:

setSym(6, true)
function myDesign(X) {
  const [a, b] = X
  cut("1", index(3), a, centerPoint())
  cut("2", index(3), 90, setSize())
  crown()
  cut("3", index(0), b, meet("1,2"))
}

const myVariableDefinitions = [      // Use any name you like
    { name: "a", initial: 40, min: 35, max: 50 },
    { name: "b", initial: 22, min: 15, max: 40 }
];

runOptimizer(myVariableDefinitions, myDesign)

Pressing the Run button will cut the design but only for the initial values. CLicking Optimize will start the optimization process.