123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511 |
- <?php
- // This file is part of GNU social - https://www.gnu.org/software/social
- //
- // GNU social is free software: you can redistribute it and/or modify
- // it under the terms of the GNU Affero General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- //
- // GNU social is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- // GNU Affero General Public License for more details.
- //
- // You should have received a copy of the GNU Affero General Public License
- // along with GNU social. If not, see <http://www.gnu.org/licenses/>.
- /**
- * Settings for email
- *
- * @category Settings
- * @package GNUsocial
- * @author Evan Prodromou <evan@status.net>
- * @author Zach Copley <zach@status.net>
- * @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->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->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->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('BEGIN');
- $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 = DB_DataObject_Cast::sql('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 = DB_DataObject_Cast::sql('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;
- }
- }
|