Formular – addCheckboxList – prazdne values v onSuccess
- Creator13
- Člen | 18
Ahojte, mam problem so spracovanim checkboxov. V onSuccess dostavam prazdne $values. Ked pozriem $form->getHttpData(), tak sa tam pole s oznacenymi checkboxami nachadza.
/**
* @param array<int, OrderProductEntity> $products
* @return Form
*/
public function create(array $products): Form
{
$form = $this->baseFormFactory->createDefaultForm();
$checkboxListData = $this->getCheckboxListData($products);
$form->addCheckboxList('products', 'Produkty', $checkboxListData)
->setHtmlAttribute('class', 'form-check-input w20 h20 order-create-control-selected-product')
->addRule($form::MinLength, 'Musíte vybrať aspoň %d položku', 1);
$form->addSubmit('send', 'Vložiť do objednávky')
->setHtmlAttribute('class', 'add-to-print-order btn btn-primary float-end button-big-1');
$form->onSuccess[] = [$this, 'onSuccess'];
return $form;
}
public function onSuccess(Form $form, $values): void
{
bdump($values);
bdump($form->getHttpData());
if (isset($form->getHttpData()['products'])) {
$products = $form->getHttpData()['products'];
$this->session->getSection('order')->products = $products;
}
}
Takisto mam problem s touto kontrolou. Funguje mi az ked zadam ako podmienku 2. Ak neoznacim ziadny checkbox, tak mi tuto podmienku neberie do uvahy.
->addRule($form::MinLength, 'Musíte vybrať aspoň %d položku', 1);
- m.brecher
- Generous Backer | 873
@Creator13
Takisto mam problem s touto kontrolou. Funguje mi az ked zadam ako podmienku 2. Ak neoznacim ziadny checkbox, tak mi tuto podmienku neberie do uvahy.
Podmínky přidané addRule() se validují až když prvek obsahuje nějaká data. Aby Ti validace fungovala i pro prázdný prvek, musíš přidat setRequired():
$form->addCheckboxList('products', 'Produkty', $checkboxListData)
->setHtmlAttribute('class', 'form-check-input w20 h20 order-create-control-selected-product')
->setRequired('Musíte vybrať aspoň 1 položku')
->addRule($form::MinLength, 'Musíte vybrať aspoň %d položku', 1);
V onSuccess dostavam prazdne $values. Ked pozriem $form->getHttpData(), tak sa tam pole s oznacenymi checkboxami nachadza.
Nevalidní data formulář do $values nepustí – může to mít souvislost s chybějícím setRequired() jak jsem uvedl výše, oprav validaci a uvidíš.
Editoval m.brecher (23. 8. 2024 13:11)
- Kamil Valenta
- Člen | 822
m.brecher napsal(a):
Nevalidní data formulář do $values nepustí – může to mít souvislost s chybějícím setRequired() jak jsem uvedl výše, oprav validaci a uvidíš.
Pokud je vidí ve $form->getHttpData(), tak je vyplnil a pak setRequired() změnu nepřinese.
Zaměřil bych se na $this->getCheckboxListData(), zda vrací stejnou množinu dat při vykreslení formuláře i při jeho odeslání. Tipuji, ale kontext není uveden, že array $products do create() se předá jen při vykreslení formu, při handlu už ne, getCheckboxListData() vrátí [] a proto všechny zvolené checkboxy budou zahozeny jako nevalidní…
- Creator13
- Člen | 18
Podarilo sa mi to vyriesit
Po odoslani formulara mi nadradena komponenta predavala prazdne pole $products
public function create(array $products): Form
Nadradena komponenta, z ktorej form berie $products.
Presunul som z render() do __construct
class CreatePrintOrderStep1Control extends Control
{
use OrderActionPanelControlTrait;
use PrintOrderCreateStep1FormControlTrait;
private ?int $userId = null;
/**
* @var array<int, OrderProductEntity>
*/
private array $products = [];
public function __construct(
private readonly UserDataProvider $userDataProvider,
private readonly OrderActionPanelControlFactory $orderActionPanelControlFactory,
private readonly OrderProductDataProvider $orderProductDataProvider,
private readonly PrintTechnologyDataProvider $printTechnologyDataProvider,
private readonly ArrayService $arrayService,
private readonly PrintOrderCreateStep1FormControlFactory $printOrderCreateStep1FormControlFactory,
)
{
$this->initialize();
}
public function initialize(): void
{
$userId = $this->userDataProvider->getUserId();
$products = $this->orderProductDataProvider->getProductsForCreatingPrintOrder($userId, true);
//products for form
$this->products = $products;
}
public function render(): void
{
$this->getTemplate()->products = $this->products;
$this->getTemplate()->setFile(__DIR__ . '/templates/createPrintOrderStep1.latte');
$this->getTemplate()->render();
}
Je toto riesenie korektne? Ma komponenta ine moznosti pre inicializaciu a kontrolu, napr. ako ma presenter startup, action?
- m.brecher
- Generous Backer | 873
@Creator13
Je toto riesenie korektne? Ma komponenta ine moznosti pre inicializaciu a kontrolu, napr. ako ma presenter startup, action?
Úplně čisté řešení to není, protože v konstruktoru by bylo lepší inicializaci nemít, ale bohužel komponentový model Nette nemá vestavěný životní cyklus pro komponenty poděděné z UI\Control, což by se hodilo a využilo třeba zrovna v tomto případě.
O něco čistší než dávat inicializaci komponenty do konstruktoru komponenty je volat inicializaci v metodě createComponent<*>() a to tak, aby fáze vytváření komponenty proběhly takto:
- vytvoření instance komponenty
- inicializace komponenty
- připojení hotové komponenty do komponentového modelu presenteru
- Kamil Valenta
- Člen | 822
m.brecher napsal(a):
Úplně čisté řešení to není, protože v konstruktoru by bylo lepší inicializaci nemít
Proč? Konstruktor je k tomu, aby inicializoval objekt se vším, co ke své existenci potřebuje.
O něco čistší než dávat inicializaci komponenty do konstruktoru komponenty je volat inicializaci v metodě createComponent<*>()
Někdy asi ano, pokud komponenta vykresluje obecná data, která pochází
z venku (i když pak bych spíš očekával setter).
Pokud ale komponenta vykresluje data dodaná modelem (z jednoho místa),
nedává mi smysl opakovat se s každým createComponent(), když to může
být v konstruktoru na jednom místě.
- m.brecher
- Generous Backer | 873
@KamilValenta
Proč? Konstruktor je k tomu, aby inicializoval objekt se vším, co ke své existenci potřebuje.
S tím bych úplně nesouhlasil, záleží na situaci, často bývá přehlednější mít dvě fáze vytvoření a inicializace objektu – třeba create a setup. Nette UI\Control ale životní cyklus nemá – to je realita, potom umístění inicializace do konstruktoru je dobrá volba – aby se předešlo nevýhodám spojeným s opakovaným psaním stejného kódu v createComponent<*> metodách.
Já používám pro formuláře vlastní abstraktní předek odvozený z UI\Control, kde jsem mimo jiné přidal jednoduchý životní cyklus setup/beforeRender a jsem s tím velmi spokojený.
Editoval m.brecher (5. 9. 2024 14:17)
- Martk
- Člen | 661
Dejme tomu, že máš 2 akce a 4 handle. Nyní se při všech 6 akcích provede dotaz do databáze (v 5 případech zbytečně, možná někdy ve všech 6 zbytečně) i když vůbec nesaháš na komponentu. Můžeš taky udělat lazy getter:
private function getProducts() {
return $this->products ??= $this->orderProductDataProvider->getProductsForCreatingPrintOrder($userId, true);
}
a ušetříš dotazy do databáze. Pokud chceš jít cestou
Konstruktor je k tomu, aby inicializoval objekt se vším, co ke své existenci potřebuje.
Tak products předávej z presenteru a udělej factory na komponentu, jen on
ví, kdy se komponenta opravdu použije.
- Kamil Valenta
- Člen | 822
Martk napsal(a):
Dejme tomu, že máš 2 akce a 4 handle. Nyní se při všech 6 akcích provede dotaz do databáze (v 5 případech zbytečně, možná někdy ve všech 6 zbytečně) i když vůbec nesaháš na komponentu.
Pokud je $products nepovinné pro fungování komponenty, tak bych to do ní cpal setterem a tahal getterem (zda lazy nebo ne není aktuálně předmětem diskuze). Což jsem ale popsal už prvně.
Pokud je $products nevyhnutelně potřeba pro existenci komponenty, připadá mi legitimní chtít to v konstruktoru a instancování objektu bez toho ani nepustit.
Tak products předávej z presenteru a udělej factory na komponentu
Někdy ano, pokud jsou data závislá ještě na něčem z presenteru. Ale pokud jsou fixně tahána z modelu, není důvod mít to na 10 místech (pokud komponentu používám 10x), to už jsem ale také psal prvně.
jen on ví, kdy se komponenta opravdu použije.
Komponenta to ví zrovna tak.
Editoval Kamil Valenta (6. 9. 2024 8:18)
- Martk
- Člen | 661
Máte pravdu, když jsem to psal, tak jsem měl asi slabší chvilku a nebral jsem ze záhadných důvodů v potaz factory komponenty.
@MajklNajt Když chceš tahat produkty z jiného zdroje a nebo chceš nějak filtrovat produkty, tak v tomto případě musíš napsat novou komponentu a nemůžeš použít stávající, když trváš na tomto přístupu. Píšu z vlastní zkušenosti, kdy jsem musel nejednou přepisovat komponentu a ano, dělám to i tak, že si tahám data v komponentě, protože jsem líný. Zároveň když ty produkty potřebuješ na 3 různých místech, tak se nevolá 3× zbytečný dotaz do databáze na to samé. To jen jako doplnění.
- MajklNajt
- Člen | 501
@Martk to je ale iný prípad, diskusia sa týkala inicializácie komponenty v konstruktore… tvoj use-case samozrejme je logicky správny, a je možné ho zrealizovať tak, že si tú kolekciu produktov vyžiadam v konstruktore… pretože ak je komponenta na nich závislá, konstruktor je jediné správne miesto, kde si ich vypýtať