Extending your Doctrine Model: Template Classes


Symfony | Technical | Development | Doctrine | January 20, 2011

A lot of times in the past, working with Symfony 1.X and Doctrine 1.X, I got into a problem. According with Model View Controller design pattern, you have to carefully choose what to put in each layer. For example if it is some business logic you have to put that in the Model, if it is some HTML as is in the web case you have to put that in the View and finally if it is some code to manage the view and model, you have to put that in the Controller.

It is easy to read that and say: "Hey I'm going to follow this cool pattern". However, when programming, against the clock it is not so simple. Maybe you put some business logic in the controller or perhaps you put some view related code into the model. The last case used to happen to me a lot of times and I'll show you why.

The Problem

Let's say you have this schema:

 
Employee:
  tableName:    t_employee
  actAs:
    Sluggable:  { fields: [ name ] }
  columns:
    id:         { type: integer  , length: 20 , primary: true, autoincrement: true }
    name:       { type: string   , length: 50                , notnull: true       }
    web:        { type: string   , length: 255                                     }
    photograph: { type: string   , length: 255                                     }
 

Here we got an Employee entity with three attributes: name, web and photograph.

In your templates you are going to use surely these two statements:

 
link_to($employee->getName(), $employee->getWeb(), array('title' => $employee->getName()))
 

This shows the name of the employee as a link to his web page.

 
link_to(image_tag($employee->getPhotograph()), $employee->getWeb(), array('title' => $employee->getName()))
 

This shows the photograph of the employee as a link to his web page.

Because you are going to use this two statements a lot let's put them in a place where its reuse is guarantee:

 
class Employee extends BaseEmployee
{
  public function getLinkToWeb()
  {
    return link_to($employee->getName(), $employee->getWeb(), array('title' => $employee->getName()));
  }
 
  public function getLinkToWebWithPhotograph()
  {
    return link_to(image_tag($employee->getPhotograph()), $employee->getWeb(), array('title' => $employee->getName()));
  }
}
 

Now you can reuse this code for all your templates ... Wrong!!!!!

We were talking about separating of concerns with MVC pattern and you mix view code with the model. The idea behind the model is to have just the logic related to the data and no more. Imagine you are going to use your model with a different markup than HTML, then these functions will make  no sense at all.

Now you know the problem I had.

 

The Solution: Template classes

It would be great if all these methods could continue in the model classes, but remember: that's wrong. Then let's create an extension of our model to work just with our HTML view.

The How

1.- Create in your project/lib/model the template directory.

2.- Add to the previous directory the EmployeeTemplate.class.php file:

 
class EmployeeTemplate extends sfDoctrineTemplateExt
{
  public function getLinkToWeb()
  {
    return link_to($employee->getName(), $employee->getWeb(), array('title' => $employee->getName()));
  }
 
  public function getLinkToWebWithPhotograph()
  {
    return link_to(image_tag($employee->getPhotograph()), $employee->getWeb(), array('title' => $employee->getName()));
  }
}
 

 

3.- Now, to properly connect your model with the template classes you need: sfDoctrineTemplateExt class

 
/**
 * sfDoctrineTemplateExt
 *
 * @package    symfext
 * @subpackage doctrine
 * @author     Jonathan Olger Nieto Lajo [johan.nieto.lajo@gmail.com]
 */
abstract class sfDoctrineTemplateExt
{
  protected
    $object = null;
 
  public function __construct(DoctrineRecord $object)
  {
    $this->object = $object;
  }
 
  public function __call($method, $arguments)
  {
    return call_user_func_array(array($this->object, $method), $arguments);
  }
}
 

 

4.- Update your sfDoctrineRecordExt file. As you may know Doctrine gives the advantage of changing the inheritance hierarchy of your model in one level. That is, you can put a class (or many) in the middle of the inheritance hierarchy. In my case, the whole hierarchy would be:

Doctrine_Record_Abstract > Doctrine_Record > sfDoctrineRecord > sfDoctrineRecordExt > BaseEmployee > Employee

 
/**
 * sfDoctrineRecordExt
 *
 * @package    symfext
 * @subpackage doctrine
 * @author     Jonathan Olger Nieto Lajo [johan.nieto.lajo@gmail.com]
 */
abstract class sfDoctrineRecordExt extends sfDoctrineRecord
{
  /**
   * Returns the corresponding template instance of this model.
   * 
   * @param boolean $force Whether or not the method should force the conversion
   * 
   * @throws RuntimeException If force and the corresponding template class does not exist
   * 
   * @return DoctrineRecord|DoctrineTemplate The result of the conversion
   */
  public function toTemplate($force = true)
  {
    $template_class = $this->getTable()->getComponentName().'Template';
 
    if (class_exists($template_class))
    {
      $template = new $template_class($this);
 
      return $template;
    }
 
    if ($force)
    {
      throw new RuntimeException(sprintf('The "%s" class does not exist. You can\'t convert a sfDoctrineRecordExt to a template without the template class', $template_class));
    }
 
    return $this;
  }
}
 

 

5.- Try your brand new functionality in the showEmployeeSuccess.php file (it is assumed an employee object is passed from the action to the view):

 
  ?php echo $employee->toTemplate()->getLinkToWeb() ?php>
 

Or

 
  ?php echo $employee->toTemplate()->getLinkToWebWithPhotograph() ?php>
 

 

Finally, the  model is free from view related code and there is the possibility to reuse methods and functionality on an entity basis.


Comments


coorubcodenda said about 11 days later

bardzo ciekawe, dzieki

Jonah Turnquist said about about 1 month later

To solve the above problem in Yii I would probably create a custom helper class.

"Helper classes: in views we often need some code snippets to do tiny tasks such as formatting data or generating HTML tags. Rather than placing this code directly into the view files, a better approach is to place all of these code snippets in a view helper class. Then, just use the helper class in your view files. Yii provides an example of this approach. Yii has a powerful CHtml helper class that can produce commonly used HTML code. Helper classes may be put in an autoloadable directory so that they can be used without explicit class inclusion."

- http://www.yiiframework.com/doc/guide/1.1/en/basics.best-practices#view



Hey Stranger, leave a comment:

Last Posts

Autoloading: Symfony vs Yii

Symfony2, PhpBB4 and Drupal8

Type and boolean columns with Doctrine and Symfony

Refactoring queries with Doctrine

Extending your Doctrine Model: Template Classes

Integrating Doctrine: Symfony vs Yii

Passing parameters from the action to the view: Symfony vs Yii

Yii framework

Adding custom information to your Doctrine schema

PHP Advent Calendar 2010


My Tweets

about 14 hours ago
Writing clean code in PHP 5.4 | Web Builder Zone: http://t.co/IAFFj3A8 via @addthis
3 months ago
Interesting tips about #scaling http://t.co/QC2paoDK
3 months ago
Learning to use #Windows #ActiveDirectory
3 months ago
Ayni - Blog: CUESTIONARIOS PARA ANÁLISIS-http://localhost:8080/AYNI-war/faces/ListarComentario.xhtml?txtIdPublicacion=2
3 months ago
Día tranquilo en casa :)#fbb
4 months ago
#Adobe Reader App crushed my new #Android movile :/
4 months ago
Testing from Android =D
5 months ago
#Refactoring code.
6 months ago
You should Snog, Marry or Avoid me http://t.co/pZCDIwW
6 months ago
Yahoo’s Options v@TechCrunchnhttp://t.co/yjzsKstKst What will Yahoo do?
6 months ago
Just realize my post http://t.co/nSuJjSp, written so many time ago, really contribute to make it happen: http://t.co/wMqYrV3
6 months ago
An Introduction to Redirecting URLs on an Apache Server http://t.co/s79ThOy via @WebmasterWorld
6 months ago
Actualizando a #Eclipse Indigo!
6 months ago
Vota x lamula.pe: http://t.co/WTWMkFG via @addthis
7 months ago
Apple Pushes Past Exxon To Become The Most Valuable Public Company In The World (Temporarily) via@TechCrunchhhttp://t.co/lowONSZZ
7 months ago
finally this class #semester is finished, new #projects in mind
8 months ago
Mark Zuckerberg Explains His Law Of Social Sharing [Video]http://t.co/sqq10Ehh via@TechCrunchh
8 months ago
retomando mi #twitter #notime
9 months ago
How Facebook Can Put Google Out of Businesshttp://t.co/HqDfQGoo via@TechCrunchh
9 months ago
So why not just cut out the middle man? Microsoft.http://techcrunch.com/2011/05/15/samsung-series-5-chromebook/
10 months ago
really like to #design class hierarchies with #compositepattern
10 months ago
Similarly, Microsoft.com started to use jQuery instead of their own ASP.NET Ajax framework. They are still using Windows, for whatever XD
10 months ago
Estudiantes de la PUCP le “voltean” campaña a esposo de Keiko | yoperiodhttp://t.co/ruI9UnmI9Unm@lamulaamula
10 months ago
Reading: Apress.-.Pro.PHP.Application.Performance.2010 - Very Insteresting #php #performance #read
10 months ago
debug_backtrace() is very important on certain situations. #php #debug
11 months ago
@alvarezrodrich felicitaciones!
11 months ago
time to do some #uml diagrams, #classdiagram
11 months ago
hoy es el día,#votaa conciente#peruu
11 months ago
@skoop @funkatron I think someone had a bad day!, #Frameworks are there but you don't have to use them.
11 months ago
making #wireframes for a new #functionality
11 months ago
so much #spam on my #blog =(
11 months ago
learning new topics and tools that I did not use before #rcp
11 months ago
aprendiendo muchos temas y herramientas que no utilizaba antes #rcp
11 months ago
#tweaking httpd.conf #virtualhost
11 months ago
thanks #symfony 1.4, even when i'm not using the entire #framework, yours classes save my life!
11 months ago
integrating with #SOA using #soap
11 months ago
It seems my most #productive working hours are on #sunday #afternoon #evening! XD
11 months ago
#ASOT 500 =)
11 months ago
installing SCA_SDO on #Centos #php
11 months ago
My web service using #soap worked!!!!! #php #SCA #SDO
11 months ago
Working in a new place since last week!, #RCP: Red Científica Peruana, the one which sells the .pe domains in#Perúú ->#happypy
12 months ago
Finally with a new #laptop: #Toshiba =)
12 months ago
@pasku1 Thanks, I will try Pivotaltracker.
12 months ago
@doolphy thanks for your answer doolphy! I'll try you!
12 months ago
@jmasson thanks for your answer! Jira + Confluence is a good combination.
12 months ago
Which is the best project management and collaboration tool right now? #projectmanagement #tool #collaboration
12 months ago
What a #voice! Sied Van Riel feat Nicole McKenna - Stealing Time (Aly & Fila Remix) + #ASOT 497 #trancefamily
12 months ago
working on a situation where #php #traits would be very useful
12 months ago
It is #awesome when you finish doing a lot of changes and nothing is broken =) #TDD #testing
12 months ago
oh, happy birthday! @mtabini o mejor dicho feliz cumpleaños!
about 1 year ago
why do #IE8 not accept #javascript "const" keyword?
about 1 year ago
Discovering there is much #more to do with #javascript ... a lot.
about 1 year ago
#composition over #inheritance: #javascript
about 1 year ago
Awesome #song!: Cerf, Mitiska & Jaren - Another World (Original Vocal Mix) #ASOT 495 #trancefamily
about 1 year ago
My legal woman is #PHP, but I have an affair with #Javascript, overall when she wears #jQuery.
about 1 year ago
This presentation is one of the best I've seen about #unit #testing http://www.slideshare.net/avalanche123/clean-code-5609451
about 1 year ago
where to do a master on #IT: US or Spain? #survey #php #master plz RT
about 1 year ago
Bobina feat. Betsie Larkin - You Belong To Me: What a #beautiful voice -> #ASOT 494 #arminvanbuuren
about 1 year ago
OH NO, IT'S MONDAY -- 2011-02-07 http://t.co/pgaIxe5 via @gojkoadzic
about 1 year ago
it was not a + b, it was parserInt(a) + parseInt(b) =(, #javascript #fail
about 1 year ago
The models are complete representations of the system, whereas an #architectural #view focuses only on what is architecturally #significant.
about 1 year ago
@jmasson that would be great and finally #wikimedia, #drupal, #wordpress and maybe #joomla would push towards the same side, the #php side.
about 1 year ago
@jmasson Thanks!, #PHP has a bright #future ahead.
about 1 year ago
A new #blog post about not reinventing the #wheel: http://www.jnieto.org/article/symfony2_phpbb4_and_drupal8 #symfony #phpbb #drupal
about 1 year ago
@giorgiosironi #indeed, that's a very good #question. I think an average of 4 but also depends on how much that #developer work.
about 1 year ago
It seems the new platform for deploying, managing and scaling PHP apps is http://orchestra.io/ #cool
about 1 year ago
#Phase project planning vs #iteration project #planning - #project #management
about 1 year ago
#jeditable with #jquery save my life =)
about 1 year ago
Amazing #song -> Sied van Riel feat. Nicole McKenna - Stealing Time #ASOT 493 #trancefamily
about 1 year ago
I simply love "offset" #jquery function =)
about 1 year ago
I really don't understand why projects like #drupal does not base their components in projects like #doctrine and #symfony
about 1 year ago
OH NO, IT'S MONDAY -- 2011-01-17 http://t.co/37pr4Bd via @gojkoadzic
about 1 year ago
@alvarezrodrich me alegra ver q ya borró su cuenta Sr. Rodrich,#twitterr es malo jajja, XD
about 1 year ago
Acabo de hacerle a mi #brother @diegonl89 un blog para que hable de #actualidad en general: http://www.elgatotechero.com #peru
about 1 year ago
I just made to my #brother a #blog to talk about current #events in #peru: http://www.elgatotechero.com
about 1 year ago
Amazing things can be done with #javascript and #css, and of course with the help of #jQuery =)
about 1 year ago
An architecturally significant element is an element that is important for #understanding the #system.
about 1 year ago
An architecturally element has a wide impact on the #structure, #performance, #robustness, #evolvability, and #scalability of a #system.
about 1 year ago
@giorgiosironi Definitely!
about 1 year ago
Playing with #table #inheritance in #Doctrine
about 1 year ago
Reading about #RUP, and how addresses the #major difficulties in a new #project.
about 1 year ago
Yandex in 2010: 43 percent revenue growth http://t.co/cpjT5Jw via @cnet
about 1 year ago
Going forward!!!!! =) poco a poco llegan los resultados de tanto esfuerzo #fb
about 1 year ago
Perfect #system with respect to the #requirements but the #wrong system with respect to the #real #problem at the time of #delivery.
about 1 year ago
Going #forward! =) #fb
about 1 year ago
@sam_dark Ok thanks!, but I don't understand why in #Yii documentantion use $_GET and $_POST instead of CHttpRequest http://bit.ly/i5emoL