In this release, new concept of multi structure is introduce. This multi structure enables snapper/entry to have nested same-classification children. For example, a multi bay has 2 sub bay children. In order to support this structure, there are few behaviors and engine functions added to handle these multi cases separately from the normal structure.
Updated logic and method to find best matched spawner by including the minimum numbers of non-matched symbols.
Removed: /** * Find best match spawner. */ extend public SnapperSpawner findBestMatchSpawner(LayerSet classification, SnapperSpawner current, MhAssortment assortment, bool &exactMatch) { if (MhSpawnerStore spawnerStore = assortment.?spawnerStore) { int maxMatchCount; // use all sorted to ensure consistency. for (_, spawner in spawnerStore.allSorted()) { if (!accepts(classification, spawner)) continue; // prioritize exact match if (classification.equal(spawner.classification)) { current = spawner; exactMatch = true; break; } LayerSet intersect = classification*spawner.classification; int layerMatchCount = intersect.count; if (layerMatchCount > maxMatchCount) { current = spawner; maxMatchCount = layerMatchCount; } } } return current; } Added: /** * Find best match spawner. */ extend public SnapperSpawner findBestMatchSpawner(LayerSet classification, MhSnapperSpawner[] spawners) { SnapperSpawner res = null; int maxMatchCount = -maxInt; int minRemainder = maxInt; // use all sorted to ensure consistency. for (spawner in spawners) { if (!accepts(classification, spawner)) continue; // prioritize exact match LayerSet ls = spawner.classification; if (classification.equal(ls)) { res = spawner; break; } LayerSet intersect = classification*ls; if (!intersect) continue; int layerMatchCount = intersect.count; int remainder = (ls - intersect).count; if (layerMatchCount > maxMatchCount or remainder < minRemainder) { res = spawner; maxMatchCount = layerMatchCount; minRemainder = remainder; } } return res; }
The method forceRebuildProps(MhStorageEditorDialog dialog)
has been moved into the parent class MhStorageEditorItem
.
Function mhFrameEditorReinsertFrames
has been made public.
Old: package void mhFrameEditorReinsertFrames(MhStorageEditorSpace space) { New: public void mhFrameEditorReinsertFrames(MhStorageEditorSpace space) {
Removed deprecated method unitLoadCountX()
.
public class MhStorageConfiguration extends MhSystemConfiguration { Removed: extend public int unitLoadCountX() : deprecated { return getValue(cMhNoOfUnitLoadsPK).?int; } }
The class MhSingleUprightFrameShape
is removed.
Removed: public class MhSingleUprightFrameShape extends MhFrameShape {
The class MhUpdateImportPrimsFunction
has been removed as it is not used in the abstract.
Removed: public class MhUpdateImportPrimsFunction extends MhSystemEngineFunction : deprecated {
The class MhRowChildrenUnlinkFunction
has been removed as it is not used in the abstract.
Removed: public class MhRowChildrenUnlinkFunction extends MhSystemEngineFunction : deprecated {
This vessel has a new field addSelection
which acts as a flag to add snapper's selection to the list of selection.
public class MhRowPickUpFrameVessel extends MhApplyColorFrameVessel { /** * Add Selection. */ public bool addSelection; /** * Constructor. */ public constructor(str key, bool active=true, CoreObject owner=null, str->Object args=null, Object env=null, symbol{} tags=null, color c=color(224, 32, 32)|, bool addSelection=true) { super(..|); this.addSelection = addSelection; } /** * Add to selection. */ public Snapper{} addToSelection(Snapper s) { if (addSelection) return super(..); return null; } }
Renamed the following method:
public class MhStorageSelectionBehavior extends MhSelectionBehavior { Old: extend public void addtionalSnapperInSelection(MhSnapper snapper, SnapperSelection sel, Line mouseLine) { } New: extend public void additionalSnapperInSelection(MhSnapper snapper, SnapperSelection sel, Line mouseLine) { } }
Renamed the following methods:
public class MhRowChildSelectionBehavior extends MhStorageSelectionBehavior { Old: extend public bool addtionalLinkedSnappers(MhSnapper snapper, SnapperSelection sel, Line mouseLine) { New: extend public bool additionalLinkedSnappers(MhSnapper snapper, SnapperSelection sel, Line mouseLine) { Old: extend public void addtionalFilteredSnappers(MhSnapper snapper, SnapperSelection sel, Line mouseLine) { New: extend public void additionalFilteredSnappers(MhSnapper snapper, SnapperSelection sel, Line mouseLine) { }
Removed the following fields and methods as well as renamed levelsEntries
to levelEntries
.
With the removal of getLevelEntry(bool first, bool last)
, replace it with getLevelEntry(bool first, bool last, int idx=-1)
.
We have undeprecated constructor(MhEngineConstructionEntry levelEntry, MhEngineConstructionEntry botLevelEntry=null, MhEngineConstructionEntry topLevelEntry=null)
and updated it to work with the field levelEntries
.
public class MhBayEntryLayout extends MhEntryLayout { Removed: public MhEngineConstructionEntry levelEntry : deprecated; Old: public MhEngineEntry[] levelsEntries; New: public MhEngineEntry[] levelEntries; Removed: extend public MhEngineConstructionEntry getLevelEntry(bool first, bool last) : deprecated { return getLevelEntry(.., idx=-1).MhEngineConstructionEntry; } }
cPredefinedConfig
has been renamed to cPredefinedBayConfigDefVal
and its value has been modified from _predefinedConfig
to _predefinedBayConfigDefault
to better reflect its usage.
Old: /** * Predefine key. */ public const str cPredefinedConfig = "_predefinedConfig"; New: /** * Predefined bay config default value. */ public const str cPredefinedBayConfigDefVal = "_predefinedBayConfigDefault";
A constant mhFrameHeighDiffKey
was moved from abstract.materialHandling
to abstract.materialHandling.storage
.
public const str mhFrameHeighDiffKey = "frameHeightDiff";
The functions removeUniqueName
, unitLoadEq
, and findEqualUnitLoad
are moved to cm/abstract/unitLoad/functions.cm
from cm/abstract/materialHandling/storage/unitLoadFunctions.cm
.
Moved: public str removeUniqueName(Object o) { public bool unitLoadEq(UnitLoad unitLoad, UnitLoad l) { public UnitLoad findEqualUnitLoad(UnitLoad unitLoad, World w=null) {
MhBayUpdateFramesFunction
now include children to framesToUpdate
to get updated if the classification is eval to its parent.
MhLevelArrangeFunction
now only sort children that are levels.
MhBayShape
will now adjust to its owner's parent if it is also a bay.
public bool adjustTo(Snapper toSnapper, Object env=null) { ... } else if (isBay(owner) and toSnapper.isBay) { box b = owner.shapeBound([sBay]); box lb = toSnapper.shapeBound([sBay]); if (adjustWidth(owner, toSnapper, b, lb, env)) adjusted = true; } ... }
Shuttle rail's shape will now only add front beam if the classification is not the rootParent
row is not front row and inner.
Old: extend public bool shoudAddFrontBeam() { return true; } New: extend public bool shoudAddFrontBeam() { if (?MhSnapper root = owner.?rootParent) { if (root.classification.isFrontRow and root.classification.isInner) return false; } return true; }
In createRowEntry(MhEngineEntryBlock block, MhEngineConstructionEntry e)
method, if entry
is in visited, it will also copy the visited entry's classification when we copy the visited entry's children. This is due to the visited entry's classification might get updated during runEntryFunctions
, so the entry
should get the same classification as well as we assume visited entry and entry
would be exactly the same.
extend public MhEngineConstructionEntry createRowEntry(MhEngineEntryBlock block, MhEngineConstructionEntry e) { ... for (MhEngineConstructionEntry ce in children) { str k = entryCacheKey(ce); if (k in visited) { ?MhEngineConstructionEntry ee = visited.get(k); for (MhEngineConstructionEntry c in ee.children(null)) ce << c.toConstructionEntry(); ce.classification = ee.classification; //copy the classification too } ... }
Introduced some new interfaces in MhStorageEditorConfiguration
to use non-serious undo when changing property values in a configuration item. This can be used in conjunction with a new UndoOp MhStorageEditorConfigPropModifyUndoOp
that can restore a configuration property value, used when a configuration value is not pushed to the snappers.
public class MhStorageEditorConfiguration extends MhSystemConfiguration { /** * Property uses serious undo? */ extend public bool propertyUsesSeriousUndo(MhStorageEditorItem item, CoreProperty property) { return true; } /** * Before property changed. */ extend public void beforePropertyChange(MhStorageEditorItem item, CoreProperty property) { ... if (propertyUsesSeriousUndo(item, property)) { space.beginSeriousUndoStep(undoSnappers, space); } else { space.beginUndoStep(); space.beginUndoRecording(); for (s in undoSnappers) space.recordForUndo(s); } ... } /** * After property changed. */ extend public void afterPropertyChanged(MhStorageEditorItem item, CoreProperty property) { ... if (propertyChangeLevel == 0) { if (propertyUsesSeriousUndo(item, property)) { if (space.seriousUndoMode) { space.?endSeriousUndoStep(undoSnappers, description=$editorItemChanged); } } else { space.endUndoRecording(); space.endUndoStep(); } ... } }
To support non-serious undo, MhFrameEditConfiguration.pushPropToPreview()
no longer disables undo. This is so that non-serious undo also records changes made to MhFrameEditConfiguration.refSpace
snappers by the engine within pushPropToPreview()
. This change was made earlier for MhBayEditConfiguration.pushPropToPreview()
but we are now making them work similarly. This should not affect you unless you are calling pushPropToPreview
in your code while recording for non-serious undo.
The behavior mhBaySelectionBehavior
has been added to MhBaySpawner
default behaviors. If your class that extends from MhBaySpawner
is adding the same behavior, you can now remove it. If your class already adds a different selection behavior and you want to retain that instead, you should override customOtherBehaviors() and exclude mhBaySelectionBehavior
after super()
, e.g. behaviors.exclude(mhBaySelectionBehavior);
.
Additionally, the behaviors mhBaySpreadPatternBehavior
and mhBayEditorMeasurementsBehavior
were previously added in gatherBehaviors(MhBehavior[] behaviors)
. They have now been moved to customOtherBehaviors()
instead.
public class MhBaySpawner extends MhStorageSpawner { /** * CustomOtherBehaviors. */ public MhBehavior[] customOtherBehaviors() { ... behaviors << mhBaySelectionBehavior; behaviors << mhBaySpreadPatternBehavior; behaviors << mhBayEditorMeasurementsBehavior; } }
Instead of putting unit load as a MemoryStream
into the propData
, we now directly put the unit load object itself.
Old: extend public void streamUnitLoad(ObjectFormatter formatter) { str key = unitLoadKey.?v; Object o = null; for (Str t, v in formatter.tempData) if (t and t.v == key) { o = v; break; } if (!o) { o = objectToMemoryStream(unitLoad); ... New: extend public void streamUnitLoad(ObjectFormatter formatter) { str key = unitLoadKey.?v; Object o = null; for (Str t, v in formatter.tempData) if (t and t.v == key) { o = v; break; } if (!o) { o = unitLoad; ...
As we are moving towards the multi layout concept, we change double deep checking to check for the child depth now instead of the classification. For example, these few places are
at MhLevelConstructionalPopulateFunction
:
Old: extend public double addStaticLevels(double z) { MhEngineEntry rowEntry = parent.rootParent(engine.MhEngine); if (rowEntry.?isDoubleDeep) return z; ... New: extend public double addStaticLevels(double z) { if (parent.childDepth(engine.MhEngine) > 1) return z; ...
at MhFrameConfigBehavior
:
Old: public bool shouldPutConfigToManager(MhSnapper z) { Snapper root = z.rootParent; if (root.isDoubleDeep) return !root.isBackRow; ... New: public bool shouldPutConfigToManager(MhSnapper z) { if (z.childDepth > 1) return false; //TODO revisit this, handle multi deep config properly ...
MhRowChildCopyPasteBehavior
has been updated to also with with cut-paste actions. This class is meant to work together with the MhRowChildFavoriteAnimationG2
animation that will generate new rows for the pasted row-child snappers. It has now been updated to also work with cut-paste, you can now cut row-child snappers and pasting should generate new row snappers for them.
Below are the updates done to support this:
public class MhRowChildCopyPasteBehavior extends MhBehavior { /** * Selected. * Put mhParentGid on selected so we can use it with userCut. */ public void selected(MhSnapper snapper) { if (Snapper parent = snapper.parent) snapper.fastPut("mhParentGid", parent.gid.hex, mhParentGidDef); } /** * Deselected. */ public void deselected(MhSnapper snapper) { snapper.fastRemove("mhParentGid"); } /** * User cut. */ public void userCut(MhSnapper snapper, SpaceSelection selection) { snapper.fastRemove("mhParentGid"); } }