Пользователь должен иметь возможность сменить свой пароль, а также поменять свои контактные данные (например, email) при необходимости. Для этого мы реализуем действие edit, которое будет в себя включать весь нужный функционал.
Начнем, пожалуй, с шаблона:
Форма регистрации и форма редактирования профиля содержит много одинаковых полей. Воспользуемся этим обстоятельством и выделим общие поля в отдельный шаблон shop/template/user/include/form_fields.phtml:
Файл shop/template/user/include/form_fields.html:
{{include file='_admin/form_errors.phtml'/}} <dl class="required field"> <dt><label for="title">Login:</label></dt> <dd>{{input name='login' type='text' title='Login'/}}</dd> </dl> <dl class="required field"> <dt><label for="name">Name:</label></dt> <dd>{{input name='name' type='text' title='Name'/}}</dd> </dl> <dl class="required field"> <dt><label for="email">Email:</label></dt> <dd>{{input name='email' type='text' title='Email' error_class='error'/}}</dd> </dl> <dl class="field"> <dt><label for="address">Delivery address:</label></dt> <dd> {{textarea name="address" id='address' type="text" title="Delivery address"/}} </dd> </dl>
Файл shop/template/user/edit.html:
<? $this->title = 'Your profile'; ?> {{wrap with="front_page_layout.phtml" in="content_zone"}} This is your profile page. You can change some of your profile fields as well as your password<br/> <h2>Profile</h2> {{form id='profile_form' name='profile_form' method='post'}} {{include file='user/include/form_fields.phtml'/}} <hr/> <input id='edit' name='edit' type='submit' value='Update profile'/> {{/form}} <h2>Password change form</h2> {{form id='change_password_form' name='change_password_form' method='post'}} {{include file='_admin/form_errors.phtml'/}} <dl class="required field"> <dt><label for="name">Old password:</label></dt> <dd>{{input name='old_password' type='text' title='Old password'/}}</dd> </dl> <dl class="required field"> <dt><label for="name">Password:</label></dt> <dd>{{input name='password' type='text' title='Password'/}}</dd> </dl> <dl class="required field"> <dt><label for="name">Repeat password:</label></dt> <dd>{{input name='repeat_password' type='text' title='Repeat password'/}}</dd> </dl> <hr/> <input id='change_password' name='change_password' type='submit' value='Change password'/> {{/form}} {{/wrap}}
Шаблон содержит 2 формы:
Теперь мы покажем, как можно обрабатывать 2 формы при помощи одного действия в контроллере.
<?php class UserController extends lmbController { [...] function doEdit() { $this->setFormDatasource($this->toolkit->getUser(), 'profile_form'); if($this->request->has('change_password')) $this->_changeUserPassword(); if($this->request->has('edit')) $this->_updateUserProfile(); } protected function _changeUserPassword() { $this->useForm('change_password_form'); $this->_validateChangePasswordForm(); if($this->error_list->isValid()) { $user = $this->toolkit->getUser(); $user->setPassword($this->request->get('password')); $user->save(); $this->flashMessage('Your password was changed'); $this->toolkit->redirect(); } } protected function _updateUserProfile() { $this->useForm('profile_form'); $this->setFormDatasource($this->toolkit->getUser()); $user_properties = $this->request->getPost( array('login', 'name', 'email', 'password', 'address') ); $user = $this->toolkit->getUser(); $user->import($user_properties); if($user->trySave($this->error_list)) { $this->flashMessage('Your profile was changed'); $this->toolkit->redirect(); } } protected function _validateChangePasswordForm() { $this->validator->addRequiredRule('old_password'); $this->_validatePasswordField(); $user = $this->toolkit->getUser(); if($old_password = $this->request->get('old_password')) { $hashed_password = User :: cryptPassword($old_password); if($user->getHashedPassword() != $hashed_password) $this->error_list->addError('Wrong old password', array('old_password')); } } [...] } ?>
Выбор той или иной функции по обработке формы по смене пароля или же по обновлению данных пользователя осуществляется очень просто - исходя из того, какая переменная пришла в request-е.
Мы также сразу передаем текущего пользователя в форму в шаблоне, так как желаем, чтобы эти данные всегда отображались в полях формы все зависимости от того, какая форма была до этого отправлена. Примем мы использовали только метод lmbController :: setFormDatasource($datasource, $form_id):
$this->setFormDatasource($this->toolkit->getUser(), 'profile_form');
Это позволит не выводить ошибки валидации сразу рядом с двумя формами, если пользователь неправильно ввел информацию в форму по смене пароля.
Вот как должна выглядеть страница /user/edit:
Для отображения информации по текущему пользователю необходимо передать объект пользователя в шаблон. Это можно сделать несколькими способами:
Мы покажем все способы и остановимся на получение данных в шаблоне через toolkit.
Для этого способа необходоимо создать базовый класс BaseShopController со следующим кодом:
class BaseShopController extends lmbController { function performAction() { $this->view->set('user', $this->toolkit->getUser()); return parent :: performAction(); } }
После этого в MACRO шаблоне объект пользователя будет доступен в корневом контейнере данных, то есть можно будет использовать такие выражения:
<? if($this->user()->is_logged_in) { ?> Logged in as :{{$#user.name}} <? } ?>
MACRO-позволяет использовать обычные php-вставки. Поэтому мы можем легко получить тулкит прямо в шаблоне и передать объект user в нужный нам контейнер данных, например:
[...] <dl id="profile"> <dt>Profile</dt> <dd> <? if(lmbToolkit :: instance()->getUser()->is_logged_in) { ?> Logged in as :{{$#user.name}} <? } ?> </dd> </dl> [...]
Если ваш верстальщик нормально относится к подобным php-вставкам в шаблонах - можете смело воспользоваться таким методом.
Этот способ предусматривает добавление нескольких строк в фильтр:
class SampleFilter implements lmbInterceptingFilter { function run($filter_chain) { $toolkit = lmbToolkit :: instance(); $user = $toolkit->getUser(); [...] $toolkit->getView()->set('user', $user); [...] } }
Данная небольшая модификация позволит использовать user из корневого контейнера данных, как в способе с базовым контроллером.
Начиная с версии пакета WEB_APP 0.4 (соответствует релизу 2007.2) toolkit поддерживает получение данных через базовые методы get(), к кому же toolkit стал доступен в шаблоне в корневой контейнере данных. Все это позволяет даже не заботиться о передаче user-а в шаблон, а действовать в стиле PullView:
файл shop/template/user/include/profile_box.phtml:
<? if($user->is_logged_in) { ?> <dd> User: {$user.name}<br/> Login: {$user.login}<br/> Email: {$user.email}<br/> <a href="/user/edit/">edit</a> <a href="/user/logout/">logout</a> </dd> <? } else {?> <dd> <form method="POST" id='login_form' action='/user/login/'> <label for='login'>Login:</label><br/> <input type="text" name="login" id="login" title="Login" class='input'/><br/> <label for='passwd'>Password:</label><br/> <input type="text" name="password" id="password" type="password" title="Password" class='input'/><br/> <p>Use <b>admin</b>/<b>secret</b> to enter in <a href="/admin/">administration panel</a>.</p> <input type='submit' name='submitted' value="Submit"/><br/> </form> <a href="/user/register/">Register</a> </dd> <? } ?>
Файл shop/template/front_page_layout.phtml:
[...] <div id="sidebar"> <div id="navigation"> <ul> <li><a href="product">Products</a></li> <li><a href="cart">Your Cart</a></li> <? if($this->toolkit->getUser()->is_logged_in) { ?> <li><a href="/user/orders/">Your Orders</a></li> <? } ?> </ul> </div> <dl id="profile"> <dt>Profile</dt> {{include file="user/include/profile_box.phtml" user="$#toolkit.user"/}} </dl> </div> [...]
В этом примере, как вы видите, мы передаем переменную user в шаблон profile_box.
Если вы модифицировали шаблон front_page_layout.phtml, то теперь после аутентификации на странице /login вы сможете увидеть данные своей учетной записи в левой части страницы.
Для удобства работы с пользователями добваим CRUD в панель управления - Шаг3.4 Управление списком пользователей
Обсуждение