Upload de imagem em um módulo personalizado

8

Estou escrevendo um módulo personalizado e preciso fazer o upload de uma imagem. Estou tendo problemas para encontrar boa documentação sobre isso, mas acho que estou perto.

o que estou perdendo? $ file retorna false no envio do formulário.

function mymodule_custom_content_block_form($form_state){
    $form = array();
    $form['custom_content_block_text'] = array(
        '#type' => 'textarea',
        '#title' => t('Block text'),
        '#default_value' => variable_get('mymodule_custom_content_block_text'),
        '#required' => true,
    );
    $form['custom_content_block_image'] = array(
        '#type' => 'file',
        '#name' => 'custom_content_block_image',
        '#title' => t('Block image'),
        '#size' => 40,
        '#description' => t("Image should be less than 400 pixels wide and in JPG format."),
    );  
    $form['submit'] = array(
        '#type' => 'submit',
        '#value' => t('Update'),
    );
    return $form;
}

function mymodule_custom_content_block_form_submit($form, &$form_state){
    if(isset($form_state['values']['custom_content_block_image'])){
        $validators = array('file_validate_extensions' => array('jpg jpeg'));
        $file = file_save_upload('custom_content_block_image', $validators, 'public://');
        if($file == false){
            drupal_set_message(t("Error saving image."), $type = "error", $repeat = false);
        }
        else{
            $file->status = FILE_STATUS_PERMANENT;
            $file = file_save($file);   
        }
    }
    variable_set('mymodule_custom_content_block_text', $form_state['values']['custom_content_block_text']);
    drupal_set_message(t('Custom Content Block has been updated.'));
}
Tim Snyder
fonte

Respostas:

19

Se você não se importa que eu diga que está fazendo isso da maneira mais difícil. O Drupal possui um managed_filetipo de elemento que lida com grande parte dessa lógica para você:

function mymodule_custom_content_block_form($form, &$form_state) {
  $form['custom_content_block_image'] = array(
    '#type' => 'managed_file',
    '#name' => 'custom_content_block_image',
    '#title' => t('Block image'),
    '#size' => 40,
    '#description' => t("Image should be less than 400 pixels wide and in JPG format."),
    '#upload_location' => 'public://'
  ); 

  return $form; 
}

function mymodule_custom_content_block_form_submit($form, &$form_state) {
  if (isset($form_state['values']['custom_content_block_image'])) {
    $file = file_load($form_state['values']['custom_content_block_image']);

    $file->status = FILE_STATUS_PERMANENT;

    file_save($file);
  }
}
Clive
fonte
Note-se que file_save só está disponível após o Drupal 6.
Será
4

Com a resposta do Clive, minha imagem foi excluída após 6 horas. Então, se alguém experimentou o mesmo problema que eu tenho. Aqui está a solução (da resposta de Clive com uma pequena adição).

function mymodule_custom_content_block_form($form, &$form_state) {
  $form['custom_content_block_image'] = array(
    '#type' => 'managed_file',
    '#name' => 'custom_content_block_image',
    '#title' => t('Block image'),
    '#size' => 40,
    '#description' => t("Image should be less than 400 pixels wide and in JPG format."),
    '#upload_location' => 'public://'
  ); 

  return $form; 
}

function mymodule_custom_content_block_form_submit($form, &$form_state) {
  if (isset($form_state['values']['custom_content_block_image'])) {
    $file = file_load($form_state['values']['custom_content_block_image']);

    $file->status = FILE_STATUS_PERMANENT;

    $file_saved =file_save($file);
    // Record that the module is using the file. 
    file_usage_add($file_saved, 'mymodule_custom_content_block_form', 'custom_content_block_image', $file_saved->fid); 
  }
}

A solução é adicionar file_usage_add. Da documentação da API:

Nota: Novos arquivos são carregados com o status 0 e são tratados como arquivos temporários que são removidos após 6 horas via cron. Seu módulo é responsável por alterar o status dos objetos de arquivo $ para FILE_STATUS_PERMANENT e salvar o novo status no banco de dados. Algo como o seguinte em seu manipulador de envio deve fazer o truque.

Consulte: https://api.drupal.org/api/drupal/developer%21topics%21forms_api_reference.html/7.x#managed_file

Gulok
fonte
1

Este atributo deve ser adicionado ao seu formulário para que ele funcione com os uploads de arquivos.

$form['#attributes']['enctype'] = "multipart/form-data";
DrupalFever
fonte