Removed redundant method selection()
. It has been replaced with Animation.getSelection()
.
Removed: public AnimationSelection selection() Replacement: public AnimationSelection getSelection()
Added default engine as argument when calling below method.
Old: public MhEngineConstructionEntry toConstructionEntry() New: public MhEngineConstructionEntry toConstructionEntry(MhEngine engine=null)
Added new argument to rebuildProp
and rebuildProps
so that we can choose which rebuildFlag to use instead of always using diffRebuildProps.domain
.
Old: extend public void rebuildProp(str k) { New: extend public void rebuildProp(str k, diffRebuildProps rebuildFlag=diffRebuildProps.domain) { Old: extend public void rebuildProps(str[] keys) { New: extend public void rebuildProps(str[] keys, diffRebuildProps rebuildFlag=diffRebuildProps.domain) {
Removed the method sortedChildren(MhEngine engine, function(MhEngineEntry, MhEngineEntry, Object):int sortFunc)
, use sortedChildren(MhEngine engine, function(MhEngineEntry, MhEngineEntry, Object):int sortFunc, Object arg=null)
instead.
public class MhEngineEntryBase : abstract, unstreamable { Removed: extend public MhEngineEntry[] sortedChildren(MhEngine engine, function(MhEngineEntry, MhEngineEntry, Object):int sortFunc) : deprecated { }
Removed the following deprecated constructors.
public class MhEngineConstructionEntry extends MhEngineSnapperEntry : inherit constructors { Removed: public constructor(box b, Transform t, LayerSet classification=null) : deprecated { Removed: public constructor(box b, LayerSet classification) : deprecated { }
Removed the following deprecated fields and methods.
private class EngineVisualiserDialog extends DialogWindow { Removed: public TreeView leftTreeView : deprecated; Removed: public REDRenderView3D preview3D : deprecated; Removed: extend public void updateEntriesInTreeView() : deprecated {} Removed: extend public void updateGraphics() : deprecated {} }
The following functions have been moved from abstract.materialHandling.storage.engine
to abstract.materialHandling
.
Moved: public MhEngineEntry[] mhImportedChildren(MhEngineEntry entry, MhEngine engine) { Moved: public MhEngineEntry[] mhPopulatedChildren(MhEngineEntry entry, MhEngine engine) { Moved: public int mhEntryXPosSort(MhEngineEntry a, MhEngineEntry b, Object env) {
The following functions have been moved from abstract.materialHandling.storage.engine
to abstract.materialHandling.engine
.
Moved: public MhEngineEntry->double mhEntryXPositions(MhEngineEntry[] entries, bool unmodify) { Moved: public MhEngineEntry[] mhXSortedEntries(MhEngineEntry[] entries, bool unmodify) { Moved: public MhEngineEntry->double mhEntryYPositions(MhEngineEntry[] entries, bool unmodify) { Moved: public MhEngineEntry[] mhYSortedEntries(MhEngineEntry[] entries, bool unmodify) {
The following functions have been added.
New: public int mhEntryPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { New: public int mhEntryYPosSort(MhEngineEntry a, MhEngineEntry b, Object env) {
The following function has been removed from abstract.materialHandling.storage.engine
. Use the functions listed instead.
Removed: public int entryYPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { Use: public int mhEntryYPosSort(MhEngineEntry a, MhEngineEntry b, Object env) {
The following functions have been removed, use the replacement functions listed instead.
Removed: public MhEngineEntry[] mhAbsImportedChildren(MhEngineEntry entry, MhEngine engine) { Use: public MhEngineEntry[] mhImportedChildren(MhEngineEntry entry, MhEngine engine) { Removed: public MhEngineEntry[] mhAbsPopulatedChildren(MhEngineEntry entry, MhEngine engine) { Use: public MhEngineEntry[] mhPopulatedChildren(MhEngineEntry entry, MhEngine engine) { Removed: public int mhAbsEntryRootXYZPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { Use: public int mhEntryRootXYZPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { Removed: public int mhAbsEntryFuncRunOrderSort(MhEngineEntry a, MhEngineEntry b, Object env) { Use: public int mhEntryFuncRunOrderSort(MhEngineEntry a, MhEngineEntry b, Object env) { Removed: public int mhAbsEntryXPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { Use: public int mhEntryXPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { Removed: public int mhAbsEntryPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { Use: public int mhEntryPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { Removed: public int mhAbsEntryYPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { Use: public int mhEntryYPosSort(MhEngineEntry a, MhEngineEntry b, Object env) { Removed: public MhEngineEntry->double mhAbsEntryXPositions(MhEngineEntry[] entries, bool unmodify) { Use: public MhEngineEntry->double mhEntryXPositions(MhEngineEntry[] entries, bool unmodify) { Removed: public MhEngineEntry[] mhAbsXSortedEntries(MhEngineEntry[] entries, bool unmodify) { Use: public MhEngineEntry[] mhXSortedEntries(MhEngineEntry[] entries, bool unmodify) { Removed: public MhEngineEntry->double mhAbsEntryYPositions(MhEngineEntry[] entries, bool unmodify) { Use: public MhEngineEntry->double mhEntryYPositions(MhEngineEntry[] entries, bool unmodify) { Removed: public MhEngineEntry[] mhAbsYSortedEntries(MhEngineEntry[] entries, bool unmodify) { Use: public MhEngineEntry[] mhYSortedEntries(MhEngineEntry[] entries, bool unmodify) {
New optional arguments are added to these functions in order to reduce number of set and seq created.
Old: public (Snapper{}) endNeighbors(MhSnapper this, LayerSet ls=null) { New: public (Snapper{}) endNeighbors(MhSnapper this, LayerSet ls=null, Snapper{} snappers=null) { Old: public (Snapper{}) nextNeighbors(MhSnapper this, LayerSet ls=null) { New: public (Snapper{}) nextNeighbors(MhSnapper this, LayerSet ls=null, Snapper{} snappers=null) { Old: public (Snapper{}) prevNeighbors(MhSnapper this, LayerSet ls=null) { New: public (Snapper{}) prevNeighbors(MhSnapper this, LayerSet ls=null, Snapper{} snappers=null) { Old: public (Snapper{}) linkedNeighbors(MhSnapper this, LayerSet ls=null) { New: public (Snapper{}) linkedNeighbors(MhSnapper this, LayerSet ls=null, Snapper{} snappers=null) {
There were various different functions and methods to retrieve bounds of snappers including their children bound. We have reduce the number of interfaces by combining them where possible.
We have consolidated the following MhSnapper
methods into one. Replace all calls to childrenBound(bool recursive=true)
, childrenShapeLocalBound(symbol[] symbols=null)
, and childrenShapeBound(symbol[] symbols=null, SnapperFilter filter=null)
with childrenShapeBound(symbol[] symbols=null, SnapperFilter filter=null, bool recursive=false)
.
Note that the old childrenBound(bool recursive=true)
method has recursive
's default argument set to true
while the new method has recursive
's default argument set to false
, so any calls to childrenBound()
should be replaced with childrenShapeBound(recursive=true)
.
Old: public class MhSnapper extends Snapper { /** * Children bound. */ extend public box childrenBound(bool recursive=true) { box bc(); bool empty=true; forChildren(MhSnapper c in this) { box b = c.localBound(); b.transform(c.rot == 0deg ? Transform(c.pos) : Transform(c.pos, c.rot)); if (b != box0) { if (empty) bc = b; else bc += b; empty = false; } if (recursive) bc += c.childrenBound(..); } return bc; } /** * Children shape local bound. */ extend public box childrenShapeLocalBound(symbol[] symbols=null) { box bc(); bool empty=true; forChildren(MhSnapper c in this) { box b = c.shapeBound(symbols); if (b != box0) { if (empty) bc = b; else bc += b; empty = false; } } if (MhSnapperBehavior b = snapperInfosBehavior()) { for (info in b.infos) { info.shape.?setOwner(this); box b = info.shape.?localBound(symbols); if (b != box0) { if (empty) bc = b; else bc += b; } info.shape.?setOwner(null); } } return bc; } /** * Children shape bound. */ extend public box childrenShapeBound(symbol[] symbols=null, SnapperFilter filter=null) { box bc(); bool empty=true; //for (MhSnapper c in children) { forChildren(MhSnapper c in this) { if (filter and !filter.accepts(c)) continue; box b = c.shapeBound(symbols); if (b != box0) { b.transform(mhTransform(c.pos, c.rot)); if (empty) bc = b; else bc += b; empty = false; } } for (info in snapperInfos()) { try { info.shape.?setOwner(this); box b = info.shape.?localBound(symbols); if (b != box0) { b.transform(mhTransform(info.pos, info.rot)); if (empty) bc = b; else bc += b; empty = false; } } finally { info.shape.?setOwner(null); } } return bc; } }
New: public class MhSnapper extends Snapper { /** * Children shape bound. */ extend public box childrenShapeBound(symbol[] symbols=null, SnapperFilter filter=null, bool recursive=false) { box bc(); bool empty = true; forChildren(MhSnapper c in this) { if (filter and !filter.accepts(c)) continue; box b = c.shapeBound(symbols); if (b.volume > 0) { b.transform(mhTransform(c.pos, c.rot)); if (empty) bc = b; else bc += b; empty = false; } if (recursive) { box cbc = c.childrenShapeBound(..); if (cbc.volume > 0) { cbc.transform(mhTransform(c.pos, c.rot)); if (empty) bc = cbc; else bc += cbc; empty = false; } } } for (info in snapperInfos()) { try { info.shape.?setOwner(this); box b = info.shape.?localBound(symbols); if (b.volume > 0) { b.transform(mhTransform(info.pos, info.rot)); if (empty) bc = b; else bc += b; empty = false; } } finally { info.shape.?setOwner(null); } } return bc; } }
Important to note is that we have also made a behavioral change. Previously in localBoundWithChildren(MhSnapper owner, symbol[] symbols=null, SnapperFilter filter=null)
and shapeBoundWithChildren(MhSnapper owner, symbol[] symbols, SnapperFilter filter=null)
we would always include the result of the children bounds regardless of what they were. Now in both of these functions, we only include the children bounds if their bound volumes are not zero. This is to fix the issue where we could include empty bounds into the resultant bound. Now you may even have a resultant bound that does not include (0, 0, 0) within it. Be aware of this as there may be some existing logic in your codebase that relied on this false positive behavior previously, you can now be more specific in the bounds you include.
Aside from that, we have updated localBoundWithChildren
and shapeBoundWithChildren
to use the newly introduced MhSnapper.childrenShapeBound(symbol[] symbols=null, SnapperFilter filter=null, bool recursive=false)
method.
Old: /** * Get localbound with children */ public box localBoundWithChildren(MhSnapper owner, symbol[] symbols=null, SnapperFilter filter=null) { if (!owner) return box(); box bc(); bool first = true; if (!filter or filter.accepts(owner)) { bc = owner.shapeBound(symbols); bc += owner.childrenShapeBound(symbols); first = false; } forChildren(MhSnapper child in owner) { if (child.includeChildrenInLocalBound) { box b = localBoundWithChildren(child, symbols, filter); Transform t = mhTransform(child.pos, child.rot); b = b.transformed(t); bc = first ? b : bc + b; if (first) first = false; } } return bc; } /** * Get shape bound with children */ public box shapeBoundWithChildren(MhSnapper owner, symbol[] symbols, SnapperFilter filter=null) { if (!owner) return box(); box bc(); bool first = true; if (!filter or filter.accepts(owner)) { bc = owner.shapeBound(symbols); bc += owner.childrenShapeBound(symbols, filter); //say what first = false; } forChildren(MhSnapper child in owner) { box b = shapeBoundWithChildren(child, symbols, filter); Transform t = mhTransform(child.pos, child.rot); b = b.transformed(t); bc = first ? b : bc + b; if (first) first = false; } return bc; }
New: /** * Localbound with children. * This differ from shapeBoundWithChildren as this recursiveness controlled by `includeChildrenInLocalBound`. */ public box localBoundWithChildren(MhSnapper owner, symbol[] symbols=null, SnapperFilter filter=null) { if (filter and !filter.accepts(owner)) return box(); if (!owner) return box(); box bc = owner.shapeBound(symbols); box b = localBoundWithChildrenRe(..); if (b.volume > 0) bc += b; return bc; } private box localBoundWithChildrenRe(MhSnapper owner, symbol[] symbols, SnapperFilter filter) { if (filter and !filter.accepts(owner)) return box(); if (!owner.includeChildrenInLocalBound) return box(); box bc = owner.childrenShapeBound(symbols, filter); forChildren(MhSnapper c in owner) { box b = localBoundWithChildrenRe(c, -..); if (b.volume > 0) { b.transform(mhTransform(c.pos, c.rot)); bc += b; } } return bc; } New: /** * Get shape bound with children. * Always recursive. */ public box shapeBoundWithChildren(MhSnapper owner, symbol[] symbols=null, SnapperFilter filter=null) { if (!owner) return box(); box bc(); if (!filter and filter.accepts(owner)) { bc = owner.shapeBound(symbols); box cbc = owner.childrenShapeBound(symbols, filter, recursive=true); if (cbc.volume > 0) bc += cbc; } return bc; }
The toConstructionEntry
method now also ensures the imported props of the snapper entry are transferred over to the newly created construction entry.
/** * To construction entry. */ public MhEngineConstructionEntry toConstructionEntry(MhEngine engine=null) { MhEngineConstructionEntry res = createConstructionEntry(); if (snapper) { ... // Transfer import props to be part of construction entry. if (!engine) engine = snapper.engine; var env = engine.?engineEnv(); for (k, _ in snapper.importProps(engine)) { if (k in [mhPrevNeighborsKey, mhNextNeighborsKey]) continue; Object v = env.getProp(snapper, k); if (v) res.putSnapperImportProp(k, v); } } return res; }
In accepts(Snapper snapper)
method, if the childDepth
is not the same, it will instantly return false.
public bool accepts(Snapper snapper) { if (owner.childDepth != snapper.childDepth) return false; ... }
We are moving away from utilizing post engine run functions. Previously when an engine is currently running functions and a new engine function was appended during this (particularly during the export function), it would have been appended to the same engine run that was being executed. This new engine function would then be executed during the post engine run. There was an issue where with a multi-assortment setup during the export function, new engine functions could be appended to the wrong engine run with the wrong assortment. Those functions would not be executed due to the mismatch of assortments.
We have now updated the logic such that when an engine is running, newly appended engine functions would be appended into a new engine run instead of the currently executing engine run. The current exception to this would be for the animation export MhSystemAnimationExportFunction
where we still allow new engine functions to be appended and executed during the post engine run.
public class MhEngine extends CxEngine { Old: /** * Engine run * Return a current run to append functions to it */ extend public MhEngineRun engineRun(MhEngineRun currentRun, MhEngineRun[] currentList, MhAssortment a) { MhEngineRun run = null; if (runByRegistrationOrder) { if (currentRun) return currentRun; if (!run) for (r in currentList) { run = r; break; } } else { if (currentRun and currentRun.assortment == a) run = currentRun; if (!run) for (r in currentList) if (r.assortment == a) { run = r; break; } } return run; } New: /** * Engine run * Return a current run to append functions to it */ extend public MhEngineRun engineRun(MhEngineRun currentRun, MhEngineRun[] currentList, MhAssortment a) { if (runByRegistrationOrder) { if (currentRun) return currentRun; for (r in currentList) { return r; } } else { if (currentRun.?assortment == a and currentRun.?allowFuncReg) return currentRun; for (r in currentList) { if (r.assortment == a and r != currentRun) { return r; } } } return null; } } public class MhEngineManager extends CoreObject { Old: /** * Register run. */ extend public void register(MhEngine engine, MhEngineFunctionRun func) { ... MhEngineRun run = engine.engineRun(currentRun, list, a); if (!run) { if (run = createEngineRun(a)) { run.assortment = a; init? list(); list << run; engineRunList.put(engine, list); } } run.?append(engine, func); } New: /** * Register run. */ extend public void register(MhEngine engine, MhEngineFunctionRun func) { ... MhEngineRun run = engine.engineRun(currentRun, list, a); if (!run) { if (run = createEngineRun(a)) { if (currentRun) run.postRun = currentRun.postRun + 1; devassert(run.postRun <= 5) { pln("Exceeding number of recursion allowed"); return; } run.assortment = a; init? list(); int index = -1; if (currentRun) index = list.indexOf(currentRun); if (index >= 0) list.insert(index + 1, run); else list << run; engineRunList.put(engine, list); } } run.?append(engine, func); } } public class MhEngineRun { New: /** * Temporarily allow same run function register. */ public bool allowFuncReg; }