2009/02/22

mergeFormかembedFormか

フォームオブジェクトを使う際にはどちらを使うのがいいのかな、と私も瞬間悩んだけど、sfForm.class.phpやsfPropelForm.class.phpを読んでみると、結構動作が違うね。embedFormForEachはembedFormをかぶせただけなので、それほど違いはないとは思う。未だにバグがあると私は思っているので、使う予定はないけども。

mergeFormの場合

  • 単純にフォームをくっつけたいときに使う
  • 具体的には、FormオブジェクトのDefault値とwidgetSchema、widgetValidatorのみmergeをしている
  • mergeをしているという名前のごとく、widgetのfield値が同じであった場合には、mergeされてしまうので、複数追加することができない
  • mergeされたFormオブジェクトの更新系のメソッド(updateObjectとかdoSave)が呼ばれることはないので、手元のフォームで呼んであげないといけないupdateDefaultFromObjectは、呼ばれる。コンストラクタ内なので

embedFormの場合

  • フォームを入れ子にしたいときに使う
  • embedされたFormオブジェクトは、sfFormクラスのインスタンスオブジェクト$embedFormsに格納される
  • 新しくfieldを指定して、その中にembedするので、embedFormの中のwidgetのfield値が同じであっても構わない(新しいfieldはユニークでなければいけないが)
  • embedされたFormオブジェクトのupdateObject, doSaveが呼ばれる

パッと見た感じ、似たようなもんかなー、と思っていたら、データの保存とかが結構違うのでmergeされたFormクラスのupdateObjectなどを独自で書いている際には、明示してあげないと呼ばれないので、ちょっとビックリする。呼ばれることを前提として書いていたので。。。

あと、sfGuardPluginのsfGuardUserAdminForm.class.phpには、mergeFormが出てくるけど、このupdateObjectってトランザクションを持たずにsaveメソッドを呼んでいるような気がするのだが。。。

というわけでちょっとした自分用のメモ。

2009/02/12

symfony1.2 embedFormでファイルアップロード

symfonyでFormを使うときって、mergeFormやらembedFormって使っていますか?

私は、最近までは別々のFormクラスを作って、アクションクラスで使うFormクラスをtemplateにrenderしていました。はい。力業力業。よくないねぇ。

最近は、その辺を激しくリファクタリングして、前から気になっていたmergeFormやembedForm使ってみました。で、いろいろ使ってみると、これいいじゃない?というわけで、そのことをブログに書いてみることにします。

環境は、1.2.5-dev。まぁ、1.2.4でも動きます。embedFormForEachというのもあって、複数のフォームを使いたいときは、そちらを使う方がスマートのような感じがしますが、バグがあるせいか、どうもうまくいかないので、embedFormをループで回して使ってみることにしました。今回は、複数個アップロードするということで、mergeFormは扱わずに説明します。まぁ、いつか説明してもいいけど、こちらはすでにいくつもサンプルがあるので、そちらを勝手に調べてくださいな。私は、embedFormで画像のアップロードを扱うことにします。

超簡単なサンプルということで、スキーマはこんな感じでどうかしら?

  1. propel:
  2.   posts:
  3.     _attributes: { phpName: Post }
  4.     id:
  5.     title:
  6.       type: varchar(128)
  7.       required: true
  8.     description:
  9.       type: longvarchar
  10.       required: true
  11.     created_at:
  12.     updated_at:
  13.   pictures:
  14.     _attributes: { phpName: Picture }
  15.     id:
  16.     post_id:
  17.       type: integer
  18.       required: true
  19.       foreignTable: posts
  20.       foreignReference: id
  21.       onDelete: cascade
  22.       onUpdate: cascade
  23.     filename:
  24.       type: varchar(255)
  25.       required: true
  26.     caption:
  27.       type: varchar(128)
  28.       required: true

相変わらずpropel使っています。説明はいらないと思うけども、一応。

postsというテーブルと、picturesというテーブルを用意して、picturesにはpost_idというカラムを持たせて、postsテーブルにリレーションをさせます。まぁ、そんだけ。つまり、postsとpicturesは、1-N関係なのですね。

databases.ymlなどの設定をしておいて、propel:build-allをするとformクラスやmodelクラスの雛形を作成してくれますね。モジュールも手で作るのが面倒なのと今回の説明には大事なところではないので、propel:generate-module post Postで作っておきます。post/templates/_form.phpは、現在のPostFormオブジェクトによって、作成されるので、echo $formだけで超シンプルにしておきます。つまり、以下のようにしておいてください。

  1. <?php if (!$form->getObject()->isNew()): ?>                                     
  2. <input type="hidden" name="sf_method" value="put" />                           
  3. <?php endif; ?>                                                                 
  4.   <table>                                                                       
  5.     <tfoot>                                                                     
  6.       <tr>                                                                     
  7.         <td colspan="2">                                                       
  8.           <?php echo $form->renderHiddenFields() ?>                             
  9.           &nbsp;<a href="<?php echo url_for('post/index') ?>">Cancel</a>       
  10.           <?php if (!$form->getObject()->isNew()): ?>                           
  11.             &nbsp;<?php echo link_to('Delete', 'post/delete?id='.$form->getObject()->getId(), array('method' => 'delete', 'confirm' => 'Are you sure?')) ?>     
  12.           <?php endif; ?>                                                       
  13.           <input type="submit" value="Save" />                                 
  14.         </td>                                                                   
  15.       </tr>                                                                     
  16.     </tfoot>                                                                   
  17.     <tbody>                                                                     
  18.       <?php echo $form->renderGlobalErrors() ?>                                 
  19.       <?php echo $form ?>                                                       
  20.     </tbody>                                                                   
  21.   </table>                                                                     
  22. </form>

では、まずPostフォームを用意します。propel:build-allをするとbuild-formもついでにしてくれるので、lib/form/PostForm.class.phpができていると思います。これをちょいといじってみます。

  1. public function configure() {
  2.     unset($this['created_at'], $this['updated_at']);
  3.     $this->setWidgets(array(
  4.       'id' => new sfWidgetFormInputHidden(),
  5.       'title' => new sfWidgetFormInput(array(
  6.         'label' => 'タイトル',
  7.       )),
  8.       'description' => new sfWidgetFormTextarea(array(
  9.         'label' => '説明',
  10.       )),
  11.     ));
  12.  
  13.     $this->setValidators(array(
  14.       'id' => new sfValidatorPropelChoice(
  15.         array('model' => 'Post', 'column' => 'id', 'required' => false)
  16.       ),
  17.       'title' => new sfValidatorString(
  18.         array('max_length' => 128, 'min_length' => 3),
  19.         array(
  20.           'max_length' => 'タイトルは128文字以内でお願いします。',
  21.           'min_length' => 'タイトルは3文字以上でお願いします。'
  22.         )
  23.       ),
  24.  
  25.       'description' => new sfValidatorString(
  26.         array('max_length' => 2048, 'min_length' => 3),
  27.         array(
  28.           'max_length' => '説明本文は2000文字以内でお願いします。',
  29.           'min_length' => '説明本文は3文字以上でお願いします。'
  30.         )
  31.       ),
  32.     ));
  33.     $this->widgetSchema->setNameFormat('post[%s]');
  34.   }
  35. }

左の画像のようになりますね。まぁ、こんなもんでしょう。titleとdescriptionだけを入力する単純なフォームですね。バリデーションは適当に書いてみました。ごくごく初歩的なsfPropelFormの使い方だと思います。説明はいらないですね。

では、PictureFormも書いてみます。

  1. class PictureForm extends BasePictureForm
  2. {
  3.   public function configure()
  4.   {
  5.     unset($this['post_id']);
  6.  
  7.     $captions = array('A', 'B', 'C', 'D');
  8.     $this->setWidgets(array(
  9.       'id' => new sfWidgetFormInputHidden(),
  10.       'filename' => new sfWidgetFormInputFileEditable(array(
  11.         'label' => false,
  12.         'delete_label' => 'ファイルを削除する',
  13.         'file_src' => '/uploads/' . $this->getObject()->getFilename(),
  14.         'is_image' => true,
  15.         'edit_mode' => !$this->isNew(),
  16.         'template' => '<div>%file%<br />%input%<br />%delete% %delete_label%</div>',
  17.       )),
  18.       'caption' => new sfWidgetFormSelect(
  19.         array(
  20.           'label' => '説明',
  21.           'choices' => array_combine($captions, $captions)
  22.         )
  23.       )
  24.     ));
  25.  
  26.     $this->setValidators(array(
  27.       'id' => new sfValidatorPropelChoice(array(
  28.         'model' => 'Picture', 'column' => 'id', 'required' => false
  29.       )),
  30.       'filename' => new sfValidatorFile(
  31.         array('required' => false, 'path' => sfConfig::get('sf_upload_dir')),
  32.         array(
  33.           'max_size' => 'ファイルサイズが大きすぎます。',
  34.           'mime_types' => '投稿できる画像フォーマットではありません。',
  35.           'partial' => 'ファイルアップロードに失敗しました。もう一度、投稿してください
  36.           。',
  37.           'no_tmp_dir' => 'システムエラーです。管理者にお伝えください。',
  38.           'cant_write' => 'システムエラーです。管理者にお伝えください。',
  39.           'extension' => 'システムエラーです。管理者にお伝えください。'
  40.         )),
  41.       'caption'    => new sfValidatorString(array(
  42.         'max_length' => 255, 'required' => false
  43.       )),
  44.       'filename_delete' => new sfValidatorPass()
  45.     ));
  46.     $this->widgetSchema->setFormFormatterName('list');
  47.   }
  48. }

これも説明の必要はないかな。symfony1.2からsfWidgetFormInputFileEditableというwidgetが追加されて、これがファイルアップロード関係でいろいろやってくれるのですね。便利になったものです。使い方はこのソースを読んでもいいですし、jobeetのadmin-generatorの章にも書いてありますので、それを参照してください。ここでは、画像だけをアップロードしてもいいのですが、せっかくなので、画像の説明としてcaptionというカラムもpictureテーブルに持たせてみることにします。A,B,C,Dというのはいいアイデアがなかったのとサンプルなので、まぁ、適当に。

これで、二つの独立したテーブルができましたね。しかし、postの投稿フォームにpictureも一緒に投稿させたいじゃないですか。さらに、pictureが複数投稿できたら尚良さそうですよね。ということで、PostFormにPictureFormをembedすることにします。しかも、複数。ここは決め打ちで3つとします。まぁ、サンプルなので。
PostFormのバリデーションの記述の後にでも、次のコードを追記してみましょう。

  1. $pictures = ($this->getObject()->isNew()) ? null: $this->getObject()->getPictures();
  2.     for ($i = 0; $i <3; $i++) {
  3.       $picture = (isset($pictures) and isset($pictures[$i])) ? $pictures[$i] : null;
  4.       $pictureForm = new PictureForm($picture);
  5.       $this->embedForm('picture_' . $i, $pictureForm, '%content%');
  6.       $this->widgetSchema['picture_' . $i]->setLabel('画像');
  7.     }

少し説明しますと、Postが新規投稿であれば、何もしないですが、編集の際には、PictureFormに初期値を設定させて、それをembedFormしていきます。embedFormをする際には、picture_0, picture_1, picture_2というnameを与えて指定してみることにします。また、embedですから入れ子になりますし、ちょっと見た目がダサくなるのでデコレーションは、%content%にしておきます。ついでにlabelも指定しておきます。Formとして表示させるには、これだけ追加するだけで、左の画像のように、title, descriptionだけではなく、3つのPictureFormつまり、filename, captionが3つあるフォームができあがります。

ただ、これだけでは、このpictureがこのpostに関連づけられているかどうかがわからないので、PostFormのupdateObjectでsetPostしてあげます。

  1. public function updateObject($values = null)
  2.   {
  3.     $object = parent::updateObject($values);
  4.     $values = $this->getValues();
  5.  
  6.     foreach ($this->embeddedForms as $i => $picture) {
  7.       if ($picture->getObject()) {
  8.         if ($picture->getObject()->getFilename() == '') {
  9.           unset($this->embeddedForms[$i]);
  10.         } else {
  11.           $picture->getObject()->setPost($object);
  12.         }
  13.       }
  14.     }
  15.   }

ここは少し説明がいりますね。まず、PostのupdateObjectはそのまま親のメソッド呼んで、設定させておきます。embedFormでembedしたFormオブジェクトは、このPostFormクラスのembedFormsフィールドに配列で保持されているので(sfForm.class.php参照)、これをループで回して、どのpostと関連つけているかを指定してあげます。ついでにファイルがアップロードされない場合も保存されてしまいますので、getFilename()が空文字列の場合は、embedFormsからその要素をunsetしておきます。これで、ファイルが入っていたときのみ、保存されることになります。

これで新規投稿に関してはファイルがアップロードがちゃんとできるようになりました。また、編集時には、ちゃんとそのフォームに画像が表示されます。
しかし、このままでは実は編集時に使うfilename_deleteを使うことができないのですね。このままですと、そのpictureオブジェクトを削除せずにpictureオブジェクトのfilenameのカラムを空にしてくれるだけです。まぁ、当たり前と言えば当たり前か。ということで、ファイルを削除できるようにしてみます。

チェックボックスには、filename_deleteという名前を与えているので、フォームから渡ってきた値にfilename_deleteがあった場合には、そのPictureのオブジェクトをインスタンス変数のremovePicturesという配列に格納しておくことにします。そして、doSaveでDBをいじるときに一緒にremovePicturesにあるPictureオブジェクトを削除することにします。ということで、最終的にできあがったPostFormは以下のようになります。

  1. class PostForm extends BasePostForm
  2. {
  3.   private $removePictures = array();
  4.   public function configure() {
  5.     unset($this['created_at'], $this['updated_at']);
  6.  
  7.     $this->setWidgets(array(
  8.       'id' => new sfWidgetFormInputHidden(),
  9.       'title' => new sfWidgetFormInput(array(
  10.         'label' => 'タイトル',
  11.       )),
  12.       'description' => new sfWidgetFormTextarea(array(
  13.         'label' => '説明',
  14.       )),
  15.     ));
  16.  
  17.     $this->setValidators(array(
  18.       'id' => new sfValidatorPropelChoice(
  19.         array('model' => 'Post', 'column' => 'id', 'required' => false)
  20.       ),
  21.       'title' => new sfValidatorString(
  22.         array('max_length' => 128, 'min_length' => 3),
  23.         array(
  24.           'max_length' => 'タイトルは128文字以内でお願いします。',
  25.           'min_length' => 'タイトルは3文字以上でお願いします。'
  26.         )
  27.       ),
  28.  
  29.       'description' => new sfValidatorString(
  30.         array('max_length' => 2048, 'min_length' => 3),
  31.         array(
  32.           'max_length' => '説明本文は2000文字以内でお願いします。',
  33.           'min_length' => '説明本文は3文字以上でお願いします。'
  34.         )
  35.       ),
  36.     ));
  37.  
  38.     $pictures = ($this->getObject()->isNew()) ? null: $this->getObject()->getPictures();
  39.     for ($i = 0; $i <3; $i++) {
  40.       $picture = (isset($pictures) and isset($pictures[$i])) ? $pictures[$i] : null;
  41.       $pictureForm = new PictureForm($picture);
  42.       $this->embedForm('picture_' . $i, $pictureForm, '%content%');
  43.       $this->widgetSchema['picture_' . $i]->setLabel('画像');
  44.     }
  45.  
  46.     $this->widgetSchema->setNameFormat('post[%s]');
  47.   }
  48.  
  49.   public function updateObject($values = null)
  50.   {
  51.     $object = parent::updateObject($values);
  52.     $values = $this->getValues();
  53.  
  54.     foreach ($this->embeddedForms as $i => $picture) {
  55.       if (isset($values[$i]['filename_delete'])) {
  56.         $this->removePictures[] = $picture->getObject();
  57.       } else if ($picture->getObject()->getFilename() == '') {
  58.         unset($this->embeddedForms[$i]);
  59.       } else {
  60.         $picture->getObject()->setPost($object);
  61.       }
  62.     }
  63.   }
  64.  
  65.   public function doSave($con = null)
  66.   {
  67.     parent::doSave($con);
  68.     foreach ($this->removePictures as $p) {
  69.       $p->delete($con);
  70.     }
  71.   }
  72. }

symfony1.2を使いこなせるかどうかのポイントの一つはsfFormがちゃんと使えるかどうかになると思います。今後もsfFormを追いかけてみようと思います。
propel:generate-module post Postで生成されたindexSuccess.phpも少しだけ修正して、ちゃんと一覧画面に表示させるようにてみますか。

  1. <h1>Post List</h1>
  2.  
  3. <table>
  4.   <thead>
  5.     <tr>
  6.       <th>Id</th>
  7.       <th>Title</th>
  8.       <th>Description</th>
  9.       <th>Created at</th>
  10.       <th>Pictures</th>
  11.     </tr>
  12.   </thead>
  13.   <tbody>
  14.     <?php foreach ($post_list as $post): ?>
  15.     <tr>
  16.       <td><a href="<?php echo url_for('post/edit?id='.$post->getId()) ?>"><?php echo $post->getId() ?></a></td>
  17.       <td><?php echo $post->getTitle() ?></td>
  18.       <td><?php echo $post->getDescription() ?></td>
  19.       <td><?php echo $post->getCreatedAt() ?></td>
  20.       <td>
  21.       <?php foreach ($post->getPictures() as $picture): ?>
  22.       <img src="/uploads/<?php echo $picture->getFilename() ?>" alt="<?php echo $picture->getCaption() ?>" width="100" />
  23.       <?php endforeach; ?>
  24.       </td>
  25.     </tr>
  26.     <?php endforeach; ?>
  27.   </tbody>
  28. </table>
  29.  
  30.   <a href="<?php echo url_for('post/new') ?>">New</a>

本当は、編集フォームページにあるDeleteボタンを押すと、ちゃんとデータベースからはPostとそれに関連するPictureが削除されるのですが、実際の画像はunlinkされないので、そこの修正も必要になりますね。編集フォームページで画像だけを削除する際には、sfPropelFormにあるremoveFileが呼ばれますので、unlinkされるのですが、単純にdeleteアクションだけを呼んだ場合には、formクラスとは別のロジックになりますので、ファイルは消えませんので、Pictureモデルのdeleteにでも書いて置かないといけなさそうですね。ちょっとredundantなので、微妙だなぁーと思って今回は、それは載せませんでした。つーか、そもそも削除させないとかw

今回は、サンプルでしたので、書きませんでしたが、画像をconvertしたりする必要があるかもしれないですね。私が作っている本番用では、同時にサムネイルを作ったり、もっとformが複雑だったり、キャプションの与え方が動的だったりします。しかし、embedFormのやり方はこのままですので、これを元にすれば、いろんなところに適応ができそうです。

今回は、symfony1.2を用いて、複数の画像投稿を実現するために、embedFormを用いて実装する方法を説明しました。確かにその複数分ループで回せばいいのですが、embedFormForEachがちゃんと動けば、そちらの方がスマートな気がします。それができるまでは、このembedFormで実現しましょう。

間違いがありましたら、教えてくださいな。

Bloglines feedburner