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


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



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 18 hours ago
@DmitryBaranovsk @avalanche123 Epic!
about 19 hours ago
@davemcclure @500Startups and what if I want to be a Tony Stark?
about 20 hours ago
Do you have a startup? Does your startup have any competence? Register them in: http://t.co/mQBsMS21dY
1 day ago
7 Simple Tips To Grow Your Social Network | LinkedIn http://t.co/QxiASZKd8T
2 days ago
@Plaza21com was added to http://t.co/D640KCry9Y . Add your local, regional or global competition and follow them w/ centralized stats.
3 days ago
@regples was added to http://t.co/D640KCry9Y . Add your local, regional or global competition and follow them w/ centralized stats.
3 days ago
@POPContacts was added to http://t.co/D640KCry9Y . Add your local, regional or global competition and follow them w/ centralized stats.
11 days ago
How Funding Works - Splitting The Equity With Investors - Infographic http://t.co/luzjgoWreg
about 1 month ago
Gran presentación de David Chandler@drfibonaccii. En el#AndroidTourPeruu Gracias!
about 1 month ago
Does somebody know about a trusted and reliable proxy provider for intensive web scraping?
2 months ago
Crean Red social exclusiva para parejas que ayudaría a combatir la infidelidad e incrementar romanticismo. http://t.co/H2mO2rIBSdd
2 months ago
Reading Startup's Owner Manual: 'Perhaps the ugliest (or most flattering) comment: “It seems to crash my computer every time I use it,”'
2 months ago
@shesho Genial presentación acerca de#UXX gracias! ;)
4 months ago
Hurray! I was one of the first 500,000 @LinkedIn members in Peru. http://t.co/utglPcjT
5 months ago
TechCrunch Giveaway: Nexus 4 And Nexus 10 #TechCrunch http://t.co/9DpeeLPZ via @TechCrunch
7 months ago
Lo bueno que Rommey no dijo en la campaña lo dijo al final.
8 months ago
A real pleasure to work with #Symfony2. :)
8 months ago
Actualizando a #Eclipse Juno.
8 months ago
At last coming back to you my dear #symfony. Updating my knowledge about #symfony2.
8 months ago
@maristiglichv cómo has estado? :)
8 months ago
Mark Zuckerberg:Facebook already processes about 1 billion search queries a day without "even trying."
8 months ago
Brazilian Startup Queremos Is A Kickstarter For Concerts And Other Live Eventshttp://t.co/zOOHt0DPP via@TechCrunchh
8 months ago
FoundationDB — Not Your Standard NoSQL Databahttp://t.co/gbSXk8Zo8Zo v@TechCrunchnch
9 months ago
The perfect match. The first glance can determine how the rest of the story ends. http://t.co/Th1V6AX6
9 months ago
One Month After Selling To Microsoft For $1.2B, Yammer CEO Predicts End Of Silicon Valley http://t.co/9mVTdqFD via @TechCrunch
10 months ago
but because she understands and is able to personally defend, own, and drive forward with implementing the decision
10 months ago
she should be saying so not because you are in a position of authority and you said so,
10 months ago
Once that person is at the point where she is willing to say “yes” to the decision,
about 1 year ago
Writing clean code in PHP 5.4 | Web Builder Zone: http://t.co/IAFFj3A8 via @addthis
about 1 year ago
Interesting tips about #scaling http://t.co/QC2paoDK
about 1 year ago
Learning to use #Windows #ActiveDirectory
about 1 year ago
Ayni - Blog: CUESTIONARIOS PARA ANÁLISIS-http://localhost:8080/AYNI-war/faces/ListarComentario.xhtml?txtIdPublicacion=2
about 1 year ago
Día tranquilo en casa :)#fbb
about 1 year ago
#Adobe Reader App crushed my new #Android movile :/
about 1 year ago
Testing from Android =D
about 1 year ago
#Refactoring code.
about 1 year ago
You should Snog, Marry or Avoid me http://t.co/pZCDIwW
about 1 year ago
Yahoo’s Options v@TechCrunchnhttp://t.co/yjzsKstKst What will Yahoo do?
about 1 year ago
Just realize my post http://t.co/nSuJjSp, written so many time ago, really contribute to make it happen: http://t.co/wMqYrV3
about 1 year ago
An Introduction to Redirecting URLs on an Apache Server http://t.co/s79ThOy via @WebmasterWorld
about 1 year ago
Actualizando a #Eclipse Indigo!
about 1 year ago
Vota x lamula.pe: http://t.co/WTWMkFG via @addthis
about 1 year ago
Apple Pushes Past Exxon To Become The Most Valuable Public Company In The World (Temporarily) via@TechCrunchhhttp://t.co/lowONSZZ
about 1 year ago
finally this class #semester is finished, new #projects in mind
about 1 year ago
Mark Zuckerberg Explains His Law Of Social Sharing [Video]http://t.co/sqq10Ehh via@TechCrunchh
about 1 year ago
retomando mi #twitter #notime
about 1 year ago
How Facebook Can Put Google Out of Businesshttp://t.co/HqDfQGoo via@TechCrunchh
over 2 years ago
So why not just cut out the middle man? Microsoft.http://techcrunch.com/2011/05/15/samsung-series-5-chromebook/
over 2 years ago
really like to #design class hierarchies with #compositepattern
over 2 years ago
Similarly, Microsoft.com started to use jQuery instead of their own ASP.NET Ajax framework. They are still using Windows, for whatever XD
over 2 years ago
Estudiantes de la PUCP le “voltean” campaña a esposo de Keiko | yoperiodhttp://t.co/ruI9UnmI9Unm@lamulaamula
over 2 years ago
Reading: Apress.-.Pro.PHP.Application.Performance.2010 - Very Insteresting #php #performance #read
over 2 years ago
debug_backtrace() is very important on certain situations. #php #debug
over 2 years ago
@alvarezrodrich felicitaciones!
over 2 years ago
time to do some #uml diagrams, #classdiagram
over 2 years ago
hoy es el día,#votaa conciente#peruu
over 2 years ago
@skoop @funkatron I think someone had a bad day!, #Frameworks are there but you don't have to use them.
over 2 years ago
making #wireframes for a new #functionality
over 2 years ago
so much #spam on my #blog =(
over 2 years ago
learning new topics and tools that I did not use before #rcp
over 2 years ago
aprendiendo muchos temas y herramientas que no utilizaba antes #rcp
over 2 years ago
#tweaking httpd.conf #virtualhost
over 2 years ago
thanks #symfony 1.4, even when i'm not using the entire #framework, yours classes save my life!
over 2 years ago
integrating with #SOA using #soap
over 2 years ago
It seems my most #productive working hours are on #sunday #afternoon #evening! XD
over 2 years ago
#ASOT 500 =)
over 2 years ago
installing SCA_SDO on #Centos #php
over 2 years ago
My web service using #soap worked!!!!! #php #SCA #SDO
over 2 years ago
Working in a new place since last week!, #RCP: Red Científica Peruana, the one which sells the .pe domains in#Perúú ->#happypy
over 2 years ago
Finally with a new #laptop: #Toshiba =)
over 2 years ago
@pasku1 Thanks, I will try Pivotaltracker.
over 2 years ago
@doolphy thanks for your answer doolphy! I'll try you!
over 2 years ago
@jmasson thanks for your answer! Jira + Confluence is a good combination.
over 2 years ago
Which is the best project management and collaboration tool right now? #projectmanagement #tool #collaboration
over 2 years ago
What a #voice! Sied Van Riel feat Nicole McKenna - Stealing Time (Aly & Fila Remix) + #ASOT 497 #trancefamily
over 2 years ago
working on a situation where #php #traits would be very useful
over 2 years ago
It is #awesome when you finish doing a lot of changes and nothing is broken =) #TDD #testing
over 2 years ago
oh, happy birthday! @mtabini o mejor dicho feliz cumpleaños!
over 2 years ago
why do #IE8 not accept #javascript "const" keyword?
over 2 years ago
Discovering there is much #more to do with #javascript ... a lot.
over 2 years ago
#composition over #inheritance: #javascript
over 2 years ago
Awesome #song!: Cerf, Mitiska & Jaren - Another World (Original Vocal Mix) #ASOT 495 #trancefamily
over 2 years ago
My legal woman is #PHP, but I have an affair with #Javascript, overall when she wears #jQuery.
over 2 years ago
This presentation is one of the best I've seen about #unit #testing http://www.slideshare.net/avalanche123/clean-code-5609451
over 2 years ago
where to do a master on #IT: US or Spain? #survey #php #master plz RT
over 2 years ago
Bobina feat. Betsie Larkin - You Belong To Me: What a #beautiful voice -> #ASOT 494 #arminvanbuuren
over 2 years ago
OH NO, IT'S MONDAY -- 2011-02-07 http://t.co/pgaIxe5 via @gojkoadzic
over 2 years ago
it was not a + b, it was parserInt(a) + parseInt(b) =(, #javascript #fail
over 2 years ago
The models are complete representations of the system, whereas an #architectural #view focuses only on what is architecturally #significant.
over 2 years ago
@jmasson that would be great and finally #wikimedia, #drupal, #wordpress and maybe #joomla would push towards the same side, the #php side.
over 2 years ago
@jmasson Thanks!, #PHP has a bright #future ahead.
over 2 years ago
A new #blog post about not reinventing the #wheel: http://www.jnieto.org/article/symfony2_phpbb4_and_drupal8 #symfony #phpbb #drupal
over 2 years ago
@giorgiosironi #indeed, that's a very good #question. I think an average of 4 but also depends on how much that #developer work.
over 2 years ago
It seems the new platform for deploying, managing and scaling PHP apps is http://orchestra.io/ #cool
over 2 years ago
#Phase project planning vs #iteration project #planning - #project #management
over 2 years ago
#jeditable with #jquery save my life =)
over 2 years ago
Amazing #song -> Sied van Riel feat. Nicole McKenna - Stealing Time #ASOT 493 #trancefamily
over 2 years ago
I simply love "offset" #jquery function =)
over 2 years ago
I really don't understand why projects like #drupal does not base their components in projects like #doctrine and #symfony
over 2 years ago
OH NO, IT'S MONDAY -- 2011-01-17 http://t.co/37pr4Bd via @gojkoadzic
over 2 years ago
@alvarezrodrich me alegra ver q ya borró su cuenta Sr. Rodrich,#twitterr es malo jajja, XD
over 2 years ago
Acabo de hacerle a mi #brother @diegonl89 un blog para que hable de #actualidad en general: http://www.elgatotechero.com #peru
over 2 years ago
I just made to my #brother a #blog to talk about current #events in #peru: http://www.elgatotechero.com
over 2 years ago
Amazing things can be done with #javascript and #css, and of course with the help of #jQuery =)
over 2 years ago
An architecturally significant element is an element that is important for #understanding the #system.
over 2 years ago
An architecturally element has a wide impact on the #structure, #performance, #robustness, #evolvability, and #scalability of a #system.
over 2 years ago
@giorgiosironi Definitely!
over 2 years ago
Playing with #table #inheritance in #Doctrine
over 2 years ago
Reading about #RUP, and how addresses the #major difficulties in a new #project.
over 2 years ago
Yandex in 2010: 43 percent revenue growth http://t.co/cpjT5Jw via @cnet
over 2 years ago
Going forward!!!!! =) poco a poco llegan los resultados de tanto esfuerzo #fb
over 2 years ago
Perfect #system with respect to the #requirements but the #wrong system with respect to the #real #problem at the time of #delivery.
over 2 years ago
Going #forward! =) #fb
over 2 years 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