2008-09-29 03:18:23 +00:00
< ? php
2010-07-07 09:17:37 +00:00
2008-09-29 03:18:23 +00:00
/**
2009-04-14 02:06:18 +00:00
* @ package userforms
2008-09-29 03:18:23 +00:00
*/
2009-04-14 02:06:18 +00:00
2008-09-29 03:18:23 +00:00
class UserDefinedForm extends Page {
2015-06-18 09:35:02 +12:00
2015-10-29 12:02:11 +13:00
/**
* @ var string
*/
private static $icon = 'userforms/images/sitetree_icon.png' ;
2015-09-11 10:20:06 +12:00
2009-10-15 00:33:13 +00:00
/**
2012-09-11 21:30:40 +12:00
* @ var string
2009-10-15 00:33:13 +00:00
*/
2013-04-10 11:35:58 +12:00
private static $description = 'Adds a customizable form.' ;
2012-09-11 21:30:40 +12:00
/**
* @ var string Required Identifier
*/
2013-04-10 11:35:58 +12:00
private static $required_identifier = null ;
2015-05-29 15:49:36 +12:00
/**
2015-07-24 14:37:48 +12:00
* @ var string
*/
private static $email_template_directory = 'userforms/templates/email/' ;
/**
* Should this module automatically upgrade on dev / build ?
2015-05-29 15:49:36 +12:00
*
* @ config
2015-07-24 14:37:48 +12:00
* @ var bool
2015-05-29 15:49:36 +12:00
*/
2015-07-24 14:37:48 +12:00
private static $upgrade_on_build = true ;
2015-07-16 15:06:53 +12:00
/**
2015-07-24 14:37:48 +12:00
* Built in extensions required by this page
* @ config
* @ var array
2015-07-16 15:06:53 +12:00
*/
2015-07-24 14:37:48 +12:00
private static $extensions = array (
'UserFormFieldEditorExtension'
);
2015-09-11 10:20:06 +12:00
2009-04-14 02:06:18 +00:00
/**
2012-09-11 21:30:40 +12:00
* @ var array Fields on the user defined form page .
2009-04-14 02:06:18 +00:00
*/
2013-04-10 11:35:58 +12:00
private static $db = array (
2008-09-29 03:18:23 +00:00
" SubmitButtonText " => " Varchar " ,
2013-11-11 15:25:31 +02:00
" ClearButtonText " => " Varchar " ,
2009-02-11 02:00:20 +00:00
" OnCompleteMessage " => " HTMLText " ,
2009-07-22 04:02:21 +00:00
" ShowClearButton " => " Boolean " ,
2012-12-06 17:25:25 +01:00
'DisableSaveSubmissions' => 'Boolean' ,
'EnableLiveValidation' => 'Boolean' ,
2014-07-27 20:51:23 +12:00
'HideFieldLabels' => 'Boolean' ,
2015-07-13 10:05:14 +12:00
'DisplayErrorMessagesAtTop' => 'Boolean' ,
2014-07-27 20:51:23 +12:00
'DisableAuthenicatedFinishAction' => 'Boolean' ,
'DisableCsrfSecurityToken' => 'Boolean'
2008-09-29 03:18:23 +00:00
);
2015-09-11 10:20:06 +12:00
2009-04-14 02:06:18 +00:00
/**
2012-09-11 21:30:40 +12:00
* @ var array Default values of variables when this page is created
2015-09-11 10:20:06 +12:00
*/
2013-04-10 11:35:58 +12:00
private static $defaults = array (
2009-04-14 02:06:18 +00:00
'Content' => '$UserDefinedForm' ,
2009-07-22 04:02:21 +00:00
'DisableSaveSubmissions' => 0 ,
2009-04-14 02:06:18 +00:00
'OnCompleteMessage' => '<p>Thanks, we\'ve received your submission.</p>'
2008-09-29 03:18:23 +00:00
);
2009-04-14 02:06:18 +00:00
/**
2013-01-29 21:44:00 +13:00
* @ var array
2009-04-14 02:06:18 +00:00
*/
2013-04-10 11:35:58 +12:00
private static $has_many = array (
2009-04-14 02:06:18 +00:00
" Submissions " => " SubmittedForm " ,
" EmailRecipients " => " UserDefinedForm_EmailRecipient "
2008-09-29 03:18:23 +00:00
);
2013-06-04 11:17:40 +02:00
2015-07-13 10:05:14 +12:00
/**
* @ var array
* @ config
*/
private static $casting = array (
'ErrorContainerID' => 'Text'
);
/**
* Error container selector which matches the element for grouped messages
*
* @ var string
* @ config
*/
private static $error_container_id = 'error-container' ;
2016-02-02 10:32:19 +11:00
/**
* The configuration used to determine whether a confirmation message is to
* appear when navigating away from a partially completed form .
*
* @ var boolean
* @ config
*/
private static $enable_are_you_sure = true ;
2016-07-28 14:09:58 +01:00
/**
* @ var bool
* @ config
2015-06-23 23:06:09 +12:00
*/
private static $recipients_warning_enabled = false ;
2013-06-04 11:17:40 +02:00
/**
* Temporary storage of field ids when the form is duplicated .
* Example layout : array ( 'EditableCheckbox3' => 'EditableCheckbox14' )
* @ var array
*/
protected $fieldsFromTo = array ();
2009-04-14 02:06:18 +00:00
/**
2013-01-29 21:44:00 +13:00
* @ return FieldList
2009-04-14 02:06:18 +00:00
*/
2015-04-20 11:28:15 +09:30
public function getCMSFields () {
2015-08-10 17:03:36 +12:00
Requirements :: css ( USERFORMS_DIR . '/css/UserForm_cms.css' );
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$self = $this ;
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$this -> beforeUpdateCMSFields ( function ( $fields ) use ( $self ) {
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
// define tabs
$fields -> findOrMakeTab ( 'Root.FormOptions' , _t ( 'UserDefinedForm.CONFIGURATION' , 'Configuration' ));
2015-07-13 15:36:20 +12:00
$fields -> findOrMakeTab ( 'Root.Recipients' , _t ( 'UserDefinedForm.RECIPIENTS' , 'Recipients' ));
2015-04-20 11:28:15 +09:30
$fields -> findOrMakeTab ( 'Root.Submissions' , _t ( 'UserDefinedForm.SUBMISSIONS' , 'Submissions' ));
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
// text to show on complete
$onCompleteFieldSet = new CompositeField (
$label = new LabelField ( 'OnCompleteMessageLabel' , _t ( 'UserDefinedForm.ONCOMPLETELABEL' , 'Show on completion' )),
$editor = new HtmlEditorField ( 'OnCompleteMessage' , '' , _t ( 'UserDefinedForm.ONCOMPLETEMESSAGE' , $self -> OnCompleteMessage ))
);
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$onCompleteFieldSet -> addExtraClass ( 'field' );
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$editor -> setRows ( 3 );
$label -> addExtraClass ( 'left' );
2015-09-11 10:20:06 +12:00
2015-07-13 15:36:20 +12:00
// Define config for email recipients
$emailRecipientsConfig = GridFieldConfig_RecordEditor :: create ( 10 );
$emailRecipientsConfig -> getComponentByType ( 'GridFieldAddNewButton' )
-> setButtonName (
_t ( 'UserDefinedForm.ADDEMAILRECIPIENT' , 'Add Email Recipient' )
);
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
// who do we email on submission
2015-07-13 15:36:20 +12:00
$emailRecipients = new GridField (
'EmailRecipients' ,
_t ( 'UserDefinedForm.EMAILRECIPIENTS' , 'Email Recipients' ),
$self -> EmailRecipients (),
$emailRecipientsConfig
2015-04-20 11:28:15 +09:30
);
2015-07-16 15:06:53 +12:00
$emailRecipients
-> getConfig ()
-> getComponentByType ( 'GridFieldDetailForm' )
2015-07-24 14:37:48 +12:00
-> setItemRequestClass ( 'UserFormRecipientItemRequest' );
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$fields -> addFieldsToTab ( 'Root.FormOptions' , $onCompleteFieldSet );
2015-07-13 15:36:20 +12:00
$fields -> addFieldToTab ( 'Root.Recipients' , $emailRecipients );
2015-04-20 11:28:15 +09:30
$fields -> addFieldsToTab ( 'Root.FormOptions' , $self -> getFormOptions ());
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
// view the submissions
$submissions = new GridField (
2015-09-11 10:20:06 +12:00
'Submissions' ,
2015-04-20 11:28:15 +09:30
_t ( 'UserDefinedForm.SUBMISSIONS' , 'Submissions' ),
$self -> Submissions () -> sort ( 'Created' , 'DESC' )
);
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
// make sure a numeric not a empty string is checked against this int column for SQL server
2015-09-24 10:40:30 +12:00
$parentID = ( ! empty ( $self -> ID )) ? ( int ) $self -> ID : 0 ;
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
// get a list of all field names and values used for print and export CSV views of the GridField below.
$columnSQL = <<< SQL
2016-03-22 12:17:04 +13:00
SELECT " SubmittedFormField " . " Name " as " Name " , " SubmittedFormField " . " Title " as " Title " , COALESCE ( " EditableFormField " . " Sort " , 999 ) AS " Sort "
2013-11-13 16:46:13 +13:00
FROM " SubmittedFormField "
LEFT JOIN " SubmittedForm " ON " SubmittedForm " . " ID " = " SubmittedFormField " . " ParentID "
2016-03-21 23:45:17 +13:00
LEFT JOIN " EditableFormField " ON " EditableFormField " . " Title " = " SubmittedFormField " . " Title " AND " EditableFormField " . " ParentID " = '$parentID'
2013-11-13 16:46:13 +13:00
WHERE " SubmittedForm " . " ParentID " = '$parentID'
2016-03-21 23:45:17 +13:00
ORDER BY " Sort " , " Title "
2013-11-13 16:46:13 +13:00
SQL ;
2015-08-28 16:21:51 +12:00
// Sanitise periods in title
$columns = array ();
foreach ( DB :: query ( $columnSQL ) -> map () as $name => $title ) {
$columns [ $name ] = trim ( strtr ( $title , '.' , ' ' ));
}
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$config = new GridFieldConfig ();
$config -> addComponent ( new GridFieldToolbarHeader ());
$config -> addComponent ( $sort = new GridFieldSortableHeader ());
$config -> addComponent ( $filter = new UserFormsGridFieldFilterHeader ());
$config -> addComponent ( new GridFieldDataColumns ());
$config -> addComponent ( new GridFieldEditButton ());
$config -> addComponent ( new GridFieldDeleteAction ());
$config -> addComponent ( new GridFieldPageCount ( 'toolbar-header-right' ));
$config -> addComponent ( $pagination = new GridFieldPaginator ( 25 ));
$config -> addComponent ( new GridFieldDetailForm ());
2015-08-21 13:41:00 +12:00
$config -> addComponent ( new GridFieldButtonRow ( 'after' ));
$config -> addComponent ( $export = new GridFieldExportButton ( 'buttons-after-left' ));
$config -> addComponent ( $print = new GridFieldPrintButton ( 'buttons-after-left' ));
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
/**
* Support for { @ link https :// github . com / colymba / GridFieldBulkEditingTools }
*/
if ( class_exists ( 'GridFieldBulkManager' )) {
$config -> addComponent ( new GridFieldBulkManager ());
}
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$sort -> setThrowExceptionOnBadDataType ( false );
$filter -> setThrowExceptionOnBadDataType ( false );
$pagination -> setThrowExceptionOnBadDataType ( false );
2015-09-11 10:20:06 +12:00
// attach every column to the print view form
2015-04-20 11:28:15 +09:30
$columns [ 'Created' ] = 'Created' ;
$filter -> setColumns ( $columns );
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
// print configuration
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$print -> setPrintHasHeader ( true );
$print -> setPrintColumns ( $columns );
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
// export configuration
$export -> setCsvHasHeader ( true );
$export -> setExportColumns ( $columns );
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$submissions -> setConfig ( $config );
$fields -> addFieldToTab ( 'Root.Submissions' , $submissions );
$fields -> addFieldToTab ( 'Root.FormOptions' , new CheckboxField ( 'DisableSaveSubmissions' , _t ( 'UserDefinedForm.SAVESUBMISSIONS' , 'Disable Saving Submissions to Server' )));
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
});
2015-09-11 10:20:06 +12:00
2015-04-20 11:28:15 +09:30
$fields = parent :: getCMSFields ();
2015-09-11 10:20:06 +12:00
2016-07-28 14:09:58 +01:00
if ( $this -> EmailRecipients () -> Count () == 0 && static :: config () -> recipients_warning_enabled ) {
2015-06-18 09:35:02 +12:00
$fields -> addFieldToTab ( " Root.Main " , new LiteralField ( " EmailRecipientsWarning " ,
" <p class= \" message warning \" > " . _t ( " UserDefinedForm.NORECIPIENTS " ,
" Warning: You have not configured any recipients. Form submissions may be missed. " )
. " </p> " ), " Title " );
}
2008-09-29 03:18:23 +00:00
return $fields ;
}
2012-12-11 17:09:02 +13:00
/**
* Allow overriding the EmailRecipients on a { @ link DataExtension }
* so you can customise who receives an email .
* Converts the RelationList to an ArrayList so that manipulation
* of the original source data isn ' t possible .
*
* @ return ArrayList
*/
2013-09-04 16:32:24 +10:00
public function FilteredEmailRecipients ( $data = null , $form = null ) {
2015-07-13 15:36:20 +12:00
$recipients = new ArrayList ( $this -> EmailRecipients () -> toArray ());
// Filter by rules
$recipients = $recipients -> filterByCallback ( function ( $recipient ) use ( $data , $form ) {
return $recipient -> canSend ( $data , $form );
});
2012-12-11 17:09:02 +13:00
2013-09-04 16:32:24 +10:00
$this -> extend ( 'updateFilteredEmailRecipients' , $recipients , $data , $form );
2012-12-11 17:09:02 +13:00
return $recipients ;
}
2009-09-22 22:36:53 +00:00
/**
2015-09-11 10:20:06 +12:00
* Custom options for the form . You can extend the built in options by
2010-09-03 05:06:13 +00:00
* using { @ link updateFormOptions ()}
2009-09-22 22:36:53 +00:00
*
2012-10-18 09:01:56 +02:00
* @ return FieldList
2009-09-22 22:36:53 +00:00
*/
2012-07-17 16:09:31 +12:00
public function getFormOptions () {
2010-09-03 05:06:13 +00:00
$submit = ( $this -> SubmitButtonText ) ? $this -> SubmitButtonText : _t ( 'UserDefinedForm.SUBMITBUTTON' , 'Submit' );
2013-11-11 15:25:31 +02:00
$clear = ( $this -> ClearButtonText ) ? $this -> ClearButtonText : _t ( 'UserDefinedForm.CLEARBUTTON' , 'Clear' );
2015-09-11 10:20:06 +12:00
2012-04-22 16:17:42 -03:00
$options = new FieldList (
2010-09-03 05:06:13 +00:00
new TextField ( " SubmitButtonText " , _t ( 'UserDefinedForm.TEXTONSUBMIT' , 'Text on submit button:' ), $submit ),
2013-11-11 15:25:31 +02:00
new TextField ( " ClearButtonText " , _t ( 'UserDefinedForm.TEXTONCLEAR' , 'Text on clear button:' ), $clear ),
2012-12-06 17:25:25 +01:00
new CheckboxField ( " ShowClearButton " , _t ( 'UserDefinedForm.SHOWCLEARFORM' , 'Show Clear Form Button' ), $this -> ShowClearButton ),
new CheckboxField ( " EnableLiveValidation " , _t ( 'UserDefinedForm.ENABLELIVEVALIDATION' , 'Enable live validation' )),
2014-07-27 20:51:23 +12:00
new CheckboxField ( " HideFieldLabels " , _t ( 'UserDefinedForm.HIDEFIELDLABELS' , 'Hide field labels' )),
2015-07-13 10:05:14 +12:00
new CheckboxField ( " DisplayErrorMessagesAtTop " , _t ( 'UserDefinedForm.DISPLAYERRORMESSAGESATTOP' , 'Display error messages above the form?' )),
2014-07-27 20:51:23 +12:00
new CheckboxField ( 'DisableCsrfSecurityToken' , _t ( 'UserDefinedForm.DISABLECSRFSECURITYTOKEN' , 'Disable CSRF Token' )),
2015-08-31 14:49:16 +12:00
new CheckboxField ( 'DisableAuthenicatedFinishAction' , _t ( 'UserDefinedForm.DISABLEAUTHENICATEDFINISHACTION' , 'Disable Authentication on finish action' ))
2009-09-22 22:36:53 +00:00
);
2015-09-11 10:20:06 +12:00
2010-09-03 05:06:13 +00:00
$this -> extend ( 'updateFormOptions' , $options );
2015-09-11 10:20:06 +12:00
2010-09-03 05:06:13 +00:00
return $options ;
2009-09-22 22:36:53 +00:00
}
2015-07-13 10:05:14 +12:00
/**
* Get the HTML id of the error container displayed above the form .
*
* @ return string
*/
public function getErrorContainerID () {
return $this -> config () -> error_container_id ;
}
2015-07-24 14:37:48 +12:00
public function requireDefaultRecords () {
parent :: requireDefaultRecords ();
if ( ! $this -> config () -> upgrade_on_build ) {
return ;
}
// Perform migrations
Injector :: inst ()
-> create ( 'UserFormsUpgradeService' )
-> setQuiet ( true )
-> run ();
2015-09-11 10:20:06 +12:00
2015-07-24 14:37:48 +12:00
DB :: alteration_message ( 'Migrated userforms' , 'changed' );
}
2015-08-12 17:18:43 +12:00
/**
* Validate formfields
*/
public function getCMSValidator () {
return new UserFormValidator ();
}
2008-09-29 03:18:23 +00:00
}
/**
* Controller for the { @ link UserDefinedForm } page type .
2009-04-14 02:06:18 +00:00
*
2012-10-18 09:01:56 +02:00
* @ package userforms
2008-09-29 03:18:23 +00:00
*/
2009-04-14 02:06:18 +00:00
2008-09-29 03:18:23 +00:00
class UserDefinedForm_Controller extends Page_Controller {
2015-09-11 10:20:06 +12:00
2014-11-28 18:16:37 +13:00
private static $finished_anchor = '#uff' ;
2013-07-15 20:49:49 +12:00
private static $allowed_actions = array (
2013-07-09 15:20:19 +12:00
'index' ,
'ping' ,
'Form' ,
'finished'
);
2009-04-21 03:44:13 +00:00
public function init () {
2009-12-07 00:37:20 +00:00
parent :: init ();
2015-09-11 10:20:06 +12:00
2009-04-21 03:44:13 +00:00
// load the jquery
2014-04-16 11:48:10 +12:00
$lang = i18n :: get_lang_from_locale ( i18n :: get_locale ());
2015-08-06 14:19:05 +12:00
Requirements :: css ( USERFORMS_DIR . '/css/UserForm.css' );
2012-04-22 16:17:42 -03:00
Requirements :: javascript ( FRAMEWORK_DIR . '/thirdparty/jquery/jquery.js' );
2014-04-16 11:48:10 +12:00
Requirements :: javascript ( USERFORMS_DIR . '/thirdparty/jquery-validate/jquery.validate.min.js' );
Requirements :: add_i18n_javascript ( USERFORMS_DIR . '/javascript/lang' );
2015-08-06 16:06:56 +12:00
Requirements :: javascript ( USERFORMS_DIR . '/javascript/UserForm.js' );
2015-08-21 11:41:42 +12:00
2014-04-16 11:48:10 +12:00
Requirements :: javascript (
USERFORMS_DIR . " /thirdparty/jquery-validate/localization/messages_ { $lang } .min.js "
);
Requirements :: javascript (
USERFORMS_DIR . " /thirdparty/jquery-validate/localization/methods_ { $lang } .min.js "
);
if ( $this -> HideFieldLabels ) {
Requirements :: javascript ( USERFORMS_DIR . '/thirdparty/Placeholders.js/Placeholders.min.js' );
}
2016-02-02 10:32:19 +11:00
// Bind a confirmation message when navigating away from a partially completed form.
$page = $this -> data ();
if ( $page :: config () -> enable_are_you_sure ) {
Requirements :: javascript ( USERFORMS_DIR . '/thirdparty/jquery.are-you-sure/jquery.are-you-sure.js' );
}
2008-09-29 03:18:23 +00:00
}
2015-09-11 10:20:06 +12:00
2009-04-14 23:50:48 +00:00
/**
* Using $UserDefinedForm in the Content area of the page shows
* where the form should be rendered into . If it does not exist
2013-01-29 21:44:00 +13:00
* then default back to $Form .
2009-04-14 23:50:48 +00:00
*
2013-01-29 21:44:00 +13:00
* @ return array
2009-04-14 23:50:48 +00:00
*/
public function index () {
2011-04-08 18:36:01 +01:00
if ( $this -> Content && $form = $this -> Form ()) {
$hasLocation = stristr ( $this -> Content , '$UserDefinedForm' );
2009-04-14 23:50:48 +00:00
if ( $hasLocation ) {
2015-08-19 14:22:03 +12:00
$content = preg_replace ( '/(<p[^>]*>)?\\$UserDefinedForm(<\\/p>)?/i' , $form -> forTemplate (), $this -> Content );
2009-04-14 23:50:48 +00:00
return array (
2012-04-22 16:25:26 -03:00
'Content' => DBField :: create_field ( 'HTMLText' , $content ),
2009-04-14 23:50:48 +00:00
'Form' => " "
);
}
}
2010-09-03 05:06:13 +00:00
2009-04-14 23:50:48 +00:00
return array (
2012-04-22 16:25:26 -03:00
'Content' => DBField :: create_field ( 'HTMLText' , $this -> Content ),
2010-09-03 05:06:13 +00:00
'Form' => $this -> Form ()
2009-04-14 23:50:48 +00:00
);
}
2009-04-27 06:00:05 +00:00
2012-04-26 12:14:27 +12:00
/**
* Keep the session alive for the user .
2013-01-29 21:44:00 +13:00
*
* @ return int
2012-04-26 12:14:27 +12:00
*/
2013-01-29 21:44:00 +13:00
public function ping () {
2012-04-26 12:14:27 +12:00
return 1 ;
}
2009-04-14 02:06:18 +00:00
/**
2010-09-03 05:06:13 +00:00
* Get the form for the page . Form can be modified by calling { @ link updateForm ()}
2013-01-29 21:44:00 +13:00
* on a UserDefinedForm extension .
2009-04-14 02:06:18 +00:00
*
2015-07-24 14:37:48 +12:00
* @ return Forms
2009-04-14 02:06:18 +00:00
*/
2012-05-04 13:39:08 +12:00
public function Form () {
2015-07-24 14:37:48 +12:00
$form = UserForm :: create ( $this );
2010-09-03 05:06:13 +00:00
$this -> generateConditionalJavascript ();
2015-07-24 14:37:48 +12:00
return $form ;
2010-09-03 05:06:13 +00:00
}
2015-07-13 10:05:14 +12:00
2010-09-03 05:06:13 +00:00
/**
* Generate the javascript for the conditional field show / hiding logic .
2013-01-29 21:44:00 +13:00
*
2010-09-03 05:06:13 +00:00
* @ return void
*/
2012-05-04 13:39:08 +12:00
public function generateConditionalJavascript () {
2010-09-03 05:06:13 +00:00
$default = " " ;
$rules = " " ;
2013-02-27 16:05:30 +13:00
$watch = array ();
$watchLoad = array ();
2010-09-03 05:06:13 +00:00
if ( $this -> Fields ()) {
foreach ( $this -> Fields () as $field ) {
2015-08-13 11:31:37 +12:00
$holderSelector = $field -> getSelectorHolder ();
2015-07-24 14:37:48 +12:00
2009-04-23 22:52:08 +00:00
// Is this Field Show by Default
2015-07-24 14:37:48 +12:00
if ( ! $field -> ShowOnLoad ) {
2015-08-17 10:43:51 +12:00
$default .= " { $holderSelector } .hide().trigger('userform.field.hide'); \n " ;
2009-04-23 22:52:08 +00:00
}
// Check for field dependencies / default
2015-12-11 17:38:31 +13:00
foreach ( $field -> EffectiveDisplayRules () as $rule ) {
2015-07-24 14:37:48 +12:00
// Get the field which is effected
$formFieldWatch = EditableFormField :: get () -> byId ( $rule -> ConditionFieldID );
2015-09-20 17:08:40 +12:00
2015-09-18 11:07:58 +12:00
// Skip deleted fields
if ( ! $formFieldWatch ) {
continue ;
}
2015-07-24 14:37:48 +12:00
2015-08-13 11:31:37 +12:00
$fieldToWatch = $formFieldWatch -> getSelectorField ( $rule );
$fieldToWatchOnLoad = $formFieldWatch -> getSelectorField ( $rule , true );
2015-07-24 14:37:48 +12:00
// show or hide?
$view = ( $rule -> Display == 'Hide' ) ? 'hide' : 'show' ;
$opposite = ( $view == " show " ) ? " hide " : " show " ;
// what action do we need to keep track of. Something nicer here maybe?
// @todo encapulsation
$action = " change " ;
2015-08-13 11:31:37 +12:00
if ( $formFieldWatch instanceof EditableTextField ) {
2015-07-24 14:37:48 +12:00
$action = " keyup " ;
}
// is this field a special option field
$checkboxField = false ;
$radioField = false ;
if ( in_array ( $formFieldWatch -> ClassName , array ( 'EditableCheckboxGroupField' , 'EditableCheckbox' ))) {
$action = " click " ;
$checkboxField = true ;
} else if ( $formFieldWatch -> ClassName == " EditableRadioField " ) {
$radioField = true ;
}
// and what should we evaluate
switch ( $rule -> ConditionOption ) {
case 'IsNotBlank' :
2015-08-13 11:31:37 +12:00
$expression = ( $checkboxField || $radioField ) ? '$(this).is(":checked")' : '$(this).val() != ""' ;
2015-07-24 14:37:48 +12:00
break ;
case 'IsBlank' :
2015-08-13 11:31:37 +12:00
$expression = ( $checkboxField || $radioField ) ? '!($(this).is(":checked"))' : '$(this).val() == ""' ;
2015-09-11 10:20:06 +12:00
2015-07-24 14:37:48 +12:00
break ;
case 'HasValue' :
if ( $checkboxField ) {
$expression = '$(this).prop("checked")' ;
} else if ( $radioField ) {
// We cannot simply get the value of the radio group, we need to find the checked option first.
$expression = '$(this).parents(".field, .control-group").find("input:checked").val()=="' . $rule -> FieldValue . '"' ;
} else {
$expression = '$(this).val() == "' . $rule -> FieldValue . '"' ;
2009-05-01 04:11:31 +00:00
}
2015-07-24 14:37:48 +12:00
break ;
case 'ValueLessThan' :
$expression = '$(this).val() < parseFloat("' . $rule -> FieldValue . '")' ;
2015-09-11 10:20:06 +12:00
2015-07-24 14:37:48 +12:00
break ;
case 'ValueLessThanEqual' :
$expression = '$(this).val() <= parseFloat("' . $rule -> FieldValue . '")' ;
2015-09-11 10:20:06 +12:00
2015-07-24 14:37:48 +12:00
break ;
case 'ValueGreaterThan' :
$expression = '$(this).val() > parseFloat("' . $rule -> FieldValue . '")' ;
break ;
case 'ValueGreaterThanEqual' :
$expression = '$(this).val() >= parseFloat("' . $rule -> FieldValue . '")' ;
break ;
default : // ==HasNotValue
if ( $checkboxField ) {
$expression = '!$(this).prop("checked")' ;
} else if ( $radioField ) {
// We cannot simply get the value of the radio group, we need to find the checked option first.
$expression = '$(this).parents(".field, .control-group").find("input:checked").val()!="' . $rule -> FieldValue . '"' ;
} else {
$expression = '$(this).val() != "' . $rule -> FieldValue . '"' ;
2009-05-08 04:27:49 +00:00
}
2014-12-02 19:49:03 +13:00
2015-07-24 14:37:48 +12:00
break ;
}
2014-12-02 19:49:03 +13:00
2015-07-24 14:37:48 +12:00
if ( ! isset ( $watch [ $fieldToWatch ])) {
$watch [ $fieldToWatch ] = array ();
}
2014-12-02 19:49:03 +13:00
2015-08-12 08:57:19 +12:00
$watch [ $fieldToWatch ][] = array (
2015-07-24 14:37:48 +12:00
'expression' => $expression ,
2015-08-13 11:31:37 +12:00
'holder_selector' => $holderSelector ,
2015-07-24 14:37:48 +12:00
'view' => $view ,
'opposite' => $opposite ,
'action' => $action
);
2013-02-27 16:05:30 +13:00
2015-07-24 14:37:48 +12:00
$watchLoad [ $fieldToWatchOnLoad ] = true ;
2009-04-23 22:52:08 +00:00
}
2009-04-21 03:44:13 +00:00
}
2008-09-29 03:18:23 +00:00
}
2015-09-11 10:20:06 +12:00
2013-02-27 16:05:30 +13:00
if ( $watch ) {
foreach ( $watch as $key => $values ) {
$logic = array ();
2014-12-02 19:49:03 +13:00
$actions = array ();
2013-02-27 16:05:30 +13:00
foreach ( $values as $rule ) {
2015-08-17 10:43:51 +12:00
// Assign action
2014-12-02 19:49:03 +13:00
$actions [ $rule [ 'action' ]] = $rule [ 'action' ];
2015-08-17 10:43:51 +12:00
// Assign behaviour
$expression = $rule [ 'expression' ];
$holder = $rule [ 'holder_selector' ];
$view = $rule [ 'view' ]; // hide or show
$opposite = $rule [ 'opposite' ];
// Generated javascript for triggering visibility
$logic [] = <<< " EOS "
if ({ $expression }) {
{ $holder }
. { $view }()
. trigger ( 'userform.field.{$view}' );
} else {
{ $holder }
. { $opposite }()
. trigger ( 'userform.field.{$opposite}' );
}
EOS ;
2013-02-27 16:05:30 +13:00
}
$logic = implode ( " \n " , $logic );
$rules .= $key . " .each(function() { \n
$ ( this ) . data ( 'userformConditions' , function () { \n
$logic\n
}); \n
}); \n " ;
2014-12-02 19:49:03 +13:00
foreach ( $actions as $action ) {
$rules .= $key . " . $action (function() {
2013-02-27 16:05:30 +13:00
$ ( this ) . data ( 'userformConditions' ) . call ( this ); \n
}); \n " ;
2014-12-02 19:49:03 +13:00
}
2013-02-27 16:05:30 +13:00
}
}
if ( $watchLoad ) {
foreach ( $watchLoad as $key => $value ) {
$rules .= $key . " .each(function() {
$ ( this ) . data ( 'userformConditions' ) . call ( this ); \n
}); \n " ;
}
}
2012-12-06 17:25:25 +01:00
// Only add customScript if $default or $rules is defined
2015-09-11 00:08:13 +12:00
if ( $default || $rules ) {
2012-12-06 17:25:25 +01:00
Requirements :: customScript ( <<< JS
( function ( $ ) {
$ ( document ) . ready ( function () {
$default
$rules
})
})( jQuery );
2009-04-21 03:44:13 +00:00
JS
2010-09-03 05:06:13 +00:00
, 'UserFormsConditional' );
2012-12-06 17:25:25 +01:00
}
2008-09-29 03:18:23 +00:00
}
2013-01-29 21:44:00 +13:00
2009-04-14 02:06:18 +00:00
/**
* Process the form that is submitted through the site
2015-08-17 17:23:45 +12:00
*
* { @ see UserForm :: validate ()} for validation step prior to processing
*
2013-01-29 21:44:00 +13:00
* @ param array $data
* @ param Form $form
*
2009-04-14 02:06:18 +00:00
* @ return Redirection
*/
2012-05-04 13:39:08 +12:00
public function process ( $data , $form ) {
2010-09-03 05:06:13 +00:00
$submittedForm = Object :: create ( 'SubmittedForm' );
2009-10-27 09:23:42 +00:00
$submittedForm -> SubmittedByID = ( $id = Member :: currentUserID ()) ? $id : 0 ;
2008-09-29 03:18:23 +00:00
$submittedForm -> ParentID = $this -> ID ;
2010-09-03 05:06:13 +00:00
// if saving is not disabled save now to generate the ID
2013-01-29 21:44:00 +13:00
if ( ! $this -> DisableSaveSubmissions ) {
$submittedForm -> write ();
}
2015-07-24 14:37:48 +12:00
2012-07-17 15:58:24 +12:00
$attachments = array ();
2012-04-22 16:25:26 -03:00
$submittedFields = new ArrayList ();
2015-07-24 14:37:48 +12:00
2009-04-14 02:06:18 +00:00
foreach ( $this -> Fields () as $field ) {
2013-01-29 21:44:00 +13:00
if ( ! $field -> showInReports ()) {
continue ;
}
2015-07-24 14:37:48 +12:00
2010-02-14 07:53:16 +00:00
$submittedField = $field -> getSubmittedFormField ();
2008-09-29 03:18:23 +00:00
$submittedField -> ParentID = $submittedForm -> ID ;
$submittedField -> Name = $field -> Name ;
2011-12-05 23:19:20 +13:00
$submittedField -> Title = $field -> getField ( 'Title' );
2015-07-24 14:37:48 +12:00
2010-09-03 05:06:13 +00:00
// save the value from the data
2009-04-27 06:17:02 +00:00
if ( $field -> hasMethod ( 'getValueFromData' )) {
$submittedField -> Value = $field -> getValueFromData ( $data );
2013-01-29 21:44:00 +13:00
} else {
if ( isset ( $data [ $field -> Name ])) {
$submittedField -> Value = $data [ $field -> Name ];
}
2009-04-14 02:06:18 +00:00
}
2015-09-11 00:08:09 +12:00
if ( ! empty ( $data [ $field -> Name ])) {
2010-02-14 07:53:16 +00:00
if ( in_array ( " EditableFileField " , $field -> getClassAncestry ())) {
2009-04-28 02:03:13 +00:00
if ( isset ( $_FILES [ $field -> Name ])) {
2013-12-20 21:07:35 +13:00
$foldername = $field -> getFormField () -> getFolderName ();
2015-09-11 10:20:06 +12:00
2009-04-28 02:03:13 +00:00
// create the file from post data
$upload = new Upload ();
$file = new File ();
2011-09-16 11:47:09 +12:00
$file -> ShowInSearch = 0 ;
2012-05-11 11:39:24 +12:00
try {
2013-12-20 21:07:35 +13:00
$upload -> loadIntoFile ( $_FILES [ $field -> Name ], $file , $foldername );
2012-05-11 11:39:24 +12:00
} catch ( ValidationException $e ) {
$validationResult = $e -> getResult ();
$form -> addErrorMessage ( $field -> Name , $validationResult -> message (), 'bad' );
Controller :: curr () -> redirectBack ();
return ;
}
2009-04-19 23:22:22 +00:00
2009-04-28 02:03:13 +00:00
// write file to form field
$submittedField -> UploadedFileID = $file -> ID ;
2015-09-11 10:20:06 +12:00
2010-09-03 05:06:13 +00:00
// attach a file only if lower than 1MB
2015-09-11 00:08:09 +12:00
if ( $file -> getAbsoluteSize () < 1024 * 1024 * 1 ) {
2009-04-28 02:03:13 +00:00
$attachments [] = $file ;
}
2015-04-20 11:28:15 +09:30
}
2008-09-29 03:18:23 +00:00
}
}
2015-07-24 14:37:48 +12:00
2014-07-06 17:03:13 +12:00
$submittedField -> extend ( 'onPopulationFromField' , $field );
2015-09-11 10:20:06 +12:00
2013-01-29 21:44:00 +13:00
if ( ! $this -> DisableSaveSubmissions ) {
$submittedField -> write ();
}
2015-07-24 14:37:48 +12:00
2009-07-22 04:02:21 +00:00
$submittedFields -> push ( $submittedField );
2010-09-03 05:06:13 +00:00
}
2015-07-24 14:37:48 +12:00
2009-02-11 02:27:55 +00:00
$emailData = array (
" Sender " => Member :: currentUser (),
2009-07-26 23:23:45 +00:00
" Fields " => $submittedFields
2009-02-11 02:27:55 +00:00
);
2015-07-24 14:37:48 +12:00
2014-02-03 12:10:44 +13:00
$this -> extend ( 'updateEmailData' , $emailData , $attachments );
2015-07-24 14:37:48 +12:00
2010-09-03 05:06:13 +00:00
// email users on submit.
2013-09-04 16:32:24 +10:00
if ( $recipients = $this -> FilteredEmailRecipients ( $data , $form )) {
2009-04-19 23:22:22 +00:00
2013-09-04 16:32:24 +10:00
foreach ( $recipients as $recipient ) {
2015-10-02 11:34:50 +13:00
$email = new UserFormRecipientEmail ( $submittedFields );
$mergeFields = $this -> getMergeFieldsMap ( $emailData [ 'Fields' ]);
2016-07-28 14:09:58 +01:00
2015-10-02 11:34:50 +13:00
if ( $attachments ) {
foreach ( $attachments as $file ) {
if ( $file -> ID != 0 ) {
$email -> attachFile (
$file -> Filename ,
$file -> Filename ,
HTTP :: get_mime_type ( $file -> Filename )
);
}
}
}
2009-04-19 23:22:22 +00:00
2015-07-16 15:06:53 +12:00
$parsedBody = SSViewer :: execute_string ( $recipient -> getEmailBodyContent (), $mergeFields );
if ( ! $recipient -> SendPlain && $recipient -> emailTemplateExists ()) {
$email -> setTemplate ( $recipient -> EmailTemplate );
}
2009-07-26 23:23:45 +00:00
$email -> populateTemplate ( $recipient );
2009-04-14 02:06:18 +00:00
$email -> populateTemplate ( $emailData );
$email -> setFrom ( $recipient -> EmailFrom );
2015-07-16 15:06:53 +12:00
$email -> setBody ( $parsedBody );
2009-04-14 02:06:18 +00:00
$email -> setTo ( $recipient -> EmailAddress );
2014-07-02 11:56:15 +00:00
$email -> setSubject ( $recipient -> EmailSubject );
2015-09-11 10:20:06 +12:00
2012-09-11 21:22:50 +12:00
if ( $recipient -> EmailReplyTo ) {
2013-01-29 22:40:28 +13:00
$email -> setReplyTo ( $recipient -> EmailReplyTo );
2012-09-11 21:22:50 +12:00
}
// check to see if they are a dynamic reply to. eg based on a email field a user selected
2009-07-05 07:14:03 +00:00
if ( $recipient -> SendEmailFromField ()) {
2009-10-27 09:23:42 +00:00
$submittedFormField = $submittedFields -> find ( 'Name' , $recipient -> SendEmailFromField () -> Name );
2012-09-11 21:22:50 +12:00
2012-07-17 15:58:24 +12:00
if ( $submittedFormField && is_string ( $submittedFormField -> Value )) {
2013-01-29 22:40:28 +13:00
$email -> setReplyTo ( $submittedFormField -> Value );
2009-07-05 07:14:03 +00:00
}
}
2009-10-27 09:23:42 +00:00
// check to see if they are a dynamic reciever eg based on a dropdown field a user selected
2009-07-05 07:14:03 +00:00
if ( $recipient -> SendEmailToField ()) {
2009-10-27 09:23:42 +00:00
$submittedFormField = $submittedFields -> find ( 'Name' , $recipient -> SendEmailToField () -> Name );
2015-09-11 10:20:06 +12:00
2012-07-17 15:58:24 +12:00
if ( $submittedFormField && is_string ( $submittedFormField -> Value )) {
2015-04-20 11:28:15 +09:30
$email -> setTo ( $submittedFormField -> Value );
2009-04-14 03:26:14 +00:00
}
}
2015-09-11 10:20:06 +12:00
2014-07-02 11:56:15 +00:00
// check to see if there is a dynamic subject
if ( $recipient -> SendEmailSubjectField ()) {
$submittedFormField = $submittedFields -> find ( 'Name' , $recipient -> SendEmailSubjectField () -> Name );
if ( $submittedFormField && trim ( $submittedFormField -> Value )) {
$email -> setSubject ( $submittedFormField -> Value );
}
}
2012-07-17 15:58:24 +12:00
$this -> extend ( 'updateEmail' , $email , $recipient , $emailData );
2012-07-17 16:09:31 +12:00
2009-05-12 06:10:13 +00:00
if ( $recipient -> SendPlain ) {
2015-07-16 15:06:53 +12:00
$body = strip_tags ( $recipient -> getEmailBodyContent ()) . " \n " ;
2010-04-25 22:36:09 +00:00
if ( isset ( $emailData [ 'Fields' ]) && ! $recipient -> HideFormData ) {
2009-05-12 06:10:13 +00:00
foreach ( $emailData [ 'Fields' ] as $Field ) {
2015-04-12 09:26:26 +10:00
$body .= $Field -> Title . ': ' . $Field -> Value . " \n " ;
2009-05-12 06:10:13 +00:00
}
}
2012-09-11 21:22:50 +12:00
2009-05-12 06:10:13 +00:00
$email -> setBody ( $body );
$email -> sendPlain ();
}
else {
2015-04-20 11:28:15 +09:30
$email -> send ();
2009-05-12 06:10:13 +00:00
}
2009-04-14 02:06:18 +00:00
}
2009-02-11 02:27:55 +00:00
}
2015-07-24 14:37:48 +12:00
2014-07-09 16:32:10 +12:00
$submittedForm -> extend ( 'updateAfterProcess' );
2014-07-12 09:43:52 +12:00
2011-04-08 18:36:01 +01:00
Session :: clear ( " FormInfo. { $form -> FormName () } .errors " );
Session :: clear ( " FormInfo. { $form -> FormName () } .data " );
2015-09-11 10:20:06 +12:00
2010-09-03 05:06:13 +00:00
$referrer = ( isset ( $data [ 'Referrer' ])) ? '?referrer=' . urlencode ( $data [ 'Referrer' ]) : " " ;
2013-12-19 15:56:10 +13:00
2015-09-11 10:20:06 +12:00
// set a session variable from the security ID to stop people accessing
2014-07-27 20:51:23 +12:00
// the finished method directly.
if ( ! $this -> DisableAuthenicatedFinishAction ) {
if ( isset ( $data [ 'SecurityID' ])) {
Session :: set ( 'FormProcessed' , $data [ 'SecurityID' ]);
} else {
// if the form has had tokens disabled we still need to set FormProcessed
// to allow us to get through the finshed method
if ( ! $this -> Form () -> getSecurityToken () -> isEnabled ()) {
$randNum = rand ( 1 , 1000 );
$randHash = md5 ( $randNum );
Session :: set ( 'FormProcessed' , $randHash );
Session :: set ( 'FormProcessedNum' , $randNum );
}
2014-02-27 12:34:57 +13:00
}
2014-01-06 15:15:01 +13:00
}
2015-07-24 14:37:48 +12:00
2014-07-12 09:43:52 +12:00
if ( ! $this -> DisableSaveSubmissions ) {
Session :: set ( 'userformssubmission' . $this -> ID , $submittedForm -> ID );
}
2014-11-28 18:16:37 +13:00
return $this -> redirect ( $this -> Link ( 'finished' ) . $referrer . $this -> config () -> finished_anchor );
2008-10-30 03:53:35 +00:00
}
2015-07-16 15:06:53 +12:00
/**
* Allows the use of field values in email body .
*
* @ param ArrayList fields
2015-08-17 13:19:25 +12:00
* @ return ArrayData
2015-07-16 15:06:53 +12:00
*/
private function getMergeFieldsMap ( $fields = array ()) {
2015-08-17 13:19:25 +12:00
$data = new ArrayData ( array ());
2015-07-16 15:06:53 +12:00
foreach ( $fields as $field ) {
$data -> setField ( $field -> Name , DBField :: create_field ( 'Text' , $field -> Value ));
}
return $data ;
}
2008-10-30 03:53:35 +00:00
/**
2015-09-11 10:20:06 +12:00
* This action handles rendering the " finished " message , which is
2013-01-29 21:44:00 +13:00
* customizable by editing the ReceivedFormSubmission template .
2008-10-30 03:53:35 +00:00
*
* @ return ViewableData
*/
2012-05-04 13:39:08 +12:00
public function finished () {
2014-07-12 09:43:52 +12:00
$submission = Session :: get ( 'userformssubmission' . $this -> ID );
if ( $submission ) {
$submission = SubmittedForm :: get () -> byId ( $submission );
}
2014-01-06 15:15:01 +13:00
$referrer = isset ( $_GET [ 'referrer' ]) ? urldecode ( $_GET [ 'referrer' ]) : null ;
2015-09-11 10:20:06 +12:00
2014-07-27 20:51:23 +12:00
if ( ! $this -> DisableAuthenicatedFinishAction ) {
$formProcessed = Session :: get ( 'FormProcessed' );
if ( ! isset ( $formProcessed )) {
return $this -> redirect ( $this -> Link () . $referrer );
} else {
$securityID = Session :: get ( 'SecurityID' );
// make sure the session matches the SecurityID and is not left over from another form
2014-02-27 12:34:57 +13:00
if ( $formProcessed != $securityID ) {
2014-07-27 20:51:23 +12:00
// they may have disabled tokens on the form
$securityID = md5 ( Session :: get ( 'FormProcessedNum' ));
if ( $formProcessed != $securityID ) {
return $this -> redirect ( $this -> Link () . $referrer );
}
2014-02-27 12:34:57 +13:00
}
2013-12-19 15:56:10 +13:00
}
2014-07-27 20:51:23 +12:00
Session :: clear ( 'FormProcessed' );
2013-12-19 15:56:10 +13:00
}
2009-07-05 07:39:28 +00:00
return $this -> customise ( array (
2014-07-12 09:43:52 +12:00
'Content' => $this -> customise ( array (
'Submission' => $submission ,
'Link' => $referrer
)) -> renderWith ( 'ReceivedFormSubmission' ),
2010-09-03 05:06:13 +00:00
'Form' => '' ,
2008-09-29 03:18:23 +00:00
));
}
}