mirror of
https://github.com/silverstripe/silverstripe-multiform
synced 2024-10-22 09:05:49 +00:00
Update README.md for conventions and 3.1 API
This commit is contained in:
parent
f309a8539a
commit
22b7031de5
426
README.md
426
README.md
@ -1,13 +1,15 @@
|
||||
# MultiForm Module #
|
||||
# MultiForm Module
|
||||
|
||||
[![Build Status](https://secure.travis-ci.org/silverstripe/silverstripe-multiform.png?branch=master)](http://travis-ci.org/silverstripe/silverstripe-multiform)
|
||||
|
||||
## Introduction ##
|
||||
## Introduction
|
||||
|
||||
MultiForm is a module for Sapphire, allowing flow control for forms, and step process to be
|
||||
automatically determined based on configuration variables on each step class. It augments the existing Form class in Sapphire.
|
||||
MultiForm is a SilverStripe module, allowing flow control for forms, and step
|
||||
process to be automatically determined based on configuration variables on each
|
||||
step class. It augments the existing Form class in SilverStripe.
|
||||
|
||||
It's goal is to allow greater flexibility than features, so each individual implementation can be customised to the project requirements.
|
||||
The goal of the module is to allow greater flexibility than features, so each
|
||||
individual implementation can be customized to the project requirements.
|
||||
|
||||
## Maintainer Contact
|
||||
|
||||
@ -22,50 +24,77 @@ SilverStripe 2.4 or higher is required.
|
||||
|
||||
|
||||
* Abstracts fields, actions and validation to each individual step.
|
||||
* Maintains flow control automatically, so it knows which steps are ahead and behind. It also can retrieve an entire step process from start to finish, useful for a step list.
|
||||
* Persists data by storing it in the session for each step, once it's completed. The session is saved into the database.
|
||||
* Allows customisation of next, previous steps, saving, loading and finalisation of the entire step process
|
||||
* Allows for basic ability to branch a step by overloading the next step method with logic to switch based on a condition (e.g. a checkbox, or dropdown in the field data).
|
||||
* Ties a user logged in who is using the step process (if applicable). This means you can build extended security or logging.
|
||||
* Basic flexibility on the URL presented to the user when they are using the forms. By default it stores an encrypted hash of the session in the URL, but you can reference it by the ID instead. It's recommend that additional security, such as checking the user who first started the session be applied if you want to reference by ID.
|
||||
* Maintains flow control automatically, so it knows which steps are ahead and
|
||||
behind. It also can retrieve an entire step process from start to finish, useful
|
||||
for a step list.
|
||||
* Persists data by storing it in the session for each step, once it's
|
||||
completed. The session is saved into the database.
|
||||
* Allows customisation of next, previous steps, saving, loading and
|
||||
finalisation of the entire step process
|
||||
* Allows for basic ability to branch a step by overloading the next step method
|
||||
with logic to switch based on a condition (e.g. a checkbox, or dropdown in the
|
||||
field data).
|
||||
* Ties a user logged in who is using the step process (if applicable). This
|
||||
means you can build extended security or logging.
|
||||
* Basic flexibility on the URL presented to the user when they are using the
|
||||
forms. By default it stores an encrypted hash of the session in the URL, but you
|
||||
can reference it by the ID instead. It's recommend that additional security,
|
||||
such as checking the user who first started the session be applied if you want to reference by ID.
|
||||
|
||||
## What it doesn't do
|
||||
|
||||
* Automatically handle relation saving, e.g. MembershipForm manages the Member
|
||||
* Provide a complete package out of the box (you must write a bit of code using the tutorial!)
|
||||
* Automatically determine what to do at the end of the process, and where to save it
|
||||
* Provide a complete package out of the box (you must write a bit of code using
|
||||
the tutorial!)
|
||||
* Automatically determine what to do at the end of the process, and where to
|
||||
save it
|
||||
* Provide nicely presented URLs of each step (an enhancement, for the future)
|
||||
|
||||
Note: The *multiform* directory should sit in your SilverStripe project directory in the file system as a sibling of cms, jsparty and sapphire.
|
||||
Note: The *multiform* directory should sit in your SilverStripe root project
|
||||
directory in the file system as a sibling of cms and framework.
|
||||
|
||||
## Reporting bugs
|
||||
|
||||
If you've found a bug that should be fixed for future releases, then please make a ticket on our open source tracker. This helps to ensure we release less buggy software in the future!
|
||||
If you've found a bug that should be fixed for future releases, then please make
|
||||
a ticket on https://github.com/silverstripe/silverstripe-multiform/issues.
|
||||
|
||||
This helps to ensure we release less buggy software in the future!
|
||||
|
||||
Go here to add a new ticket to our tracker: http://open.silverstripe.com/newticket
|
||||
## Tutorial
|
||||
|
||||
The assumption is the developer who is starting this tutorial has an intermediate level of knowledge in SilverStripe, understands what "run dev/build?flush=1" means, and has written some custom PHP code in SilverStripe before.
|
||||
The assumption is the developer who is starting this tutorial has an
|
||||
intermediate level of knowledge in SilverStripe, understands what "run
|
||||
dev/build?flush=1" means, and has written some custom PHP code in SilverStripe
|
||||
before.
|
||||
|
||||
If you are not familiar with SilverStripe, it is highly recommended you run through the tutorials before attempting to start with this one.
|
||||
If you are not familiar with SilverStripe, it is highly recommended you run
|
||||
through the tutorials before attempting to start with this one.
|
||||
|
||||
* [View a listing of all available tutorials](/tutorials)
|
||||
|
||||
### 1. Installing
|
||||
|
||||
[Download a version of multiform](multiform#downloading). Install it as a sibling of the top level directories (sapphire, jsparty and cms) in your installation.
|
||||
[Download a version of multiform](multiform#downloading). Install it as a
|
||||
sibling of the top level directories (framework and cms) in your installation.
|
||||
|
||||
If you use composer to manage your SilverStripe installation add the following
|
||||
line to your composer.json file
|
||||
|
||||
"silverstripe/multiform": "dev-master"
|
||||
|
||||
### 2. Create subclass of MultiForm
|
||||
|
||||
First of all, we need to create a new subclass of *MultiForm*, which can be called anything.
|
||||
First of all, we need to create a new subclass of *MultiForm*, which can be
|
||||
called anything.
|
||||
|
||||
A suggestion for a good name, is an abbreviation of the project this multi-form is for, with
|
||||
"MultiForm" appended to the end. For example, if I was doing a site called "Bob's Chicken Shack"
|
||||
then a good name would be *BCSMultiForm*. This makes sense, as it is specific to the project
|
||||
that you're developing a multi-form for.
|
||||
A suggestion for a good name, is an abbreviation of the project this multi-form
|
||||
is for, with "MultiForm" appended to the end. For example, if I was doing a
|
||||
site called "Bob's Chicken Shack" then a good name would be *BCSMultiForm*.
|
||||
This makes sense, as it is specific to the project that you're developing a
|
||||
multi-form for.
|
||||
|
||||
Keep in mind the PHP file you create must be in the mysite/code directory, and the filename must
|
||||
be the same as the class name, with the .php extension.
|
||||
Keep in mind the PHP file you create must be in the mysite/code directory, and
|
||||
the filename must be the same as the class name, with the .php extension.
|
||||
|
||||
For the above example, our multi-form will be called *BCSMultiForm.php*
|
||||
|
||||
@ -77,16 +106,17 @@ For the above example, our multi-form will be called *BCSMultiForm.php*
|
||||
}
|
||||
|
||||
|
||||
|
||||
### 3. Set up first step
|
||||
|
||||
Now that we've created our new MultiForm subclass step, we need to define what form is going
|
||||
to be the first step in our multi-step process.
|
||||
Now that we've created our new MultiForm subclass step, we need to define what
|
||||
form is going to be the first step in our multi-step process.
|
||||
|
||||
Each form step must subclass MultiFormStep. This is so the multi-form can identify that this step is part of our step process, and is not just a standard form.
|
||||
Each form step must subclass MultiFormStep. This is so the multi-form can
|
||||
identify that this step is part of our step process, and is not just a standard
|
||||
form.
|
||||
|
||||
So, for example, if we were going to have a first step which collects the personal details of the
|
||||
form user, then we might have this class:
|
||||
So, for example, if we were going to have a first step which collects the
|
||||
personal details of the form user, then we might have this class:
|
||||
|
||||
:::php
|
||||
<?php
|
||||
@ -96,34 +126,37 @@ form user, then we might have this class:
|
||||
}
|
||||
|
||||
|
||||
Now that we've got our first step of the form defined, we need to go back to our subclass of MultiForm, BCSMultiForm, and tell it that BCSPersonalDetailsFormStep is the first step.
|
||||
Now that we've got our first step of the form defined, we need to go back to our
|
||||
subclass of MultiForm, BCSMultiForm, and tell it that BCSPersonalDetailsFormStep
|
||||
is the first step.
|
||||
|
||||
:::php
|
||||
<?php
|
||||
|
||||
class BCSMultiForm extends MultiForm {
|
||||
|
||||
public static $start_step = 'BCSPersonalDetailsFormStep';
|
||||
private static $start_step = 'BCSPersonalDetailsFormStep';
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
### 4. Define next step, and final step
|
||||
|
||||
We've managed to set up the basics of multi-form, but it's not very useful, as there's only one
|
||||
step!
|
||||
We've managed to set up the basics of multi-form, but it's not very useful, as
|
||||
there's only one step!
|
||||
|
||||
To get more than one step, each step needs to know what it's next step is in order to use flow control in our system.
|
||||
To get more than one step, each step needs to know what it's next step is in
|
||||
order to use flow control in our system.
|
||||
|
||||
To let the step know what step is next in the process, we do the same as setting the ''$start_step'' variable *BCSMultiForm*, but we call it ''$next_steps''.
|
||||
To let the step know what step is next in the process, we do the same as setting
|
||||
the `$start_step` variable *BCSMultiForm*, but we call it `$next_steps`.
|
||||
|
||||
:::php
|
||||
<?php
|
||||
|
||||
class BCSPersonalDetailsFormStep extends MultiFormStep {
|
||||
|
||||
public static $next_steps = 'BCSOrganisationDetailsFormStep';
|
||||
private static $next_steps = 'BCSOrganisationDetailsFormStep';
|
||||
|
||||
function getFields() {
|
||||
return new FieldSet(
|
||||
@ -135,34 +168,41 @@ To let the step know what step is next in the process, we do the same as setting
|
||||
}
|
||||
|
||||
|
||||
At the very least, each step also has to have a ''getFields()'' method returning a *FieldSet* with some form field objects. These are the fields that the form will render for the given step.
|
||||
At the very least, each step also has to have a ''getFields()'' method returning
|
||||
a *FieldSet* with some form field objects. These are the fields that the form
|
||||
will render for the given step.
|
||||
|
||||
Keep in mind that our multi-form also requires an end point. This step is the final, and needs to
|
||||
have another variable set to let the multi-form system know this is the final step.
|
||||
Keep in mind that our multi-form also requires an end point. This step is the
|
||||
final, and needs to have another variable set to let the multi-form system know
|
||||
this is the final step.
|
||||
|
||||
So, if we assume that the last step in our process is BCSOrganisationDetailsFormStep, then we can do
|
||||
something like this:
|
||||
So, if we assume that the last step in our process is
|
||||
BCSOrganisationDetailsFormStep, then we can do something like this:
|
||||
|
||||
:::php
|
||||
<?php
|
||||
|
||||
class BCSOrganisationDetailsFormStep extends MultiFormStep {
|
||||
|
||||
public static $is_final_step = true;
|
||||
private static $is_final_step = true;
|
||||
|
||||
}
|
||||
|
||||
|
||||
### 5. Run database integrity check
|
||||
|
||||
We need to run *dev/build?flush=1* now, so that the classes are available to the SilverStripe manifest builder, and to ensure that the database is up to scratch with all the latest tables. So you can go ahead and do that.
|
||||
We need to run *dev/build?flush=1* now, so that the classes are available to the
|
||||
SilverStripe manifest builder, and to ensure that the database is up to scratch
|
||||
with all the latest tables. So you can go ahead and do that.
|
||||
|
||||
*Note: Whenever you add a new step, you **MUST** run dev/build?flush=1 or you may receive errors.*
|
||||
*Note: Whenever you add a new step, you **MUST** run dev/build?flush=1 or you
|
||||
may receive errors.*
|
||||
|
||||
However, we've forgotten one thing. We need to create a method on a page-type so that the form can
|
||||
be rendered into a given template.
|
||||
However, we've forgotten one thing. We need to create a method on a page-type so
|
||||
that the form can be rendered into a given template.
|
||||
|
||||
So, if we want to render our multi-form as ''$BCSMultiForm'' in the *Page.ss* template, we need to create a BCSMultiForm method (function) on Page_Controller:
|
||||
So, if we want to render our multi-form as `$BCSMultiForm` in the *Page.ss*
|
||||
template, we need to create a BCSMultiForm method (function) on Page_Controller:
|
||||
|
||||
:::php
|
||||
<?php
|
||||
@ -184,7 +224,7 @@ So, if we want to render our multi-form as ''$BCSMultiForm'' in the *Page.ss* te
|
||||
function finished() {
|
||||
return array(
|
||||
'Title' => 'Thank you for your submission',
|
||||
'Content' => `<p>`You have successfully submitted the form. Thanks!`</p>`
|
||||
'Content' => `<p>`You have successfully submitted the form!`</p>`
|
||||
);
|
||||
}
|
||||
|
||||
@ -193,106 +233,118 @@ So, if we want to render our multi-form as ''$BCSMultiForm'' in the *Page.ss* te
|
||||
}
|
||||
|
||||
|
||||
The BCSMultiForm() function will create a new instance our subclass of MultiForm, which in this example, is *BCSMultiForm*. This in turn will then set up all the form fields, actions, and validation available to each step, as well as the session.
|
||||
The BCSMultiForm() function will create a new instance our subclass of
|
||||
MultiForm, which in this example, is *BCSMultiForm*. This in turn will then set
|
||||
up all the form fields, actions, and validation available to each step, as well
|
||||
as the session.
|
||||
|
||||
You can of course, put the *BCSMultiForm* method on any controller class you like.
|
||||
You can of course, put the *BCSMultiForm* method on any controller class you
|
||||
like.
|
||||
|
||||
Your template should look something like this, to render the form in:
|
||||
|
||||
:::html
|
||||
`<div id="content">`
|
||||
`<% if Content %>`
|
||||
`<div class="typography">`
|
||||
$Content
|
||||
`</div>`
|
||||
`<% end_if %>`
|
||||
`<% if BCSMultiForm %>`
|
||||
$BCSMultiForm
|
||||
`<% end_if %>`
|
||||
`<% if Form %>`
|
||||
$Form
|
||||
`<% end_if %>`
|
||||
`</div>`
|
||||
<div id="content">
|
||||
<% if Content %>
|
||||
$Content
|
||||
<% end_if %>
|
||||
|
||||
<% if BCSMultiForm %>
|
||||
$BCSMultiForm
|
||||
<% end_if %>
|
||||
|
||||
<% if Form %>
|
||||
$Form
|
||||
<% end_if %>
|
||||
</div>
|
||||
|
||||
|
||||
In this case, the above template example is a *sub-template* inside the *Layout* directory for the templates. Note that we have also included ''$Form'', so standard forms are still able to be used alongside our multi-step form.
|
||||
|
||||
If you've forgotten to add something to one of your form steps, you should get a fatal error complaining that there's a problem. If you just see "This site is unable to respond to your request" or any other vague error, then the website is most likely in "Live" mode, meaning it won't output any errors to you, as that is a function for "Dev" or "Test" sites only.
|
||||
|
||||
To set the environment for your project, make sure your server is made known to SS that is indeed a development server. Add a line to the set_dev_servers() function in the _config.php located in the *mysite* directory:
|
||||
|
||||
:::php
|
||||
Director::set_dev_servers(array(
|
||||
'localhost',
|
||||
'localhost:8888',
|
||||
));
|
||||
|
||||
|
||||
Make sure that when your site eventually goes live, that the server it's hosted on doesn't show up in the set_dev_servers array, otherwise the world will be exposed to all fatal errors from the site.
|
||||
In this case, the above template example is a *sub-template* inside the *Layout*
|
||||
directory for the templates. Note that we have also included ''$Form'', so
|
||||
standard forms are still able to be used alongside our multi-step form.
|
||||
|
||||
### 6. Adding a step indicator
|
||||
|
||||
By default, we include a couple of basic progress indicators which could be useful, out of the box.
|
||||
By default, we include a couple of basic progress indicators which could be
|
||||
useful, out of the box.
|
||||
|
||||
Two of them, as of the time of writing this are:
|
||||
|
||||
|
||||
* Progress list (multiform/templates/Includes/MultiFormProgressList.ss)
|
||||
|
||||
* Progress complete percent (multiform/templates/Includes/MultiFormProgressPercent.ss)
|
||||
|
||||
They are designed to be used either by themselves, or alongside eachother. For example, the percentage could compliment the progress list to give an indication of completion status.
|
||||
They are designed to be used either by themselves, or alongside each other.
|
||||
For example, the percentage could compliment the progress list to give an
|
||||
indication of completion status.
|
||||
|
||||
To include these with our instance of multiform, we just need to add an `<% include %>` statement into the template.
|
||||
To include these with our instance of multiform, we just need to add an
|
||||
`<% include %>` statement into the template.
|
||||
|
||||
For example:
|
||||
|
||||
:::html
|
||||
`<% control BCSMultiForm %>``<% include MultiFormProgressList %>``<% end_control %>`
|
||||
<% with BCSMultiForm %>
|
||||
<% include MultiFormProgressList %>
|
||||
<% end_with %>
|
||||
|
||||
|
||||
This means the included template is rendered within the scope of the BCSMultiForm instance returned, instead of the top level controller context. This gives us the data to show the progression of the steps.
|
||||
This means the included template is rendered within the scope of the
|
||||
BCSMultiForm instance returned, instead of the top level controller context.
|
||||
This gives us the data to show the progression of the steps.
|
||||
|
||||
Putting it together, we might have something looking like this:
|
||||
|
||||
:::html
|
||||
`<div id="content">`
|
||||
`<% if Content %>`
|
||||
`<div class="typography">`
|
||||
$Content
|
||||
`</div>`
|
||||
`<% end_if %>`
|
||||
`<% if BCSMultiForm %>`
|
||||
`<% control BCSMultiForm %>``<% include MultiFormProgressList %>``<% end_control %>`
|
||||
$BCSMultiForm
|
||||
`<% end_if %>`
|
||||
`<% if Form %>`
|
||||
$Form
|
||||
`<% end_if %>`
|
||||
`</div>`
|
||||
<div id="content">
|
||||
<% if Content %>
|
||||
$Content
|
||||
<% end_if %>
|
||||
|
||||
<% if BCSMultiForm %>
|
||||
<% with BCSMultiForm %>
|
||||
<% include MultiFormProgressList %>
|
||||
<% end_with %>
|
||||
|
||||
$BCSMultiForm
|
||||
<% end_if %>
|
||||
<% if Form %>
|
||||
$Form
|
||||
<% end_if %>
|
||||
</div>
|
||||
|
||||
|
||||
Feel free to play around with the progress indicators. If you need something specific to your project, just create a new "Include" template inside your own project templates directory, and include that instead. Some helpful methods to use on the MultiForm would be:
|
||||
Feel free to play around with the progress indicators. If you need something
|
||||
specific to your project, just create a new "Include" template inside your own
|
||||
project templates directory, and include that instead. Some helpful methods to
|
||||
use on the MultiForm would be:
|
||||
|
||||
|
||||
* ''AllStepsLinear()'' (which also makes use of ''getAllStepsRecursive()'' to produce a list of steps)
|
||||
* `AllStepsLinear()` (which also makes use of `getAllStepsRecursive()` to
|
||||
produce a list of steps)
|
||||
* `getCompletedStepCount()`
|
||||
* `getTotalStepCount()`
|
||||
* `getCompletedPercent()`
|
||||
|
||||
* ''getCompletedStepCount()''
|
||||
The default progress indicators make use of the above functions in the
|
||||
templates.
|
||||
|
||||
* ''getTotalStepCount()''
|
||||
|
||||
* ''getCompletedPercent()''
|
||||
|
||||
The default progress indicators make use of the above functions in the templates.
|
||||
|
||||
To use a custom method of your own, simply create a new method on your subclass of MultiForm. In this example, *BCSMultiForm* would be the one to customise. This new method you create would then become available in the progress indicator template.
|
||||
To use a custom method of your own, simply create a new method on your subclass
|
||||
of MultiForm. In this example, *BCSMultiForm* would be the one to customise.
|
||||
This new method you create would then become available in the progress indicator
|
||||
template.
|
||||
|
||||
|
||||
### 7. Finishing it up
|
||||
|
||||
Now that we've got a structure set up to collect form data along each step, and progress through successfully, we need to customise what happens at the end of the last step.
|
||||
Now that we've got a structure set up to collect form data along each step, and
|
||||
progress through successfully, we need to customise what happens at the end of
|
||||
the last step.
|
||||
|
||||
On the final step, the ''finish()'' method is called to finalise all the data from the steps we completed. This method can be found on *MultiForm*. However, we cannot automatically save each step, because we don't know where to save it. So, we must write some code on our subclass of *MultiForm*, overloading ''finish()'' to tell it what to do at the end.
|
||||
On the final step, the ''finish()'' method is called to finalise all the data
|
||||
from the steps we completed. This method can be found on *MultiForm*. However,
|
||||
we cannot automatically save each step, because we don't know where to save it.
|
||||
So, we must write some code on our subclass of *MultiForm*, overloading
|
||||
`finish()` to tell it what to do at the end.
|
||||
|
||||
Here is an example of what we could do here:
|
||||
|
||||
@ -301,16 +353,22 @@ Here is an example of what we could do here:
|
||||
|
||||
class BCSMultiForm extends MultiForm {
|
||||
|
||||
public static $start_step = 'BCSPersonalDetailsForm';
|
||||
private static $start_step = 'BCSPersonalDetailsForm';
|
||||
|
||||
public function finish($data, $form) {
|
||||
parent::finish($data, $form);
|
||||
$steps = DataObject::get('MultiFormStep', "SessionID = {$this->session->ID}");
|
||||
|
||||
$steps = DataObject::get(
|
||||
'MultiFormStep',
|
||||
"SessionID = {$this->session->ID}"
|
||||
);
|
||||
|
||||
if($steps) {
|
||||
foreach($steps as $step) {
|
||||
if($step->class == 'BCSPersonalDetailsFormStep') {
|
||||
$member = new Member();
|
||||
$data = $step->loadData();
|
||||
|
||||
if($data) {
|
||||
$member->update($data);
|
||||
$member->write();
|
||||
@ -320,47 +378,55 @@ Here is an example of what we could do here:
|
||||
if($step->class == 'BCSOrganisationDetailsFormStep') {
|
||||
$organisation = new Organisation();
|
||||
$data = $step->loadData();
|
||||
|
||||
if($data) {
|
||||
$organisation->update($data);
|
||||
if($member && $member->ID) $organisation->MemberID = $member->ID;
|
||||
|
||||
if($member && $member->ID) {
|
||||
$organisation->MemberID = $member->ID;
|
||||
}
|
||||
|
||||
$organisation->write();
|
||||
}
|
||||
}
|
||||
|
||||
// Debug::show($step->loadData()); // Shows the step data (unserialized by loadData)
|
||||
// Shows the step data (unserialized by loadData)
|
||||
// Debug::show($step->loadData());
|
||||
}
|
||||
}
|
||||
|
||||
$controller = $this->getController();
|
||||
$controller->redirect($controller->Link() . 'finished');
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
#### Organisation
|
||||
#### Organization
|
||||
|
||||
The class Organisation() is mentioned in the above example but doesn't exist at the moment (unlike the existing Member() class which looks after the member groups in SilverStripe) so we need to create it:
|
||||
The class Organisation() is mentioned in the above example but doesn't exist at
|
||||
the moment (unlike the existing Member() class which looks after the member
|
||||
groups in SilverStripe) so we need to create it:
|
||||
|
||||
(I have chosen in this example to create Organisation as a separate DataObject but you may wish to change the code and add the data to the Member class).
|
||||
(I have chosen in this example to create Organisation as a separate DataObject
|
||||
but you may wish to change the code and add the data to the Member class).
|
||||
|
||||
:::php
|
||||
<?php
|
||||
|
||||
class Organisation extends DataObject {
|
||||
|
||||
static $db = array(
|
||||
private static $db = array(
|
||||
// Add your Organisation fields here
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
?>
|
||||
|
||||
|
||||
#### Warning
|
||||
|
||||
If you're dealing with sensitive data, it's best to delete the session and step data immediately after the form is successfully submitted.
|
||||
If you're dealing with sensitive data, it's best to delete the session and step
|
||||
data immediately after the form is successfully submitted.
|
||||
|
||||
You can delete it by calling this method on the finish() for your MultiForm subclass:
|
||||
You can delete it by calling this method on the finish() for your MultiForm
|
||||
subclass:
|
||||
|
||||
:::php
|
||||
$this->session->delete();
|
||||
@ -370,38 +436,46 @@ This will also go through each of it's steps and delete them as well.
|
||||
|
||||
## Customising
|
||||
|
||||
Because the multi-form system doesn't magically do everything for you, although it does provide
|
||||
a sensible set of defaults, it means you need to customise certain aspects of it. Here are some
|
||||
useful methods that can be customised (although you can technically overload anything available
|
||||
on MultiForm and MultiFormStep!):
|
||||
Because the multi-form system doesn't magically do everything for you, although
|
||||
it does provide a sensible set of defaults, it means you need to customise
|
||||
certain aspects of it. Here are some useful methods that can be customised
|
||||
(although you can technically overload anything available on MultiForm and
|
||||
MultiFormStep!):
|
||||
|
||||
### Templates
|
||||
|
||||
The best part about this system is the freedom to customise each form step template.
|
||||
|
||||
In order, when you have a page with a multi-form rendering into it, it chooses which template to render that form in this order, within the context of the MultiForm class:
|
||||
The best part about this system is the freedom to customise each form step
|
||||
template.
|
||||
|
||||
In order, when you have a page with a multi-form rendering into it, it chooses
|
||||
which template to render that form in this order, within the context of the
|
||||
MultiForm class:
|
||||
|
||||
* $this->getCurrentStep()->class (the current step class)
|
||||
|
||||
* MultiFormStep
|
||||
|
||||
* $this->class (your subclass of MultiForm)
|
||||
|
||||
* MultiForm
|
||||
|
||||
* Form
|
||||
|
||||
More than likely, you'll want the first one to be available when the form renders. To that effect, you can start placing templates in the *templates/Includes* directory for your project. You need to call them the same as the class name for each step. For example, if you want *MembershipForm*, a subclass of *MultiFormStep* to have it's own template, you would put *MembershipForm.ss* into that directory, and run *?flush=1*.
|
||||
More than likely, you'll want the first one to be available when the form
|
||||
renders. To that effect, you can start placing templates in the
|
||||
*templates/Includes* directory for your project. You need to call them the same
|
||||
as the class name for each step. For example, if you want *MembershipForm*, a
|
||||
subclass of *MultiFormStep* to have it's own template, you would put
|
||||
*MembershipForm.ss* into that directory, and run *?flush=1*.
|
||||
|
||||
If you'd like a pre-existing template on how to customise the form step, have a look at Form.ss that's found within the sapphire module. Use that template, as a base for your new MembershipForm.ss template in your project templates.
|
||||
If you'd like a pre-existing template on how to customise the form step, have a
|
||||
look at Form.ss that's found within the sapphire module. Use that template, as
|
||||
a base for your new MembershipForm.ss template in your project templates.
|
||||
|
||||
For more information on this, please [look at the Form documentation](/form#using_a_custom_template) under "Using a custom template".
|
||||
|
||||
### getNextStep()
|
||||
|
||||
If you are wanting to override the next step (so, if you want the next step to be something
|
||||
different based on a user's choice of input during the step, you can override getNextStep() on
|
||||
any given step to manually override what the next step should be. An example:
|
||||
If you are wanting to override the next step (so, if you want the next step to
|
||||
be something different based on a user's choice of input during the step, you
|
||||
can override getNextStep() on any given step to manually override what the next
|
||||
step should be. An example:
|
||||
|
||||
:::php
|
||||
class MyStep extends MultiFormStep
|
||||
@ -421,16 +495,11 @@ any given step to manually override what the next step should be. An example:
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
### Validation
|
||||
|
||||
To define validation on a step-by-step basis, please define getValidator() and return a Validator object, such as RequiredFields - for more information on form validation see [:form](/form).
|
||||
To define validation on a step-by-step basis, please define getValidator() and
|
||||
return a Validator object, such as RequiredFields - for more information on form
|
||||
validation see [:form](/form).
|
||||
|
||||
e.g.
|
||||
|
||||
@ -453,11 +522,14 @@ e.g.
|
||||
|
||||
### finish()
|
||||
|
||||
''finish()'' is the final call in the process. At this step, all the form data would most likely be
|
||||
unserialized, and saved to the database in whatever way the developer sees fit. By default, we
|
||||
have a ''finish()'' method on *MultiForm* which serializes the last step form data into the database, and that's it.
|
||||
`finish()` is the final call in the process. At this step, all the form data
|
||||
would most likely be unserialized, and saved to the database in whatever way the
|
||||
developer sees fit. By default, we have a `finish()` method on *MultiForm* which
|
||||
serializes the last step form data into the database, and that's it.
|
||||
|
||||
''finish()'' should be overloaded onto your subclass of *MultiForm*, and ''parent::finish()'' should be called first, otherwise the last step form data won't be saved.
|
||||
`finish()` should be overloaded onto your subclass of *MultiForm*, and
|
||||
`parent::finish()` should be called first, otherwise the last step form data
|
||||
won't be saved.
|
||||
|
||||
For example:
|
||||
|
||||
@ -466,53 +538,62 @@ For example:
|
||||
|
||||
class BCSMultiForm extends MultiForm {
|
||||
|
||||
public static $start_step = 'BCSPersonalDetailsForm';
|
||||
private static $start_step = 'BCSPersonalDetailsForm';
|
||||
|
||||
public function finish($data, $form) {
|
||||
parent::finish($data, $form);
|
||||
$steps = DataObject::get('MultiFormStep', "SessionID = {$this->session->ID}");
|
||||
|
||||
$steps = MultiFormStep::get()->filter(array(
|
||||
"SessionID" => $this->session->ID
|
||||
));
|
||||
|
||||
if($steps) {
|
||||
foreach($steps as $step) {
|
||||
Debug::show($step->loadData()); // Shows the step data (unserialized by loadData)
|
||||
// Shows the step data (unserialized by loadData)
|
||||
Debug::show($step->loadData());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
The above is a sample bit of code that simply fetches all the steps in the database that were
|
||||
saved. Further refinement could include getting steps only if the Data (serialized raw form data)
|
||||
is set, as the above example doesn't respect branching of steps (multiple next steps on a given
|
||||
form step).
|
||||
The above is a sample bit of code that simply fetches all the steps in the
|
||||
database that were saved. Further refinement could include getting steps only
|
||||
if the Data (serialized raw form data) is set, as the above example doesn't
|
||||
respect branching of steps (multiple next steps on a given form step).
|
||||
|
||||
## Best practices
|
||||
|
||||
### Delete session after submission
|
||||
|
||||
If you're dealing with sensitive data, such as credit card fields, or personal fields that shouldn't be lying around in the session database, then it's a good idea to immediately delete this data after the user has submitted.
|
||||
If you're dealing with sensitive data, such as credit card fields, or personal
|
||||
fields that shouldn't be lying around in the session database, then it's a good
|
||||
idea to immediately delete this data after the user has submitted.
|
||||
|
||||
This can be easily achieved by adding the following line at the end of your finish() method on your MultiForm subclass.
|
||||
This can be easily achieved by adding the following line at the end of your
|
||||
`finish()` method on your MultiForm subclass.
|
||||
|
||||
:::php
|
||||
$this->session->delete();
|
||||
|
||||
|
||||
|
||||
|
||||
### Expiring old session data
|
||||
|
||||
Included with the MultiForm module is a class called *MultiFormPurgeTask*. This task can be used to purge expired session data on a regular basis. The date of expiry can be customised, and is given a default of 7 days to delete sessions after their creation.
|
||||
Included with the MultiForm module is a class called *MultiFormPurgeTask*. This
|
||||
task can be used to purge expired session data on a regular basis. The date of
|
||||
expiry can be customised, and is given a default of 7 days to delete sessions
|
||||
after their creation.
|
||||
|
||||
You can run the task from the URL, by using http://mysite.com/MultiFormPurgeTask/run
|
||||
|
||||
MultiFormPurgeTask is a subclass of *DailyTask*, so can be used alongside the cli-script.php automated tasks which are included with SilverStripe.
|
||||
MultiFormPurgeTask is a subclass of *DailyTask*, so can be used alongside the
|
||||
cli-script.php automated tasks which are included with SilverStripe.
|
||||
|
||||
One way of automatically running this on a UNIX based machine is by cron. For more information, see [:scheduledtask](/scheduledtask) for more information.
|
||||
One way of automatically running this on a UNIX based machine is by cron. For
|
||||
more information, see [:scheduledtask](/scheduledtask) for more information.
|
||||
|
||||
## TODO
|
||||
|
||||
|
||||
* Code example on how to use ''$form->saveInto()'' with MultiForm, as it doesn't have all steps in the $form context at ''finish()''
|
||||
|
||||
* Allowing a user to click a link, and have an email sent to them with the current state, so they can come back and use the form exactly where they left off
|
||||
@ -528,7 +609,6 @@ One way of automatically running this on a UNIX based machine is by cron. For mo
|
||||
|
||||
## Related
|
||||
|
||||
|
||||
* [Form](/form)
|
||||
* [Form field types](/form-field-types)
|
||||
* [Form recipes](recipes/forms)
|
||||
|
Loading…
x
Reference in New Issue
Block a user