Deklarace persistentních parametrů a komponent bez anotací
- David Grudl
- Nette Core | 8239
Odedneška lze deklarovat persistentní parametry a komponenty bez použití anotací. Slouží k tomu dvě nové statické metody getPersistentParams() a getPersistentComponents(). Každá vrací pole, jehož klíče nebo hodnoty tvoří jména parametrů resp. komponent.
Protože statické metody v PHP < 5.3 jsou opruz, dostávají tyto metody jako parametr jméno třídy. Také není potřeba, aby vracely seznam parametrů nebo komponent svých předků.
Příklad:
// pomocí anotací:
class FifteenControl extends /*Nette\Application\*/Control
{
/** @persistent */
public $order;
/** @persistent */
public $round = 0;
}
// pomocí metody getPersistentParams
class FifteenControl extends /*Nette\Application\*/Control
{
public $order;
public $round = 0;
public static function getPersistentParams()
{
return array('order', 'round');
}
}
A persistentní komponenty:
// pomocí anotací:
/**
* @persistent(game)
*/
class DefaultPresenter extends /*Nette\Application\*/Presenter
{
public function actionDefault()
{
$game = new FifteenControl($this, 'game');
...
// pomocí metody getPersistentComponents
class DefaultPresenter extends /*Nette\Application\*/Presenter
{
public static function getPersistentComponents()
{
return array('game');
}
public function actionDefault()
{
$game = new FifteenControl($this, 'game');
...
Vyšší prioritu má funkce nad anotacemi. Přesněji řečeno, výchozí implementace každé z funkcí vrací pole detekované z anotací. Tím, že ji přepíšete, tak vlastně zrušíte toto chování a anotace určujete manuálně.
- PetrP
- Člen | 587
Davide jsi skutečně blesk. Jak se ale budou řešit další meta informace v daleké budoucnosti (doufám že neprozrazuju nějaké tajemství). Myslel jsem že přidáš jednotný způsob přidání všech takovýchto informací. Nebo chceš jít cestou kdy pro každej typ budeš mít takhle vlastní metodu.
Mimochodem je možný způsoby kombinovat? A jak? Nebude to dělat někde nějaké problémy?
- David Grudl
- Nette Core | 8239
PetrP napsal(a):
Davide jsi skutečně blesk. Jak se ale budou řešit další meta informace v daleké budoucnosti (doufám že neprozrazuju nějaké tajemství). Myslel jsem že přidáš jednotný způsob přidání všech takovýchto informací. Nebo chceš jít cestou kdy pro každej typ budeš mít takhle vlastní metodu.
Současné řešení je asi nejjednodušší a nejsrozumitelnější pro uživatele. V podstatě i nějak takto by mohl vypadat „jednotný“ způsob. To ukáže čas.
Mimochodem je možný způsoby kombinovat? A jak? Nebude to dělat někde nějaké problémy?
Ano, obojí lze libovolně kombinovat.
- David Grudl
- Nette Core | 8239
Wosonj napsal(a):
Super!!! Díky moc!
Tenhle příspěvek má za účel přidat do threadu klíčové slovo eAccelerator, aby byl na eAccelerator vyhledatelný a nikdo se už neztrapnil podobně jako já včera
Neztrapnil, jen jsi demonstroval, že v Nette se požadavky řeší o 13 hodin dřív, než se vznesou :-)))
- Jakub Šulák
- Člen | 222
Tak jsem asi špatně Davida pochopil. Píše, že řešení lze kombinovat, což ale není tak úplně pravda – kombinovat to jde jen za cenu přetížení metody getPersistentParams().
Myslel jsem si, že to bude fungovat trochu jinak – budu mít nadefinované persistení proměnné přes anotace a budu moci některé proměnné definovat navíc přes tuto metodu.
Nešlo by přidat metodu:
addPersistentParams($arrayOfParams)
která by vložila další parametry?
Jedná se mi o to, když člověk potřebuje programově rozhodnout, zda bude/nebude proměnná persistentní – nemohu to zapsat logicky do anotace. Ale pokud si chci (z historických důvodů, nebo kvůli přehlednosti) nechat jiné deklarace v anotacích, nemusel bych přetěžovat metodu getPersistentParams().
Navíc v okamžiku, kdy se v ní něco změní, tak to zapomene člověk upravit, jedině v ní volat natvrdo PresenterComponent::getPersistentParams().
- morousej
- Člen | 18
<?php
private static $paramsToAdd = array();
public static function addPersistantParams($arrayOfParams)
{
self::$paramsToAdd = array_merge(self::$paramsToAdd, $arrayOfParams);
}
public static function getPersistantParams($className)
{
return array_merge(parent::getPersistantParams($className), self::$paramsToAdd);
}
?>
Editoval morousej (10. 6. 2009 18:18)
- Jakub Šulák
- Člen | 222
nevím proč, ale předtím mi to psalo chybu když jsem volal parent:: –
hmmm.
ale díky – takto to je bez problému.
- DocX
- Člen | 154
Měl bych k této problematice ještě jeden dotaz. Jde nějak programově určit defaultní hodnotu parametru? Já jsem zkusil upravit PresenterHelper::getPersistentParams :
<?php
public static function getPersistentParams($class)
{
...
foreach (call_user_func(array($class, 'getPersistentParams'), $class) as $name => $meta) {
if (is_string($name)) $default[$name] = $meta; else if (is_string($meta)) $name = $meta;
$params[$name] = array(
'def' => $defaults[$name],
'since' => $class,
);
}
...
}
?>
a potom použít GetPersistentParams ve vlastní třídě následovně:
<?php
class MyClass extends Presenter
public $lang;
public static function getPersistentParams()
{
return array('lang' => Environment::getConfig('langs.default', 'en'));
}
?>
ale tvorba odkazů se stále chová jako když parametr nema výchozí hodnotu. (pokud by v konfiguraci bylo „cs“ tak i při url ?lang=cs se $this->canonicalize() nepřesměruje na url bez lang). Pokud výchozí přiřazení připíšu, chová se jak má. Takže jde spíš o to, zjistit, podle čeho to adresy s persistentnímy parametry zkracuje, když to není výstup funkce GetPersistentParams()
- DocX
- Člen | 154
PetrP napsal(a):
A normální „přiřazení“ defaultní hodnoty nefunguje?
Ano to funguje. Já bych ale potřeboval tuto defaultní hodnotu určit až v runtime. Protože se podle ní potom zkracuje url, když je hodnota parametru stejná jako jeho defaultní hodnota.
Je mi proto divné, že i když jsem upravil metodu PresenterHelper::getPersistentParams() tak, aby to jako defaultni hodnotu považovalo tu, kterou předám mojí override metodou getPersistentParams, stále to někde jinde pozná ještě přímo z definice té proměnné v kódu třídy. Nemohu ale nalézt kde se tak děje :(