Depois de ler e depurar os arquivos principais do Magento 2 , encontrei uma solução limpa e simples sobre esse problema. A transmissão de dados de um formulário personalizado para uma grade personalizada usando o UIComponent insertListing é realmente difícil e não está documentada.
O objeto InsertListing possui dois parâmetros na tag: exportações e importações que usei na minha listagem:
<fieldset name="relatedto" >
<settings>
<label>Related to</label>
<componentType>fieldset</componentType>
</settings>
<insertListing name="threadrelated_listing">
<settings>
<dataLinks>
<exports>false</exports>
<imports>true</imports>
</dataLinks>
<externalProvider>mycompany_helpdesk_threadrelated_listing.mycompany_helpdesk_threadrelated_listing_data_source</externalProvider>
<selectionsProvider>mycompany_helpdesk_threadrelated_listing.mycompany_helpdesk_threadrelated_listing.mycompany_helpdesk_threadrelated_columns.ids</selectionsProvider>
<autoRender>true</autoRender>
<dataScope>mycompany_helpdesk_threadrelated_listing</dataScope>
<ns>mycompany_helpdesk_threadrelated_listing</ns>
<exports>
<link name="ticket_id">${ $.externalProvider }:params.ticket_id</link>
</exports>
<imports>
<link name="ticket_id">${ $.provider }:data.ticket_id</link>
</imports>
</settings>
</insertListing>
</fieldset>
e depois de horas para entender e encontrar uma solução na web, não encontrei nenhuma pista!
Então, eu li o arquivo Magento Core e descobri que o Magento combina a maneira de criar as grades de listagem aninhadas no projeto. Às vezes, ele usa o antigo método de inserção de bloco e algumas vezes o novo método de listagem UIComponent.
Encontrei a grade de listagem de endereços do cliente em customer_address_listing.xml (/vendor/magento/module-customer/view/adminhtml/ui_component/customer_address_listing.xml) e obtém a variável parent_id definida em customer_form.xml (/ vendor / magento /module-customer/view/base/ui_component/customer_form.xml), mas a pergunta é:
Como o Magento passa os dados do formulário para a grade de listagem aninhada?
Magento passa os dados pelo QUERYSTRING PARAMETER!
Se você ler o arquivo DataProvider.php, ficará surpreso porque ele obtém a variável parent_id (customer) pela QUERYSTRING! Veja /vendor/magento/module-customer/Ui/Component/Listing/Address/DataProvider.php linha 58:
/**
* Add country key for default billing/shipping blocks on customer addresses tab
*
* @return array
*/
public function getData(): array
{
$collection = $this->getCollection();
$data['items'] = [];
if ($this->request->getParam('parent_id')) {
$collection->addFieldToFilter('parent_id', $this->request->getParam('parent_id'));
$data = $collection->toArray();
}
foreach ($data['items'] as $key => $item) {
if (isset($item['country_id']) && !isset($item['country'])) {
$data['items'][$key]['country'] = $this->countryDirectory->loadByCode($item['country_id'])->getName();
}
}
return $data;
}
mas como faço para definir o parâmetro no URL da lista de listagem? Eu encontrei o parâmetro filterUrlParams, mas há um problema estranho também aqui! Vamos dar uma olhada neste trecho de código dataSource:
<dataSource name="mycompany_helpdesk_threadrelated_listing_data_source" component="Magento_Ui/js/grid/provider">
<settings>
<filterUrlParams>
<param name="ticket_id">*</param>
</filterUrlParams>
<storageConfig>
<param name="indexField" xsi:type="string">threadrelated_id</param>
</storageConfig>
<updateUrl path="mui/index/render"/>
</settings>
<dataProvider class="mycompany\Helpdesk\Ui\DataProvider\Threadrelated\ThreadRelatedDataProvider" name="mycompany_helpdesk_threadrelated_listing_data_source">
<settings>
<requestFieldName>id</requestFieldName>
<primaryFieldName>threadrelated_id</primaryFieldName>
</settings>
</dataProvider>
</dataSource>
Eu configurei o ticket_id com um curinga (*) que significa: obtenha todos os tickets! mas se você não definir nenhum ID no filterUrlParams, o URL insertListing NÃO TEM QUALQUER ticket_id SET! Então por que?!
A solução oferecida pelo @ hashish-raj não funciona para mim.
Estes são todos os posts que li sobre esse problema:
No final, encontrei uma solução temporária usando a sessão principal e armazene o parâmetro ticket_id na sessão. Em seguida, no provedor de dados personalizado, verifiquei e apliquei na coleção:
/***
* @return array
*/
public function getData()
{
$collection = $this->getSearchResult();
/** see: check Mycompany\Helpdesk\Controller\Adminhtml\Ticket\Edit **/
if($this->coreSession->getTicketId()){
$collection->addFieldToFilter('ticket_id', ['eq' => $this->coreSession->getTicketId()]);
}
return $this->searchResultToOutput($collection);
}
Se você tem uma solução alternativa ou entende como o Magento lida com esse relacionamento entre o UIComponent , compartilhe seu conhecimento!
Abri uma "recompensa" aqui: https://magento.stackexchange.com/a/306537/2004