| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 |
- <?xml version="1.0" encoding="UTF-8"?>
- <!-- Reviewed: no -->
- <sect1 id="learning.plugins.usage">
- <title>Using Plugins</title>
- <para>
- Components that make use of plugins typically use
- <classname>Zend_Loader_PluginLoader</classname> to do their work. This class has you
- register plugins by specifying one or more "prefix paths". The component will then call the
- PluginLoader's <methodname>load()</methodname> method, passing the plugin's short name to
- it. The PluginLoader will then query each prefix path to see if a class matching that short
- name exists. Prefix paths are searched in LIFO (last in, first out) order, so it will match
- those prefix paths registered last first -- allowing you to override existing plugins.
- </para>
- <para>
- Some examples will make all of this more clear.
- </para>
- <example id="learning.plugins.usage.basic">
- <title>Basic Plugin Example: Adding a single prefix path</title>
- <para>
- In this example, we will assume some validators have been written and placed in the
- directory <filename>foo/plugins/validators/</filename>, and that all these classes share
- the class prefix "Foo_Validate_"; these two bits of information form our "prefix path".
- Furthermore, let's assume we have two validators, one named "Even" (ensuring a number to
- be validated is even), and another named "Dozens" (ensuring the number is a multiple of
- 12). The tree might look like this:
- </para>
- <programlisting language="text"><![CDATA[
- foo/
- |-- plugins/
- | |-- validators/
- | | |-- Even.php
- | | |-- Dozens.php
- ]]></programlisting>
- <para>
- Now, we'll inform a <classname>Zend_Form_Element</classname> instance of this prefix
- path. <classname>Zend_Form_Element</classname>'s
- <methodname>addPrefixPath()</methodname> method expects a third argument that indicates
- the type of plugin for which the path is being registered; in this case, it's a
- "validate" plugin.
- </para>
- <programlisting language="php"><![CDATA[
- $element->addPrefixPath('Foo_Validate', 'foo/plugins/validators/', 'validate');
- ]]></programlisting>
- <para>
- Now we can simply tell the element the short name of the validators we want to use. In
- the following example, we're using a mix of standard validators ("NotEmpty", "Int") and
- custom validators ("Even", "Dozens"):
- </para>
- <programlisting language="php"><![CDATA[
- $element->addValidator('NotEmpty')
- ->addValidator('Int')
- ->addValidator('Even')
- ->addValidator('Dozens');
- ]]></programlisting>
- <para>
- When the element needs to validate, it will then request the plugin class from the
- PluginLoader. The first two validators will resolve to
- <classname>Zend_Validate_NotEmpty</classname> and
- <classname>Zend_Validate_Int</classname>, respectively; the next two will resolve to
- <classname>Foo_Validate_Even</classname> and <classname>Foo_Validate_Dozens</classname>,
- respectively.
- </para>
- </example>
- <note>
- <title>What happens if a plugin is not found?</title>
- <para>
- What happens if a plugin is requested, but the PluginLoader is unable to find a class
- matching it? For instance, in the above example, if we registered the plugin "Bar" with
- the element, what would happen?
- </para>
- <para>
- The plugin loader will look through each prefix path, checking to see if a file matching
- the plugin name is found on that path. If the file is not found, it then moves on to the
- next prefix path.
- </para>
- <para>
- Once the stack of prefix paths has been exhausted, if no matching file has been found,
- it will throw a <exceptionname>Zend_Loader_PluginLoader_Exception</exceptionname>.
- </para>
- </note>
- <example id="learning.plugins.usage.override">
- <title>Intermediate Plugin Usage: Overriding existing plugins</title>
- <para>
- One strength of the PluginLoader is that its use of a LIFO stack allows you to override
- existing plugins by creating your own versions locally with a different prefix path, and
- registering that prefix path later in the stack.
- </para>
- <para>
- For example, let's consider <classname>Zend_View_Helper_FormButton</classname> (view
- helpers are one form of plugin). This view helper accepts three arguments, an element
- name (also used as the element's DOM identifier), a value (used as the button label),
- and an optional array of attributes. The helper then generates <acronym>HTML</acronym>
- markup for a form input element.
- </para>
- <para>
- Let's say you want the helper to instead generate a true <acronym>HTML</acronym>
- <constant>button</constant> element; don't want the helper to generate a DOM identifier,
- but instead use the value for a CSS class selector; and that you have no interest in
- handling arbitrary attributes. You could accomplish this in a couple of ways. In both
- cases, you'd create your own view helper class that implements the behavior you want;
- the difference is in how you would name and invoke them.
- </para>
- <para>
- Our first example will be to name the element with a unique name:
- <classname>Foo_View_Helper_CssButton</classname>, which implies the plugin name
- "CssButton". While this certainly is a viable approach, it poses several issues: if
- you've already used the Button view helper in your code, you now have to refactor;
- alternately, if another developer starts writing code for your application, they may
- inadvertently use the Button view helper instead of your new view helper.
- </para>
- <para>
- So, the better example is to use the plugin name "Button", giving us the class name
- <classname>Foo_View_Helper_Button</classname>. We then register the prefix path with the
- view:
- </para>
- <programlisting language="php"><![CDATA[
- // Zend_View::addHelperPath() utilizes the PluginLoader; however, it inverts
- // the arguments, as it provides a default value of "Zend_View_Helper" for the
- // plugin prefix.
- //
- // The below assumes your class is in the directory 'foo/view/helpers/'.
- $view->addHelperPath('foo/view/helpers', 'Foo_View_Helper');
- ]]></programlisting>
- <para>
- Once done, anywhere you now use the "Button" helper will delegate to your custom
- <classname>Foo_View_Helper_Button</classname> class!
- </para>
- </example>
- </sect1>
|