Hi Everyone,
Today I want to share with you one way how you can keep your code simpler.
Lets assume you have some big data view declaration – PXSelect<Table, <InnerJoin, …. <AnotherJoin, … <MoreJoins, …. <Where<, …. <MoreWheres< …. SomeDataView
And something like this for 10-20-30 lines of code. I’m pretty sure that you can understand what do i mean.
And also this select declaration may be inside Acumatica base code and you want to customize it.
Your task is to create a new data view delegate and select the same data with some additional filtering or dynamic calculations. But we cannot just create delegate and call dataview data, as we will have stack-overflow exception.
First idea to resolve it is to just copy the PXSelect statement and put it into your delegate. But this idea is not perfect, becouse:
- You will store BQL command twice, and if you need to modify it, you have to do it twice
- If this statement is from Acumatica source code you have a risk that Acumatica will update this command in the future, but your copy will be untouched.
So to resolve this problem we can use PXView declaration inside the data view delegate with reference to the original BQL Statement.
public IEnumerable dataView() { PXView select = new PXView(this, true, DataView.View.BqlSelect); Int32 totalrow = 0; Int32 startrow = PXView.StartRow; List<object> result = select.Select(PXView.Currents, PXView.Parameters, PXView.Searches, PXView.SortColumns, PXView.Descendings, PXView.Filters, ref startrow, PXView.MaximumRows, ref totalrow); PXView.StartRow = 0; foreach (PXResult<Contract, ContractBillingSchedule, CSAnswers> row in result) { //Do any dynamic calculations } return result; }
Some important points here:
- DataView.View.BqlSelect – using this construction you can get access to the BQL command that lies under the PXSelect<>.
- Int32 startrow = PXView.StartRow – is required to support proper paging. By default Acumatica will select just some rows that can be shown on user interface. Other rows should be skipped for better performance.
- PXView.MaximumRows – will be exactly top xx records that will be selected from database.
- PXView.StartRow = 0; – required to notify base code that you have already filtered data and select required records count. Otherwise base code will trim records one more time, because by default system thinks that you will select all records and return untrimmed list.
Full code snippet:
Have a nice Development!
Hello,
I cant create new records it returns the first record. it is possible that I can create new record if I use PXView ?
You can return not existing records using view like here: https://asiablog.acumatica.com/2015/09/using-virtual-data-access-class.html
But you also can use cache.Insert to insert new records if needed.
Hi Sergey,
Will this approach work for the data views which already have delegates with additional filtering?
For example, the activity view of the EmployeeActivitiesEntry.
I need to ignore the filtering by filterRow.OwnerID for some cases and add filtering by Customer.
Hi Samvel, as you create a new view, your view will replace the original including delegate.
You actually can use 4th parameter in the PXView constructor to specify delegate, but please note, that delegates usually create own query, so if you modify something in the view with delegate, your modifications will be replaced by the delegate. So basically it may not do what you want.