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.
There is a change on how we stream UnitLoadContainer
and its unit loads. This is so that failing to unstream one unit load will not cause the entire UnitLoadContainer
to fail to unstream.
As of 16.0, preview image support has been implemented for core Part
s and is no longer exclusive to DsPart
s.
The system for generating preview images is:
cmWritable("tmp/partPreviewImageCache/")
str->Url
map is stored in Space cacheData
for easy access cm/core/part/hooks.cm
)renderPartPreviewImagesTask
task begins (cm/core/part/tasks.cm
)As a part of the effort to standardize and improve query dialogs utilized in CET, we have created a core implementation of Query Dialogs for Part
s and PropObj
s.
This new implementation is meant to phase out the current implementation in cm/abstract/tools/queryDialog/abstractQueryDialog.cm
.
This Query dialog is designed to be opt-in and it should not require migration effort.
The code for the QueryDialog
system can be found in the cm.core.part.query
package.
An overridden implementation for handling ProdPart
s and SpecOption
s can be found in cm.abstract.party.query
.
Doing a static method call without the first argument being an instance of the class or an instance of a derived class, would incorrectly be able to bind to a function without class as first parameter.
private class Foo {} private void bar(str k) {} Foo.bar("");//this call would incorrectly bind to the function above
A repeated case in a case expression list would fail to be detected
switch("foo") { case "foo": {} case "foo", "bar": {} //repeated case "foo" in case expression list would incorrectly be allowed }
Field showPreview
has been renamed to activate
.
# CardConfig Old: public bool showPreview=true; New: public bool activate = true;
As of 16.0, preview image support has been implemented for core Part
s and is no longer exclusive to DsPart
s.
As a result, the global constant dsPreviewColumn
has been removed from cm.abstract.dataSymbol
.
A new constant of the column (called partPreviewColumn
) has been added and is registered in cm/core/init/corePartColumns.cm
.
In cm/abstract/dataSymbol/init.cm Removed: public const DsPreviewColumn dsPreviewColumn(); In cm/core/init/corePartColumns.cm Added: public const PartPreviewColumn partPreviewColumn();
DsOptionInfoTree
now extends SpecOptionInfoTree
. It no longer stores values for str description
, str price
, str groupDescription
, or SpecOption specOpt
.
SpecOptionInfoTree
stores the SpecOption
for the tree and values like description and price are acquired directly from the SpecOption
.
The following has been removed/added in DsOptionInfoTree
:
// FIELDS Removed: public str description; Removed: public str price; Removed: public str groupDescription; Removed: public SpecOption specOpt; // CONSTRUCTORS Removed: public constructor(str key, str description, str price, Part owner, DsPDataOption option, str header="", bool choosableInCalculation=true) {} Removed: public constructor(str key, str description, str groupDescription, str price, Part owner, DsPDataOption option, str header="", bool choosableInCalculation=true) {} Removed: public constructor(PartInfoTree parent, str key, str description, str price, Part owner, DsPDataOption option, str header="", bool choosableInCalculation=true) {} Removed: public constructor(PartInfoTree parent, str key, str description, str groupDescription, str price, Part owner, DsPDataOption option, str header="", bool choosableInCalculation=true) {} Removed: public constructor(PartInfoTree parent, str key, str description, str groupDescription, str price, Part owner, DsPDataOption option, SpecOption specOpt, str header="", bool choosableInCalculation=true) {} Added: public constructor(Part owner, DsPDataOption option, PartInfoTree parent=null, str key=null, SpecOption specOpt=null, str header="", bool choosableInCalculation=true)
DsPartTagColumn
has been deprecated.Use AbsTagPartColumn
n in cm.abstract.part.partColumn
.
As of 16.0, preview image support has been implemented for core Part
s and is no longer exclusive to DsPart
s.
As a result, the global constant dsPreviewColumn
has been removed from cm.abstract.dataSymbol
as well as the DsPreviewColumn
itself.
The core version of this column is in cm/core/calc/init/partPreviewColumn.cm
. A renamer has also been added to direct loaded DsPreviewColumn
types to the new PartPreviewColumn
type.
A new constant of the column (called partPreviewColumn
) has been added and is registered in cm/core/init/corePartColumns.cm
.
In cm/abstract/dataSymbol/partColumns/prdPreviewColumn.cm Removed: public class DsPreviewColumn extends BasicPartColumn : inherit constructors {} Removed: public class DsPreviewImageGridCell extends DsImageGridCell {} In cm/core/init/partPreviewColumn.cm and cm/core/calc/previewImageGridCell.cm Added: public class PartPreviewColumn extends BasicPartColumn : inherit constructors {} Added: public class PreviewImageGridCell extends ImageGridCell : inherit constructors {}
Argument str cardProductLevelCode
has been changed to ProductLevel cardLevel
. The default value remains as null
.
Old: public constructor(Window parent, str label, symbol pkg, str name, DataCatalog catalog, ProductCatalog productCatalog, str sortKey=null, bool useInPaperSpace=false, str cardProductLevelCode=null) { New: public constructor(Window parent, str label, symbol pkg, str name|, DataCatalog catalog, ProductCatalog productCatalog, str sortKey=null, bool useInPaperSpace=false, ProductLevel cardLevel=null) {
Removed unused argument Brush brush=null
from the constructor.
Old: public constructor(Window parent, Font font=systemFont(), Brush brush=null, // MergeTODO 16.0 Prism: Remove as this is not used! Brush stdBrush=dsGradientBrush, Brush hoverBrush=dsLightGradientBrush, Brush mouseDownBrush=dsSuperLightGradientBrush, FrameStyle frameStyle=dsRoundedFrame, frame3DState frameState=frameStateDown, pointI pos=(0, 0), sizeI size=(0, 0), pointI margins=(6, 4), alignment align=left, bool popup=false, bool shadow=false, function (Control control) callback=null, bool alwaysMultiSelect=false, bool allowMultiSelect=false, bool directSelect=false, bool listBoxStyle=true, treeViewSelectionStyle selectionStyle=treeViewSelectionStyle.undefined, int minWidth=-1, int autoOpenDelay=500, bool autoGestures=false, Image selectionImage=null, str selectionLabel=null, Brush selectionBrush=null, str key=null, str label=null, color labelColor=color(0, 0, 0), Image image=null, Image disabledImage=null, bool useFocusRectangle=true, bool keepInsideScreen=true) { New: public constructor(Window parent, Font font=systemFont(), Brush stdBrush=dsGradientBrush, Brush hoverBrush=dsLightGradientBrush, Brush mouseDownBrush=dsSuperLightGradientBrush, FrameStyle frameStyle=dsRoundedFrame, frame3DState frameState=frameStateDown, pointI pos=(0, 0), sizeI size=(0, 0), pointI margins=(6, 4), alignment align=left, bool popup=false, bool shadow=false, function (Control control) callback=null, bool alwaysMultiSelect=false, bool allowMultiSelect=false, bool directSelect=false, bool listBoxStyle=true, treeViewSelectionStyle selectionStyle=treeViewSelectionStyle.undefined, int minWidth=-1, int autoOpenDelay=500, bool autoGestures=false, Image selectionImage=null, str selectionLabel=null, Brush selectionBrush=null, str key=null, str label=null, color labelColor=color(0, 0, 0), Image image=null, Image disabledImage=null, bool useFocusRectangle=true, bool keepInsideScreen=true) {
The cosntructor has a new argument buttonType btnType
with a default value of buttonType.undefined
Old: public constructor(Window parent, pointI pos=(0, 0), sizeI size=(0, 0), pointI margins=(0, 0), DsDragAnimationEnv dragAnimationEnv=null, FrameStyle frameStyle=toolboxButtonFrameStyle, frame3DState frameState=frameStateDown, bool facelift=false, SrcRef src=#:src) { New: public constructor(Window parent, pointI pos=(0, 0), sizeI size=(0, 0), pointI margins=(0, 0), DsDragAnimationEnv dragAnimationEnv=null, FrameStyle frameStyle=toolboxButtonFrameStyle, frame3DState frameState=frameStateDown, bool facelift=false, buttonType btnType=buttonType.undefined, SrcRef src=#:src) {
There were some references to legacy frame logic that was removed.
Removed: extend public void paintFrame3D(DsGridWindowPaintArgs args) {
Previously the defaultFrameStyle method applied a WinFrame with thin3DFrameOld
for selected
and label
. This has now been updated to thin3DFrame
.
extend public void defaultFrameStyle(DsFrameStyle style) { style."default" = WinFrame(PenFrame(linePen), frame3DState.frameStateFlat); Old: style."selected" = WinFrame(thin3DFrameOld, frameStateDown); New: style."selected" = WinFrame(thin3DFrame, frameStateDown); Old: style."label" = WinFrame(thin3DFrameOld, frameStateUp); New: style."label" = WinFrame(thin3DFrame, frameStateUp); }
Method appendThumbnails
now passes in dsUILevel lastNonIcon
and is final.
Old: extend public void appendThumbnails(LibraryLimb parent, ProductLevel productLevel) {} New: final public void appendThumbnails(LibraryLimb parent, ProductLevel productLevel, dsUILevel lastNonIcon) {
Method createThumbnail
now passes in a str thumbDesc
.
Old: extend public DsBasicThumbnail createThumbnail(ProductLevel parent, DsPData data) { New: extend public DsBasicThumbnail createThumbnail(ProductLevel parent, DsPData data, str thumbDesc=null) {
The constructor now takes in an extra argument buttonType btnType
with a default value of buttonType.medium
.
Old: public constructor(LibraryLimb parent, symbol pkg, str key, DsBasicThumbnail[] thumbnails=null, LibraryLimbVisibility vs=null, Image image=null, bool frame=true, str label=null, str toolTipText=null, Image toolTipImage=null, int maxRows=3/*-1 == full size*/) { New: public constructor(LibraryLimb parent, symbol pkg, str key, DsBasicThumbnail[] thumbnails=null, LibraryLimbVisibility vs=null, Image image=null, bool frame=true, str label=null, str toolTipText=null, Image toolTipImage=null, int maxRows=3/*-1 == full size*/, buttonType btnType=buttonType.medium) {
This function has been removed. Instead call ApplianceEnv.buildQuickProperties(..) directly.
Removed: public void buildWGAQuickProperty(Snapper snapper, ApplianceEnv applianceEnv, QuickProperties props) {
This function has been removed. Instead call ApplianceEnv.quickPropertiesChanged(..) directly.
Removed: public bool quickPropertyWGAChanged(Snapper snapper, ApplianceEnv applianceEnv, str key, Object value) {
Added a defaut hidden
argument to constructor.
Old: public constructor(..) { New: public constructor(.., bool hidden=false) {
Removed method hidden=
. Instead use constructor.
Removed: extend public bool hidden=(bool v) {
Removed two obsolete constructors. Instead use constructor which covers both obsolete constructors.
Removed: public constructor(str article, str brand=null, str model3DStr=null, str eanCode=null, str extProductSheetLink=null, str cetType=null, size snapperSize = (0, 0, 0), size nicheSize = (0, 0, 0), double depthOffset=0, double heightOffset=0, fanCm3DType fanType=fanCm3DType.undefined, str imageStr=null, symbol creatorPkg=#:package : caller eval, bool hidden=false) { Removed: public constructor(str article, str brand=null, str model3DStr=null, str eanCode=null, str extProductSheetLink=null, str cetType=null, size snapperSize = (0, 0, 0), size nicheSize = (0, 0, 0), double depthOffset=0, double heightOffset=0, fanCm3DType fanType=fanCm3DType.undefined, symbol creatorPkg=#:package : caller eval, bool hidden = false) {
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; }
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) {
This class is now removed.
Removed: public class MhShuttleRowPickUpFrameVessel extends MhRowPickUpFrameVessel {
AbsTagPartColumn
Made AbsTagPartColumn
public for use.
public class AbsTagPartColumn extends BasicPartColumn
SpecOptionInfoTree
has had constructors removed and consolidated into one constructor.
Added: public constructor(Part owner, SpecOption specOpt, PartInfoTree parent=null, str key=null, PartColumn[] columns=null) {} Removed: public constructor(Part owner, SpecOption specOpt) {} Removed: public constructor(Part owner, SpecOption specOpt, rest: PartColumn[] columns) {} Removed: public constructor(Part owner, SpecOption specOpt, PartColumn[] columns) {}
In 15.5 Minor we introduced a new UnitLoadDialog design while still retaining all the logic used for the old design. In 16.0 Major we are now removing all the old logic and will only be maintaining the new design. If you would like to maintain the old design in your extension, you will need to copy over the removed interfaces and logic into your dialog class.
The following fields and methods were only used for the old design and have been removed:
public class UnitLoadDialog extends DialogWindow { //Fields. public TreeView userRegisteredTree; public DropDownTreeView systemRegisteredTree; // Methods. extend public SubWindow buildAddTemplateWin(Window parent) { extend public void buildUserUnitLoadWin(Window sub, Window templateInsert) { extend public void buildApplyButtons() { extend public pointI paneMargin() { extend public void updateUnitLoadSelectorWin() { extend public void updatePreviewWin() { extend public void updatePropsWin() { extend public void updatePanes() { extend public void addSelectedTemplateToUserRegistry(bool refresh) { extend public str propsLabel(UnitLoad unitLoad) { extend public TreeViewItem getSelectedSystemUnitLoad() { }
The following fields and methods were temporarily added in 15.5 Minor to support the new design without removing the old design. They have been removed or modified (renamed) now that the old design is no longer supported:
public class UnitLoadDialog extends DialogWindow { //Fields. Removed: public bool useV2; // Methods. Removed: public constructor(bool useV2, Window parent=null, pointI pos=(-1, -1)) { Removed: extend public void buildUnitLoadSelectorWin2() { Replacement: extend public void buildUnitLoadSelectorWin() { Removed: extend public void buildPropertiesWin2() { Replacement: extend public void buildPropertiesWin() { Removed: extend public void buildPreviewWin2() { Replacement: extend public void buildPreviewWin() { Removed: extend public void renameUnitLoadItem(UnitLoadTreeViewItem2 item, str newName) { Replacement: extend public void renameUnitLoadItem(UnitLoadTreeViewItem item, str newName) { Removed: extend public UnitLoadTreeViewItem2 getSelectedUnitLoadTVI() { Replacement: extend public UnitLoadTreeViewItem getSelectedUnitLoadTVI() {
The logic of the following methods in UnitLoadDialog
have been modified to fit the new design. We have removed the logic used for the old design.
public class UnitLoadDialog extends DialogWindow { extend public void build() { extend public void buildUnitLoadSelectorWin() { extend public void buildPropertiesWin() { extend public void buildPreviewWin() { extend public void unitLoadSelectionChanged() { public void rebuild() { extend public void populatePropsWindow(UnitLoad mtbh) { extend public void removeProps() { extend public void createPropsUI(UnitLoad mtbh) { extend public void animApplyCallback() { extend public void applyAllCallback(UnitLoad[] applyUnitLoads) { extend public void afterPropertyChanged(CoreProperty property, Object oldValue) { extend public void clearModifiedUnitLoads(str->UnitLoad newLoads=null) { extend public UnitLoad getSelectedUserUnitLoad(bool actual=false) { extend public bool selectUserUnitLoad(UnitLoad mtbh) { extend public void validateUserUnitLoadsInUse() { }
Removed the temporary function introduced in 15.5 Minor to choose between using the old or new dialog design.
Removed: /** * Temporary interface to useV2 dialog. FIXME 16.0 remove */ public void showUnitLoadDialog(bool useV2, bool closeIfValid=false, UnitLoad selectUnitLoad=null) { Replacement: /** * Show the current Material to be Handled dialog or * create a new one and show that. */ public void showUnitLoadDialog(bool closeIfValid=false, UnitLoad selectUnitLoad=null) {
Removed the following globals:
Removed: public pointI unitLoadGetLastDialogPos() : deprecated { return lastDialogPos; } Removed: public const str cUnitLoadDialogTreeKey = "mhmtbhDialogTree";
The class UnitLoadTreeViewItem2
has been removed and the current state of UnitLoadTreeViewItem
is now identical to UnitLoadTreeViewItem2
in 15.5 Minor.
Removed: /** * Unit load tree view item. * FIXME 16.0 Rename to UnitLoadTreeViewItem. */ public class UnitLoadTreeViewItem2 extends TreeViewItem : inherit constructors {
Some other details to be mentioned:
registeredUnitLoadTV
instead for the list of unit loads in the drawing.public class UnitLoadDialog extends DialogWindow { Old: public TreeView userRegisteredTree; New: public UnitLoadDropDownTreeView registeredUnitLoadTV; }
systemRegisteredTree
is not used in the new design and has been replaced with a new dialog UnitLoadTemplateSelectorDialog
.public class UnitLoadDialog extends DialogWindow { Old: public DropDownTreeView systemRegisteredTree; New: /** * Open unit load template selector dialog. */ extend public void openTemplateSelectorDialog() { UnitLoadTemplateSelectorDialog(this); } Old: extend public void addSelectedTemplateToUserRegistry(bool refresh) { New: extend public void addSelectedTemplateToUserRegistry(UnitLoad ul, UnitLoadGroup grp) { }
UnitLoadTreeViewItem
into UnitLoadDialog
for the new design. If you have extended UnitLoadTreeViewItem
and modified these features, considering extending these methods in your UnitLoadDialog
class.public class UnitLoadDialog extends DialogWindow { // Methods. extend public bool allowModifyUnitLoad() { extend public bool allowRemoveUnitLoad() { extend public void openRenamePopup() { extend public void renameUnitLoadItem(UnitLoadTreeViewItem item, str newName) { extend public void duplicateUnitLoad() { extend public void removeUnitLoad() { extend public int promptUserRemoveUnitLoad() { }
This class has been greatly simplified. We have removed the custom draw logic, ability to drag and reposition UnitLoadTreeViewItem
objects in the TreeView
, and the custom drawn buttons used to rename/copy/remove the UnitLoadTreeViewItem
object. Below are the removed fields and methods in UnitLoadTreeViewItem
.
public class UnitLoadTreeViewItem extends TreeViewItem : inherit constructors { // Fields. public rectI editButtonRect; public rectI cpyButtonRect; public rectI rmButtonRect; public byte over = 0; // Methods. extend public void drawColoredRect(PixelDevice c, rectI r, rectI clipRect, treeViewItemState state, treeViewSelectionStyle style) { extend public void drawMainArrow(PixelDevice d, rectI r, rectI clipRect) { extend public int btnMult(str btnName) { extend public bool allowModify() { extend public bool allowRemove() { extend public rectI drawBtn(str btnK, Image img, PixelDevice d, rectI r, rectI clipRect) { extend public rectI drawBtn(str btnK, Image img, PixelDevice d, rectI r, rectI clipRect, bool disable) { extend public pointI getBtnPos(str btnK, Image img, rectI r) { extend public Brush backgroundBrush(treeViewItemState state) { extend public bool showSelectedAsWhite(treeViewItemState state, treeViewSelectionStyle style) { extend public ScrollBar vScrollBar(bool ifVisible=false) { extend public void updateApplyButtons(TreeView tv) { extend public void openEditNamePopup(TreeView tv) { extend public void editName(str newName, TreeView tv) { extend public void copyUnitLoadDown(TreeView tv) { extend public void removeUnitLoad(TreeView tv) { extend public void updateTreeView(TreeView tv) { extend public bool isMainSelection() { extend public bool isMultiSelect() {
Removed the following deprecated interfaces
Removed: public void simulateMove(Animation a, line[] moveLines, WindowView view=null) { Removed: public void simulateMove(Animation a, line moveLine, WindowView view=null) { Removed: public void simulateMove(Animation a, point2D[] ps, WindowView view=null) { Removed: public void simulateMove(Animation a, point2D p, WindowView view=null) { Removed: public void simulateClick(Animation a, line[] clickLines, WindowView view=null) { Removed: public void simulateClick(Animation a, line clickLine, WindowView view=null) { Removed: public void simulateClick(Animation a, point2D[] clickPos, WindowView view=null) { Removed: public void simulateClick(Animation a, point2D clickPos=point2D(), WindowView view=null) {
The reason for this is that the Animation argument is redundant, since there is only one active animation at any given moment. As replacement, use the corresponding simulateAnimationMove(..)
or simulateAnimationClicks(..)
function. They work in the same way, with the exception that they always target the active animation (animation()
).
ApplicationManager has been moved to cm.application
. If you get any build/compile errors,you can include the following in your package.cm :
use cm.application;
The following function has been removed, its replacement lives in cm.core.toolbox
:
Old: public Animation getLazyToolboxAnimation(symbol snapperPkg, str snapperName, str c3CatalogId) : deprecated { New: public Animation lazyGetToolboxAnimation(symbol animationPkg, str animationName, AnimationSpawner spawner=null, str c3CatalogId=null) {
We have introduced a new argument bool insidePopupView
with a default value of false
.
Old: public constructor(Window parent) { New: public constructor(Window parent, bool insidePopupView=false) {
This file and all content in it has been removed. It contained an experimental "auto categorization" system that has not been released and is now superseded by a similar system in cm/core/visibility/categorize/ that was released in 15.5 minor.
The following public interfaces were removed:
Removed: public void autoAddUserCategoriesFromToolbar(Object o) { Removed: public class CategorizationToolbar extends ContextualViewToolbar { Removed: public class CATContextualViewToolbarScrollWin extends ContextualViewToolbarScrollWin { Removed: public class CategorizationToolbarModel extends ContextualViewToolbarModel { Removed: public class CategoryRecordToolbarItem extends ToolbarModelItem {
To improve performance we have implemented a cache for picking in 3d. This means the cost of calling objectsAt
multiple times on the same mouseline is greatly reduced. This means we can make the call to objectsAt
when it is guaranteed that the result will be used instead of doing it once and passing the result along to where it perhaps will be used.
Old: extend public void move(View3D view, Object[] objectsAtPos) { New: extend public void move(View3D view, WindowViewMouseInfo mi) {
In this case, the argument objectsAtPos
was computed as follows:
Object[] objects = view.objectsAt(mi.pos, selectionOnTop=true, onlySelectable=true); view.removeObjects(objects, [Class: ViewFocusPoint, DropArea]);
Consider if there are cases when you don't need objectsAt
, e.g., during animations.
Removed interfaces related to the collection of usage and drawing statistics which is now deprecated.
Removed: public void initCoreUsageLog() public CoreUsageSampler coreUsageSampler() public StatsUsageDB coreUsageSamplerDB() public class CoreUsageSampler public bool dbgTestStatisticsReporting=(bool v)
The Snapper mirror functionality uses the space coordinates of the Snapper and the space coordinates of the mirroring plane to calculate the mirrored Snapper position and rotation. However, this was not implemented completely for Child Snappers.
Starting 16.0, Child Snappers will now mirror based on its local coordinates, and uses its parent's symmetry line as the reflecting plane instead. The new strategy ignores the need to have the parent snappers coordinate to be correct, also removing the hassle to translate coordinates to space and back to the child's local coordinates.
To ensure your child snappers work correctly with mirror, make sure the parent snappers have their mirrorPoint()
and angleOfSymmetry()
methods overriden to return the proper values. mirrorPoint()
still return a value in space coordinates for child snappers. Please refer to this article for an explanation of what each method should return - Mirroring Snappers
We added a tool to aid in finding the correct value in Debug Graphics Toolbar : Mirror
If you have implemented fixes for mirroring child snappers in your extension previously, we recommend to remove any code that perform previously unhandled transformations for child snappers, and instead return corresponding values for mirrorPoint()
and angleOfSymmetry()
if the default values do not work for your case.
We also removed some code which do not have any use anymore in MirrorEnv
:
Removed: public int->Point2D oldMirrorPoints(); Removed: public int->Angle oldAnglesOfSymmetry();
Old: extend public void cleanUpSelectionIds(int{} ids) { New: extend public void cleanUpSelectionIds(int{} ids, bool undoable=true) {
The following classes and functions have been removed as they have been superseeded by FrameWindow's ability to automatically save size and position. You can see more details under the Runtime / Behavior Changes for cm.win
.
Removed: public BoundSaver coreBoundSaver() { Removed: public class CoreSingleBoundSaver extends BoundSaver { Removed: public class CoreMultiBoundSaver extends BoundSaver { # CoreAppWindow Removed: final public void saveWindowBound() {
The following functionality has been renamed and moved to cm.application
Old: public bool isPrimaryCoreAppWindow(CoreAppWindow this) { New: public bool isPrimaryAppWindow(AppWindow appWindow) {
As of 16.0, preview image support has been implemented for core Part
s and is no longer exclusive to DsPart
s.
As a result, a new PreviewImageGridCell
type has been made to replace the old DsPreviewImageGridCell
type.
In cm/core/calc/previewImageGridCell.cm Added: public class PreviewImageGridCell extends ImageGridCell : inherit constructors {} In cm/abstract/dataSymbol/partColumns/prdPreviewColumn.cm Removed: public class DsPreviewImageGridCell extends DsImageGridCell {}
Old: extend public void removeDwgSnapper(Snapper candidate, bool silent=false) { New: extend public void removeDwgSnapperAndXrefs(Snapper candidate, bool silent=false) {
Changed to use a TreeView rather than creating subwindows. Many definitions have been changed or moved.
Fields:
Removed: private ScrollableSubWindow mainContainer; Removed: public DwgLayer focusLayer; Removed: public LayerBarSelectButton->DwgLayer barToLayer; Old: public DwgLayer{} selectedLayers; New: extend public DwgLayer{} selectedLayers() {
Methods:
Removed: final public void clearLayerButtonStates() { Removed: final public DwgLayer getLayer(Window window) { Removed: final public void scrollToLayer(DwgLayer layer) { Removed: final public void selectedLayer(DwgLayer z, bool v) { Old: final public void updateLayerStates(bool updateVisibility=true) { New: final public void updateLayerStates() { Old: final public void updateContent(bool clear=true) { New: final public void updateContent() { Old: final public void appendToSelection(DwgLayer z, bool pushdown=false) { New: final public void appendToSelection(DwgLayer z) { Old: final public void setLayerColor(DwgLayer z, color c, bool ignoreUserMaterial=false) { New: final public void setLayerColor(DwgLayer layer, color c, bool ignoreUserMaterial) {
Old: public constructor(Url url, str originalFileName = null, LayerExpr layer=null, int previewSize=125) { New: public constructor(Url url, str originalFileName = null, LayerExpr layer=null, int previewSize=125, bool filled=true) { Old: public constructor(str key, LayerExpr layer=null, int previewSize=125) { New: public constructor(str key, LayerExpr layer=null, int previewSize=125, bool filled=true) {
As of 16.0, preview image support has been implemented for core Part
s and is no longer exclusive to DsPart
s.
As a result, the global constant dsPreviewColumn
has been removed from cm.abstract.dataSymbol
as well as the DsPreviewColumn
itself.
The core version of this column is in cm/core/calc/init/partPreviewColumn.cm
. A renamer has also been added to direct loaded DsPreviewColumn
types to the new PartPreviewColumn
type.
A new constant of the column (called partPreviewColumn
) has been added and is registered in cm/core/init/corePartColumns.cm
.
In cm/abstract/dataSymbol/partColumns/prdPreviewColumn.cm Removed: public class DsPreviewColumn extends BasicPartColumn : inherit constructors {} Removed: public class DsPreviewImageGridCell extends DsImageGridCell {} In cm/core/init/partPreviewColumn.cm and cm/core/calc/previewImageGridCell.cm Added: public class PartPreviewColumn extends BasicPartColumn : inherit constructors {} Added: public class PreviewImageGridCell extends ImageGridCell : inherit constructors {}
LibraryLimb.cm + LibraryLimb Constructor now has an additional argument `SrcRef src`. uiHint.cm + UIImageHint constructor now accepts an additional argument `bool prefer3D=false` + Field `Image image` is now `copy=reference`.
Added argument hideSendToButton to LayoutGroupLimb to hide send to component tab button.
Old: public constructor(LibraryLimb parent, symbol pkg, str key, bool frame=true, function(Window window) layout=null, LibraryLimbVisibility vs=null) New: public constructor(LibraryLimb parent, symbol pkg, str key, bool frame=true, function(Window window) layout=null, LibraryLimbVisibility vs=null, bool hideSendToButton=false)
BasicLibraryHeaderBuilder now supports passing in brandingImage.
Old: public constructor(Image icon, str title, str schemeKey=null) { New: public constructor(Image icon, str title, Image brandingImage=null, str schemeKey=null) {
TabbedLibraryHeaderBuilder now supports passing in brandingImage
Old: public constructor(Image icon, str title, str schemeKey=null, function(Control, int) callback=null) { New: public constructor(Image icon, str title, Image brandingImage=null, str schemeKey=null, function(Control, int) callback=null) { Old: public constructor(Image icon, str title, bool useDropDownStyle=false, str dropDownLabel=$tab # ':', function(Control, int) callback=null) { New: public constructor(Image icon, str title, Image brandingImage=null, bool useDropDownStyle=false, str dropDownLabel=$tab # ':', function(Control, int) callback=null) {
There is a new argument bool toggleStyle
. This argument determines if a checkbox for toggle is shown for BoolCallbackLimb.
Old: public constructor(LibraryLimb parent, symbol pkg, str key, function(bool v) callback, bool initial=false, LibraryLimbVisibility vs=null, Image image=null, str label=null, UIHint hint=null, SrcRef src=#:src) { New: public constructor(LibraryLimb parent, symbol pkg, str key, function(bool v) callback, bool initial=false, LibraryLimbVisibility vs=null, Image image=null, str label=null, UIHint hint=null, bool toggleStyle=false, SrcRef src=#:src) {
This deprecated function did nothing, calls to formatToolboxUI9
can be safely removed.
Removed: public void formatToolboxUI9(Library lib) : deprecated {}
Part
completeSpecifiedValue()
for returning the value of what complete means.New: extend public str completeSpecifiedValue()
Default return values were moved to constants in cm.core.part.cm
.
New: public const str completeSpecYes = "Y"; public const str completeSpecNo = "N";
completedSpec()
for indicating whether there is a completed specification of this part.New: extend public bool completedSpec()
completedSpec()
needs to return true or false in order to get the default complete values 'Y' or 'N' in the Complete column in Calculations. completedSpec()
will also drive the output for Status column and Error Description in Spec PMX export.
The following functions have been added to PartInfoTree
s:
Added: extend public bool containsSpecial() {} Added: extend public PartSpecial special() {} Added: extend public str code() {} Added: extend public str description() {} Added: extend public double price() {} Added: extend public str groupDescription() {}
Over the years, various efforts were made to unify the way texts were shortened. Calls to the deprecated psShortenLabel
should be replaced with shortenText
. Note the position of the first two arguments have swapped.
Old: public str psShortenLabel(Font font, str oriLabel, int allowedW) : deprecated { New: public str shortenText(str text, Font font, int width) {
The deprecated field perspective
has been removed. You can use the method persective() instead.
Old: public bool perspective = true : deprecated; New: extend public bool perspective() {
All of the redRenderSnapshot
functions at the bottom of cm/core/red3D/redRenderSnapperEnv.cm
have received an extra
default parameter (bool retire=true
). It serves as a flag to indicate whether or not to retire the REDThumbnailEnv
after rendering.
The dbg_enableQuickSpaceVolumeChange()
-function has been removed since the "quick space volume change" system has been enabled and stable for many years now. The function always returned true.
Removed: public bool dbg_enableQuickSpaceVolumeChange() {
We have streamlined the interfaces to save a world, moving bool template
argument to be after Url target
.
Old: public bool saveWorld(World world, Url target=null, userSaveAction action=userSaveAction.user, int generation=3, // new in 3.2 bool progress=false, bool searchKeywords=false, function(int progress):bool interrupt=null) { Old: public bool saveWorld(World world, Url target=null, userSaveAction action=userSaveAction.user, int generation=3, // new in 3.2 bool progress=false, bool searchKeywords=false, function(int progress):bool interrupt=null, bool template=false) { New: public bool saveWorld(World world, Url target=null, bool template=false, userSaveAction action=userSaveAction.user, int generation=3, // new in 3.2 bool progress=false, bool searchKeywords=false, function(int progress):bool interrupt=null) {
The following deprecated function has been removed:
Removed: public bool saveCopyOfWorld(World world, Url target=null) : deprecated {
The adjustPicturePos
method has been tweaked to remove the argument sizeI textSize
Old: extend public void adjustPicturePos(sizeI textSize=(-1, -1)) { New: extend public void adjustPicturePos() {
The field current
has its type changed to a more generic type.
Old: public LazyBasicSnapperButton current : deprecated; New: public Control currentButton;
The functionality to addSubButton has been adjusted to accomodate generic button.
Old: extend public void addSubButton(LazyBasicSnapperButton button) : deprecated { New: extend public void addButton(Control button) {
The arguments verticalBar, horizontalBar, handScrollX, handScrollY
have been removed from the constructor.
Old: public constructor(Window parent, bool verticalBar=true, bool horizontalBar=false, // different from parent // inherited key arguments. Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=thick3DFrame, frame3DState frameState=frameStateDown, pointI origin=(1, 1), bool selectable=true, pointI pos=(0, 0), sizeI size=(0, 0), bool passive=false, bool handScrollX=false, // different from parent bool handScrollY=true, bool slimScrollBars=false, bool showStepButtons=false, str scrollPosId=null, SrcRef src=#:src) { New: public constructor(Window parent, Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=thick3DFrame, frame3DState frameState=frameStateDown, pointI origin=(1, 1), bool selectable=true, pointI pos=(0, 0), sizeI size=(0, 0), bool passive=false, bool slimScrollBars=false, bool showStepButtons=false, str scrollPosId=null, SrcRef src=#:src) {
Old: extend public void updateCheckedState(TreeViewItem item = null, bool disabled = false, Object[] picked = null) { New: extend public void updateCheckedState() {
Removed: extend public void setParentCheckState(TriStateComboTreeViewItem item) {
The following field methods in cm.downloader have been removed.
Removed:
public str sendBlockingMessage(str topic, str message, bool unicode, timespan ttl=500ms)
- Changed: public void ultraEndSnooze(ExtensionManager em, ExtensionState state) + Added: public void ultraEndSnooze(ExtensionManager em, ExtensionState state, BasicProgressEnv env=null) {
Old: public bool cpp_rayMeshToleranceIntersect(pointF& rayOrigin, pointF& rayDirection, fix float[] verts, int vc, fix int[] triangles, int tc, TransformF transform, pointF& ip, int& triangle, angle tolerance, pointF& cameraLoc); New: public bool cpp_rayMeshToleranceIntersect(pointF& rayOrigin, pointF& rayDirection, fix float[] verts, int vc, fix int[] triangles, int tc, TransformF transform, pointF& ip, int& triangle, angle tolerance);
Old: public bool rayMeshToleranceIntersect(pointF& rayOrigin, pointF& rayDirection, fix float[] verts, int vc, fix int[] triangles, int tc, TransformF transform, pointF& ip, int& triangle, angle tolerance, pointF& cameraLoc) { New: public bool rayMeshToleranceIntersect(pointF& rayOrigin, pointF& rayDirection, fix float[] verts, int vc, fix int[] triangles, int tc, TransformF transform, pointF& ip, int& triangle, angle tolerance) {
A new argument, o
, has been added to the intersection
function. o
is the origin/owner of the red shape z
and is required for caching purposes.
Old: final public REDPick intersection(REDShape z, lineF ln, REDShape[] path, pointF lineOriginInWC, bool xtrace=false, angle coneAngle=0deg, pointF cameraLoc=pointF(), REDPick[] intersections=null) { New: final public REDPick intersection(Object o, REDShape z, lineF ln, REDShape[] path, pointF lineOriginInWC, bool xtrace=false, angle coneAngle=0deg, pointF cameraLoc=pointF(), REDPick[] intersections=null) {
Usage and drawing statistics are no longer logged and the following interfaces have been removed.
Removed: public class JsonDrawingReport extends JsonStatisticsReport public class JsonUsageReport extends JsonStatisticsReport public class JsonUsageReporterSP extends JsonReporterSP public class StatsUploader public class StatsUsageDB public class UsageSnapshot public class UsageData public void testUsageSaveAndLoad(UsageSnapshot snapshot)
The "blocker" lines concept has been removed and the interfaces below have changed as a result. Note that while many methods had a "blockers" argument, it wasn't actually used for anything.
Removed: extend public line2D[] blockers(FeatureLineSearch fs, point fp, SimpleWallLine{} walls) { Old: extend public void appendCandidates(SimpleWallLine wall, FeatureLineSearch fs, GraphPrimitive{} currentFT, GraphPrimitive{} otherFT, line2D[] blockers) { New: extend public void appendCandidates(SimpleWallLine wall, FeatureLineSearch fs, GraphPrimitive{} currentFT, GraphPrimitive{} otherFT) { Old: extend public void appendPickupPointCandidate(SimpleWallLine wall, SpecialWallFeaturePrimitive current, FeatureLineSearch fs, line2D[] blockers) { New: extend public void appendPickupPointCandidate(SimpleWallLine wall, SpecialWallFeaturePrimitive current, FeatureLineSearch fs) { Old: extend public void appendIntersectionCandidate(SimpleWallLine wall, SpecialWallFeaturePrimitive current, SpecialWallFeaturePrimitive other, FeatureLineSearch fs, line2D[] blockers) { New: extend public void appendIntersectionCandidate(SimpleWallLine wall, SpecialWallFeaturePrimitive current, SpecialWallFeaturePrimitive other, FeatureLineSearch fs) { Old: extend public void appendPerpendicularToOtherCandidate(SimpleWallLine wall, SpecialWallFeaturePrimitive current, SpecialWallFeaturePrimitive other, FeatureLineSearch fs, line2D[] blockers) { New: extend public void appendPerpendicularToOtherCandidate(SimpleWallLine wall, SpecialWallFeaturePrimitive current, SpecialWallFeaturePrimitive other, FeatureLineSearch fs) {
The volatile()
-method has been removed. To get the same behavior, override the status()
-method instead and return testCaseStatus.volatile
.
Removed: extend public bool volatile() {
Replacement example:
Old: public bool volatile() { return true; } New: public testCaseStatus status() { return testCaseStatus.volatile; }
Changed error types from simple strings to a newly introduced TestCaseError
object. As a result, the following two interfaces have changed:
Old: extend public str[] validate(str->Object res) { New: extend public TestCaseError[] validate(str->Object res) { Old: extend public void handleErrors(str[] errors) { New: extend public void handleErrors(TestCaseError[] errors) {
Error messages can still be appended using the appendError(..)
or validate(..)
help functions, the same way as before. But if you had previously manually added any error strings to the blackboard, those would have to be changed from str
to BasicTestCaseError("error message")
. And reading error messages through the new class is done with the message()
function.
The constructors of all TestInstruction's have been changed to include the default argument SrcRef src=#:src
. The reason for this is to improve the debugging of errors by enabling developers to jump directly to the test instruction that caused an error. Be sure to include this default argument in all classes extending TestInstruction
and pass it along to super
.
The following utility functions have been updated due to test case error messages being changed from simple strings to the new object TestCaseError
. The error message is now instead contained in the new class.
Old: public str[] getErrors(str->Object blackboard) { New: public TestCaseError[] getErrors(str->Object blackboard) { Old: public void appendError(str->Object blackboard, str error) { New: public void appendError(str->Object blackboard, str error, SrcRef src=#:src) {
Added a source reference to the constructor in order to know where in the code a TestInstruction instance is constructed. This is used when printing test errors and allows developers to jump directly to the source of the error. Be sure to include the default argument SrcRef src=#:src
in the constructors of all TestInstruction
-subclasses and pass it along to super
.
Old: public constructor() { New: public constructor(SrcRef src=#:src) {
Made the execute
function abstract since it's required to be implemented by subclasses.
Old: extend public void execute(str->Object blackboard) { } New: extend public void execute(str->Object blackboard) : abstract { }
The class has been made abstract as there is no reason to instantiate it by itself.
Old: public class ValidateInstruction extends TestInstruction { New: public class ValidateInstruction extends TestInstruction : abstract {
This class has been removed. As replacement, use a combination of CopyInstruction
, CutInstruction
, PasteInstruction
and PasteSelectionInstruction
. These new instructions more closely mimic the default behaviour of the copy, cut and paste actions in CET.
Removed: class CopySnapperInstruction Added: class CopyInstruction Added: class CutInstruction Added: class PasteInstruction Added: class PasteSelectionInstruction
Since the default copy, cut and paste behaviour use the active selection and animations, you may also want to combine them with SelectSnapperInstruction
and SimulateAnimationClickInstruction
.
All fields have been changed from public
to private
. The reason for this is that all of them can already be assigned via the constructors, and TestInstruction fields are in general not intended to be changeable afterwards.
Old: public Snapper snapper; New: private Snapper snapper; Old: public line mouseLine; New: private line[] mouseLines; Old: public str outputKey; New: private str outputKey; Old: public bool skipSnap; New: private bool skipSnap;
InsertSnapperInstruction now supports simulating multiple clicks when inserting snappers, making it possible to insert things like lines that require both a start pos and end pos. To handle this, the line mouseLine
field has been changed to a sequence, and the following constructors have been added:
Added: public constructor(Snapper snapper, line[] mouseLines, str outputKey=null, bool skipSnap=false) Added: public constructor(SnapperSpawner spawner, line[] mouseLines, str outputKey=null, bool skipSnap=false) Added: public constructor(Snapper snapper, point2D[] pos, str outputKey=null, bool skipSnap=false) Added: public constructor(SnapperSpawner spawner, point2D[] pos, str outputKey=null, bool skipSnap=false)
This class has been removed. Instead use the more general InsertSnapperInstruction.
Removed: class InsertSimpleWallLineInstruction
Example:
Old: InsertSimpleWallLineInstruction((0.0, 0.0, 0.0), (2.0, 0.0, 0.0), outputKey="wall"); New: InsertSnapperInstruction(SimpleWallLine(), [(0.0, 0.0), (2.0, 0.0)], outputKey="wall");
This class has been removed. Instead use the more general InsertSnapperInstruction.
Removed: class InsertWallLineInstruction
Example:
Old: InsertWallLineInstruction((0.0, 0.0, 0.0), (2.0, 0.0, 0.0), outputKey="wall"); New: InsertSnapperInstruction(WallLine(), [(0.0, -0.1), (2.0, -0.1)], outputKey="wall", skipSnap=true);
To set the thickness and height of the wall, use the new PutQuickPropInstruction
.
PutQuickPropInstruction("wall", "d", Distance(0.2.distance)); // Set thickness to 0.2 PutQuickPropInstruction("wall", "h", Distance(1.8.distance), ensureValidProps=true); // Set thickness to 1.8
Please note that the insert position may vary slightly because the InsertSnapperInstruction
utilizes the standard insert animation, whereas InsertWallLineInstruction
did not. Normally, you need to offset the position by the thickness of the wall to end up at the same position as before.
Since InsertWallLineInsruction
did not use an insert animation, it also ignored snapping. To get the same behaviour, set the parameter skipSnap=true
.
This class has been removed and should be replaced with the new SwitchToViewInstruction.
A 1 to 1 replacement would look like this:
SwitchToView2DInstruction()
-> SwitchToViewInstruction(viewOption.splitHorizontal, viewType.view2DType)
But if the test doesn't require a split view, the following is sufficient SwitchToViewInstruction(viewOption.threeD)
This class has been removed and should be replaced with the new SwitchToViewInstruction.
A 1 to 1 replacement would look like this:
SwitchToView3DInstruction()
-> SwitchToViewInstruction(viewOption.splitHorizontal, viewType.view3DType)
But if the test doesn't require a split view, the following is sufficient SwitchToViewInstruction(viewOption.twoD)
This class has been removed. Instead, use GetFieldValueInstruction
followed by ValidateValueInstruction
.
Removed: class ValidateFieldValueInstruction Added: class GetFieldValueInstruction
This class has been removed. Instead, use ToggleFreezeInstruction
.
ToggleFreezeInstruction
freezes or unfreezes items in the active selection (same as using the ctrl+e shortcut). If you need to update the active selection, use SelectSnapperInstruction
.
Removed: class FreezeSnappersInstruction Added: class ToggleFreezeInstruction
This class has been removed. Instead, use ToggleFreezeInstruction
.
ToggleFreezeInstruction
freezes or unfreezes items in the active selection (same as using the ctrl+e shortcut). If you need to update the active selection, use SelectSnapperInstruction
.
Removed: class UnfreezeSnappersInstruction Added: class ToggleFreezeInstruction
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Combined the str snapperKey
and str connectorKey
into one. If the key refers to a connector, the connector will be clicked. If it refers to a snapper, all connectors on that snapper will be clicked.
Old: public constructor(str snapperKey) { New: public constructor(str connectorKey, SrcRef src=#:src) { Removed: public constructor(str snapperKey, str connectorKey, SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str listKey1; New: private public str listKey1; Old: public str listKey2; New: private public str listKey2; Old: public function(Object, Object):bool equalsFunction; New: private public function(Object, Object):bool equalsFunction; Old: public function(Object):str itemToLabelFunction; New: private function(Object):str itemToLabelFunction;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str : public str snapperKey; New: private str : public str snapperKey; Old: public line : public line mouseLine; New: private line : public line mouseLine; Old: public str->Object : public str->Object props New: private str->Object : public str->Object props Old: public bool : public bool skipSnap; New: private bool : public bool skipSnap;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public str outputKey New: private str outputKey Old: public SnapperFilter filter; New: private SnapperFilter filter; Old: public bool recursive; New: private bool recursive;
The class has been renamed from GetConnectorOfClassInstruction
to GetConnectorInstruction
and a new constructor has been added.
Old: public constructor(str snapperKey, Class connectorClass, str outputKey) { New: public constructor(str snapperKey, Class connectorClass, str outputKey, SrcRef src=#:src) { Added: public constructor(str snapperKey, str connectorClassName, str outputKey, SrcRef src=#:src) {
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str objectKey; New: private str objectKey; Old: public str fieldName; New: private str fieldName; Old: public str outputKey; New: private str outputKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str outputKey; New: private str outputKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public bool onlyMain; New: private bool onlyMain; Old: public str outputKey; New: private str outputKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public SnapperFilter filter; New: private SnapperFilter filter; Old: public str outputKey; New: private str outputKey; Old: public rect r; New: private rect r;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public SnapperFilter filter; New: private SnapperFilter filter; Old: public str outputKey; New: private str outputKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str listKey; New: private str listKey; Old: public str outputKey; New: private str outputKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public Url url; New: private Url url; Old: public str outputKey; New: private str outputKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str propKey; New: private str propKey; Old: public Object value; New: private Object value;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public str propKey; New: private str propKey; Old: public Object value; New: private Object value;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public angle rotAngle; New: private angle rotAngle;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public Url url; New: private Url url; Old: public str dataKey; New: private str dataKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str listKey; New: private str listKey; Old: public function(Snapper, Snapper, Object):int sortFunction; New: private function(Snapper, Snapper, Object):int sortFunction; Old: public Object sortEnv; New: private Object sortEnv;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public str propKey; New: private str propKey; Old: public Object{} expectedValues; New: private Object{} expectedValues;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public Vector expectedSize; New: private Vector expectedSize;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str fromSnapperKey; New: private str fromSnapperKey; Old: public str toSnapperKey; New: private str toSnapperKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey;
Added a "expectedValue" argument.
Old: public constructor(str snapperKey) { New: public constructor(str snapperKey, bool expectedValue=true, SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public Int expectedConnections; New: private Int expectedConnections; Old: public SnapperFilter filter; New: private SnapperFilter filter;
This class has been changed to support lists other than sequences and sets of snappers.
Note that due compiler-technical reasons, not all lists are supported. It currently supports Snappers and TreeViewItems. If you need this expanded, please contact Configura Support.
Old: public str snapperKey; New: private str key; Old: public Int expectedListCount; New: private Int expectedListCount; Old: public constructor(str snapperKey, Int expectedListCount) { New: public constructor(str key, Int expectedListCount, SrcRef src=#:src) {
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public Vector expectedSize; New: private Vector expectedSize;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public Int expectedChildCount; New: private Int expectedChildCount; Old: public SnapperFilter filter; New: private SnapperFilter filter; Old: public bool recursive; New: private bool recursive;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str articleCode; New: private str articleCode; Old: public double expectedQuantity; New: private double expectedQuantity;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public str propKey; New: private str propKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str articleCode; New: private str articleCode; Old: public str expectedCode; New: private str expectedCode; Old: public Str expectedDescription; New: private Str expectedDescription; Old: public Int expectedSequence; New: private Int expectedSequence; Old: public Int expectedLevel; New: private Int expectedLevel; Old: public Str expectedGroupDescription; New: private Str expectedGroupDescription; Old: public <str, str> expectedUserDefined; New: private <str, str> expectedUserDefined; Old: public Bool expectedExportable; New: private Bool expectedExportable;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str expectedKey; New: private str expectedKey; Old: public str treeViewItemKey; New: private str treeViewItemKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public Object expected; New: private Object expected; Old: public str itemKey; New: private str itemKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public Orientation expectedRot; New: private Orientation expectedRot;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public str propKey; New: private str propKey; Old: public Object expectedValue; New: private Object expectedValue;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public orientation expectedRot; New: private orientation expectedRot;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public GMaterial3D expectedMaterial; New: private GMaterial3D expectedMaterial;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str snapperKey; New: private str snapperKey; Old: public Point expectedPos; New: private Point expectedPos;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public bool expectedEnabled; New: private bool expectedEnabled; Old: public str itemKey; New: private str itemKey;
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Changed fields to private as they are not intended to be assigned outside of the constructor.
Old: public str propKey; New: private str propKey; Old: public Object expectedValue; New: private Object expectedValue;
cm/win/winPainter.cm - Old: extend public sorted str->str getPainterData() + New: extend public str->Object inspectorData() cm/win/window.cm - Old: extend public sorted str->str getPainterData() + New: extend public str->Object inspectorData()
We have revised argument update
to resize
while adding a new argument refresh
to control repaints. This is a far reaching change, affecting over 88 files in the base repo.
Old: extend public void setImage(Image image, bool update=true) { New: extend public void setImage(Image image, bool resize, bool refresh) {
We have introduced a new argument bool transparentBG
defaulting to false
. This argument allows us to draw scrollbars in an experiemntal way that is transparent.
Old: public constructor(Window parent, bool showStepButtons=false, bool slim=false) { New: public constructor(Window parent, bool showStepButtons=false, bool slim=false, bool transparentBG=false) {
This is also exposed in TreeView's createDoubleScrollBar
method:
Old: extend public DoubleScrollBar createDoubleScrollBar() { New: extend public DoubleScrollBar createDoubleScrollBar(bool transparentBG=false) {
TreeView's constructor also exposes this functionality with the bool transparentScrollBars
argument.
Old: public constructor(Window parent, // inherited key arguments Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=stdLightFrame, frame3DState frameState=frameStateDown, pointI pos=(0, 0), sizeI size=(100, 100), pointI margins=(2, 2), alignment align=middle, bool popup=false, bool shadow=false, function (Control control) callback=null, // extended key arguments bool alwaysMultiSelect=false, bool allowMultiSelect=false, bool directSelect=false, bool listBoxStyle=false, treeViewSelectionStyle selectionStyle=treeViewSelectionStyle.undefined, function (Control control) click2Callback=null, bool noScrollBars=false, str key=null, SrcRef src=#:src) { New: public constructor(Window parent, // inherited key arguments Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=stdLightFrame, frame3DState frameState=frameStateDown, pointI pos=(0, 0), sizeI size=(100, 100), pointI margins=(2, 2), alignment align=middle, bool popup=false, bool shadow=false, function (Control control) callback=null, // extended key arguments bool alwaysMultiSelect=false, bool allowMultiSelect=false, bool directSelect=false, bool listBoxStyle=false, treeViewSelectionStyle selectionStyle=treeViewSelectionStyle.undefined, function (Control control) click2Callback=null, bool transparentScrollBars=false, bool noScrollBars=false, str key=null, SrcRef src=#:src) {
Privatized a member and added setters/getters.
Old: public color _color; New: private color _color; New: extend public color color() { New: extend public void setColor(color c) {
As an ongoing effort of modernizing cm.win
, we have removed a few more key classes in this release.
The following methods have been removed from Window:
Removed: final public void setFrameOLD(frame3DStyle oldStyle, frame3DState state) : deprecated { // OLD Removed: final public void setFrameStyleOLD(frame3DStyle oldStyle) : deprecated { Old: extend public void eraseFrame() { New: extend public void refreshFrame() {
The field frame3DStyle oldStyle
and related methods have been removed from WinFrame:
Removed: public frame3DStyle oldStyle; Removed: final public void setOldStyle(frame3DStyle fstyle) { Removed: final public void drawFrameOLD(PixelDevice d, rectI r) { Removed: final public void invalidateFrame(Window parent, bool rightDownOnly=true) { Removed: final public void invalidateFrame(Window parent, sizeI r, bool rightDownOnly=true) { Removed: final public void drawFramePoly(PixelDevice d, PolylineI poly, color c) { Old: public constructor(frame3DStyle oldStyle, frame3DState state) { New: public constructor(FrameStyle style, frame3DState state) {
With these changes, LegacyFrame has no more purpose and has been removed.
Removed: public class LegacyFrame extends FrameStyle {
Argument bool allowReselect
with a default value of false
has added to the constructor.
Old: public constructor(function(Control c) callback=null, // extends arguments bool exclusiveChoice=true, bool noSelectionPossible=false) { New: public constructor(function(Control c) callback=null, // extends arguments bool exclusiveChoice=true, bool noSelectionPossible=false, bool allowReselect=false) {
Argument bool showUnderline
with a default value of false
has been added to control the underline state.
Old: public constructor(Window parent, Font font=underlineSystemFont, Brush brush=null, FrameStyle frameStyle=noFrame, str label="", color labelColor=color(0, 0, 240), alignment textSide=undefinedAlignment, Image image=null, Image disabledImage=null, color color=nocolor, sizeI size=(-1, -1), pointI margins=(0, 0), alignment align=middle, str link=null, function(Control button) callback=null, SrcRef src=#:src) { New: public constructor(Window parent, Font font=underlineSystemFont, Brush brush=null, FrameStyle frameStyle=noFrame, str label="", color labelColor=color(0, 0, 240), alignment textSide=undefinedAlignment, Image image=null, Image disabledImage=null, color color=nocolor, sizeI size=(-1, -1), pointI margins=(0, 0), alignment align=middle, str link=null, bool showUnderline=false, function(Control button) callback=null, SrcRef src=#:src) {
We have added arguments str key, bool faceliftScrollBars, bool transparentScrollBars
to ScrollableSubWindow's constructor. This allows the caller to specify some additional scroll bar parameters.
Old: public constructor(Window parent, bool verticalBar=true, bool horizontalBar=true, // inherited key arguments. Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=thick3DFrame, frame3DState frameState=frameStateDown, pointI origin=(1, 1), bool selectable=true, pointI pos=(0, 0), sizeI size=(0, 0), bool passive=false, bool handScrollX=true, bool handScrollY=true, bool slimScrollBars=false, bool showStepButtons=false, str scrollPosId=null, SrcRef src=#:src) { New: public constructor(Window parent, bool verticalBar=true, bool horizontalBar=true, // inherited key arguments. Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=thick3DFrame, frame3DState frameState=frameStateDown, pointI origin=(1, 1), bool selectable=true, pointI pos=(0, 0), sizeI size=(0, 0), bool passive=false, str key=null, // extended key arguments. bool handScrollX=true, bool handScrollY=true, bool slimScrollBars=false, bool faceliftScrollBars=false, bool transparentScrollBars=false, bool showStepButtons=false, str scrollPosId=null, SrcRef src=#:src) {
The init scroll methods have also been revised, taking in two additional arguments bool faceliftScrollBars, bool transparentScrollBars
, both defaulting to false
.
Old: extend public ScrollBar initVScroll(sizeI sz, bool showStepButtons) { New: extend public ScrollBar initVScroll(sizeI sz, bool showStepButtons, bool faceliftScrollBars=false, bool transparentScrollBars=false) { Old: extend public ScrollBar initHScroll(sizeI sz, bool showStepButtons) { New: extend public ScrollBar initHScroll(sizeI sz, bool showStepButtons, bool faceliftScrollBars=false, bool transparentScrollBars=false) {
Due to the addition of str key
argument into ScrollableSubWindow, there is no purpose for ScrollableSubWindowWithKey and it has been removed
Removed: public class ScrollableSubWindowWithKey extends ScrollableSubWindow {
Over the years, various efforts were made to unify the way texts were shortened. Calls to the deprecated truncateWithEllipsis
should be replaced with shortenText
.
Old: public str truncateWithEllipsis(str text, Font font, int width) : deprecated { New: public str shortenText(str text, Font font, int width) {
A new argument bool windows11
has been added.
Old: public void configureCustomCaptionWindow(mwnd m, lparam lParam) = win_configureCustomCaptionWindow; New: public void configureCustomCaptionWindow(mwnd m, lparam lParam, bool windows11) = win_configureCustomCaptionWindow;
The following unused fields and methods are removed from ModalDialog
Removed: public Window{} previouslyDisabled; Removed: final public void enablePreviouslyDisabled() {
This dialog is used by cm.core.debug
, so it has been moved to that package.
This class was previously used by Window Inspector to perform double buffered repainting pre 15.0. As of 15.0, TreeView has its own method of double buffered, so this class has been redundant and removed.
Removed: public class PreITreeView extends TreeView {
We have added an argument str loc
defaulting to callerLoc()
to contructors of FrameWindow subclasses. This makes it easier to figure out which the call site of a constructed FrameWindow. Useful to identify the call site of a ModalDialog / message boxes.
# AppWindow Old: public constructor(str label="Configura Virtual Machine", // inherited key arguments Font font=systemFont, Brush brush=null, pointI origin=(0, 0), pointI pos=(0, 0), sizeI size=(300, 200), bool initVisible=true, bool noMinimize=false, bool noMaximize=false, // extended key arguments str name=null, Window appWinAsDialogParent=null) { New: public constructor(str label="Configura Virtual Machine", // inherited key arguments Font font=systemFont, Brush brush=null, pointI origin=(0, 0), pointI pos=(0, 0), sizeI size=(300, 200), bool initVisible=true, bool noMinimize=false, bool noMaximize=false, // extended key arguments str name=null, Window appWinAsDialogParent=null, str loc=callerLoc()) { Old: public constructor(TitleBarConfig tbConfig, str label="Configura Virtual Machine", // inherited key arguments Font font=systemFont, Brush brush=null, pointI origin=(0, 0), pointI pos=(0, 0), sizeI size=(300, 200), bool initVisible=true, // extended key arguments str name=null, Window appWinAsDialogParent=null) { New: public constructor(TitleBarConfig tbConfig, str label="Configura Virtual Machine", // inherited key arguments Font font=systemFont, Brush brush=null, pointI origin=(0, 0), pointI pos=(0, 0), sizeI size=(300, 200), bool initVisible=true, // extended key arguments str name=null, Window appWinAsDialogParent=null, str loc=callerLoc()) { Old: package constructor(bool noBorder, bool noCaptionStyle, bool shadow=false, // Same as above str label="CustomCaption", Font font=systemFont, Brush brush=null, pointI origin=(0, 0), pointI pos=(0, 0), sizeI size=(300, 200), bool initVisible=true, bool noMinimize=false, bool noMaximize=false, str name=null) { New: package constructor(bool noBorder, bool noCaptionStyle, bool shadow=false, // Same as above str label="CustomCaption", Font font=systemFont, Brush brush=null, pointI origin=(0, 0), pointI pos=(0, 0), sizeI size=(300, 200), bool initVisible=true, bool noMinimize=false, bool noMaximize=false, str name=null, str loc=callerLoc()) {
# FormattedTextMessageBox Old: public constructor(Window parent, str header, FormattedText content, Font font=systemFont(), Brush brush=null, pointI origin=(0, 0), bool selectable=true, str label="Message", pointI pos=(-1, -1), sizeI size=(280, 80), bool framed=false, bool initVisible=true, bool readOnly=true) { New: public constructor(Window parent, str header, FormattedText content, Font font=systemFont(), Brush brush=null, pointI origin=(0, 0), bool selectable=true, str label="Message", pointI pos=(-1, -1), sizeI size=(280, 80), bool framed=false, bool initVisible=true, bool readOnly=true, str loc=callerLoc()) { Old: public void makeFormattedTextMessageBox(Window parent, str message, FormattedText content, bool framed=false, bool readOnly=true) { New: public void makeFormattedTextMessageBox(Window parent, str message, FormattedText content, bool framed=false, bool readOnly=true, str loc=callerLoc()) { Old: public void makeFormattedTextMessageBox(Window parent, str label, str message, FormattedText content, bool framed=false, bool readOnly=true) { New: public void makeFormattedTextMessageBox(Window parent, str label, str message, FormattedText content, bool framed=false, bool readOnly=true, str loc=callerLoc()) {
# OneButtonBox Old: public constructor(Window parent, str message, Font font=systemFont(), Brush brush=null, pointI origin=(0, 0), bool selectable=true, str label=$messageDialogCaption, pointI pos=(-1, -1), sizeI size=(280, 80), bool initVisible=true) { New: public constructor(Window parent, str message, Font font=systemFont(), Brush brush=null, pointI origin=(0, 0), bool selectable=true, str label=$messageDialogCaption, pointI pos=(-1, -1), sizeI size=(280, 80), bool initVisible=true, str loc=callerLoc()) {
A few changes have been made to the constructor.
bool transparentScrollBars
defaulting to false
has been addedSrcRef src
defaulting to #:src
has been moved to be the last argument.Old: public constructor(Window parent, // inherited key arguments Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=stdLightFrame, frame3DState frameState=frameStateDown, pointI pos=(0, 0), sizeI size=(100, 100), pointI margins=(2, 2), alignment align=middle, bool popup=false, bool shadow=false, function (Control control) callback=null, bool alwaysMultiSelect=false, bool allowMultiSelect=false, bool directSelect=false, bool listBoxStyle=false, treeViewSelectionStyle selectionStyle=treeViewSelectionStyle.undefined, function (Control control) click2Callback=null, bool noScrollBars=false, str key=null, SrcRef src=#:src|, // extended key arguments function(TreeViewItem, int):Brush itemBackgroundCB=null) { New: public constructor(Window parent, // inherited key arguments Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=stdLightFrame, frame3DState frameState=frameStateDown, pointI pos=(0, 0), sizeI size=(100, 100), pointI margins=(2, 2), alignment align=middle, bool popup=false, bool shadow=false, function (Control control) callback=null, bool alwaysMultiSelect=false, bool allowMultiSelect=false, bool directSelect=false, bool listBoxStyle=false, treeViewSelectionStyle selectionStyle=treeViewSelectionStyle.undefined, function (Control control) click2Callback=null, bool transparentScrollBars=false, bool noScrollBars=false, str key=null|, // extended key arguments function(TreeViewItem, int):Brush itemBackgroundCB=null, SrcRef src=#:src) {
Similar to AxTreeView, a few changes have been made to the constructor.
bool transparentScrollBars
defaulting to false
has been addedSrcRef src
defaulting to #:src
has been moved to be the last argument.Old: public constructor(Window parent, // inherited key arguments Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=stdLightFrame, frame3DState frameState=frameStateDown, pointI pos=(0, 0), sizeI size=(100, 100), pointI margins=(2, 2), alignment align=middle, bool popup=false, bool shadow=false, function (Control control) callback=null, bool alwaysMultiSelect=false, bool allowMultiSelect=false, bool directSelect=false, bool listBoxStyle=false, treeViewSelectionStyle selectionStyle=treeViewSelectionStyle.undefined, function (Control control) click2Callback=null, bool noScrollBars=false, str key=null, SrcRef src=#:src, function(TreeViewItem, int):Brush itemBackgroundCB=null|, // extended key arguments AxColumnsTVIColumnInfos infos=null, AxColumnsTVIColumnHeaders headers=null) { New: public constructor(Window parent, // inherited key arguments Font font=systemFont(), Brush brush=null, FrameStyle frameStyle=stdLightFrame, frame3DState frameState=frameStateDown, pointI pos=(0, 0), sizeI size=(100, 100), pointI margins=(2, 2), alignment align=middle, bool popup=false, bool shadow=false, function (Control control) callback=null, bool alwaysMultiSelect=false, bool allowMultiSelect=false, bool directSelect=false, bool listBoxStyle=false, treeViewSelectionStyle selectionStyle=treeViewSelectionStyle.undefined, function (Control control) click2Callback=null, bool transparentScrollBars=false, bool noScrollBars=false, str key=null, function(TreeViewItem, int):Brush itemBackgroundCB=null|, // extended key arguments AxColumnsTVIColumnInfos infos=null, AxColumnsTVIColumnHeaders headers=null, SrcRef src=#:src) {
Changed the inheritance of DcAssignExtRefKeysDialog in custom.dataCatalog.builder.geometry from ModalDialog to DcBulkAssignDialog.
Old: public class DcAssignExtRefKeysDialog extends ModalDialog New: public class DcAssignExtRefKeysDialog extends DcBulkAssignDialog
The following interfaces have been affected.
The following interfaces in DcAssignExtRefKeysDialog have been changed. Removed fields:
public CatalogTreeView targetPrdTV public DsDropDownTreeView prdFilterDD public DsDropDownTreeView levelDD public DsButtonDropDownTreeView visibilityDD public DsFormattedTextField prdFilterField public DsButton expandAllBtn public DsButton hideAllBtn public DsButton closeBtn public DsGridLayoutHelper layoutHelper
Removed methods:
public bool autoSavePos() public bool autoSaveSize() extend public DcDBBuilderGeometryCard card() extend public DataCatalog catalog() extend public void build() extend public Window buildVisibilitySection(FrameStyle frameStyle=dsRoundedFrame) extend public void buildTargetProductsSource(FrameStyle frameStyle=dsRoundedFrame) extend public void arrangeVisibilitySection() public void setSize(sizeI s) public sizeI maximalSize() public sizeI minimalSize() public void resizeEnd() extend public void populateLevelDD() extend public void populateVisibilityDD() extend public bool modeFeatures() extend public void populateTargetPrds(bool doFilter=false) extend public void updateVisibility() extend public void updateTargetPrdTV(bool doFilter=true) extend public dcCatalogElementType catalogElementType() extend public void updatePrdFilter() extend public void prdFilterChanged() extend public void update(DcDBBuilderUpdateEnv env) extend public void populateFilterDD(DsDropDownTreeView filterDD) extend public filterType[] validFilterTypes() extend public TreeViewItem[] gatherCheckedPrds(TreeViewItem t, Class{} xClass=null) public bool keyCode(Window originator, KeyCode keyCode) extend public void beginOperation(str pdLabel, int taskCount) extend public void incrementTasks(int count=1) extend public bool abortOperation() extend public void endOperation() final public Class{} acceptedCatalogTvi() public void beforeRemove() public void close() public void loadSize(sizeI default) public void loadPos(pointI default) extend public void saveSize(sizeI s) extend public void savePos(pointI p)
Changed methods:
Old: extend public CatalogTreeViewEnv getCatalogTVEnv() New: public CatalogTreeViewEnv getCatalogTVEnv()
Old: extend public void handleEvent(str key, str msg=null, Object value=null) New: public void handleEvent(str key, str msg=null, Object value=null)
Removed the class DcAssignExtRefKeysDialog in custom.dataCatalog.builder.geometry.
Removed:
public class DcAssignEtkCatalogTreeView extends CatalogTreeView
Interface changes in the class DcCatalogItemFilterEnv in custom.dataCatalog.builder.geometry.
Removed:
extend public bool visitChild(Object obj) extend public bool acceptChildAny(Object obj, bool useCached=true, str{} visited=null)
Interface changes in the class DcCatalogItemAndExternalRefKeyFilterEnv in custom.dataCatalog.builder.geometry.
Changed methods:
Old: public bool visitChild(Object obj) New: extend public bool visitChild(Object obj)
Interface changes in the class DcCatalogItemAndMeasurementFilterEnv in custom.dataCatalog.builder.geometry.
Removed:
public bool visitChild(Object obj)
Changed methods:
Old: public bool acceptChildAny(Object obj, bool useCached=true, str{} visited=null) New: public bool acceptNode(Object obj, bool useCached=true, str{} visited=null)
Interface changes in the class DcCopyConnectorsDialog in custom.dataCatalog.builder.geometry.
Removed:
extend public bool accepts(filterType fType, str filterStr, str target)
Interface changes in the class DcMeasuresTreeViewModule in custom.dataCatalog.builder.geometry.
Removed:
extend public <str{}, str{}, str{}, str{}, str{}> gatherAllProductMeasures(DsProductType product=null, <str{}, str{}, str{}, str{}, str{}> measures=null, str{} visitedFeatures=null)
The following field methods in custom.dataCatalog.builder.geometry have been removed.
Removed:
public void dcUpdateAssignExtRefKeysDialog(DcDBBuilderUpdateEnv env)
The following functions related to storing window position have been removed. They have been superseeded by FrameWindow automatic saving of size and position.
Removed: public bool dcCheckSingleScreen(PointI p, Window parent) { Removed: public str dcGetDialogPosKey(Class class) {
This class has been removed. Instead use InsertSnapperInstruction.
Removed: class K2InsertWallLineInstruction
This class has been changed from public to package. Instead use InsertSnapperInstruction.
Old: public class K2InsertWallBumpInstruction extends TestInstruction : deprecated { New: package class K2InsertWallBumpInstruction extends TestInstruction : deprecated {
This class has been changed from public to package. Instead use InsertSnapperInstruction.
Old: public class K2InsertWallWindowInstruction extends TestInstruction { New: package class K2InsertWallWindowInstruction extends TestInstruction {
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Added a default SrcRef
argument to all constructors.
Old: public constructor(..) { New: public constructor(.., SrcRef src=#:src) {
Maps are now sorted on export to sql and xml to ensure that ordering is consistent for tracking in version control.
The following functions on DsPart
account for PartSpecial
information now.
// adds specialFlattenableKey() to tail of key public str flattenableKey() {} // returns PartSpecial price if found public double basePrice() {
As of 16.0, preview image support has been implemented for core Part
s and is no longer exclusive to DsPart
s.
As a result, the following changes have been made to DsPart
:
dsPreviewColumn
has been removed from DsPart
s overridePartColumns()
functionpreviewImageUrl()
function overrides the core Part
version and now has an added default parameterOld: public PartColumn[] overridePartColumns() { return [PartColumn: dsPrdCatalogColumn, dsPreviewColumn, dsTagPartColumn ]; } New: public PartColumn[] overridePartColumns() { return [PartColumn: dsPrdCatalogColumn, dsTagPartColumn ]; } Old: extend public Url previewImageUrl() {} New: public Url previewImageUrl(bool generateIfNotFound=false) {} Added: public Image previewImage(sizeI size=cPartPreviewImageSize, bool generateIfNotFound=false, bool regenerate=false) {} Added: public bool allowRenderPreviewImage() {}
DsPData
has undergone a change in it's public double price(SpecOption[] specOs=null)
function. The function now returns a base price
that does not include SpecOption
prices. As of 16.0, upcharge()
has been implemented in cm/abstract/part/part.cm
and SpecOption
s contain a price value.
customListPrice
now returns the base price plus the upcharge value.
The public void queryDialog(Snapper snppr) {}
function has also been changed with the new QueryDialog
implementation. It no longer opens the old DsQueryDialog
implementation
but utilizes the new QueryDialog
.
The following has been changed in DsPData
:
// price function Old: extend public double price(SpecOption[] specOs=null) { double price; price = basePrice(); if (!specOs) specOs = specOptions().:SpecOption[]; for (DsSpecOption specOpt in specOs) price += specOpt.price; return price; } New: extend public double price(SpecOption[] specOs=null) { return basePrice(); } // queryDialog function extend public void queryDialog(Snapper snppr) { if (proxy) { ... } Old: DsQueryDialog(anyFrameWindow, snppr); New: dsOpenQueryDialog(snppr); }
The code of DsSpecOption is now constructed based on the option code instead of the option valueS to fix a bug where the dropdown selection of options in sub-features of numeric order features is not populated in the calculation dialog.
DsPart
Added optional parameter to the constructor for partSourceId
.
Added new constructor for convenience.
Old: public constructor(Snapper snapper, DsPData data, double price, double qty=1) New: public constructor(Snapper snapper, DsPData data, double price, double qty=1, str partSourceId=null) New: public constructor(Snapper snapper, DsPData data, str articleCode, str description, double listPrice, double quantity=1, double inPrice=0, bool flattenable=true, bool group=false, partStyle style=psNone, str flattenableKey=null, bool allowOtherParent=true, str partSourceId=null)
Removed: private str _itemTagInfoKey;
_itemTagInfoKey
has been moved to ProdPart
in cm.abstract.part
.
User modifications to Ind. Tag are part of the flattenable key now when the user turns that setting on.
ItemTagInfo
Methods for managing Ind. Tags were moved to ProdPart
.
DsSpecOption
price
Removed: public double price;
DsRextileLengthPartColumn
is renamed to DsMaterialQtyPartColumn
MR: Rename Textile Length column to Material Quantity
Previous: public class DsTextileLengthPartColumn extends BasicPartInfoTreeColumn New: public class DsMaterialQtyPartColumn extends BasicPartInfoTreeColumn
With the new QueryDialog
, interface has been added to DataSymbol
to opt-in to utilize the new dialog. DataSymbol
s are defaulted to utilize the new QueryDialog
.
Any subclasses of DataSymbol
must opt-out if the new dialog is not to be utilized.
The acceptQuery
function is checked before adding the query context-menu item to a DataSymbol
and the openQueryDialog
function
is utilized to open a QueryDialog
for a DataSymbol
.
The acceptQuery
function returns true
and openQueryDialog
opens the new QueryDialog
.
The following interface has been added to DataSymbol
:
// QueryDialog helper functions Added: public bool acceptQuery() {} Added: public void openQueryDialog() {}
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; }
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"); } }
Previously the behavior mhShuttleBaySpreadPatternBehavior
was added in gatherBehaviors(MhBehavior[] behaviors)
. It has now been moved to customOtherBehaviors()
instead.
public class MhShuttleBaySpawner extends MhRackBaySpawner { /** * CustomOtherBehaviors. */ public MhBehavior[] customOtherBehaviors() { ... behaviors.exclude(mhBaySpreadPatternBehavior); behaviors << mhShuttleBaySpreadPatternBehavior; return behaviors; } }
AODataPanelJunctionSnapper
Modified createInitialData()
to set a main data via the mainDataKey()
method.
Before: extend public void createInitialData() { putData(createData(partNumber())); } New: extend public void createInitialData() { putData(createData(partNumber()), dataKey=mainDataKey()); }
Updated areas referencing cAOMainData
to reference mainDataKey()
method.
Relavent Merge Request:
DsPart
have been moved to ProdPart
in cm.abstract.part./*********************************************************************** * Item Tags ***********************************************************************/ /** * Create a new item tag */ extend public str itemTagInfoKey=(str k) /** * Item Tag info Key */ public str itemTagInfoKey() /** * Item tag key. */ public str itemTagKey() /** * Item tag info. */ public ItemTagInfo itemTagInfo() /** * Finalize after merge. * Done after dividing parts into groups and merging them, but before the post merge hooks. */ public void finalizeAfterMerge() /** * Re-link any saved ItemTagInfo on the owner that should be shown with this part. */ extend public void reLinkItemTagInfo() /** * Set the item tag info key. */ extend public void setItemTagInfoKey(str key) /** * Reset the ItemTagInfoKey. */ extend public void resetItemTagInfoKey() { _itemTagInfoKey = null; } /** * Set new info to item tag. */ public ItemTagInfo setItemTagInfo(str text) /** * Reset ItemTagInfo. */ public ItemTagInfo resetItemTagInfo()
New: public bool ownerHasStickyTag(Snapper owner) public bool stickyIndTags()
AbsPart
now overrides customListPrice
to include upcharge()
.
/** * Custom (overidable) list price. */ public double customListPrice(bool includeChildren=false, str currency=null, Space space=null) { return super(..) + upcharge().?v; }
AbsTagPartColumn
ItemTag adjustments are now will be set through setItemTagInfo
on cm.core.Part
.
completeSpecified()
Equivalent functionality moved to Part
in cm.core.part
. See completedSpec
and completeSpecifiedValue()
.
MR: Preserve Ind. Tag customizations
While this new setting in CET is activated, any user modifications to Ind. Tag will make those parts unique to that Ind. Tag.
Added all the methods below related to Item Tag from DsPart
have been moved to ProdPart
in cm.abstract.part.
/*********************************************************************** * Item Tags ***********************************************************************/ /** * Create a new item tag */ extend public str itemTagInfoKey=(str k) /** * Item Tag info Key */ public str itemTagInfoKey() /** * Item tag key. */ public str itemTagKey() /** * Item tag info. */ public ItemTagInfo itemTagInfo() /** * Finalize after merge. * Done after dividing parts into groups and merging them, but before the post merge hooks. */ public void finalizeAfterMerge() /** * Re-link any saved ItemTagInfo on the owner that should be shown with this part. */ extend public void reLinkItemTagInfo() /** * Set the item tag info key. */ extend public void setItemTagInfoKey(str key) /** * Reset the ItemTagInfoKey. */ extend public void resetItemTagInfoKey() { _itemTagInfoKey = null; } /** * Set new info to item tag. */ public ItemTagInfo setItemTagInfo(str text) /** * Reset ItemTagInfo. */ public ItemTagInfo resetItemTagInfo()
price
New: public double price;
SpecOption
from PartOptionItem
SpecOption
now extends PartOptionItem
to allow for a singular class hierarchy for PartOption
. This is needed to establish consistent order output via multiple formats.
New: public constructor(str code, str description, str groupDescription=null, int level=1, double price=0.0, bool exportable=true)
PartOptionItem
methods for option output/** * Code */ public str code() { return code; } /** * Group description. (OPTIONAL) */ public str groupDescription() /** * Description string. (OPTIONAL) */ public str description() /** * Level Attribute. (OPTIONAL) */ public int level() /** * Price string. (OPTIONAL) */ public Double optionPrice()
ProdPart
_itemTagInfoKey
to ProdPart
.New: /** * Cached ItemTagInfo key. */ private str _itemTagInfoKey : public readable;
_partSourceID
to ProdPart
Added new constructor with optional parameter partSourceId
added. This new field is intended for use to identify more generically the part source on the owning Snapper
. It should not be set to the Part Number as it will use that by default when not supplied, but something more generic like "leftTableLeg".
New: public constructor(PartData data, str partSourceId=null) New: public constructor(Snapper snapper, str articleCode, str description, double listPrice, double quantity=1, double inPrice=0, bool flattenable=true, bool group=false, partStyle style=psNone, str flattenableKey=null, bool allowOtherParent=true, str partSourceId=null)
_partOptions
to ProdPart
for holding PartOption
items in abstractImplemented methods on ProdPart
as a baseline for PartOption
management, as well as options()
here for overriding as needed.
/*********************************************************************** * Part Option API ***********************************************************************/ /** * Options */ public PartOption[] options() /** * Spec options */ final public SpecOption[] specOptions() /** * Append spec option. * * Note: Ensure if you are building these that level on the * SpecOption is set properly. * */ final public SpecOption[] specOptions=(SpecOption[] newOptions)
User modifications to Ind. Tag are part of the flattenable key.
Instead of saving UnitLoadContainer
in world's auxillary, now we save it in world's cache instead. This is because we dont want to stream UnitLoadContainer
together with the world.
Old: public UserUnitLoadContainer userUnitLoadContainer(World w=null) { if (!w or w.nonDrawing) w = mainWorld(selectWorldIfNull=false); if (!w) return null; UserUnitLoadContainer res = w.getAuxillaryObject(cUnitLoadContainer).UserUnitLoadContainer; if (!res) { res = UserUnitLoadContainer(); w.putAuxillaryObject(cUnitLoadContainer, res); ... New: public UserUnitLoadContainer userUnitLoadContainer(World w=null) { if (!w or w.nonDrawing) w = mainWorld(selectWorldIfNull=false); if (!w) return null; // handle old drawing - old way of keeping contianer if (?UserUnitLoadContainer cont = w.getAuxillaryObject(cUnitLoadContainer)) { w.putCached(cUnitLoadContainer, cont); w.removeAuxillaryObject(cUnitLoadContainer); } UserUnitLoadContainer res = w.getCached(cUnitLoadContainer).UserUnitLoadContainer; if (!res) { res = UserUnitLoadContainer(); w.putCached(cUnitLoadContainer, res); ...
Since now we dont stream UnitLoadContainer
anymore, we need to stream the unit loads inside the container individually in the world. We put the unit loads into auxillary during saveWorldHooks
. Then we remove them in postSaveWorldHooks
so that we dont have double instances of the unit loads in auxillary and container.
New: /** * Append individual uls to aux. */ package void putUlOnAuxHook(World world, userSaveAction action) { for (ul in userUnitLoadContainer(world).unitLoads) { world.putAuxillaryObject(cUnitLoadGrpsKey#";"#ul.gid.hex, ul); } } /** * Remove uls from aux. */ package void removeUlFromAuxHook(World world, userSaveAction action) { for (ul in userUnitLoadContainer(world).unitLoads) { world.removeAuxillaryObject(cUnitLoadGrpsKey#";"#ul.gid.hex); } }
The unit loads will then get appended back to the container during earlyLoadWorldHooks
.
New: /** * Unload individual ul from aux hook. */ package void unloadUlFromAuxHook(World world) { UserUnitLoadContainer cont = userUnitLoadContainer(world); for (k, o in world.allAuxillary) { if (o as MemoryStream) { if (k.beginsWith(cUnitLoadGrpsKey)) { if (?UnitLoad ul = memoryStreamToObject(o)) { str name = userFriendlyUnitLoadName(ul.?name); UnitLoad load = findEqualUnitLoad(ul, world); if (load) { load.gid = ul.gid; } else if (!unitLoadFromName(name, world)) { cont << ul; } } world.removeAuxillaryObject(k); } } } }
Usage and drawing statistics are no longer logged.
Previously, flushDeadPool had a default budget of 100 snappers which incorrectly flushed every snapper on each call. We have now fixed flushDeadPool to correctly consider the budget. To maintain compatibility with the old behavior, we have tweaked the default arguments. It also now validates the snapeprs to be processed by the CET query language with a budget of 1ms.
Old: extend public void flushDeadPool(RemoveSnappersEnv env=null, Int budget=Int(100), bool updateDialogs=true) { New: extend public void flushDeadPool(RemoveSnappersEnv env=null, Int budget=null, bool updateDialogs=true) {
TaggableSnapper
Added setItemTagInfo
method for use with Part
to push ItemTag updates through. This facilitated the ability to mark a Ind. Tag as user modified for the Preserve Ind. Tag customizations setting introduced to user in the Control Panel for Calculations.
New: extend public void setItemTagInfo(Part part, str key, ItemTagInfo info, bool userMod=false)
afterGetParts
was modified to never remove a user modified tag from the tag collection and space.
As of 15.5, a new summation type setting was added to the Summary Control Panel in the Calculations dialog. It allows the user to choose the price type that their summary Sum items display (Sell, Buy, List, Profit).
As a result, SummaryPriceInfo
has been changed in the following ways:
public double profit
field was added to reflect total profitsum
no longer defaults to sell
GlobalPartAdjustmentSum
instance to get the sum priceIn SummaryPriceInfo... Added: public double profit; Added: public constructor(double list, double buy, double sell, double profit, GlobalPartAdjustmentSum summationType) {}
We have fixed the LibraryLimb syntax such that the correct SrcRefs
are now passed correctly to the Control
created. This means that CM Inspector (alt-click) and Window Inspector can now trace the correct SrcRef and open up the correct line in corresponding library.cm
(or any relevant CM source files) of which the LibraryLimb was built.
The following functions on Part
account for PartSpecial
information now.
// adds specialFlattenableKey() to tail of key extend public str flattenableKey() {} // returns PartSpecial part number if found extend public str articleCode() {} // returns PartSpecial description if found extend public str description(Space space=null) {} // returns PartSpecial price if found extend public double customListPrice(bool includeChildren=false, str currency=null, Space space=null) {
ToolboxButtonPainter
now copies the margins of the painter it replaces.ToolboxButtonPainter
, ImagePainter
, and ToolboxResizingImagePainter
to visualize painter bounds.ToolboxResizingImagePainter
will not resize if the MemoryImage is not editable.LazyBasicSnapperButton
are buttons that generate icons automatically from its 2D graphs or 3D models, of both snappers and animations, and they are usually utilized by developers through SnapperLimb
and AnimationLimb
and UIHint
argument prefer3D
. We have done some improvements and bugfixing overall to improve the way they work under New UI.
Note: There's also a relatively new skipRenderImage
param available for SnapperLimb
and AnimationLimb
that will skip the rendering of images from 2D graphs or 3D models, just in case your buttons started showing unwanted renderings.
LazyBasicSnapperButton + extend public str cachedGfxUrl(bool ignoreReplacement=false, bool writable=false) Note: This replaces the dual-use of getSnapperSourceS() for naming local icon-cache file paths. LazyAnimation2DButton = Constructor has added a 'facelift' argument. LazySnapper2DButton = Constructor has added a 'facelift' argument. LazySnapper3DMemoryButton
LazySnapper3DMemoryButton
now uses ToolboxButtonPainter
under New UI mode, this replaces the old LabelPainterEx which did not respect the New UI guidelines and margins. This new painter has a automatic resizing behavior which may impact your icons, please double check the behavior and report any discrepancies.
These buttons automatically generate icons from the 2D graphs of snappers and animations (through SnapperLimb
and AnimationLimb
), however they do not currently work very well depending on your snapper bounds and structure. We have done some bugfixing to bounds & view centering and improvements to caching so that it adhere to the Facelift guidelines, however we discourage the use of this button type over the long term.
SnapperImageButton, DataSnapperImageButton = Constructor has added a 'facelift' argument.
Functions like showToolboxHelp()
used to create help text dialogs has been unified with the implementation in cm.core.ui.showLibHelpText()
, and it is recommended to switch to that function.
We had a few bug reports caused by extensions modifying the colors of LineType
's after they had been placed in a cache.
To prevent this from happening again we added an assert in private LineType getCachedLineType(str key)
running in developMode
that will produce an assertion failure (crash) if you do this in your extension and then attempt to retrieve the LineType
from the cache.
You should never be modifying the members of an already cached LineType. If you run into this error, please copy the LineType
before attempting to modify it, so that you don't update the cached instance.
This function used to take in a reference and altered the object in place. The original rect is no longer altered, only the returned rect will have the shrunken (correct) size.
Old: public rectI shrunken(rectI& this, int x, int y) : inline { New: public rectI shrunken(rectI this, int x, int y) : inline {
Usage and drawing statistics are no longer logged.
The click functionality now more closely mimics what happens when a user clicks a connector in CET, in other words it reads the snapClickAnimation(..)
from the snapper and calls animate(..)
on it.
if (Animation a = c.snapper.snapClickAnimation(c)) { animate(a); simulateAnimationClick(line()); }
Previously, it was written specifically to handle SnapClickAnimation
's and would call executeToggleConnection(..)
directly on them instead of going through the proper animation chain.
if (?SnapClickAnimation clickAnimation = c.clickAnimation()) { Connector attach = c.connection(); if (!attach) { for (a in c.connections()) { attach = a; break; } } if (attach) { clickAnimation.currentSpace.beginUndoStep(); clickAnimation.executeToggleConnection(attach, true); clickAnimation.currentSpace.endUndoStep(); } }
As a result of this change, more of the click animation's code, such as begin()
and end()
, will be executed which can lead to behaviour changes. However, it is not likely to require any code adjustments.
cm/win/control.cm = setLabel(str label, bool update=true) now calls setFont to ensure the correct font is propagated to the winPainter. cm/win/display.cm + Added: final public void setLabelAndTruncate(str label, int maxW, int maxWrap, bool update=true) {
Existing email interfaces uses MAPI which is not currently supported by the new Outlook app. We have added new interfaces that uses .eml files instead:
public bool createEmail(str subject, str text, str[] to, str[] cc=null, Url[] attachments=null, Url path=null, bool open=true) public bool createEmail(str subject, str text, str to, str[] cc=null, Url[] attachments=null, Url path=null, bool open=true) public bool createEmail(str subject, str text, str->Object args, Url path=null, bool open=true) public bool createEmail(str subject, str text, str->Object args, bool open=true)
Improvements have been made to how we store and restore window position and sizes.
To opt-in to this, your dialog has to override these methods:
/** * Auto save position or size? */ public bool autoSavePos() { return true; } public bool autoSaveSize() { return true; }
Previously, most FrameDialogs required to override resizing1
in order for childWindows that are not inside a SubWindow to reposition and resize correctly. This is no longer required, as we have moved some resizing logic from SubWindow
to Window
.
# SubWindow public void ancestorResizing() { Removed: redoLastPosition(); Removed: redoLastSize(); super(); refreshFrame(); } # Window extend public void ancestorResizing() { Added: redoLastPosition(); Added: redoLastSize(); for (c in _childWindows) { if (!c.visible) continue; if (c in PaneContainer) continue; c.ancestorResizing(); } // Second pass to process PaneContainer. for (c in _childWindows) { if (!c.visible) continue; if (c !in PaneContainer) continue; c.ancestorResizing(); } }
We now officially target .NET Framework 4.8 assemblies for CM interop. As there are no version compatibility changes or behavioral changes, there should be no action required from your end. Note that Windows 10 and 11 has shipped with .NET Framework version 4.8 since 2019, and Microsoft officially has no plans to remove it as of September 2024.
We have also resolved an issue with COM interop type lookup when using certain system libraries like System.Net.Http
.
The old UI in CET is now deprecated and enters maintenance support period. This implies the following: