. /** * Settings for email * * @category Settings * @package GNUsocial * @author Evan Prodromou * @author Zach Copley * @copyright 2008-2009 StatusNet, Inc. * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ defined('GNUSOCIAL') || die(); /** * Settings for email * * @copyright 2008-2009 StatusNet, Inc. * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later * * @see Widget */ class EmailsettingsAction extends SettingsAction { /** * Title of the page * * @return string Title of the page */ public function title() { // TRANS: Title for e-mail settings. return _('Email settings'); } /** * Instructions for use * * @return instructions for use */ public function getInstructions() { // XXX: For consistency of parameters in messages, this should be a // regular parameters, replaced with sprintf(). // TRANS: E-mail settings page instructions. // TRANS: %%site.name%% is the name of the site. return _('Manage how you get email from %%site.name%%.'); } public function showScripts() { parent::showScripts(); $this->script('emailsettings.js'); $this->autofocus('email'); } /** * Content area of the page * * Shows a form for adding and removing email addresses and setting * email preferences. * * @return void */ public function showContent() { $user = $this->scoped->getUser(); $this->elementStart('form', array('method' => 'post', 'id' => 'form_settings_email', 'class' => 'form_settings', 'action' => common_local_url('emailsettings'))); $this->elementStart('fieldset'); $this->elementStart('fieldset', array('id' => 'settings_email_address')); // TRANS: Form legend for e-mail settings form. $this->element('legend', null, _('Email address')); $this->hidden('token', common_session_token()); if (!$user->isNull('email')) { $this->element('p', array('id' => 'form_confirmed'), $user->email); // TRANS: Form note in e-mail settings form. $this->element('p', array('class' => 'form_note'), _('Current confirmed email address.')); $this->hidden('email', $user->email); // TRANS: Button label to remove a confirmed e-mail address. $this->submit('remove', _m('BUTTON', 'Remove')); } else { try { $confirm = $this->getConfirmation(); $this->element('p', array('id' => 'form_unconfirmed'), $confirm->address); $this->element( 'p', ['class' => 'form_note'], // TRANS: Form note in e-mail settings form. _('Awaiting confirmation on this address. '. 'Check your inbox (and spam box!) for a message '. 'with further instructions.') ); $this->hidden('email', $confirm->address); // TRANS: Button label to cancel an e-mail address confirmation procedure. $this->submit('cancel', _m('BUTTON', 'Cancel')); } catch (NoResultException $e) { $this->elementStart('ul', 'form_data'); $this->elementStart('li'); // TRANS: Field label for e-mail address input in e-mail settings form. $this->input( 'email', _('Email address'), $this->trimmed('email') ?: null, // TRANS: Instructions for e-mail address input form. Do not translate // TRANS: "example.org". It is one of the domain names reserved for // TRANS: use in examples by http://www.rfc-editor.org/rfc/rfc2606.txt. // TRANS: Any other domain may be owned by a legitimate person or // TRANS: organization. _('Email address, like "UserName@example.org"') ); $this->elementEnd('li'); $this->elementEnd('ul'); // TRANS: Button label for adding an e-mail address in e-mail settings form. $this->submit('add', _m('BUTTON', 'Add')); } } $this->elementEnd('fieldset'); if (common_config('emailpost', 'enabled') && $user->email) { $this->elementStart('fieldset', array('id' => 'settings_email_incoming')); // TRANS: Form legend for incoming e-mail settings form. $this->element('legend', null, _('Incoming email')); $this->elementStart('ul', 'form_data'); $this->elementStart('li'); $this->checkbox( 'emailpost', // TRANS: Checkbox label in e-mail preferences form. _('I want to post notices by email.'), $user->emailpost ); $this->elementEnd('li'); $this->elementEnd('ul'); // Our stylesheets make the form_data list items all floats, which // creates lots of problems with trying to wrap divs around things. // This should force a break before the next section, which needs // to be separate so we can disable the things in it when the // checkbox is off. $this->elementStart('div', array('style' => 'clear: both')); $this->elementEnd('div'); $this->elementStart('div', array('id' => 'emailincoming')); if (!$user->isNull('incomingemail')) { $this->elementStart('p'); $this->element('span', 'address', $user->incomingemail); // @todo XXX: Looks a little awkward in the UI. // Something like "xxxx@identi.ca Send email ..". Needs improvement. $this->element( 'span', 'input_instructions', // TRANS: Form instructions for incoming e-mail form in e-mail settings. _('Send email to this address to post new notices.') ); $this->elementEnd('p'); // TRANS: Button label for removing a set sender e-mail address to post notices from. $this->submit('removeincoming', _m('BUTTON', 'Remove')); } $this->elementStart('p'); if (!$user->isNull('incomingemail')) { // TRANS: Instructions for incoming e-mail address input form, when an address has already been assigned. $msg = _('Make a new email address for posting to; '. 'cancels the old one.'); } else { // TRANS: Instructions for incoming e-mail address input form. $msg = _('To send notices via email, we need to create a unique email address for you on this server:'); } $this->element('span', 'input_instructions', $msg); $this->elementEnd('p'); // TRANS: Button label for adding an e-mail address to send notices from. $this->submit('newincoming', _m('BUTTON', 'New')); $this->elementEnd('div'); // div#emailincoming $this->elementEnd('fieldset'); } $this->elementStart('fieldset', array('id' => 'settings_email_preferences')); // TRANS: Form legend for e-mail preferences form. $this->element('legend', null, _('Email preferences')); $this->elementStart('ul', 'form_data'); if (Event::handle('StartEmailFormData', array($this, $this->scoped))) { $this->elementStart('li'); $this->checkbox( 'emailnotifysub', // TRANS: Checkbox label in e-mail preferences form. _('Send me notices of new subscriptions through email.'), $user->emailnotifysub ); $this->elementEnd('li'); $this->elementStart('li'); $this->checkbox( 'emailnotifymsg', // TRANS: Checkbox label in e-mail preferences form. _('Send me email when someone sends me a private message.'), $user->emailnotifymsg ); $this->elementEnd('li'); $this->elementStart('li'); $this->checkbox( 'emailnotifyattn', // TRANS: Checkbox label in e-mail preferences form. _('Send me email when someone sends me an "@-reply".'), $user->emailnotifyattn ); $this->elementEnd('li'); $this->elementStart('li'); $this->checkbox( 'emailnotifynudge', // TRANS: Checkbox label in e-mail preferences form. _('Allow friends to nudge me and send me an email.'), $user->emailnotifynudge ); $this->elementEnd('li'); Event::handle('EndEmailFormData', array($this, $this->scoped)); } $this->elementEnd('ul'); // TRANS: Button label to save e-mail preferences. $this->submit('save', _m('BUTTON', 'Save')); $this->elementEnd('fieldset'); $this->elementEnd('fieldset'); $this->elementEnd('form'); } /** * Gets any existing email address confirmations we're waiting for * * @return Confirm_address Email address confirmation for user, or null */ public function getConfirmation() { $confirm = new Confirm_address(); $confirm->user_id = $this->scoped->getID(); $confirm->address_type = 'email'; if ($confirm->find(true)) { return $confirm; } throw new NoResultException($confirm); } protected function doPost() { if ($this->arg('save')) { return $this->savePreferences(); } elseif ($this->arg('add')) { return $this->addAddress(); } elseif ($this->arg('cancel')) { return $this->cancelConfirmation(); } elseif ($this->arg('remove')) { return $this->removeAddress(); } elseif ($this->arg('removeincoming')) { return $this->removeIncoming(); } elseif ($this->arg('newincoming')) { return $this->newIncoming(); } // TRANS: Message given submitting a form with an unknown action in e-mail settings. throw new ClientException(_('Unexpected form submission.')); } /** * Save email preferences * * @return void */ public function savePreferences() { if (Event::handle('StartEmailSaveForm', array($this, $this->scoped))) { $emailnotifysub = $this->boolean('emailnotifysub'); $emailnotifymsg = $this->boolean('emailnotifymsg'); $emailnotifynudge = $this->boolean('emailnotifynudge'); $emailnotifyattn = $this->boolean('emailnotifyattn'); $emailpost = $this->boolean('emailpost'); $user = $this->scoped->getUser(); $user->query('START TRANSACTION'); $original = clone($user); $user->emailnotifysub = $emailnotifysub; $user->emailnotifymsg = $emailnotifymsg; $user->emailnotifynudge = $emailnotifynudge; $user->emailnotifyattn = $emailnotifyattn; $user->emailpost = $emailpost; $result = $user->update($original); if ($result === false) { common_log_db_error($user, 'UPDATE', __FILE__); $user->query('ROLLBACK'); // TRANS: Server error thrown on database error updating e-mail preferences. throw new ServerException(_('Could not update user.')); } $user->query('COMMIT'); Event::handle('EndEmailSaveForm', array($this, $this->scoped)); } // TRANS: Confirmation message for successful e-mail preferences save. return _('Email preferences saved.'); } /** * Add the address passed in by the user * * @return void */ public function addAddress() { $user = $this->scoped->getUser(); $email = $this->trimmed('email'); // Some validation if (empty($email)) { // TRANS: Message given saving e-mail address without having provided one. throw new ClientException(_('No email address.')); } $email = common_canonical_email($email); if (empty($email)) { // TRANS: Message given saving e-mail address that cannot be normalised. throw new ClientException(_('Cannot normalize that email address.')); } if (!Validate::email($email, common_config('email', 'check_domain'))) { // TRANS: Message given saving e-mail address that not valid. throw new ClientException(_('Not a valid email address.')); } elseif ($user->email === $email) { // TRANS: Message given saving e-mail address that is already set. throw new ClientException(_('That is already your email address.')); } elseif ($this->emailExists($email)) { // TRANS: Message given saving e-mail address that is already set for another user. throw new ClientException(_('That email address already belongs to another user.')); } if (Event::handle('StartAddEmailAddress', array($user, $email))) { $confirm = new Confirm_address(); $confirm->address = $email; $confirm->address_type = 'email'; $confirm->user_id = $user->getID(); $confirm->code = common_confirmation_code(64); $result = $confirm->insert(); if ($result === false) { common_log_db_error($confirm, 'INSERT', __FILE__); // TRANS: Server error thrown on database error adding e-mail confirmation code. throw new ServerException(_('Could not insert confirmation code.')); } $confirm->sendConfirmation(); Event::handle('EndAddEmailAddress', array($user, $email)); } // TRANS: Message given saving valid e-mail address that is to be confirmed. return _('A confirmation code was sent to the email address you added. '. 'Check your inbox (and spam box!) for the code and instructions '. 'on how to use it.'); } /** * Handle a request to cancel email confirmation * * @return void */ public function cancelConfirmation() { $email = $this->trimmed('email'); try { $confirm = $this->getConfirmation(); if ($confirm->address !== $email) { // TRANS: Message given canceling e-mail address confirmation for the wrong e-mail address. throw new ClientException(_('That is the wrong email address.')); } } catch (NoResultException $e) { // TRANS: Message given canceling e-mail address confirmation that is not pending. throw new AlreadyFulfilledException(_('No pending confirmation to cancel.')); } $confirm->delete(); // TRANS: Message given after successfully canceling e-mail address confirmation. return _('Email confirmation cancelled.'); } /** * Handle a request to remove an address from the user's account * * @return void */ public function removeAddress() { $user = common_current_user(); $email = $this->trimmed('email'); // Maybe an old tab open...? if ($user->email !== $email) { // TRANS: Message given trying to remove an e-mail address that is not // TRANS: registered for the active user. throw new ClientException(_('That is not your email address.')); } $original = clone($user); $user->email = $user->sqlValue('NULL'); // Throws exception on failure. Also performs it within a transaction. $user->updateWithKeys($original); // TRANS: Message given after successfully removing a registered e-mail address. return _('The email address was removed.'); } /** * Handle a request to remove an incoming email address * * @return void */ public function removeIncoming() { $user = common_current_user(); if (empty($user->incomingemail)) { // TRANS: Form validation error displayed when trying to remove an incoming e-mail address while no address has been set. throw new AlreadyFulfilledException(_('No incoming email address.')); } $orig = clone($user); $user->incomingemail = $user->sqlValue('NULL'); $user->emailpost = false; // Throws exception on failure. Also performs it within a transaction. $user->updateWithKeys($orig); // TRANS: Message given after successfully removing an incoming e-mail address. return _('Incoming email address removed.'); } /** * Generate a new incoming email address * * @return void */ public function newIncoming() { $user = common_current_user(); $orig = clone($user); $user->incomingemail = mail_new_incoming_address(); $user->emailpost = true; // Throws exception on failure. Also performs it within a transaction. $user->updateWithKeys($orig); // TRANS: Message given after successfully adding an incoming e-mail address. return _('New incoming email address added.'); } /** * Does another user already have this email address? * * Email addresses are unique for users. * * @param string $email Address to check * * @return boolean Whether the email already exists. */ public function emailExists($email) { $user = common_current_user(); $other = User::getKV('email', $email); if (!$other instanceof User) { return false; } return $other->id != $user->id; } }