today I want to write a few words on how to modify behavior of buttons Add task, Add event, Add email, Add activity, ..., Add work item of Business Accounts page.
The main issue of chaning it is in the fact, that it is not just ordinary buttons, but separate class, which has injection of logic. Part of it's declaration goes below:
public class CRActivityList<TPrimaryView> : CRActivityListBase<TPrimaryView, CRPMTimeActivity>
where TPrimaryView : class, IBqlTable, new()
{
public CRActivityList(PXGraph graph)
: base(graph)
{
}
public CRActivityList(PXGraph graph, Delegate handler)
: base(graph, handler)
{
}
}
To my surprise, in order to override it's behavior you'll need to inherit from CRActivityList, and add your logic, for example like this:
public class BusinessAccountMaintExt : PXGraphExtension<BusinessAccountMaint>
{
[PXViewName(Messages.Activities)]
[PXFilterable]
[CRReference(typeof(BAccount.bAccountID), Persistent = true)]
public CRActivityListModified<BAccount> Activities;
}
public class CRActivityListModified<TPrimaryView> : CRActivityList<TPrimaryView>
where TPrimaryView : class, IBqlTable, new()
{
public CRActivityListModified(PXGraph graph)
: base(graph)
{
}
public CRActivityListModified(PXGraph graph, Delegate handler)
: base(graph, handler)
{
}
public override IEnumerable NewTask(PXAdapter adapter)
{
try
{
base.NewTask(adapter); // throws exception, catch it and add you needed logic
}
catch (PXRedirectRequiredException ex)
{
var g = (CRTaskMaint)ex.Graph;
var a = g.Tasks.Current;
a.Subject = "Test Subject";
a = g.Tasks.Update(a);
var timeactivity = g.TimeActivity.Current;
throw;
}
return adapter.Get();
}
public override IEnumerable NewEvent(PXAdapter adapter)
{
try
{
base.NewEvent(adapter);
}
catch (PXRedirectRequiredException ex)
{
var g = (EPEventMaint)ex.Graph;
var a = g.Events.Current;
a.Subject = "Test Subject";
a = g.Events.Update(a);
var timeactivity = g.TimeActivity.Current;
throw;
}
return adapter.Get();
}
public override IEnumerable NewMailActivity(PXAdapter adapter)
{
try
{
base.NewMailActivity(adapter);
}
catch (PXRedirectRequiredException ex)
{
var g = (CREmailActivityMaint)ex.Graph;
var a = g.Message.Current;
a.Subject = "Test Subject";
a = g.Message.Update(a);
var timeactivity = g.TimeActivity.Current;
throw;
}
return adapter.Get();
}
public override IEnumerable NewActivity(PXAdapter adapter)
{
return base.NewActivity(adapter);
}
protected override IEnumerable NewActivityByType(PXAdapter adapter, string type)
{
try
{
base.NewActivityByType(adapter, type);
}
catch (PXRedirectRequiredException ex)
{
var g = (CRActivityMaint)ex.Graph;
var a = g.Activities.Current;
a.Subject = "Test Subject";
a = g.Activities.Update(a);
var timeactivity = g.TimeActivity.Current;
throw;
}
return adapter.Get();
}
}
Few comments to presented code.
- All pop ups appear after exception, so you'll not be able to avoid exceptions, no matter what
- Your addendums should be modified in catch
- throw; will re-throw exception one more time. This is particularly interesitng detail, which you can re-use in some other scenarios when you need to deal with pop ups
Summary
Acumatica is very flexible, and the base flexibility mechanism is inheritance and polymorphism, two pillars of OOP. In case if you need to change behavior of Cases screen, you'll need to inherit-override CRPMTimeActivity class.
today I want to leave a short code sample on how to modify PXIntList or dropdown list in Acumatica. Below goes code sample of it:
protected virtual void _(Events.RowSelected<CROpportunity> e)
{
if (e.Row == null)
return;
var opportunityExtension = e.Row.GetExtension<CROpportunityExt>();
if (opportunityExtension.UsrProduct == 0)
{
var listInts = new List<int>();
var listStrings = new List<String>();
listInts.Add(0);
listInts.Add(1);
listInts.Add(2);
listStrings.Add("String 1");
listStrings.Add("String 2");
listStrings.Add("String 3");
PXIntListAttribute.SetList<CROpportunityExt.usrProposition>(e.Cache, e.Row, listInts.ToArray(), listStrings.ToArray());
}
if (opportunityExtension.UsrProduct == 1)
{
var listInts = new List<int>();
var listStrings = new List<String>();
listInts.Add(0);
listInts.Add(3);
listInts.Add(5);
listStrings.Add("String 2");
listStrings.Add("String 3");
listStrings.Add("String 4");
PXIntListAttribute.SetList<CROpportunityExt.usrProposition>(e.Cache, e.Row, listInts.ToArray(), listStrings.ToArray());
}
}
This code sample has two most important parts:
- RowSelected ( declared over new syntax )
- PXIntListAttribute.SetList<CROpportunityExt.usrProposition> call
With usage of those two principles you can easily get modifiable collection accoding to necessary conditions in your code.
Hello everybody,
today I want to speak about one very interesting feature of FBQL, which I don't know if exists in BQL. Function Brackets!
Take a look on following code sample:
var bracketsDemo = SelectFrom<SOOrder>.InnerJoin<SOLine>.On<SOLine.orderNbr.IsEqual<SOOrder.orderNbr>>.InnerJoin<SOShipLine>
.On<SOShipLine.origOrderNbr.IsEqual<SOOrder.orderNbr>>.Where<
Brackets<SOShipLine.confirmed.IsNotNull.
And<SOShipLine.baseOrigOrderQty.IsNotNull>.
And<SOShipLine.completeQtyMin.IsNotNull>.
And<SOShipLine.confirmed.IsEqual<True>.
Or<SOShipLine.confirmed.IsNull>>.
And<SOShipLine.baseOriginalShippedQty.IsGreater<SOShipLine.unassignedQty>.
Or<SOShipLine.baseOrigOrderQty.IsLess<SOShipLine.baseOriginalShippedQty>>>>.
Or<SOShipLine.baseOrigOrderQty.IsNotNull>>.AggregateTo<GroupBy<SOOrder.orderNbr>,
Min<SOOrder.curyDiscTot>>.OrderBy<SOOrder.curyFreightAmt.Asc, SOOrder.curyDocDisc.Desc>.View.Select(Base);
as you can see from the code, now in FBQL if you need to have Or operator, then as anohter option you may use Brackets.
Hello everybody,
today I want to leave a short note on how to pass some additional parameters into Processing method of processing page with help of lambda expression. Syntax is pretty simple:
public class SSShipmentDateResetter : PXGraph<SSShipmentDateResetter>
{
public PXFilter<ShipmentFilter> ShipmentFilter;
public PXFilteredProcessing<SOShipment, ShipmentFilter> ShipmentsForProcessing;
public SSShipmentDateResetter()
{
var shipmentFilter = ShipmentFilter.Current;
ShipmentsForProcessing.SetProcessDelegate(shipments =>
{
ResetDate(shipmentFilter, shipments);
});
}
public static void ResetDate(ShipmentFilter filter, List<SOShipment> shipments)
{
}
}
for me such approach is a bit easier to use as it involes a bit less amout of typing.
If you want to have even simpler syntax, then following sample also may work for you:
public SSShipmentDateResetter()
{
var filter = ShipmentFilter.Current;
ShipmentsForProcessing.SetProcessDelegate(shipments => ResetDate(filter, shipments));
}
For simplicity sake, I've showed only constructor.
today I want to leave a post on how to restore Acumatica snapshots with help of Acumatica Wizard.
Before creating snapshot you may need to switch Acumatica to maintenance mode. Below are the steps for achieveing it:
1. Go to System --> Management --> Apply Updates, click on the button "Schedule Lockout" and specify the reason for lockout
2. Don't forget to remove the lockout after you'll restore Snapshot.
Steps are the following:
1. Create a snapshot of the tenant you wish to back up
2. Export that snapshot (I recoment xml format).
3. In the folder where the corresponding Acumatica Wizard is installed (usually c:\program files\Acumatica ERP\), find the folder called \Database\Data, where there are individual folders with snapshot data.
4. Create a new folder for your new snapshot
5. Extract the contents of the .zip exported snapshot into the new folder you created
6. Run the Acumatica Wizard
7. Choose the "Perform Application Maintenance" option
8. Select the site into which you wish to import this data
9. Choose the "Tenant Maintenance" option or "Company Maintenance":
10. Click on the "Advanced Settings" button at the bottom
12. Click on the "Insert Data" option next to the tenant where the data should be imported and select the new folder/snapshot you added in step #4 above:
13. Proceed through the wizard as normal. When the install process is complete you should have a tenant with restored data.
Hello everybody,
today I want to write a few words about changes of PXUIFieldAttribute.SetVisibility method.
In the past, if you wanted to turn on some column, you could write in RowSelected event something like this:
PXUIFieldAttribute.SetVisibility<APRegister.finPeriodID>(Documents.Cache, null, true);
If you want to turn it off, then this:
PXUIFieldAttribute.SetVisibility<APRegister.finPeriodID>(Documents.Cache, null, false);
But for quite a long time ( even in 2017 version ), Acumatica team introduced PXUIVisibility enum. In code it looks like this:
[Flags]
public enum PXUIVisibility
{
Undefined = 0,
Invisible = 1,
Visible = 3,
SelectorVisible = 7,
Dynamic = 9,
Service = 19,
HiddenByAccessRights = 33,
}
For now I can comment on Visible and InVisible which correspond to false and true in the past implemtations. And in future posts I'll describe other selectors.