mirror of
https://github.com/silverstripe/silverstripe-cms
synced 2024-10-22 08:05:56 +02:00
elofgren: Track whether mail() returns TRUE or FALSE when a Newsletter is sent and record the result in a new Newsletter_SentRecipient table.
Add a 'Sent Status Report' tab to Newsletters that shows the following 4 reports: * Sending to the Following Recipients Failed * Sending to the Following Recipients Bounced (@TODO: Make bounces actually show up here) * The Newsletter has Never Been Sent to Following Subscribers * Sending to the Following Recipients was Successful More info: http://support.silverstripe.com/gsoc/wiki/GSoc07UsabilityElijah (merged from branches/gsoc) git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/cms/trunk@41995 467b73ca-7a2a-4603-9d3b-597d59a354a9
This commit is contained in:
parent
07c4301ca7
commit
4deaf09212
@ -11,12 +11,15 @@ class Newsletter extends DataObject {
|
|||||||
require_once("forms/Form.php");
|
require_once("forms/Form.php");
|
||||||
|
|
||||||
$group = DataObject::get_by_id("Group", $this->Parent()->GroupID);
|
$group = DataObject::get_by_id("Group", $this->Parent()->GroupID);
|
||||||
|
$sent_status_report = $this->renderWith("Newsletter_SentStatusReport");
|
||||||
$ret = new FieldSet(
|
$ret = new FieldSet(
|
||||||
new TabSet("Root",
|
new TabSet("Root",
|
||||||
$mailTab = new Tab("Newsletter",
|
$mailTab = new Tab("Newsletter",
|
||||||
new TextField("Subject", "Subject", $this->Subject),
|
new TextField("Subject", "Subject", $this->Subject),
|
||||||
new HtmlEditorField("Content", "Content")
|
new HtmlEditorField("Content", "Content")
|
||||||
|
),
|
||||||
|
$sentToTab = new Tab("Sent Status Report",
|
||||||
|
new LiteralField("Sent Status Report", $sent_status_report)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
@ -29,6 +32,57 @@ class Newsletter extends DataObject {
|
|||||||
return $ret;
|
return $ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a DataObject listing the recipients for the given status for this newsletter
|
||||||
|
*
|
||||||
|
* @param string $result 3 possible values: "Sent", (mail() returned TRUE), "Failed" (mail() returned FALSE), or "Bounced" ({@see $email_bouncehandler}).
|
||||||
|
*/
|
||||||
|
function SentRecipients($result) {
|
||||||
|
$SQL_result = Convert::raw2sql($result);
|
||||||
|
return DataObject::get("Newsletter_SentRecipient",array("ParentID='".$this->ID."'", "Result='".$SQL_result."'"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a DataObjectSet containing the subscribers who have never been sent this Newsletter
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
function UnsentSubscribers() {
|
||||||
|
// Get a list of everyone who has been sent this newsletter
|
||||||
|
$sent_recipients = DataObject::get("Newsletter_SentRecipient","ParentID='".$this->ID."'");
|
||||||
|
// If this Newsletter has not been sent to anyone yet, $sent_recipients will be null
|
||||||
|
if ($sent_recipients != null) {
|
||||||
|
$sent_recipients_array = $sent_recipients->toNestedArray('MemberID');
|
||||||
|
} else {
|
||||||
|
$sent_recipients_array = array();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get a list of all the subscribers to this newsletter
|
||||||
|
$subscribers = DataObject::get( 'Member', "`GroupID`='".$this->Parent()->GroupID."'", null, "INNER JOIN `Group_Members` ON `MemberID`=`Member`.`ID`" );
|
||||||
|
// If this Newsletter has no subscribers, $subscribers will be null
|
||||||
|
if ($subscribers != null) {
|
||||||
|
$subscribers_array = $subscribers->toNestedArray();
|
||||||
|
} else {
|
||||||
|
$subscribers_array = array();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get list of subscribers who have not been sent this newsletter:
|
||||||
|
$unsent_subscribers_array = array_diff_key($subscribers_array, $sent_recipients_array);
|
||||||
|
|
||||||
|
// Create new data object set containing the subscribers who have not been sent this newsletter:
|
||||||
|
$unsent_subscribers = new DataObjectSet();
|
||||||
|
foreach($unsent_subscribers_array as $key => $data) {
|
||||||
|
$record = array(
|
||||||
|
'ID' => $data['ID'],
|
||||||
|
'FirstName' => $data['FirstName'],
|
||||||
|
'Surname' => $data['Surname'],
|
||||||
|
'Email' => $data['Email']
|
||||||
|
);
|
||||||
|
$unsent_subscribers->push(new ArrayData($record));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $unsent_subscribers;
|
||||||
|
}
|
||||||
|
|
||||||
function getTitle() {
|
function getTitle() {
|
||||||
return $this->getField('Subject');
|
return $this->getField('Subject');
|
||||||
}
|
}
|
||||||
@ -51,6 +105,7 @@ class Newsletter extends DataObject {
|
|||||||
|
|
||||||
static $has_many = array(
|
static $has_many = array(
|
||||||
"Recipients" => "Newsletter_Recipient",
|
"Recipients" => "Newsletter_Recipient",
|
||||||
|
"SentRecipients" => "Newsletter_SentRecipient",
|
||||||
);
|
);
|
||||||
|
|
||||||
static function newDraft( $parentID, $subject, $content ) {
|
static function newDraft( $parentID, $subject, $content ) {
|
||||||
@ -69,6 +124,21 @@ class Newsletter extends DataObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class Newsletter_SentRecipient extends DataObject {
|
||||||
|
/**
|
||||||
|
* The DB schema for Newsletter_SentRecipient.
|
||||||
|
*
|
||||||
|
* ParentID is the the Newsletter
|
||||||
|
* Email and MemberID keep track of the recpients information
|
||||||
|
* Result has 3 possible values: "Sent", (mail() returned TRUE), "Failed" (mail() returned FALSE), or "Bounced" ({@see $email_bouncehandler}).
|
||||||
|
*/
|
||||||
|
static $db = array(
|
||||||
|
"ParentID" => "Int",
|
||||||
|
"Email" => "Varchar(255)",
|
||||||
|
"MemberID" => "Int",
|
||||||
|
"Result" => "Enum('Sent, Failed, Bounced', 'Sent')",
|
||||||
|
);
|
||||||
|
}
|
||||||
class Newsletter_Recipient extends DataObject {
|
class Newsletter_Recipient extends DataObject {
|
||||||
static $db = array(
|
static $db = array(
|
||||||
"ParentID" => "Int",
|
"ParentID" => "Int",
|
||||||
|
@ -59,7 +59,7 @@ class NewsletterEmailProcess extends BatchProcess {
|
|||||||
|
|
||||||
|
|
||||||
$e->populateTemplate( array( 'Member' => $member, 'FirstName' => $member->FirstName ) );
|
$e->populateTemplate( array( 'Member' => $member, 'FirstName' => $member->FirstName ) );
|
||||||
$this->sendToAddress( $e, $address, $this->messageID );
|
$this->sendToAddress( $e, $address, $this->messageID, $member);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -69,9 +69,28 @@ class NewsletterEmailProcess extends BatchProcess {
|
|||||||
return parent::next();
|
return parent::next();
|
||||||
}
|
}
|
||||||
|
|
||||||
private function sendToAddress( $email, $address, $messageID = null ) {
|
/*
|
||||||
|
* Sends a Newsletter email to the specified address
|
||||||
|
*
|
||||||
|
* @param $member The object containing information about the member being emailed
|
||||||
|
*/
|
||||||
|
private function sendToAddress( $email, $address, $messageID = null, $member) {
|
||||||
$email->setTo( $address );
|
$email->setTo( $address );
|
||||||
$email->send( $messageID );
|
$result = $email->send( $messageID );
|
||||||
|
// Log result of the send
|
||||||
|
$newsletter = new Newsletter_SentRecipient();
|
||||||
|
$newsletter->Email = $address;
|
||||||
|
$newsletter->MemberID = $member->ID;
|
||||||
|
// If Sending is successful
|
||||||
|
if ($result == true) {
|
||||||
|
$newsletter->Result = 'Sent';
|
||||||
|
} else {
|
||||||
|
$newsletter->Result = 'Failed';
|
||||||
|
}
|
||||||
|
$newsletter->ParentID = $this->newsletter->ID;
|
||||||
|
$newsletter->write();
|
||||||
|
// Adding a pause between email sending can be useful for debugging purposes
|
||||||
|
// sleep(10);
|
||||||
}
|
}
|
||||||
|
|
||||||
function complete() {
|
function complete() {
|
||||||
|
@ -326,7 +326,7 @@ class NewsletterAdmin extends LeftAndMain {
|
|||||||
* Reloads the list of recipients via ajax
|
* Reloads the list of recipients via ajax
|
||||||
*/
|
*/
|
||||||
function getrecipientslist() {
|
function getrecipientslist() {
|
||||||
if( $_REQUEST['ajax'] ) {
|
if(Director::is_ajax()) {
|
||||||
$newsletterType = DataObject::get_by_id('NewsletterType', $this->urlParams['ID'] );
|
$newsletterType = DataObject::get_by_id('NewsletterType', $this->urlParams['ID'] );
|
||||||
$fields = new FieldSet($memberList = new MemberTableField($this, "Recipients", $newsletterType->Group() ));
|
$fields = new FieldSet($memberList = new MemberTableField($this, "Recipients", $newsletterType->Group() ));
|
||||||
$memberList->setController($this);
|
$memberList->setController($this);
|
||||||
@ -336,6 +336,17 @@ class NewsletterAdmin extends LeftAndMain {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reloads the "Sent Status Report" tab via ajax
|
||||||
|
*/
|
||||||
|
function getsentstatusreport($params) {
|
||||||
|
if(Director::is_ajax()) {
|
||||||
|
$newsletter = DataObject::get_by_id( 'Newsletter', $params['ID'] );
|
||||||
|
$sent_status_report = $newsletter->renderWith("Newsletter_SentStatusReport");
|
||||||
|
return $sent_status_report;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static function template_path() {
|
public static function template_path() {
|
||||||
if(self::$template_path) return self::$template_path;
|
if(self::$template_path) return self::$template_path;
|
||||||
else return self::$template_path = project() . '/templates/email';
|
else return self::$template_path = project() . '/templates/email';
|
||||||
|
@ -323,6 +323,17 @@ Behaviour.register({
|
|||||||
// $('SendProgressBar').setText( 'Done!' );
|
// $('SendProgressBar').setText( 'Done!' );
|
||||||
$('SendProgressBar').reset();
|
$('SendProgressBar').reset();
|
||||||
// this.elements.Message.value = '';
|
// this.elements.Message.value = '';
|
||||||
|
|
||||||
|
// Reload the "Sent Status Report" tab after a Newsletter is sent
|
||||||
|
var id = this.elements.NewsletterID.value;
|
||||||
|
var request = new Ajax.Request(baseHref() + 'admin/newsletter/getsentstatusreport/' + id + '?ajax=1', {
|
||||||
|
onSuccess: function( response ) {
|
||||||
|
$('Root_SentStatusReport').innerHTML = response.responseText;
|
||||||
|
},
|
||||||
|
onFailure: function(response) {
|
||||||
|
statusMessage('Could not automatically refresh Sent Status Report tab', 'bad');
|
||||||
|
}
|
||||||
|
});
|
||||||
this.close();
|
this.close();
|
||||||
if( response )
|
if( response )
|
||||||
Ajax.Evaluator(response);
|
Ajax.Evaluator(response);
|
||||||
@ -359,6 +370,7 @@ Behaviour.register({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// @TODO: Check if this code is used anymore now that NewsletterList was removed.
|
||||||
NewsletterList = Class.create();
|
NewsletterList = Class.create();
|
||||||
NewsletterList.applyTo('table.NewsletterList');
|
NewsletterList.applyTo('table.NewsletterList');
|
||||||
NewsletterList.prototype = {
|
NewsletterList.prototype = {
|
||||||
|
93
templates/Newsletter_SentStatusReport.ss
Normal file
93
templates/Newsletter_SentStatusReport.ss
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
<% if SentRecipients(Failed) %>
|
||||||
|
<h2 class="error">Sending to the Following Recipients Failed</h2>
|
||||||
|
<table class="CMSList">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th class="Email">Email</th>
|
||||||
|
<th>Result</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<% control SentRecipients(Failed) %>
|
||||||
|
<tr>
|
||||||
|
<td>$Email</td>
|
||||||
|
<td>$LastEdited</td>
|
||||||
|
<td>$Result</td>
|
||||||
|
</tr>
|
||||||
|
<% end_control %>
|
||||||
|
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<% end_if %>
|
||||||
|
<% if SentRecipients(Bounced) %>
|
||||||
|
<h2 class="error">Sending to the Following Recipients Bounced</h2>
|
||||||
|
<table class="CMSList">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th class="Email">Email</th>
|
||||||
|
<th>Date</th>
|
||||||
|
<th>Result</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
|
||||||
|
<tbody>
|
||||||
|
<% control SentRecipients(Bounced) %>
|
||||||
|
<tr>
|
||||||
|
<td>$Email</td>
|
||||||
|
<td>$LastEdited</td>
|
||||||
|
<td>$Result</td>
|
||||||
|
</tr>
|
||||||
|
<% end_control %>
|
||||||
|
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<% end_if %>
|
||||||
|
|
||||||
|
<% if UnsentSubscribers %>
|
||||||
|
<h2>The Newsletter has Never Been Sent to Following Subscribers</h2>
|
||||||
|
<table class="CMSList">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th class="FirstName">Firstname</th>
|
||||||
|
<th class="Surname">Surname</th>
|
||||||
|
<th class="Email">Email</th>
|
||||||
|
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
|
||||||
|
<tbody>
|
||||||
|
<% control UnsentSubscribers %>
|
||||||
|
<tr id="unsent-member-$ID">
|
||||||
|
<td>$FirstName</td>
|
||||||
|
<td>$Surname</td>
|
||||||
|
<td>$Email</td>
|
||||||
|
</tr>
|
||||||
|
<% end_control %>
|
||||||
|
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<% end_if %>
|
||||||
|
|
||||||
|
<% if SentRecipients(Sent) %>
|
||||||
|
<h2>Sending to the Following Recipients was Successful</h2>
|
||||||
|
<table class="CMSList">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th class="Email">Email</th>
|
||||||
|
<th>Date</th>
|
||||||
|
<th>Result</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
|
||||||
|
<tbody>
|
||||||
|
<% control SentRecipients(Sent) %>
|
||||||
|
<tr id="sent-member-$ID">
|
||||||
|
<td>$Email</td>
|
||||||
|
<td>$LastEdited</td>
|
||||||
|
<td>$Result</td>
|
||||||
|
</tr>
|
||||||
|
<% end_control %>
|
||||||
|
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<% end_if %>
|
Loading…
Reference in New Issue
Block a user