Is there a way to connect a single scale, maybe two or three in time, to Devicehub without having to purchase "Automated Warehouse Operations"? It seems kind of silly that we can’t even connect one.
Will we ever purchase "Automated Warehouse Operations"? Likely in time but we have a long way to go before getting into the Manufacturing and Warehousing aspects of Acumatica and we’d like to be utilizing this in Shipping now.
Is there a compromise or work around anyone’s thought of that works?
Thanks ;-)
Best answer by Gabriel Michaud
@jmckinnon I think DeviceHub is available regardless of your license; that’s all you need to integrate with the scale. The problem is that the standard Shipments screen doesn’t support the scale, but a developer can create a customization that pulls in the current weight from the user’s scale and inputs it in the current row in the Packages tab. Here’s a mockup of what I have in mind:
Do you work with a consultant that is familiar with the Acumatica framework? I don’t have a USB scale to test, but sample code to read the current scale weight can be found in PickPackShip.cs, in the ProcessScaleWeight function (extracted here in case anyone is interested) -- basically it just reads from the SMScale table to get the last received weight, and does some sanity checks in case the weight hasn’t changed in the last 30 seconds:
@jmckinnon I think DeviceHub is available regardless of your license; that’s all you need to integrate with the scale. The problem is that the standard Shipments screen doesn’t support the scale, but a developer can create a customization that pulls in the current weight from the user’s scale and inputs it in the current row in the Packages tab. Here’s a mockup of what I have in mind:
Do you work with a consultant that is familiar with the Acumatica framework? I don’t have a USB scale to test, but sample code to read the current scale weight can be found in PickPackShip.cs, in the ProcessScaleWeight function (extracted here in case anyone is interested) -- basically it just reads from the SMScale table to get the last received weight, and does some sanity checks in case the weight hasn’t changed in the last 30 seconds:
I thought the error was on the client app itself, I’ll do some due diligence and follow-up with some details to be sure and I’ll be crossing my fingers that you’re right.
Regarding the customization…
This is C#, right? I haven’t gotten into that part yet. I see there’s a lot of checking / error trapping going on… guess that’s par for the course lol In addition to this then I suspect I’d also have to focus on the readings from the specific scale, just in case there’s two of them being used. on two different stations.
Thanks again for your guidance! maybe this will serve as my introduction into C#. ;-)
@jmckinnon@Gabriel Michaud - do you know if any company has deployed the customization idea that Gabriel proposed? This could be an extremely useful tool for us to start doing all of our packing and shipping within Acumatica directly.
Is it possible to capture weights from a device-hub connected scale for inventory transactions, e.g. when issuing a raw material ingredient to a Production Order, or receipting a PO quantity? Does this require a customisation too?
@fionabruce Yes, it requires a customization. But a DeviceHub connected scale should work for Production Orders or PO Receipts.
I needed a similar customization as described by @Gabriel Michaud. My company ships within Acumatica, but we don’t use Automated Warehouse Operations for pick/pack/ship. I was able to add a “Get Weight” action button to the Packages tab/grid of the Shipment Entry screen and have it use a Device Hub connected scale (e.g. Mettler Toledo PS60.) I’ve pasted the code below for anyone who wants a head-start.
If Acumatica would add support to DeviceHub for dimensioners... then I'd have something. We have 12 packaging lines and our box sizes are variable, so we are hand-keying W,L,H for every box. It's error prone.
Note: DeviceHub uses an API connection for scales, which appear to stay logged-in. I'm assuming this is for each connected scale. We have 15+ scales and several other connected applications which also rely on API connections (e.g. e-commerce, analytics, etc.)
I don't have all the scales online yet with DeviceHub, but I'm concerned I'm going to create a denial-of-service situation whereby the scales are constraining other higher priority API calls. Seems silly (to me) that Acumatica treats DeviceHub calls against your API limits.
I added a non-persisting boolean usrScaleRegistered field to the packages grid to allow me to (en|dis)able the action button based on if the scale is present for the logged-in user.
publicclass SOShipmentEntry_Extension : PXGraphExtension<SOShipmentEntry>
{
protected SMScale UserScale { get; set; }
publicvirtualint ScaleWeightValiditySeconds => 150;
publicoverridevoidInitialize()
{
base.Initialize();
this.UserScale = GetScale();
}
protectedvirtual SMScale GetScale()
{
// get default scale value from user profilevar userPrefs = UserPreferences.PK.Find(Base, Base.Accessinfo.UserID);
Guid? scaleDeviceID = userPrefs?.DefaultScalesID;
Base.Caches<SMScale>().ClearQueryCache();
SMScale scale = SMScale.PK.Find(Base, scaleDeviceID);
return scale;
}
#region Actions & Buttonspublic PXAction<SOShipment> getWeight;
[PXUIField(DisplayName = "Get Weight", MapViewRights = PXCacheRights.Select, MapEnableRights = PXCacheRights.Select)]
[PXButton(ImageKey = PX.Web.UI.Sprite.Main.DataEntryF, Tooltip = "Get Weight from Scale")]
protectedvirtual IEnumerable GetWeight(PXAdapter adapter)
{
SOPackageDetailEx current = Base.Packages.Current;
if (current != null)
{
// begin sanity checksif (UserScale == null)
{
Base.Document.Ask("Scale", Msg.ScaleMissing, MessageButtons.OK);
}
DateTime now = GetServerTime();
if (UserScale.LastModifiedDateTime.Value.AddHours(1) < now)
{
Base.Document.Ask("Scale", Msg.ScaleDisconnected, MessageButtons.OK);
}
if (UserScale.LastModifiedDateTime.Value.AddSeconds(ScaleWeightValiditySeconds) < now)
{
Base.Document.Ask("Scale", Msg.ScaleTimeout, MessageButtons.OK);
}
elseif (UserScale.LastWeight.GetValueOrDefault() != 0m)
{
decimal weight = Math.Round(ConvertKilogramToWeightUnit(UserScale.LastWeight.GetValueOrDefault(), CommonSetupUOM.Current.WeightUOM), 2);
Base.Packages.Cache.SetValueExt<SOPackageDetail.weight>(current, weight);
Base.Packages.Cache.SetValueExt<SOPackageDetail.confirmed>(current, true);
Base.Packages.Update(current);
}
}
return adapter.Get();
}
#endregion#region EventsprotectedvirtualvoidSOPackageDetailEx_RowSelecting(PXCache sender, PXRowSelectingEventArgs e)
{
SOPackageDetail row = e.Row as SOPackageDetail;
if (row == null) return;
SetScaleRegistered(sender, row);
}
protectedvirtualvoidSOPackageDetailEx_BoxID_FieldUpdated(PXCache sender, PXFieldUpdatedEventArgs e)
{
SOPackageDetail row = e.Row as SOPackageDetail;
if (row == null) return;
SetScaleRegistered(sender, row);
}
#endregion#region UtilityprotectedvoidSetScaleRegistered(PXCache sender, SOPackageDetail package)
{
// until proven otherwisebool scaleRegistered = false;
if (UserScale != null && package.BoxID != null)
{
scaleRegistered = true;
}
// used to (dis|en)able Get Weight action button
sender.SetValueExt(package, "usrScaleRegistered", scaleRegistered);
}
protectedvirtual DateTime GetServerTime()
{
DateTime dbNow;
PXDatabase.SelectDate(out DateTime _, out dbNow);
dbNow = PXTimeZoneInfo.ConvertTimeFromUtc(dbNow, LocaleInfo.GetTimeZone());
return dbNow;
}
protectedvirtualdecimalConvertKilogramToWeightUnit(decimal weight, string weightUnit)
{
weightUnit = weightUnit.Trim().ToUpperInvariant();
decimal conversionFactor =
weightUnit == "KG" ? 1m :
weightUnit == "LB" ? 0.453592m :
thrownew PXException(Msg.BoxWeightWrongUOM, weightUnit);
return weight / conversionFactor;
}
#endregion#region Messages
[PXLocalizable]
publicabstractclass Msg
{
publicconststring ScaleMissing = "The scale is not found in the database.";
publicconststring ScaleDisconnected = "Scale data expired. Check scale connections.";
publicconststring ScaleTimeout = "Scale measurement has expired. Remove the package from the scale and try again.";
publicconststring BoxWeightWrongUOM = "Only KG and LB are supported.";
}
#endregion
}
It seems like the Scales (SM206530) screen is only available with the WMS license feature enabled. Were you able to add the scale for use in DeviceHub without this screen?
We use 3 different kinds of cookies. You can choose which cookies you want to accept. We need basic cookies to make this site work, therefore these are the minimum you can select. Learn more about our cookies.