Проблема с FAPI. Когда друпал строит форму вызывает drupal_get_form.
При построении формы он ей выдаёт идентификатор #form_build_id. Его он в дальнейшем использует для восстановления формы из кэша, сохранения в кэш, передавая этот параметр как аргумент $_POST в запросах.
Если мы обновим страницу с формой, FORMS API 'забудет' про предыдущий #form_build_id и построит форму снова. Как этого избежать? Логично что нужно как-то выдывать свой #forms_build_id с захешированным uid юзера, но как?
Комментарии
Как вам такой вариант?
Вставить что-то вроде этого до вызова drupal_get_form():
if(isset($_SESSION['your_form_id']['form_build_id']))
{
$_POST['form_build_id'] = $_SESSION['your_form_id']['form_build_id'];
}
а в hook_form_alter():
$_SESSION['your_form_id']['form_build_id'] = $form['#build_id'];
будет работать? Как считаете?
Форум мёртв или у меня вопросы сложные? Может быть не понятные?
Проблема то распространённая сто пудов.
Короче решил так, весьма красиво получилось:
Перед вызовом drupal_get_form вставляю код:
$path = drupal_get_path('module', 'ctools'); //путь до функции ctools_object_cache_get
include_once $path.'/includes/object-cache.inc';
global $user;
if (isset($_GET['from_cache'])) { //если в строке запроса есть разрекшение на чтение формы из кэша
$data = ctools_object_cache_get('post_data', $user->uid.'_post'); //читаем данные из кэша
$_POST = $data->post; //подменяем текущий $_POST на $_POST из кэша.
}
В hook_form_alter пишу это:
$obj = new stdClass(); //новый объект в котором будем хранить $_POST
$obj->post = $_POST;
ctools_object_cache_set('post_data', $user->uid.'_post' , $obj); //сохранчем кэш
Использовал 2 функции для работы с кэшем из модуля ctools.
Они очень простые:
* Get an object from the non-volatile ctools cache.
*
* This function caches in memory as well, so that multiple calls to this
* will not result in multiple database reads.
*
* @param $obj
* A 32 character or less string to define what kind of object is being
* stored; primarily this is used to prevent collisions.
* @param $name
* The name of the object being stored.
* @param $skip_cache
* Skip the memory cache, meaning this must be read from the db again.
*
* @return
* The data that was cached.
*/
function ctools_object_cache_get($obj, $name, $skip_cache = FALSE) {
$cache = &ctools_static(__FUNCTION__, array());
$key = "$obj:$name";
if ($skip_cache) {
unset($cache[$key]);
}
if (!array_key_exists($key, $cache)) {
$data = db_fetch_object(db_query("SELECT * FROM {ctools_object_cache} WHERE sid = '%s' AND obj = '%s' AND name = '%s'", session_id(), $obj, $name));
if ($data) {
$cache[$key] = unserialize(db_decode_blob($data->data));
}
}
return isset($cache[$key]) ? $cache[$key] : NULL;
}
/**
* Store an object in the non-volatile ctools cache.
*
* @param $obj
* A 32 character or less string to define what kind of object is being
* stored; primarily this is used to prevent collisions.
* @param $name
* The name of the object being stored.
* @param $cache
* The object to be cached. This will be serialized prior to writing.
*/
function ctools_object_cache_set($obj, $name, $cache) {
// Store the CTools session id in the user session to force a
// session for anonymous users in Drupal 7 and Drupal 6 Pressflow.
// see http://drupal.org/node/562374, http://drupal.org/node/861778
if (empty($GLOBALS['user']->uid) && empty($_SESSION['ctools_session_id'])) {
$_SESSION['ctools_hold_session'] = TRUE;
}
ctools_object_cache_clear($obj, $name);
db_query("INSERT INTO {ctools_object_cache} (sid, obj, name, data, updated) VALUES ('%s', '%s', '%s', %b, %d)", session_id(), $obj, $name, serialize($cache), time());
}
вот и всё, теперь если мы хотим чтобы форма была восстановлена из кэша при переходе на неё по ссылке в адресной строке ($_GET) нужно указать аргумент from_cache = true.
Отпишите если хоть кто-то понял