), on restreindra les enfants et (parents) possibles à quelque chose de sensé. // Plutôt que d'avoir plein de sous-classes, ElementDocument a une méthode __call(), qui vérifie ce qu'on peut appeller en fonction du type de l'élément. // Propriété "url" pour le document ? Probablement innutile, pusiqu'on bosse principalement avec des uid (le href d'un est un uid). class ElementDocument { public $espaceCss = null; private static $types = array(); private static $widgets = array(); private $type = null; private $enfants = array(); private $attr = array(); protected $document = null; public static function add_type($singleton, $type, $typesEnfants = "", $attributs_oblig = "", $attributs_opt = "") { if ($singleton !== true && $singleton !== false) { $attributs_opt = $attributs_oblig; $attributs_oblig = $typesEnfants; $typesEnfants = $type; $type = $singleton; $singleton = false; } self::$types[$type] = array( "singleton" => $singleton, "enfants" => qw($typesEnfants), "attributs_oblig" => qw($attributs_oblig), "attributs_opt" => qw($attributs_opt) ); } public static function add_widget($nom) { self::$widgets["w_" . $nom] = "fn_w_" . $nom; } public function type() { return $this->type; } /* public function inclure($elem) { // Tente de fusionner $elem avec $this // Très mauvaise fonction car l'inclusion peut planter bien après la définition des deux parties. niy("inclure"); }*/ public function attr($nom, $valeur) { $this->attr[$nom] = $valeur; } public function to_XHTML_5() { return applyXSLT($this->to_XML(), dirname(__FILE__) . "/xslt/xhtml5.xsl"); } public function to_XML($indent = "") { if ($this->type == "litteral") { return $this->attr['valeur']; } $ret = ""; $ret .= "$indent<" . $this->type; foreach ($this->attr as $k => $v) { $ret .= " " . htmlspecialchars($k) . '="' . htmlspecialchars($v) . '"'; } if (count($this->enfants) == 0) { $ret .= "/>\n"; } else { $ret .= ">\n"; foreach ($this->enfants as $k => $v) { $ret .= $v->to_XML($indent . " "); } $ret .= "$indenttype . ">\n"; } return $ret; } public function to_HTML_5() { niy("to_HTML_5"); } public function to_HTML_4_01() { niy("to_HTML_4_01"); } public function to_XHTML_1_1() { niy("to_XHTML_1_1"); } public function url() { return $this->document->page->url(); } public function __construct($type = "document", &$doc = null) { $this->type = $type; $this->document = $doc; } public static function has_widget($w) { return array_key_exists($w, self::$widgets); } public function type_autorisé($t) { return array_key_exists($t, self::$types) && in_array($t, self::$types[$this->type]["enfants"]); } public function singleton_élément($type, $args) { if (!array_key_exists($type, $this->document->singletons)) { $this->document->singletons[$type] = $this->créer_élément($type, $args); } return $this->document->singletons[$type]; } public function créer_élément($type, $args) { $elem = new self($type, $this->document); $max = 0; foreach (self::$types[$type]["attributs_oblig"] as $i => $nom) { if (!isset($args[$i])) { Debug("erreur", "Argument manquant : $nom pour " . $elem->type); } $elem->attr($nom, $args[$i]); $max = $i; } foreach (self::$types[$type]["attributs_opt"] as $i => $nom) { if (isset($args[$i])) { $elem->attr($nom, $args[$i]); } } $this->enfants[] = $elem; return $elem; } public function créer_widget($nom, $args) { $f = self::$widgets[$nom]; array_unshift($args, $this); return call_user_func_array($f, $args); } public function __call($fn, $args) { if (self::type_autorisé($fn)) { if (self::$types[$fn]["singleton"]) return $this->singleton_élément($fn, $args); else return $this->créer_élément($fn, $args); } elseif (self::has_widget($fn)) { return $this->créer_widget($fn, $args); } else { Debug("erreur", "Impossible d'ajouter un élément $fn à " . $this->type); return null; } } } class Document extends ElementDocument { protected $singletons = array(); protected $page = null; public function __construct($page) { parent::__construct("document", $this); $this->erreurs(); $this->header(); $this->nav(); $this->article(); $this->footer(); $this->page = $page; } } $inline_elems = "span text a strong em img form"; ElementDocument::add_type("document", "erreurs header footer nav article script style"); ElementDocument::add_type(true, "header", "title"); ElementDocument::add_type(true, "erreurs", "litteral"); ElementDocument::add_type("title", "text"); ElementDocument::add_type("litteral", "", "valeur"); ElementDocument::add_type(true, "footer", ""); ElementDocument::add_type(true, "nav", "ul"); ElementDocument::add_type(true, "article", "ul hX table p form span"); // span ? ElementDocument::add_type("hX", $inline_elems); ElementDocument::add_type("script", "", "src"); ElementDocument::add_type("style", "", "src"); ElementDocument::add_type("ul", "li"); ElementDocument::add_type("table", "thead tbody tfoot"); ElementDocument::add_type("tbody", "tr"); ElementDocument::add_type("tr", "td th"); ElementDocument::add_type("td", $inline_elems, "", "colspan rowspan"); ElementDocument::add_type("th", $inline_elems); ElementDocument::add_type("li", $inline_elems); ElementDocument::add_type("form", $inline_elems . " input_text_line input_text_multi input_text_rich input_file input_submit", "action"); ElementDocument::add_type("input_text_line", "", "name value"); ElementDocument::add_type("input_text_rich", "", "name value"); ElementDocument::add_type("input_file", "name"); ElementDocument::add_type("input_submit", "", "label"); ElementDocument::add_type("a", $inline_elems, "href"); ElementDocument::add_type("span", $inline_elems, "", "class"); ElementDocument::add_type("img", "", "alt src"); ElementDocument::add_type("p", $inline_elems); ElementDocument::add_type("text", "", "text"); function fn_w_titre($d, $cell) { // renvoie un

ou un selon les droits $d->header()->title()->text(toString($cell)); // TODO : modification si on a les droits. $d->article()->hX()->text(toString($cell)); } function fn_w_en_tete($d, $cell_titre, $cell_description) { $d->w_titre($cell_titre); $d->w_description($cell_description); } function fn_w_description($d, $cell) { // TODO : modification si on a les droits. return $d->article()->p()->text(toString($cell)); } function fn_w_bouton($d, $texte, $page_callback, $ressource_callback, $action_callback) { // afficher un input[type=button] // lors du clic, appeller $action_callback sur $page_callback/?res=$ressource_callback ? $a = $d->a($page_callback->url($ressource_callback, "act_" . $page_callback->uid() . "_" . $action_callback)); $a->text($texte); return $a; } function fn_w_liste($d, $liste_pages, $function_formattage_elements) { $ul = $d->ul(); foreach ($liste_pages as $page) { $li = $ul->li(); $function_formattage_elements($page, $li); } return $ul; } function fn_w_tableau($d, $select, $function_formattage_elements) { $t = $d->table(); $tr = $t->tbody()->tr(); $tr->td()->text("Not Implemented Yet"); return $t; } function fn_w_img_file_desc($d, $cell_img, $cell_description) { // TODO : modification si on a les droits. $d->w_img_file($cell_img); $d->w_description($cell_description); return $img; } function fn_w_field($d, $cell) { if ($cell->page()->if_perm("w", $cell->nom_attribut())) { return call_user_func(array($d, "w_w_" . $cell->type()), $cell); } else { return call_user_func(array($d, "w_r_" . $cell->type()), $cell); } } function fn_w_r_field($d, $cell) { return call_user_func(array($d, "w_r_" . $cell->type()), $cell); } /* ==== Types ==== */ // text_line function fn_w_r_text_line($d, $cell) { return $d->text(toString($cell)); } function fn_w_w_text_line($d, $cell) { $f = $d->form($d->url()); $f->input_text_line($cell->name_for_set(), toString($cell)); $f->input_submit("Ok"); return $f; } function fn_serialize_text_line($input) { return preg_replace("/\n/", " ", $input); } // text_nix function fn_w_r_text_nix($d, $cell) { // Texte naze (sans espaces etc.) à la *nix. return $d->text(toString($cell)); } function fn_w_w_text_nix($d, $cell) { return $d->form()->input_text($cell->name_for_set(), toString($cell)); // TODO !!! } function fn_serialize_text_nix($input) { return str_to_nix($input); } // text_rich function fn_w_r_text_rich($d, $cell) { // TODO : rendu du texte riche. return $d->p()->text(toString($cell)); } function fn_w_w_text_rich($d, $cell) { // TODO : modification du texte riche. return $d->form()->input_text_rich($cell->name_for_set(), toString($cell)); } function fn_serialize_text_rich($input) { return $input; } // bool function fn_w_r_bool($d, $cell) { return $d->text("w_bool(" . toString($cell) . ")"); } function fn_w_w_bool($d, $cell) { // TODo : checkbox } function fn_serialize_bool($input) { return ($input == "true") ? "true" : "false"; } // img_file function fn_w_r_img_file($d, $cell) { return $d->img(toString($cell)); } function fn_w_w_img_file($d, $cell) { $f = $d->form($d->url()); fn_w_r_img_file($f, $cell); $f->input_file($cell->name_for_set()); return $f; } function fn_serialize_img_file($input) { // Doit gérer le stockage de l'image dans un dossier, // la création de la miniature et renvoyer le chemin vers l'image. // TODO !!! comment faire ? return $input; } // date function fn_w_r_date($d, $cell) { // affichage localisé. return $d->text(strftime("%Y-%m-%d %H:%M:%S", toString($cell))); } function fn_w_w_date($d, $cell) { // affichage localisé. return $d->form()->input_text_line($cell->name_for_set(), strftime("%Y-%m-%d %H:%M:%S", toString($cell))); } function fn_serialize_date($input) { // TODO return toString(strtotime($input)); } ElementDocument::add_widget("titre", "fn_w_titre"); ElementDocument::add_widget("en_tete", "fn_w_en_tete"); ElementDocument::add_widget("description", "fn_w_description"); ElementDocument::add_widget("bouton", "fn_w_bouton"); ElementDocument::add_widget("liste", "fn_w_liste"); ElementDocument::add_widget("tableau", "fn_w_tableau"); ElementDocument::add_widget("img_file_desc", "fn_w_img_file_desc"); ElementDocument::add_widget("field"); ElementDocument::add_widget("r_field"); Module::add_type("text_line"); Module::add_type("text_nix"); Module::add_type("text_rich"); Module::add_type("bool"); Module::add_type("img_file"); Module::add_type("date"); ?>