Model-driven types

In the previous section, we defined two types by using Zope schema. In this section, we're going to define a type's fields using an XML model file.

The great advantage of using a model file is that we can prototype the content type in Dexterity's through-the-web field editor, then export the XML model file for incorporation into our package.

XML may be used to do pretty much anything you could do via Zope schema. Many users not already schooled in Zope schema will find this by far the easiest and fastest way to create Dexterity content types.

Adding the type

As in the previous section, we'll use addcontent to add our content type to the project. This type will be for conference presenters.

$ ../../bin/paster addcontent dexterity_content
Enter contenttype_name (Content type name ) ['Example Type']: Presenter
Enter contenttype_description (Content type description ) ['Description of the Example Type']: A person presenting a conference session
Enter folderish (True/False: Content type should act as a container ) [False]: False
Enter global_allow (True/False: Globally addable ) [True]:
Enter allow_discussion (True/False: Allow discussion ) [False]:

Setting the field model

Look in example.conference/example/conference/models/presenter.xml for a bare model file created by addcontent. Let's elaborate it:

<model xmlns:form="http://namespaces.plone.org/supermodel/form"
       xmlns:security="http://namespaces.plone.org/supermodel/security"
       xmlns:marshal="http://namespaces.plone.org/supermodel/marshal"
       xmlns="http://namespaces.plone.org/supermodel/schema">
  <schema>
    <field name="name" type="zope.schema.TextLine">
      <description/>
      <title>Name</title>
    </field>
    <field name="description" type="zope.schema.Text">
      <description/>
      <title>A short summary</title>
    </field>
    <field name="bio" type="plone.app.textfield.RichText">
      <description/>
      <required>False</required>
      <title>Bio</title>
    </field>
    <field name="photo" type="plone.namedfile.field.NamedBlobImage">
      <description>Please upload an image.</description>
      <required>False</required>
      <title>Photo</title>
    </field>I
  </schema>
</model>

The XML name spaces we use are described in the Dexterity XML reference section.

That's all we need! To see why, look in the generated file presenter.py:

from example.conference import MessageFactory as _
from plone.supermodel import model
from zope import schema


class IPresenter(model.Schema):

    """Schema for Conference Presenter content type."""

    model.load('models/presenter.xml')

Note the model.load directive. We'd deleted that when we created schema-driven field sets. Now, we leave it in to automatically load our model file.

Setting Factory Type Information

This part of the process is identical to what we explained for schema-driven type.

Look in the types.xml file in your packages example/conference/profiles/default directory. It should now contain:

<object name="portal_types">
  <object name="example.conference.program" meta_type="Dexterity FTI" />
  <object name="example.conference.session" meta_type="Dexterity FTI" />
  <object name="example.conference.presenter" meta_type="Dexterity FTI" />
</object>

For the Presenter type, we have example.conference.presenter.xml:

<?xml version="1.0"?>
<object name="example.conference.presenter" meta_type="Dexterity FTI"
     xmlns:i18n="http://xml.zope.org/namespaces/i18n"
    i18n:domain="example.conference">

  <!-- Basic metadata -->
  <property name="title" i18n:translate="">Presenter</property>
  <property name="description" i18n:translate="">Conference Presenter</property>
  <property name="icon_expr">string:${portal_url}/document_icon.png</property>
  <property name="factory">example.conference.presenter</property>
  <property name="global_allow">True</property>
  <property name="filter_content_types">True</property>
  <property name="allowed_content_types" />
  <property name="allow_discussion">False</property>

  <!-- schema and class used for content items -->
  <property name="schema">example.conference.presenter.IPresenter</property>
  <property name="klass">example.conference.presenter.Presenter</property>

  <property name="behaviors">
    <element value="plone.app.content.interfaces.INameFromTitle" />
   </property>

  <!-- View information -->
  <property name="link_target"></property>
  <property name="immediate_view">view</property>
  <property name="default_view">view</property>
  <property name="view_methods">
    <element value="view"/>
  </property>
  <property name="default_view_fallback">False</property>
  <property name="add_permission">cmf.AddPortalContent</property>

  <!-- Method aliases -->
  <alias from="(Default)" to="(dynamic view)" />
  <alias from="view" to="(selected layout)" />
  <alias from="edit" to="@@edit" />
  <alias from="sharing" to="@@sharing" />

  <!-- Actions -->
  <action title="View" action_id="view" category="object" condition_expr=""
      url_expr="string:${object_url}/" visible="True">
    <permission value="View" />
  </action>
  <action title="Edit" action_id="edit" category="object" condition_expr=""
      url_expr="string:${object_url}/edit" visible="True">
    <permission value="Modify portal content" />
  </action>
</object>

Note that this is addable anywhere.