<?xml version="1.0" encoding="utf-8"?><?xml-stylesheet title="XSL formatting" type="text/xsl" href="http://prendreuncafe.com/blog/feed/rss2/xslt" ?><rss version="2.0"
  xmlns:dc="http://purl.org/dc/elements/1.1/"
  xmlns:wfw="http://wellformedweb.org/CommentAPI/"
  xmlns:content="http://purl.org/rss/1.0/modules/content/"
  xmlns:atom="http://www.w3.org/2005/Atom">
<channel>
  <title>Prendre un Café - Tag - bestpractices</title>
  <link>http://prendreuncafe.com/blog/</link>
  <atom:link href="http://prendreuncafe.com/blog/feed/tag/bestpractices/rss2" rel="self" type="application/rss+xml"/>
  <description></description>
  <language>fr</language>
  <pubDate>Tue, 01 Feb 2011 14:49:24 +0100</pubDate>
  <copyright>Contenus sous licence Creative Commons BY-SA</copyright>
  <docs>http://blogs.law.harvard.edu/tech/rss</docs>
  <generator>Dotclear</generator>
  
    
  <item>
    <title>Optimize your Doctrine Workflow with Specialized Queries</title>
    <link>http://prendreuncafe.com/blog/post/Optimize-your-Doctrine-Workflow-with-Specialized-Queries</link>
    <guid isPermaLink="false">urn:md5:07be0ff35e1f2e047f2faca25323c26e</guid>
    <pubDate>Tue, 15 Sep 2009 19:14:00 +0200</pubDate>
    <dc:creator>NiKo</dc:creator>
        <category>Dev</category>
        <category>bestpractices</category><category>doctrine</category><category>php</category><category>query</category><category>sql</category><category>symfony</category><category>tips</category>    
    <description>    &lt;p&gt;I&amp;#8217;m currently working on a big &lt;a href=&quot;http://www.symfony-project.org/&quot; hreflang=&quot;en&quot;&gt;Symfony&lt;/a&gt; project, with a lot of &lt;a href=&quot;http://www.doctrine-project.org/&quot; hreflang=&quot;en&quot;&gt;Doctrine&lt;/a&gt; models and complex queries to write. I found a way to organize all of them in an object-oriented and cleaner way than using the traditionnal &lt;code&gt;addNamedQuery()&lt;/code&gt; and &lt;code&gt;createNamedQuery()&lt;/code&gt; methods workflow&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/Optimize-your-Doctrine-Workflow-with-Specialized-Queries#pnote-1123-1&quot; id=&quot;rev-pnote-1123-1&quot;&gt;1&lt;/a&gt;]&lt;/sup&gt;.&lt;/p&gt;


&lt;p&gt;The idea is to create dedicated &lt;a href=&quot;http://www.doctrine-project.org/Doctrine_Query/1_2&quot; hreflang=&quot;en&quot;&gt;query&lt;/a&gt; classes for a given model&amp;#160;; this way, you can provide useful methods to build the business-related parts of your query.&lt;/p&gt;


&lt;p&gt;As usual, the theory is more understandable with a concrete example. Let&amp;#8217;s consider this simple Doctrine model&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/Optimize-your-Doctrine-Workflow-with-Specialized-Queries#pnote-1123-2&quot; id=&quot;rev-pnote-1123-2&quot;&gt;2&lt;/a&gt;]&lt;/sup&gt;&amp;#160;:&lt;/p&gt;


&lt;p&gt;&lt;strong&gt;Disclaimer:&lt;/strong&gt; The provided examples have been written in a hurry, so mistakes might have been not detected by my attentive proof-reading &lt;img src=&quot;/blog/themes/battlestar/smilies/wink.gif&quot; alt=&quot;;)&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;

&lt;pre&gt; yaml
BlogAuthor:
  columns:
    id:
      type: integer(4)
      primary: true
      autoincrement: true  
    name:
      type: string(255)
  relations:
    Post:
      type: one
      class: BlogPost
      local: id
      foreign: author_id

BlogPost:
  columns:
    id:
      type: integer(4)
      primary: true
      autoincrement: true
    author_id:
      type: integer(4)
      notnull: true
    title:
      type: string(255)
    content:
      type: string(65535)
  relations:
    Author:
      type: one
      class: BlogAuthor
      local: author_id
      foreign: id
    Comments:
      type: many
      class: BlogComment
      local: id
      foreign: post_id

BlogComment:
  columns:
    id:
      type: integer(4)
      primary: true
      autoincrement: true
    post_id:
      type: integer(4)
      notnull: true
    author:
      type: string(255)
    content:
      type: string(5000)
  relations:
    Post:
      type: one
      class: BlogPost
      local: post_id
      foreign: id
&lt;/pre&gt;


&lt;p&gt;Now let&amp;#8217;s imagine a Query class dedicated to query the BlogPost table:&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php 
class BlogPostQuery extends Doctrine_Query
{
  static public function create($conn = null, $class = null)
  {
    return parent::create($conn, 'BlogPostQuery')
      -&amp;gt;from('BlogPost p');
  }
  
  public function addPosts($fields = 'p.*')
  {
    return $this-&amp;gt;addSelect('p.*');
  }
  
  public function addComments($fields = 'c.*')
  {
    return $this
      -&amp;gt;addSelect($fields)
      -&amp;gt;leftJoin('p.Comments c')
      -&amp;gt;addGroupBy('c.id');
  }
  
  public function addAuthors($fields = 'a.*')
  {
    return $this
      -&amp;gt;addSelect($fields)
      -&amp;gt;leftJoin('p.Author a')
      -&amp;gt;addGroupBy('a.id');
  }
  
  public function addCommentsCount($alias = 'nb_comments')
  {
    return $this
      -&amp;gt;addSelect(sprintf('COUNT(c.id) as %s', $alias))
      -&amp;gt;addGroupBy('c.id');
  }
  
  public function filterByAuthorName($authorName)
  {
    return $this
      -&amp;gt;andWhere('a.name = ?', $authorName);
  }
}
&lt;/pre&gt;


&lt;p&gt;So how can we use this query object? Here are some sample uses:&lt;/p&gt;

&lt;pre&gt; php
// Retrieve all posts
$posts = BlogPostQuery::create()
  -&amp;gt;addPosts()
  -&amp;gt;fetchArray();

// Retrieve all posts with comments
$posts = BlogPostQuery::create()
  -&amp;gt;addPosts()
  -&amp;gt;addComments()
  -&amp;gt;fetchArray();

// Retrieve all posts with comments and their count per post
$posts = BlogPostQuery::create()
  -&amp;gt;addPosts()
  -&amp;gt;addComments()
  -&amp;gt;addCommentsCount('yataa')
  -&amp;gt;fetchArray();

// Retrieve all post with chuck as its author and related comments
$posts = BlogPostQuery::create()
  -&amp;gt;addAuthors()
  -&amp;gt;addPosts()
  -&amp;gt;addComments()
  -&amp;gt;filterByAuthorName('chuck')
  -&amp;gt;fetchArray();

// and so on...
&lt;/pre&gt;


&lt;p&gt;Of course, this example of use is not really relevant as our model is really simple, but when you&amp;#8217;re dealing with dozens of internationalized objects, it can help cleaning your model classes, controllers and improving the organization of your work.&lt;/p&gt;


&lt;h3&gt;Update and important precisions&lt;/h3&gt;


&lt;p&gt;Some people are having negative feedback regarding this technique, claiming it will encourage people using the custom query object directly in the controllers; that&amp;#8217;s absolutely not the case as the queries are to be used only within the model layer, for example in the &lt;code&gt;BlogPostTable&lt;/code&gt; class:&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php
class BlogPostTable extends Doctrine_Table
{
  static public function getPostsWithCommentsByAuthor($authorName)
  {
    return BlogPostQuery::create()
      -&amp;gt;addPosts()
      -&amp;gt;addComments()
      -&amp;gt;filterByAuthorName($authorName)
      -&amp;gt;fetchArray()
    ;
  }
}
&lt;/pre&gt;


&lt;p&gt;And in a controller:&lt;/p&gt;

&lt;pre&gt; php
class blogActions extends sfActions
{
  public function executeListByAuthor(sfWebRequest $request)
  {
    $this-&amp;gt;posts = BlogPostTable::getPostsWithCommentsByAuthor($request-&amp;gt;getParameter('author'));
  }
}
&lt;/pre&gt;
&lt;div class=&quot;footnotes&quot;&gt;&lt;h4&gt;Notes&lt;/h4&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/Optimize-your-Doctrine-Workflow-with-Specialized-Queries#rev-pnote-1123-1&quot; id=&quot;pnote-1123-1&quot;&gt;1&lt;/a&gt;] &amp;#8230; or raw queries written directly within controllers, but you may know that &lt;a href=&quot;http://www.slideshare.net/nperriault/30-symfony-best-practices&quot; hreflang=&quot;en&quot;&gt;this is really bad&lt;/a&gt; &lt;img src=&quot;/blog/themes/battlestar/smilies/wink.gif&quot; alt=&quot;;)&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/Optimize-your-Doctrine-Workflow-with-Specialized-Queries#rev-pnote-1123-2&quot; id=&quot;pnote-1123-2&quot;&gt;2&lt;/a&gt;] I&amp;#8217;m using Doctrine 1.2 beta (bundled with upcoming symfony 1.3) in the provided example.&lt;/p&gt;&lt;/div&gt;&lt;hr/&gt;&lt;p style=&quot;margin:.5em 0;padding:.5em;border:1px solid #333;background:#eee;color:#222&quot;&gt;&lt;small&gt;Ce billet intitulé &lt;a href=&quot;http://prendreuncafe.com/blog/post/Optimize-your-Doctrine-Workflow-with-Specialized-Queries&quot;&gt;Optimize your Doctrine Workflow with Specialized Queries&lt;/a&gt; a été rédigé par &lt;a href=&quot;http://prendreuncafe.com/cv&quot;&gt;Nicolas Perriault&lt;/a&gt; et publié sur le blog &lt;a href=&quot;http://prendreuncafe.com/blog/&quot;&gt;Prendre un Café&lt;/a&gt; sous licence &lt;a href=&quot;http://creativecommons.org/licenses/by-nc-sa/2.5/&quot;&gt;Creative Commons BY-NC-SA&lt;/a&gt;.&lt;/small&gt;&lt;/p&gt;</description>
    
    
    
      </item>
    
  <item>
    <title>30 Symfony Best Practices, the slides from my talk at SymfonyDay Cologne '09</title>
    <link>http://prendreuncafe.com/blog/post/2009/09/04/30-Symfony-Best-Practices%2C-the-slides-from-my-talk-at-SymfonyDay-Cologne-09</link>
    <guid isPermaLink="false">urn:md5:b3db81a6e20a143a4e03c71d6b81c16f</guid>
    <pubDate>Fri, 04 Sep 2009 20:50:00 +0200</pubDate>
    <dc:creator>NiKo</dc:creator>
        <category>Dev</category>
        <category>bestpractices</category><category>php</category><category>sfdaycgn</category><category>symfony</category>    
    <description>    &lt;p&gt;Hey there, it&amp;#8217;s been a while, huh?&lt;/p&gt;


&lt;p&gt;Today I gave a talk about Symfony best practices at the &lt;a href=&quot;http://www.symfonyday.com/&quot; hreflang=&quot;en&quot;&gt;Symfony Day event in Cologne, Germany&lt;/a&gt;; you can &lt;a href=&quot;http://www.slideshare.net/nperriault/30-symfony-best-practices&quot; hreflang=&quot;en&quot;&gt;get the slides on slideshare&lt;/a&gt; or directly browse them below:&lt;/p&gt;

&lt;object type=&quot;application/x-shockwave-flash&quot; data=&quot;http://static.slidesharecdn.com/swf/ssplayer2.swf?doc=30symfonybestpractices-090904074841-phpapp01&amp;amp;stripped_title=30-symfony-best-practices&quot; width=&quot;425&quot; height=&quot;344&quot; style=&quot;width:425px;height:344px;margin:0 auto&quot;&gt;
  &lt;param name=&quot;movie&quot; value=&quot;http://static.slidesharecdn.com/swf/ssplayer2.swf?doc=30symfonybestpractices-090904074841-phpapp01&amp;amp;stripped_title=30-symfony-best-practices&quot; /&gt;
  &lt;param name=&quot;wmode&quot; value=&quot;transparent&quot; /&gt;
  &lt;param name=&quot;allowFullScreen&quot; value=&quot;true&quot; /&gt;
  &lt;param name=&quot;allowScriptAccess&quot; value=&quot;always&quot; /&gt;
  
&lt;/object&gt;



&lt;p&gt;The Symfony Day event has been purely awesomely incredibly well organized, many kudos and thanks to &lt;a href=&quot;http://www.interlutions.de/&quot; hreflang=&quot;de&quot;&gt;Interlutions&lt;/a&gt; and to all the attendees. You&amp;#8217;re all great people. Thanks.&lt;/p&gt;

&lt;a href=&quot;http://www.flickr.com/photos/n1k0/3886662176/&quot; title=&quot;Symfony Day '09 Cologne by Nicolas Perriault, on Flickr&quot;&gt;&lt;img src=&quot;http://farm3.static.flickr.com/2555/3886662176_8f0826e610.jpg&quot; width=&quot;500&quot; height=&quot;333&quot; alt=&quot;Symfony Day '09 Cologne&quot; /&gt;&lt;/a&gt;&lt;hr/&gt;&lt;p style=&quot;margin:.5em 0;padding:.5em;border:1px solid #333;background:#eee;color:#222&quot;&gt;&lt;small&gt;Ce billet intitulé &lt;a href=&quot;http://prendreuncafe.com/blog/post/2009/09/04/30-Symfony-Best-Practices%2C-the-slides-from-my-talk-at-SymfonyDay-Cologne-09&quot;&gt;30 Symfony Best Practices, the slides from my talk at SymfonyDay Cologne '09&lt;/a&gt; a été rédigé par &lt;a href=&quot;http://prendreuncafe.com/cv&quot;&gt;Nicolas Perriault&lt;/a&gt; et publié sur le blog &lt;a href=&quot;http://prendreuncafe.com/blog/&quot;&gt;Prendre un Café&lt;/a&gt; sous licence &lt;a href=&quot;http://creativecommons.org/licenses/by-nc-sa/2.5/&quot;&gt;Creative Commons BY-NC-SA&lt;/a&gt;.&lt;/small&gt;&lt;/p&gt;</description>
    
    
    
      </item>
    
  <item>
    <title>Comment faire fuir à coup sûr vos utilisateurs ?</title>
    <link>http://prendreuncafe.com/blog/post/2008/08/24/Comment-faire-fuir-a-coup-sur-vos-utilisateurs</link>
    <guid isPermaLink="false">urn:md5:3a3c403335eef28f603dc8c9c9ec3e78</guid>
    <pubDate>Sun, 24 Aug 2008 15:45:00 +0200</pubDate>
    <dc:creator>NiKo</dc:creator>
        <category>Divers</category>
        <category>bestpractices</category><category>ergonomie</category><category>formulaires</category><category>grumph</category><category>nimp</category><category>web</category>    
    <description>    &lt;p&gt;C&amp;#8217;est bien connu, il est déjà sufisament dur de faire (re)venir des utilisateurs sur vos services en ligne pour ne pas les faire partir à la moindre occasion. C&amp;#8217;est pourtant ce que fait un site dédié à la guitare que j&amp;#8217;ai tenté de consulter récemment. Connaissant les tendances dérivistes des services juridiques de certaines entités, je m&amp;#8217;abstiendrai donc de faire un lien hypertexte direct vers ce dernier, et même de le citer textuellement&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/08/24/Comment-faire-fuir-a-coup-sur-vos-utilisateurs#pnote-990-1&quot; id=&quot;rev-pnote-990-1&quot;&gt;1&lt;/a&gt;]&lt;/sup&gt;.&lt;/p&gt;


&lt;p&gt;Je tombe sur ce site, qui propose apparemment moult ressources sur l&amp;#8217;instrument et qui m&amp;#8217;intéressent potentiellement, mais au moment de consulter l&amp;#8217;article objet de ma convoitise, on me notifie du message suivant&amp;#160;:&lt;/p&gt;


&lt;p&gt;&lt;img src=&quot;http://prendreuncafe.com/blog/public/images/Ergonomie/GuitareLive/registration_obligatoire.png&quot; alt=&quot;registration_obligatoire.png&quot; style=&quot;display:block; margin:0 auto;&quot; /&gt;&lt;/p&gt;


&lt;p&gt;Jusque-là, rien de profondément choquant même si on peut imaginer que nombre d&amp;#8217;internautes ne s&amp;#8217;embêteront pas à créer un compte pour consulter la ressource et passeront à la concurrence&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/08/24/Comment-faire-fuir-a-coup-sur-vos-utilisateurs#pnote-990-2&quot; id=&quot;rev-pnote-990-2&quot;&gt;2&lt;/a&gt;]&lt;/sup&gt;.&lt;/p&gt;


&lt;p&gt;Allez, je suis dans un bon jour, je demande la création d&amp;#8217;un compte. Je me retrouve face à un formulaire assez copieux, exigeant bien entendu des informations hautement nécessaires pour consulter un tutoriel de guitare comme ma localisation géographique, mon nom de famille, ma date de naissance, sans oublier bien entendu les inévitables propositions d&amp;#8217;abonnement à des newsletter très étroitement liées à la pratique de la guitare&amp;#160;:&lt;/p&gt;


&lt;p&gt;&lt;img src=&quot;http://prendreuncafe.com/blog/public/images/Ergonomie/GuitareLive/pollution.png&quot; alt=&quot;pollution.png&quot; style=&quot;display:block; margin:0 auto;&quot; /&gt;&lt;/p&gt;


&lt;p&gt;Sentant le SPAM arriver à grand pas, je rentre mon email suffixé pour &lt;a href=&quot;http://www.seandeasy.com/multiple-email-addresses-for-one-gmail-account/&quot; hreflang=&quot;en&quot;&gt;détecter a posteriori le service à l&amp;#8217;origine d&amp;#8217;éventuels courriels indésirables reçus&lt;/a&gt;. Ben non&amp;#160;:&lt;/p&gt;


&lt;p&gt;&lt;img src=&quot;http://prendreuncafe.com/blog/public/images/Ergonomie/GuitareLive/wrong_email_validation.png&quot; alt=&quot;wrong_email_validation.png&quot; style=&quot;display:block; margin:0 auto;&quot; /&gt;&lt;/p&gt;


&lt;p&gt;Vous noterez le plus fort, c&amp;#8217;est que non seulement le service n&amp;#8217;est pas à même de valider convenablement une adresse email (le signe + est acceptable), mais en plus on me soupçonne ouvertement de SPAM potentiel.&lt;/p&gt;


&lt;p&gt;Normalement, à ce stade, &lt;em&gt;je me casse sur Mars&lt;/em&gt;, comme on dit. Mais pour la beauté de l&amp;#8217;exercice, je &lt;em&gt;corrige&lt;/em&gt; mon adresse email, pérsévère en entrant un mot de passe et valide à nouveau le formulaire&amp;#160;:&lt;/p&gt;


&lt;p&gt;&lt;img src=&quot;http://prendreuncafe.com/blog/public/images/Ergonomie/GuitareLive/restricted_chars.png&quot; alt=&quot;restricted_chars.png&quot; style=&quot;display:block; margin:0 auto;&quot; /&gt;&lt;/p&gt;


&lt;p&gt;Oui, cela commence à faire beaucoup. Je ne réexpliquerai pas ici pourquoi &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/06/04/Demander-a-restreindre-les-caracteres-utilises-dans-les-mots-de-passe-est-idiot&quot; hreflang=&quot;fr&quot;&gt;limiter les caractères dans les mots de passe est idiot&lt;/a&gt;, cela a déjà été fait ici-même précédemment.&lt;/p&gt;


&lt;p&gt;Toujours dans la perspective de poursuivre cette expérience amusante, j&amp;#8217;arrive enfin à valider le formulaire sans retours d&amp;#8217;erreur et m&amp;#8217;identifie donc sur le service pour consulter la ressource en question. Patatra&amp;#160;:&lt;/p&gt;


&lt;p&gt;&lt;img src=&quot;http://prendreuncafe.com/blog/public/images/Ergonomie/GuitareLive/abo_obligatoire.png&quot; alt=&quot;abo_obligatoire.png&quot; style=&quot;display:block; margin:0 auto;&quot; /&gt;&lt;/p&gt;


&lt;p&gt;Non seulement la création d&amp;#8217;un compte était obligatoire, mais maintenant il faut s&amp;#8217;abonner (c&amp;#8217;est à dire &lt;em&gt;payer&lt;/em&gt;) pour consulter la ressource en question, chose qui n&amp;#8217;avait été aucunement stipulée auparavant. Sachant bien entendu qu&amp;#8217;à ce stade, je n&amp;#8217;ai toujours &lt;strong&gt;aucune&lt;/strong&gt; idée de la qualité des contenus proposés par le site.&lt;/p&gt;


&lt;p&gt;Bref, je leur souhaite malgré tout un avenir comptable radieux, même si je ne peux m&amp;#8217;empêcher de pouffer à l&amp;#8217;idée du chiffre d&amp;#8217;affaire généré par le se(r)vice.&lt;/p&gt;
&lt;div class=&quot;footnotes&quot;&gt;&lt;h4&gt;Notes&lt;/h4&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/08/24/Comment-faire-fuir-a-coup-sur-vos-utilisateurs#rev-pnote-990-1&quot; id=&quot;pnote-990-1&quot;&gt;1&lt;/a&gt;] Mais les curieux trouveront bien entendu sans peine le site en question.&lt;/p&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/08/24/Comment-faire-fuir-a-coup-sur-vos-utilisateurs#rev-pnote-990-2&quot; id=&quot;pnote-990-2&quot;&gt;2&lt;/a&gt;] Sur le Web, vous avez quasiment systématiquement un concurrent qui propose la même chose que vous gratuitement. C&amp;#8217;est comme ça, et c&amp;#8217;est aussi un peu pour ça que c&amp;#8217;est chouette.&lt;/p&gt;&lt;/div&gt;&lt;hr/&gt;&lt;p style=&quot;margin:.5em 0;padding:.5em;border:1px solid #333;background:#eee;color:#222&quot;&gt;&lt;small&gt;Ce billet intitulé &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/08/24/Comment-faire-fuir-a-coup-sur-vos-utilisateurs&quot;&gt;Comment faire fuir à coup sûr vos utilisateurs ?&lt;/a&gt; a été rédigé par &lt;a href=&quot;http://prendreuncafe.com/cv&quot;&gt;Nicolas Perriault&lt;/a&gt; et publié sur le blog &lt;a href=&quot;http://prendreuncafe.com/blog/&quot;&gt;Prendre un Café&lt;/a&gt; sous licence &lt;a href=&quot;http://creativecommons.org/licenses/by-nc-sa/2.5/&quot;&gt;Creative Commons BY-NC-SA&lt;/a&gt;.&lt;/small&gt;&lt;/p&gt;</description>
    
    
    
      </item>
    
  <item>
    <title>Mes conventions de codage...</title>
    <link>http://prendreuncafe.com/blog/post/2008/07/23/Mes-conventions-de-codage</link>
    <guid isPermaLink="false">urn:md5:b94a814a2f724b5abe17d299a6dfc06a</guid>
    <pubDate>Wed, 23 Jul 2008 21:21:00 +0200</pubDate>
    <dc:creator>NiKo</dc:creator>
        <category>Dev</category>
        <category>bestpractices</category><category>php</category>    
    <description>    &lt;p&gt;... sont celles des projets sur lesquels je me greffe. C'est en effet pour moi une forme de respect que d'appliquer les standards de codage partagés par une communauté (ou une équipe) de développeurs : ainsi, on maximise les chances de se comprendre et on minimise les coûteuses phases de communication entre geeks introvertis&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/07/23/Mes-conventions-de-codage#pnote-988-1&quot; id=&quot;rev-pnote-988-1&quot;&gt;1&lt;/a&gt;]&lt;/sup&gt; &lt;img src=&quot;/blog/themes/battlestar/smilies/tongue.gif&quot; alt=&quot;:p&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;


&lt;p&gt;En effet, rien de plus pénible que de reprendre le code de quelqu'un qui a pris des libertés avec des conventions établies à ce niveau, l'apothéose étant obtenue avec ce genre de code :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php
  class Ma_superClasse {

    function dire_coucou ( $popol) {
  echo 'coucou ' . $popol   . ' !' ;
}
     function DireAuRevoir($Popol )
{ print &amp;quot;Au revoir $Popol !&amp;quot;;
     }
  }
&lt;/pre&gt;


&lt;p&gt;Je force bien évidemment ici le trait, mais tout le monde est déjà tombé sur ce genre de code illisible, qui multiplie par 10 votre temps d'intervention sur ce dernier et divise par 1000 votre passion pour la &lt;acronym title=&quot;Tierce Maintenance Applicative&quot;&gt;TMA&lt;/acronym&gt;.&lt;/p&gt;


&lt;p&gt;Bien entendu, il peut arriver de produire du code sur un projet ne nécessitant l'utilisation d'aucune brique logicielle existante. Auquel cas vous pouvez librement appliquer vos propres standards de codage, l'important étant ici qu'ils soient cohérents et constamment appliqués. S'il peuvent être ceux d'un projet open source existant reconnu, cela augmentera la sympathie potentielle à votre égard de futurs intervenants sur votre code &lt;img src=&quot;/blog/themes/battlestar/smilies/wink.gif&quot; alt=&quot;;)&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;


&lt;p&gt;Je noterai quand même en vrac quelques bonnes pratiques générales globalement reconnues et appréciées :&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;être explicite,&lt;/li&gt;
&lt;li&gt;indenter son code,&lt;/li&gt;
&lt;li&gt;documenter son code,&lt;/li&gt;
&lt;li&gt;à choisir entre les deux, privilégier la lisibilité à la concision,&lt;/li&gt;
&lt;li&gt;utiliser des noms de variables, de classes, de méthodes, de fonctions et d'arguments parlants,&lt;/li&gt;
&lt;li&gt;utiliser des noms anglophones,&lt;/li&gt;
&lt;li&gt;utiliser des motifs de conception connus.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Personnellement, j'ai mes petites préférences et tout comme &lt;a href=&quot;http://case.oncle-tom.net/&quot; hreflang=&quot;fr&quot;&gt;Oncle Tom&lt;/a&gt; - qui m'a gentiment refilé &lt;a href=&quot;http://case.oncle-tom.net/2008/07/23/conventions-de-programmation-necessaire-maturite/&quot; hreflang=&quot;fr&quot;&gt;cette chaîne&lt;/a&gt;&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/07/23/Mes-conventions-de-codage#pnote-988-2&quot; id=&quot;rev-pnote-988-2&quot;&gt;2&lt;/a&gt;]&lt;/sup&gt; - j'ai tendance à appliquer les &lt;a href=&quot;http://trac.symfony-project.org/wiki/HowToContributeToSymfony#CodingStandards&quot; hreflang=&quot;en&quot;&gt;standards de codage de symfony&lt;/a&gt;, que je trouve homogènes et cohérents. Mais ce sont là bien évidemment essentiellement des questions de goûts et de couleurs.&lt;/p&gt;
&lt;div class=&quot;footnotes&quot;&gt;&lt;h4&gt;Notes&lt;/h4&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/07/23/Mes-conventions-de-codage#rev-pnote-988-1&quot; id=&quot;pnote-988-1&quot;&gt;1&lt;/a&gt;] Voire les trolls genre &lt;em&gt;les tabulations ça pue, vive l'indentation à trois espaces&lt;/em&gt;...&lt;/p&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/07/23/Mes-conventions-de-codage#rev-pnote-988-2&quot; id=&quot;pnote-988-2&quot;&gt;2&lt;/a&gt;] Salopard, ça va se payer ! &lt;img src=&quot;/blog/themes/battlestar/smilies/wink.gif&quot; alt=&quot;;-)&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;&lt;/div&gt;&lt;hr/&gt;&lt;p style=&quot;margin:.5em 0;padding:.5em;border:1px solid #333;background:#eee;color:#222&quot;&gt;&lt;small&gt;Ce billet intitulé &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/07/23/Mes-conventions-de-codage&quot;&gt;Mes conventions de codage...&lt;/a&gt; a été rédigé par &lt;a href=&quot;http://prendreuncafe.com/cv&quot;&gt;Nicolas Perriault&lt;/a&gt; et publié sur le blog &lt;a href=&quot;http://prendreuncafe.com/blog/&quot;&gt;Prendre un Café&lt;/a&gt; sous licence &lt;a href=&quot;http://creativecommons.org/licenses/by-nc-sa/2.5/&quot;&gt;Creative Commons BY-NC-SA&lt;/a&gt;.&lt;/small&gt;&lt;/p&gt;</description>
    
    
    
      </item>
    
  <item>
    <title>Demander à restreindre les caractères utilisés dans les mots de passe est idiot</title>
    <link>http://prendreuncafe.com/blog/post/2008/06/04/Demander-a-restreindre-les-caracteres-utilises-dans-les-mots-de-passe-est-idiot</link>
    <guid isPermaLink="false">urn:md5:f0b1a6866f144db2f77ede15d40d6c37</guid>
    <pubDate>Wed, 04 Jun 2008 12:25:00 +0200</pubDate>
    <dc:creator>NiKo</dc:creator>
        <category>Divers</category>
        <category>bestpractices</category><category>ergonomie</category><category>password</category><category>security</category>    
    <description>    &lt;p&gt;Voulant créer un compte sur un nouveau service en ligne, j'ai eu la désagréable surprise d'obtenir ce message d'erreur au moment de la validation de mon mot de passe :&lt;/p&gt;


&lt;blockquote&gt;&lt;p&gt;Password may only contain alphanumeric characters&lt;/p&gt;&lt;/blockquote&gt;


&lt;p&gt;Pour les anglophobes, on me demande de n'utiliser que des caractères alphanumériques pour mon mot de passe.&lt;/p&gt;


&lt;p&gt;Ce n'est pas la première fois que je tombe sur la mise en place de pareil procédé. Quel est l'intérêt d'imposer cette limitation ? Je pense que plus un mot de passe utilise une plage de caractères diversifiée, plus il est sécurisé. Je peux éventuellement imaginer les problèmes que pourraient poser l'utilisation de caractères codés sur deux octets, ou encore l'angoisse d'une faille XSS potentielle chez les plus paranos, mais de là à interdire purement et simplement les tirets, underscores, espaces&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/06/04/Demander-a-restreindre-les-caracteres-utilises-dans-les-mots-de-passe-est-idiot#pnote-974-1&quot; id=&quot;rev-pnote-974-1&quot;&gt;1&lt;/a&gt;]&lt;/sup&gt;, je trouve ça tout simplement &lt;strong&gt;idiot à tomber par terre&lt;/strong&gt;.&lt;/p&gt;


&lt;p&gt;Voila, c'est dit.&lt;/p&gt;


&lt;p&gt;&lt;strong&gt;Edit :&lt;/strong&gt; &lt;a href=&quot;http://googleblog.blogspot.com/2008/06/does-your-password-pass-test.html&quot; hreflang=&quot;en&quot;&gt;Quelques astuces sur la sécurisation des mots de passe&lt;/a&gt;.&lt;/p&gt;
&lt;div class=&quot;footnotes&quot;&gt;&lt;h4&gt;Notes&lt;/h4&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/06/04/Demander-a-restreindre-les-caracteres-utilises-dans-les-mots-de-passe-est-idiot#rev-pnote-974-1&quot; id=&quot;pnote-974-1&quot;&gt;1&lt;/a&gt;] Alors que &lt;a href=&quot;http://www.baekdal.com/articles/usability/password-security-usability/&quot; hreflang=&quot;en&quot;&gt;l'utilisation de deux espaces dans un mot de passe est très sécurisante&lt;/a&gt;&lt;/p&gt;&lt;/div&gt;&lt;hr/&gt;&lt;p style=&quot;margin:.5em 0;padding:.5em;border:1px solid #333;background:#eee;color:#222&quot;&gt;&lt;small&gt;Ce billet intitulé &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/06/04/Demander-a-restreindre-les-caracteres-utilises-dans-les-mots-de-passe-est-idiot&quot;&gt;Demander à restreindre les caractères utilisés dans les mots de passe est idiot&lt;/a&gt; a été rédigé par &lt;a href=&quot;http://prendreuncafe.com/cv&quot;&gt;Nicolas Perriault&lt;/a&gt; et publié sur le blog &lt;a href=&quot;http://prendreuncafe.com/blog/&quot;&gt;Prendre un Café&lt;/a&gt; sous licence &lt;a href=&quot;http://creativecommons.org/licenses/by-nc-sa/2.5/&quot;&gt;Creative Commons BY-NC-SA&lt;/a&gt;.&lt;/small&gt;&lt;/p&gt;</description>
    
    
    
      </item>
    
  <item>
    <title>QOTD</title>
    <link>http://prendreuncafe.com/blog/post/2008/05/04/QOTD</link>
    <guid isPermaLink="false">urn:md5:7b40bbbd9d40f96105e19c63b5b0fc59</guid>
    <pubDate>Sun, 04 May 2008 16:38:00 +0200</pubDate>
    <dc:creator>NiKo</dc:creator>
        <category>Dev</category>
        <category>architecture</category><category>bestpractices</category><category>framework</category><category>quote</category><category>scale</category>    
    <description>    &lt;p&gt;Une belle quote comme ça, je me la note dans un coin pour la ressortir la prochaine fois qu'on trolle sur les performances du &lt;em&gt;framework X&lt;/em&gt; ou du &lt;em&gt;langage Y&lt;/em&gt; :&lt;/p&gt;


&lt;blockquote&gt;&lt;p&gt;Languages, libraries and frameworks don't scale. Architectures do.&lt;/p&gt;&lt;/blockquote&gt;


&lt;p&gt;En français approchant :&lt;/p&gt;


&lt;blockquote&gt;&lt;p&gt;Les langages de programmation, les librairies et les frameworks ne tiennent pas la charge. Les architectures, si.&lt;/p&gt;&lt;/blockquote&gt;


&lt;p&gt;(&lt;a href=&quot;http://pownce.com/ubernostrum/notes/2043798/&quot; hreflang=&quot;en&quot;&gt;via&lt;/a&gt;, apparement attribuable à &lt;a href=&quot;http://www.iamcal.com/&quot; hreflang=&quot;en&quot;&gt;Cal Henderson&lt;/a&gt;)&lt;/p&gt;&lt;hr/&gt;&lt;p style=&quot;margin:.5em 0;padding:.5em;border:1px solid #333;background:#eee;color:#222&quot;&gt;&lt;small&gt;Ce billet intitulé &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/05/04/QOTD&quot;&gt;QOTD&lt;/a&gt; a été rédigé par &lt;a href=&quot;http://prendreuncafe.com/cv&quot;&gt;Nicolas Perriault&lt;/a&gt; et publié sur le blog &lt;a href=&quot;http://prendreuncafe.com/blog/&quot;&gt;Prendre un Café&lt;/a&gt; sous licence &lt;a href=&quot;http://creativecommons.org/licenses/by-nc-sa/2.5/&quot;&gt;Creative Commons BY-NC-SA&lt;/a&gt;.&lt;/small&gt;&lt;/p&gt;</description>
    
    
    
      </item>
    
  <item>
    <title>Symfony 1.1 beta, tour du propriétaire - L'internationalisation (i18n)</title>
    <link>http://prendreuncafe.com/blog/post/2008/03/13/Symfony-11-beta-tour-du-proprietaire-Linternationalisation-i18n</link>
    <guid isPermaLink="false">urn:md5:4646e60529b4ae4cc53445b5d6cf0103</guid>
    <pubDate>Thu, 13 Mar 2008 11:52:00 +0100</pubDate>
    <dc:creator>NiKo</dc:creator>
        <category>Dev</category>
        <category>bestpractices</category><category>framework</category><category>i18n</category><category>php</category><category>symfony</category><category>tutoriel</category>    
    <description>    &lt;p&gt;Dans la liste des tâches nouvellement ajoutées en Symfony 1.1, on remarque une section dédiée à l'internationalisation :&lt;/p&gt;


&lt;pre&gt;i18n
  :extract            Extracts i18n strings from php files
  :find               Finds non &amp;quot;i18n ready&amp;quot; strings in an application&lt;/pre&gt;


&lt;p&gt;Et oui, la fonctionnalité dont tous les gens qui ont un jour travaillé sur des applications internationalisées en Symfony 1.0 ont rêvé a enfin été ajoutée : une tâche d'extraction des chaînes de caractères à traduire, avec génération et mise à jour des fichiers de traductions &lt;a href=&quot;http://developers.sun.com/dev/gadc/technicalpublications/articles/xliff.html&quot; hreflang=&quot;en&quot;&gt;XLIFF&lt;/a&gt; &lt;img src=&quot;/blog/themes/battlestar/smilies/smile.gif&quot; alt=&quot;:)&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;


&lt;p&gt;Si vous avez suivi les &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/10/Symfony-11-beta-tour-du-proprietaire-Installation&quot; hreflang=&quot;fr&quot;&gt;précédents&lt;/a&gt; &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires&quot; hreflang=&quot;fr&quot;&gt;tutoriels&lt;/a&gt;, vous devez disposer d'un projet &lt;code&gt;sf11test&lt;/code&gt;, d'une application &lt;code&gt;main&lt;/code&gt; et d'un module &lt;code&gt;contact&lt;/code&gt;.&lt;/p&gt;


&lt;p&gt;On va activer la gestion de l'internationalisation dans l'application en éditant le fichier de configuration &lt;code&gt;apps/main/config/settings.yml&lt;/code&gt; comme suit :&lt;/p&gt;

&lt;pre&gt;
[...]
all:
  [...]
  .settings:
    [...]
    i18n:                   on
    [...]
    standard_helpers:       [Partial, Cache, Form, I18N]
    [...]
    default_culture:        en
&lt;/pre&gt;


&lt;p&gt;On part du principe que la langue par défaut sera l'anglais (en). Ajoutons quelques chaînes internationalisées dans le template &lt;code&gt;apps/main/modules/contact/templates/indexSuccess.php&lt;/code&gt; au moyen du helper &lt;code&gt;__()&lt;/code&gt; :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;h2&amp;gt;&amp;lt;?php echo __('Contact us') ?&amp;gt;&amp;lt;/h2&amp;gt;
&amp;lt;p&amp;gt;&amp;lt;?php echo __('Drop us a message using the form below:') ?&amp;gt;&amp;lt;/p&amp;gt;
&amp;lt;form action=&amp;quot;&amp;lt;?php echo url_for('contact/index') ?&amp;gt;&amp;quot; method=&amp;quot;post&amp;quot;&amp;gt;
  &amp;lt;table&amp;gt;
    &amp;lt;?php echo $form ?&amp;gt;
    &amp;lt;tr&amp;gt;
      &amp;lt;td&amp;gt;&amp;lt;/td&amp;gt;
      &amp;lt;td&amp;gt;&amp;lt;input type=&amp;quot;submit&amp;quot; value=&amp;quot;&amp;lt;?php echo __('Send your message') ?&amp;gt;&amp;quot; /&amp;gt;&amp;lt;/td&amp;gt;
    &amp;lt;/tr&amp;gt;
  &amp;lt;/table&amp;gt;
&amp;lt;/form&amp;gt;
&lt;/pre&gt;


&lt;p&gt;Maintenant, lançons la commande d'extraction des chaînes à traduire pour notre future version française, en lui demandant poliment de générer automatiquement le fichier de traduction et de supprimer automatiquement les entrées orphelines :&lt;/p&gt;

&lt;pre&gt;
 $ ./symfony i18n:extract --auto-save --auto-delete main fr
&amp;gt;&amp;gt; i18n      extracting i18n strings for the &amp;quot;main&amp;quot; application
&amp;gt;&amp;gt; i18n      found &amp;quot;3&amp;quot; new i18n strings
&amp;gt;&amp;gt; i18n      found &amp;quot;0&amp;quot; old i18n strings
&amp;gt;&amp;gt; i18n      saving new i18n strings
&amp;gt;&amp;gt; i18n      deleting old i18n strings
&lt;/pre&gt;


&lt;p&gt;Le fichier &lt;code&gt;apps/main/i18n/fr/messages.xml&lt;/code&gt; a été généré, examinons son contenu :&lt;/p&gt;

&lt;pre&gt; xml
&amp;lt;?xml version=&amp;quot;1.0&amp;quot;?&amp;gt;
&amp;lt;xliff version=&amp;quot;1.0&amp;quot;&amp;gt;
  &amp;lt;file source-language=&amp;quot;EN&amp;quot; target-language=&amp;quot;fr&amp;quot; datatype=&amp;quot;plaintext&amp;quot;
    original=&amp;quot;messages&amp;quot; date=&amp;quot;2008-03-13T11:13:45Z&amp;quot;
    product-name=&amp;quot;messages&amp;quot;&amp;gt;
    &amp;lt;body&amp;gt;
      &amp;lt;trans-unit id=&amp;quot;1&amp;quot;&amp;gt;
        &amp;lt;source&amp;gt;Contact us&amp;lt;/source&amp;gt;
        &amp;lt;target&amp;gt;&amp;lt;/target&amp;gt;
      &amp;lt;/trans-unit&amp;gt;
      &amp;lt;trans-unit id=&amp;quot;2&amp;quot;&amp;gt;
        &amp;lt;source&amp;gt;Drop us a message using the form below:&amp;lt;/source&amp;gt;
        &amp;lt;target&amp;gt;&amp;lt;/target&amp;gt;
      &amp;lt;/trans-unit&amp;gt;
      &amp;lt;trans-unit id=&amp;quot;3&amp;quot;&amp;gt;
        &amp;lt;source&amp;gt;Send your message&amp;lt;/source&amp;gt;
        &amp;lt;target&amp;gt;&amp;lt;/target&amp;gt;
      &amp;lt;/trans-unit&amp;gt;
    &amp;lt;/body&amp;gt;
  &amp;lt;/file&amp;gt;
&amp;lt;/xliff&amp;gt;
&lt;/pre&gt;


&lt;p&gt;Il ne nous reste &lt;em&gt;plus qu'à&lt;/em&gt; traduire nos chaînes en remplissant les balises &lt;code&gt;&amp;lt;target&amp;gt;&amp;lt;/target&amp;gt;&lt;/code&gt; en conséquence pour traduire notre application en français. Si l'on venait à modifier notre template en supprimant, modifiant ou ajoutant de nouvelles chaînes, la tâche d'extraction se chargerait de mettre à jour nos fichiers de traduction en conséquence, tout en préservant le travail déjà effectué &lt;img src=&quot;/blog/themes/battlestar/smilies/smile.gif&quot; alt=&quot;:)&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;&lt;hr/&gt;&lt;p style=&quot;margin:.5em 0;padding:.5em;border:1px solid #333;background:#eee;color:#222&quot;&gt;&lt;small&gt;Ce billet intitulé &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/13/Symfony-11-beta-tour-du-proprietaire-Linternationalisation-i18n&quot;&gt;Symfony 1.1 beta, tour du propriétaire - L'internationalisation (i18n)&lt;/a&gt; a été rédigé par &lt;a href=&quot;http://prendreuncafe.com/cv&quot;&gt;Nicolas Perriault&lt;/a&gt; et publié sur le blog &lt;a href=&quot;http://prendreuncafe.com/blog/&quot;&gt;Prendre un Café&lt;/a&gt; sous licence &lt;a href=&quot;http://creativecommons.org/licenses/by-nc-sa/2.5/&quot;&gt;Creative Commons BY-NC-SA&lt;/a&gt;.&lt;/small&gt;&lt;/p&gt;</description>
    
    
    
      </item>
    
  <item>
    <title>Symfony 1.1 beta, tour du propriétaire - Les formulaires</title>
    <link>http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires</link>
    <guid isPermaLink="false">urn:md5:06e57c127e9ce12fd2905da7c9f68edc</guid>
    <pubDate>Tue, 11 Mar 2008 10:50:00 +0100</pubDate>
    <dc:creator>NiKo</dc:creator>
        <category>Dev</category>
        <category>bestpractices</category><category>forms</category><category>framework</category><category>html</category><category>php</category><category>symfony</category><category>tutoriel</category>    
    <description>    &lt;p&gt;Nous venons de voir la &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/10/Symfony-11-beta-tour-du-proprietaire-Installation&quot; hreflang=&quot;fr&quot;&gt;procédure d'installation de la beta1 de Symfony 1.1&lt;/a&gt;. Nous allons maintenant rentrer un peu plus dans les détails des nouvelles fonctionnalités en commençant par les formulaires.&lt;/p&gt;


&lt;h3&gt;La gestion des formulaires avec Symfony 1.1&lt;/h3&gt;


&lt;p&gt;La nouvelle gestion des formulaires est l'une des fonctionnalités majeures de cette nouvelle mouture du framework. Elle propose une séparation claire entre couche de contrôle, couche de définition et couche de présentation des données, soit un bon vieux pattern &lt;acronym title=&quot;Modèle Vue Contrôleur&quot;&gt;MVC&lt;/acronym&gt; des familles.&lt;/p&gt;


&lt;p&gt;Pour illustrer ces fonctionnalités, nous allons retrousser nos manches et créer un formulaire de contact basique. On commence par initialiser un nouveau module &lt;code&gt;contact&lt;/code&gt; dans l'application &lt;code&gt;main&lt;/code&gt; de notre projet &lt;code&gt;sf11test&lt;/code&gt; initié &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/10/Symfony-11-beta-tour-du-proprietaire-Installation&quot; hreflang=&quot;fr&quot;&gt;précédemment&lt;/a&gt; :&lt;/p&gt;


&lt;pre&gt;$ ./symfony generate:module main contact&lt;/pre&gt;


&lt;h4&gt;Création d'une classe de formulaire&lt;/h4&gt;


&lt;p&gt;Nous allons créer une classe qui représentera notre formulaire de contact, que nous stockerons dans le fichier &lt;code&gt;apps/main/lib/ContactForm.class.php&lt;/code&gt;. Cette classe étendra la classe de base &lt;code&gt;sfForm&lt;/code&gt; et surchargera sa méthode de configuration afin de définir les champs de formulaire et les différents validateurs associés. Nous définirons quatre champs (appelés &lt;em&gt;widgets&lt;/em&gt; en Symfony 1.1) :&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Le nom de l'expéditeur, sous la forme d'un champs de saisie textuelle (&lt;code&gt;&amp;lt;input type=&amp;quot;text&amp;quot;/&amp;gt;&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;Son adresse email, également sous la forme d'un champs de saisie textuelle&lt;/li&gt;
&lt;li&gt;Le sujet de son message, sous la forme d'une boîte de sélection (&lt;code&gt;&amp;lt;select/&amp;gt;&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;Le texte de son message, sous la forme d'un champs texte multilignes (&lt;code&gt;&amp;lt;textarea/&amp;gt;&lt;/code&gt;)&lt;/li&gt;
&lt;/ul&gt;
&lt;pre&gt; php
&amp;lt;?php
class ContactForm extends sfForm
{
  public function configure()
  {
    // Widgets
    $topics = sfConfig::get('app_contact_topics', array());
    $widgetSchema = new sfWidgetFormSchema(array(
      'topic'   =&amp;gt; new sfWidgetFormSelect(array('choices' =&amp;gt; $topics)),
      'name'    =&amp;gt; new sfWidgetFormInput(),
      'email'   =&amp;gt; new sfWidgetFormInput(),
      'message' =&amp;gt; new sfWidgetFormTextarea()
    ));
    $widgetSchema-&amp;gt;setNameFormat('contact[%s]'); // HTML field names format
    $this-&amp;gt;setWidgetSchema($widgetSchema);

    // Validators
    $this-&amp;gt;setValidators(array(
      'topic'   =&amp;gt; new sfValidatorRegex(array('pattern' =&amp;gt; '/[a-z_]/')),
      'name'    =&amp;gt; new sfValidatorString(array('min_length' =&amp;gt; 2,
                                               'max_length' =&amp;gt; 45)),
      'email'   =&amp;gt; new sfValidatorAnd(array(new sfValidatorEmail(),
                                            new sfValidatorString(array('max_length' =&amp;gt; 100)))),
      'message' =&amp;gt; new sfValidatorString(array('min_length' =&amp;gt; 10)),
    ));
  }
}
&lt;/pre&gt;


&lt;p&gt;Au passage, nous ajoutons les sujets possibles de message dans le fichier de configuration &lt;code&gt;apps/main/config/app.yml&lt;/code&gt; :&lt;/p&gt;

&lt;pre&gt;
all:
  contact:
    topics:
      carrots_request: Do you have carrots?
      eggs_request:    Do you have eggs?
&lt;/pre&gt;


&lt;p&gt;Marquons un arrêt pour examiner de plus près ce que nous venons d'écrire :&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Nous avons ajouté 4 &lt;em&gt;widgets&lt;/em&gt; à notre formulaires,&lt;/li&gt;
&lt;li&gt;Nous avons défini et paramétré leurs validateurs associés,
&lt;ul&gt;
&lt;li&gt;Au passage, vous noterez qu'il est possible de spécifier plusieurs validateurs pour un même champs, comme c'est ici le cas pour le champs &lt;code&gt;email&lt;/code&gt;&lt;/li&gt;
&lt;/ul&gt;&lt;/li&gt;
&lt;li&gt;Nous avons défini le format de nommage des champs de formulaire et choisi une syntaxe à base de tableau pour manipuler plus aisément les données dans notre action,&lt;/li&gt;
&lt;li&gt;Nous avons déporté une partie de la configuration textuelle dans un fichier externe dédié, pour en faciliter la maintenance.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;N'oublions pas de purger le cache de Symfony, car nous venons d'ajouter un nouvel objet php :&lt;/p&gt;


&lt;pre&gt;$ ./symfony cc&lt;/pre&gt;


&lt;h4&gt;Interaction avec le formulaire depuis le contrôleur de l'application&lt;/h4&gt;


&lt;p&gt;Éditons maintenant le fichier &lt;code&gt;apps/main/modules/contact/actions/actions.class.php&lt;/code&gt; pour y définir l'action par défaut du module&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires#pnote-948-1&quot; id=&quot;rev-pnote-948-1&quot;&gt;1&lt;/a&gt;]&lt;/sup&gt; :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php
class contactActions extends sfActions
{
  public function executeIndex(sfWebRequest $request)
  {
    $form = new ContactForm();
    if ($request-&amp;gt;isMethod('post')) // If HTTP method is POST
    {
      // Bind submitted values to the contact form instance
      $form-&amp;gt;bind($request-&amp;gt;getParameter('contact'));
      // Validate the form
      if ($form-&amp;gt;isValid()) 
      {
        // Retrieve submitted values
        $values = $form-&amp;gt;getValues(); 
        // ... Send your message here using submitted values
        // Then redirect user to the homepage with a one-shot message
        $this-&amp;gt;getUser()-&amp;gt;setFlash('notice', 'Message sent');
        $this-&amp;gt;redirect('@homepage');
      }
    }
    // Publish form instance to the view
    $this-&amp;gt;form = $form;
  }
}
&lt;/pre&gt;


&lt;p&gt;Ici, dans l'ordre :&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;On instancie un objet de formulaire de contact&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Si&lt;/strong&gt; la méthode HTTP est POST :
&lt;ul&gt;
&lt;li&gt;On assigne les paramètres de la requête correspondant aux champs du formulaire de contact à ce dernier&lt;/li&gt;
&lt;li&gt;On lance la validation, et &lt;strong&gt;si&lt;/strong&gt; c'est valide :
&lt;ul&gt;
&lt;li&gt;On effectue les opérations nécessaires (traitement, redirection, etc.)&lt;/li&gt;
&lt;/ul&gt;&lt;/li&gt;
&lt;/ul&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Vous noterez que tous les &lt;em&gt;sinon&lt;/em&gt; sont gérés automatiquement par le framework de façon transparente (mais ces comportements par défaut sont toujours surchargeables) :&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Par défaut le formulaire présenté sera vierge&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires#pnote-948-2&quot; id=&quot;rev-pnote-948-2&quot;&gt;2&lt;/a&gt;]&lt;/sup&gt;,&lt;/li&gt;
&lt;li&gt;Les champs seront  automatiquement préremplis en cas d'erreur de validation,&lt;/li&gt;
&lt;li&gt;Les erreurs seront contextualisées par rapport aux champs.&lt;/li&gt;
&lt;/ul&gt;

&lt;h4&gt;Rendu du formulaire dans la vue&lt;/h4&gt;


&lt;p&gt;Enfin, il nous reste à créer un template pour présenter le formulaire et éventuellement afficher un message à l'utilisateur, dans notre vue gérée au travers du fichier &lt;code&gt;apps/main/modules/contact/templates/indexSuccess.php&lt;/code&gt; :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php if ($sf_user-&amp;gt;hasFlash('notice')): ?&amp;gt;
  &amp;lt;p class=&amp;quot;notice&amp;quot;&amp;gt;&amp;lt;?php echo $sf_user-&amp;gt;getFlash('notice') ?&amp;gt;&amp;lt;/p&amp;gt;
&amp;lt;?php endif; ?&amp;gt;

&amp;lt;form action=&amp;quot;&amp;lt;?php echo url_for('contact/index') ?&amp;gt;&amp;quot; method=&amp;quot;post&amp;quot;&amp;gt;
  &amp;lt;table&amp;gt;
    &amp;lt;?php echo $form ?&amp;gt;
    &amp;lt;tr&amp;gt;
      &amp;lt;td&amp;gt;&amp;lt;/td&amp;gt;
      &amp;lt;td&amp;gt;&amp;lt;input type=&amp;quot;submit&amp;quot; /&amp;gt;&amp;lt;/td&amp;gt;
    &amp;lt;/tr&amp;gt;
  &amp;lt;/table&amp;gt;
&amp;lt;/form&amp;gt;
&lt;/pre&gt;


&lt;p&gt;Si on charge la page &lt;code&gt;/main_dev.php/contact&lt;/code&gt;, on a le résultat suivant :&lt;/p&gt;


&lt;p&gt;&lt;img src=&quot;http://prendreuncafe.com/blog/public/images/ScreenShots/Symfony/sf11test_form.png&quot; alt=&quot;Symfony 1.1 test form&quot; style=&quot;display:block; margin:0 auto;&quot; /&gt;&lt;/p&gt;


&lt;p&gt;Au passage si vous regardez la source, une protection anti &lt;a href=&quot;http://fr.wikipedia.org/wiki/Cross-Site_Request_Forgeries&quot; hreflang=&quot;fr&quot;&gt;CSRF&lt;/a&gt; est automatiquement gérée de façon transparente.&lt;/p&gt;


&lt;p&gt;Bien entendu, le mode de rendu par défaut utilise un tableau pour la mise en forme, mais il existe un autre décorateur plus &lt;em&gt;sémantique&lt;/em&gt; à base de liste. Pour l'utiliser, modifions notre classe &lt;code&gt;ContactForm&lt;/code&gt; :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php
class ContactForm extends sfForm
{
  public function configure()
  {
   // ...
    $this-&amp;gt;widgetSchema-&amp;gt;setFormFormatterName('list');
  }
}
&lt;/pre&gt;


&lt;p&gt;Il conviendra bien entendu d'adapter le template pour enlever les balises tables. On peut aussi imaginer de créer notre propre décorateur HTML. Par exemple, créeons la classe &lt;code&gt;sfWidgetFormSchemaFormatterDiv&lt;/code&gt; comme suit :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php
class sfWidgetFormSchemaFormatterDiv extends sfWidgetFormSchemaFormatter
{
  protected
    $rowFormat       = &amp;quot;&amp;lt;div class=\&amp;quot;form-row\&amp;quot;&amp;gt;\n  %error%%label%\n  %field%%help%\n%hidden_fields%&amp;lt;/div&amp;gt;\n&amp;quot;,
    $errorRowFormat  = &amp;quot;&amp;lt;div class=\&amp;quot;form-errors\&amp;quot;&amp;gt;\n%errors%&amp;lt;/div&amp;gt;\n&amp;quot;,
    $helpFormat      = '&amp;lt;div class=&amp;quot;form-help&amp;quot;&amp;gt;%help%&amp;lt;/div&amp;gt;',
    $decoratorFormat = &amp;quot;&amp;lt;div&amp;gt;\n  %content%&amp;lt;/div&amp;gt;&amp;quot;;
}
&lt;/pre&gt;


&lt;p&gt;Encore une fois, modifions notre classe &lt;code&gt;ContactForm&lt;/code&gt; pour l'utiliser (après avoir purgé le cache symfony, comme il se doit) :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php
class ContactForm extends sfForm
{
  public function configure()
  {
   // ...
    $this-&amp;gt;widgetSchema-&amp;gt;setFormFormatterName('div');
  }
}
&lt;/pre&gt;


&lt;h4&gt;Rendu des widgets&lt;/h4&gt;


&lt;p&gt;Vous me direz, un simple &lt;code&gt;&amp;lt;?php echo $form ?&amp;gt;&lt;/code&gt; n'est pas très satisfaisant du point de vue de l'intégration HTML, qui nécessite bien souvent de prendre la main finement sur le code html généré. Le framework de formulaires de Symfony 1.1 apporte une solution en permettant de générer le rendu de chacun des widgets de façon indépendante&lt;/p&gt;


&lt;p&gt;Imaginons par exemple que nous souhaitions gérer spécifiquement la présentation du champ &lt;code&gt;name&lt;/code&gt; de notre formulaire ; notre template devient alors :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php if ($sf_user-&amp;gt;hasFlash('notice')): ?&amp;gt;
  &amp;lt;p class=&amp;quot;notice&amp;quot;&amp;gt;&amp;lt;?php echo $sf_user-&amp;gt;getFlash('notice') ?&amp;gt;&amp;lt;/p&amp;gt;
&amp;lt;?php endif; ?&amp;gt;

&amp;lt;form action=&amp;quot;&amp;lt;?php echo url_for('contact/index') ?&amp;gt;&amp;quot; method=&amp;quot;post&amp;quot;&amp;gt;
  &amp;lt;table&amp;gt;
    &amp;lt;?php echo $form['topic']-&amp;gt;renderRow() ?&amp;gt;
    &amp;lt;tr&amp;gt;
      &amp;lt;th&amp;gt;&amp;lt;?php echo $form['name']-&amp;gt;renderLabel() ?&amp;gt;&amp;lt;/th&amp;gt;
      &amp;lt;td&amp;gt;
        &amp;lt;?php echo $form['name']-&amp;gt;renderError() ?&amp;gt;
        &amp;lt;?php echo $form['name']-&amp;gt;render(array('class' =&amp;gt; 'toto')) ?&amp;gt;
      &amp;lt;/td&amp;gt;
    &amp;lt;/tr&amp;gt;
    &amp;lt;?php echo $form['email']-&amp;gt;renderRow() ?&amp;gt;
    &amp;lt;?php echo $form['message']-&amp;gt;renderRow() ?&amp;gt;
    &amp;lt;tr&amp;gt;
      &amp;lt;td&amp;gt;&amp;lt;/td&amp;gt;
      &amp;lt;td&amp;gt;&amp;lt;input type=&amp;quot;submit&amp;quot; /&amp;gt;&amp;lt;/td&amp;gt;
    &amp;lt;/tr&amp;gt;
  &amp;lt;/table&amp;gt;
&amp;lt;/form&amp;gt;
&lt;/pre&gt;


&lt;p&gt;Vous noterez qu'on accède ici aux différents widgets du formulaire au moyen de clés de tableaux classiques ; c'est tout simplement car la classe &lt;code&gt;sfForm&lt;/code&gt; implémente l'interface &lt;code&gt;&lt;a href=&quot;http://www.php.net/~helly/php/ext/spl/interfaceArrayAccess.html&quot; hreflang=&quot;en&quot;&gt;ArrayAccess&lt;/a&gt;&lt;/code&gt; de la &lt;a href=&quot;http://www.php.net/~helly/php/ext/spl/&quot; hreflang=&quot;en&quot;&gt;SPL&lt;/a&gt;. C'est très pratique !&lt;/p&gt;


&lt;h4&gt;Le nerf de la guerre : la génération de formulaires à partir d'objets Propel&lt;/h4&gt;


&lt;p&gt;Ayant été à une lointaine époque un fervent adepte de &lt;a href=&quot;http://pear.php.net/package/DB_DataObject_FormBuilder&quot; hreflang=&quot;en&quot;&gt;PEAR::FormBuilder&lt;/a&gt;, brique permettant de générer des formulaires à partir d'instance d'objets de données &lt;acronym title=&quot;Object Relational Mapping&quot;&gt;ORM&lt;/acronym&gt;, j'étais curieux de voir comment cette fonctionnalité demandée par les développeurs depuis longtemps allait être implémentée dans Symfony 1.1 au travers de l'ORM &lt;a href=&quot;http://propel.phpdb.org/&quot; hreflang=&quot;en&quot;&gt;Propel&lt;/a&gt;, bundlé par défaut&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires#pnote-948-3&quot; id=&quot;rev-pnote-948-3&quot;&gt;3&lt;/a&gt;]&lt;/sup&gt;.&lt;/p&gt;


&lt;p&gt;Après avoir configuré un accès à notre &lt;acronym title=&quot;Système de Gestion de Bases de Données&quot;&gt;SGBD&lt;/acronym&gt; préféré&lt;sup&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires#pnote-948-4&quot; id=&quot;rev-pnote-948-4&quot;&gt;4&lt;/a&gt;]&lt;/sup&gt;, nous allons définir une table &lt;code&gt;contact_demand&lt;/code&gt; dans notre fichier &lt;code&gt;config/schema.yml&lt;/code&gt; pour y stoker nos demandes de contacts :&lt;/p&gt;

&lt;pre&gt; php
propel:
  contact_demand:
    id:
    name:    { type: varchar, size: 45, required: true }
    email:   { type: varchar, size: 100, required: true }
    topic:   { type: varchar, size: 255, required: true }
    message: { type: longvarchar, required: true }
    created_at:
&lt;/pre&gt;


&lt;p&gt;On lance rapidement la tâche de création de la table et des objets ORM associés :&lt;/p&gt;


&lt;pre&gt;$ ./symfony propel:build-all
$ ./symfony cc&lt;/pre&gt;


&lt;p&gt;Maintenant, allons faire un tour dans le répertoire &lt;code&gt;lib/model&lt;/code&gt; pour voir ce qui a été généré. Surprise, des objets de formulaires ont été automatiquement créés pour nous !&lt;/p&gt;


&lt;p&gt;Jetons un oeil plus particulièrement au fichier &lt;code&gt;lib/form/base/BaseContactDemandForm.class.php&lt;/code&gt; :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php
class BaseContactDemandForm extends BaseFormPropel
{
  public function setup()
  {
    $this-&amp;gt;setWidgets(array(
      'id'         =&amp;gt; new sfWidgetFormInputHidden(),
      'name'       =&amp;gt; new sfWidgetFormInput(),
      'email'      =&amp;gt; new sfWidgetFormInput(),
      'topic'      =&amp;gt; new sfWidgetFormInput(),
      'message'    =&amp;gt; new sfWidgetFormTextarea(),
      'created_at' =&amp;gt; new sfWidgetFormDateTime(),
    ));

    $this-&amp;gt;setValidators(array(
      'id'         =&amp;gt; new sfValidatorPropelChoice(array('model' =&amp;gt; 'ContactDemand', 'column' =&amp;gt; 'Id', 'required' =&amp;gt; false)),
      'name'       =&amp;gt; new sfValidatorString(),
      'email'      =&amp;gt; new sfValidatorString(),
      'topic'      =&amp;gt; new sfValidatorString(),
      'message'    =&amp;gt; new sfValidatorString(),
      'created_at' =&amp;gt; new sfValidatorDateTime(array('required' =&amp;gt; false)),
    ));

    $this-&amp;gt;widgetSchema-&amp;gt;setNameFormat('contact_demand[%s]');

    $this-&amp;gt;errorSchema = new sfValidatorErrorSchema($this-&amp;gt;validatorSchema);

    parent::setup();

  }

  public function getModelName()
  {
    return 'ContactDemand';
  }
}
&lt;/pre&gt;


&lt;p&gt;Cela ne vous rappelle rien ? C'est presque quasiment ce que nous avions écrit manuellement précédemment dans notre classe &lt;code&gt;ContactForm&lt;/code&gt;. On notera également la présence dans le fichier &lt;code&gt;lib/form/ContactDemandForm.class.php&lt;/code&gt; de la classe &lt;code&gt;ContactDemandForm&lt;/code&gt;, cette dernière héritant de &lt;code&gt;BaseContactDemandForm.class.php&lt;/code&gt;, ce qui nous permettra de surcharger tout ou partie de ses méthodes pour adapter le formulaire généré à nos besoins.&lt;/p&gt;


&lt;p&gt;En l'occurrence, il nous faut adapter un peu la méthode &lt;code&gt;configure()&lt;/code&gt; pour retrouver notre &lt;em&gt;sélecteur de sujets&lt;/em&gt; et réappliquer nos validateurs. Voici le code de la classe &lt;code&gt;ContactDemandForm&lt;/code&gt; modifiée en conséquence :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php
class ContactDemandForm extends BaseContactDemandForm
{
  public function configure()
  {
    // Widgets
    $topics = sfConfig::get('app_contact_topics', array());
    $this-&amp;gt;setWidgets(array(
      'id'      =&amp;gt; new sfWidgetFormInputHidden(),
      'topic'   =&amp;gt; new sfWidgetFormSelect(array('choices' =&amp;gt; $topics)),
      'name'    =&amp;gt; new sfWidgetFormInput(),
      'email'   =&amp;gt; new sfWidgetFormInput(),
      'message' =&amp;gt; new sfWidgetFormTextarea()
    ));

    // Validators
    $this-&amp;gt;setValidators(array(
      'id'         =&amp;gt; new sfValidatorPropelChoice(array('model'    =&amp;gt; 'ContactDemand',
                                                        'column'   =&amp;gt; 'Id',
                                                        'required' =&amp;gt; false)),
      'topic'      =&amp;gt; new sfValidatorRegex(array('pattern' =&amp;gt; '/[a-z_]/')),
      'name'       =&amp;gt; new sfValidatorString(array('min_length' =&amp;gt; 2,
                                                  'max_length' =&amp;gt; 45)),
      'email'      =&amp;gt; new sfValidatorAnd(array(new sfValidatorEmail(),
                                               new sfValidatorString(array('max_length' =&amp;gt; 100)))),
      'message'    =&amp;gt; new sfValidatorString(array('min_length' =&amp;gt; 10)),
      'created_at' =&amp;gt; new sfValidatorDateTime(array('required' =&amp;gt; false)),
    ));

    $this-&amp;gt;widgetSchema-&amp;gt;setNameFormat('contact_demand[%s]');
    $this-&amp;gt;errorSchema = new sfValidatorErrorSchema($this-&amp;gt;validatorSchema);
  }
}
&lt;/pre&gt;


&lt;p&gt;On va maintenant modifier notre action pour prendre en compte notre nouvelle classe de formulaire liée à notre objet Propel :&lt;/p&gt;

&lt;pre&gt; php
&amp;lt;?php
class contactActions extends sfActions
{
  public function executeIndex(sfWebRequest $request)
  {
    $form = new ContactDemandForm();
    if ($request-&amp;gt;isMethod('post')) // If HTTP method is POST
    {
      // Bind submitted values to the contact form instance
      $form-&amp;gt;bind($request-&amp;gt;getParameter('contact_demand'));
      // Validate the form
      if ($form-&amp;gt;isValid())
      {
        // Save submitted values in form's ContactDemand object
        $form-&amp;gt;save();
        // Then redirect user to the homepage with a one-shot message
        $this-&amp;gt;getUser()-&amp;gt;setFlash('notice', 'Message sent');
        $this-&amp;gt;redirect('@homepage');
      }
    }
    // Publish form instance to the view
    $this-&amp;gt;form = $form;
  }
}
&lt;/pre&gt;


&lt;p&gt;Ainsi, un simple &lt;code&gt;$form-&amp;gt;save()&lt;/code&gt; nous permet de persister en base les données soumises par l'utilisateur au travers du formulaire autogénéré. Un sacré gain de productivité en phase de prototypage, assurement !&lt;/p&gt;


&lt;h4&gt;En conclusion&lt;/h4&gt;


&lt;p&gt;Voilà, ce n'est bien évidemment là qu'une infime partie de ce que peut faire le nouveau système de gestion de formulaires de Symfony 1.1, mais c'est un peu à reculons que l'on retourne à l'ancien système &lt;img src=&quot;/blog/themes/battlestar/smilies/wink.gif&quot; alt=&quot;;-)&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;
&lt;div class=&quot;footnotes&quot;&gt;&lt;h4&gt;Notes&lt;/h4&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires#rev-pnote-948-1&quot; id=&quot;pnote-948-1&quot;&gt;1&lt;/a&gt;] Nous ne créons pas de route pour le moment, même si ce serait là une bonne pratique, afin de ne pas surcharger inutilement le contenu de ce tutorial, déjà bien assez dense comme ça &lt;img src=&quot;/blog/themes/battlestar/smilies/smile.gif&quot; alt=&quot;:-)&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires#rev-pnote-948-2&quot; id=&quot;pnote-948-2&quot;&gt;2&lt;/a&gt;] On aurait bien entendu pu proposer des valeurs par défaut.&lt;/p&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires#rev-pnote-948-3&quot; id=&quot;pnote-948-3&quot;&gt;3&lt;/a&gt;] Le mécanisme de génération de code devrait permettre de proposer facilement la même fonctionnalité pour Doctrine prochainement.&lt;/p&gt;
&lt;p&gt;[&lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires#rev-pnote-948-4&quot; id=&quot;pnote-948-4&quot;&gt;4&lt;/a&gt;] Je vous laisse le soin de vous référer au &lt;a href=&quot;http://prendreuncafe.com/blog/post/2007/04/28/Gagnez-du-temps-avec-Symfony-et-son-generateur-de-back-office&quot; hreflang=&quot;fr&quot;&gt;tutoriel existant&lt;/a&gt; sur ce même blog &lt;img src=&quot;/blog/themes/battlestar/smilies/wink.gif&quot; alt=&quot;;-)&quot; class=&quot;smiley&quot; /&gt;&lt;/p&gt;&lt;/div&gt;&lt;hr/&gt;&lt;p style=&quot;margin:.5em 0;padding:.5em;border:1px solid #333;background:#eee;color:#222&quot;&gt;&lt;small&gt;Ce billet intitulé &lt;a href=&quot;http://prendreuncafe.com/blog/post/2008/03/11/Symfony-11-beta-tour-du-proprietaire-Les-formulaires&quot;&gt;Symfony 1.1 beta, tour du propriétaire - Les formulaires&lt;/a&gt; a été rédigé par &lt;a href=&quot;http://prendreuncafe.com/cv&quot;&gt;Nicolas Perriault&lt;/a&gt; et publié sur le blog &lt;a href=&quot;http://prendreuncafe.com/blog/&quot;&gt;Prendre un Café&lt;/a&gt; sous licence &lt;a href=&quot;http://creativecommons.org/licenses/by-nc-sa/2.5/&quot;&gt;Creative Commons BY-NC-SA&lt;/a&gt;.&lt;/small&gt;&lt;/p&gt;</description>
    
    
    
      </item>
    
</channel>
</rss>

