Archive for category Open Source
Hide Input Format Options from Drupal Node Create Form
Posted by anil in Drupal, Open Source on May 12th, 2009
Normally you don’t want to show “Input Format” options available in node create form to your website users. You can easily hide them using CSS. We are hiding these options only from normal users and not from admin user.

Input Format
1. Create a hook_form_FormID_alter callback function to add the ID to DIV that containing the Input Format options. Suppose our module name is “story” and we want to hide Input Format options from “Create Story” form. So, our callback function name should be story_form_story_node_form_alter(). Where “story_node_form” is Form ID of create story form.
function story_form_story_node_form_alter(&$form, &$form_state) { global $user; if ($user->uid != 1) { $form['format']['#attributes'] = array('id' => 'fieldset-input-format'); } }
2. We added the ID “fieldset-input-format” to DIV containing Input Format options. Now, we need to hide it using CSS.
#fieldset-input-format { display:none; }
Create image upload field using Drupal Form API
Posted by anil in Drupal, Open Source on April 28th, 2009

Product Form with Image Field
We can create a image/file upload field using Drupal Form API. Here I am considering an example of product form so we can easily understand its use and its implementation. Our product form have two fields product name and product image. I am considering only three fields productid, product name and product image in our product table. The productid is primary key and set to auto increment. Our module name is ‘product.module’.
STEP 1.
function new_product_form() { $form['product_name'] = array( '#title' => t('Product Name'), '#type' => 'textfield', '#required' => TRUE, '#description' => t('Please enter product name.'), ); // Product picture $form['picture'] = array('#type' => 'fieldset', '#title' => t('Product image')); $form['picture']['picture_upload'] = array('#type' => 'file', '#title' => t('Upload product image'), '#size' => 48, '#description' => t('Maximum dimensions are %dimensions and the maximum size is %size kB.', array('%dimensions' => '250x250', '%size' => '30'))); $form['#validate'][] = 'product_validate_picture'; $form['submit'] = array( '#type' => 'submit', '#value' => t('Submit'), ); $form['#redirect'] = 'product_list'; $form['#attributes']['enctype'] = 'multipart/form-data'; return $form; }
We have set the enctype of form as multipart/form-data because we are uploading the file. We also set a validation function product_validate_picture() for validating our uploaded image.
STEP 2.
Now we are creating product_validate_picture() function.
function product_validate_picture(&$form, &$form_state) { $validators = array( 'file_validate_is_image' => array(), 'file_validate_image_resolution' => array('250x250')), 'file_validate_size' => array(30 * 1024), ); if ($file = file_save_upload('picture_upload', $validators)) { // Remove the old picture. if (isset($form_state['values']['_product']->image_path) && file_exists($form_state['values']['_product']->image_path)){ file_delete($form_state['values']['_product']->image_path); } $productid = 0; if (!isset($form_state['values']['productid'])) { // Execute in case of new product $query = "SHOW TABLE STATUS LIKE 'product'"; $rs = db_query($query); $row = db_fetch_object($rs); $productid = isset($row->Auto_increment) ? $row->Auto_increment : 0; } else { $productid = $form_state['values']['productid']; } $info = image_get_info($file->filepath); $destination = variable_get('product_picture_path', 'product_pictures') .'/picture-'. $productid .'.'. $info['extension']; if (file_copy($file, $destination, FILE_EXISTS_REPLACE)) { $form_state['values']['picture'] = $file->filepath; } else { form_set_error('picture_upload', t("Failed to upload the picture image; the %directory directory doesn't exist or is not writable.", array('%directory' => variable_get('product_picture_path', 'product_pictures')))); } } }
Drupal provide a function file_save_upload() for saving the uploaded file to a new location. Its also take associative array of callback functions used to validate the file as optional argument. Here we passing three callback functions file_validate_is_image, file_validate_image_resolution, and file_validate_size for validating file type as image, image resolution and file size respectively. This function return an object containing the file information or 0 in case of error.
We are using this product_validate_picture() function in both new product and edit product form. So, if ‘_product’ is not set in form values then its fetch the id of next auto_increment value set in product table. In case of edit product form it replace the old product image with new image if user uploaded a new image.
We assign the uploaded image path in our form values so we can store that path in out product table.
$form_state['values']['picture'] = $file->filepath;
STEP 3.
function new_product_form_submit($form_id, $form) { $form_values = $form['values']; $product_name = trim($form_values['product_name']); $image_path = $form_values['picture']; $query = "INSERT INTO product(product_name, image_path) VALUES('%s', '%s')"; db_query($query, $product_name, $image_path); drupal_set_message(t('Product has been added successfully.')); }
Then we will save the product imformation in out product table.
STEP 4.
In case of edit product form we are passing productid as arg(2). We fetch the product imformation using this productid and set the default values in form. We also save the product information and productid in form values.
function edit_product_form() { $productid = arg(2); $query = "SELECT * FROM {product} WHERE productid = '%d'"; $rs = db_query($query, $productid); $product_data = db_fetch_object($rs); $form['_product'] = array('#type' => 'value', '#value' => $product_data); $form['productid'] = array('#type' => 'value', '#value' => $product_data->productid); $form['product_name'] = array( '#title' => t('Product Name'), '#type' => 'textfield', '#default_value' => stripslashes($product_data->product_name), '#required' => TRUE, '#description' => t('Please enter product name.'), ); // Product picture $form['picture'] = array('#type' => 'fieldset', '#title' => t('Product image')); $picture = theme('product_picture', (object)$product_data); if ($product_data->image_path) { $form['picture']['current_picture'] = array('#value' => $picture); $form['picture']['picture_delete'] = array('#type' => 'checkbox', '#title' => t('Delete picture'), '#description' => t('Check this box to delete your current picture.')); } else { $form['picture']['picture_delete'] = array('#type' => 'hidden'); } $form['picture']['picture_upload'] = array('#type' => 'file', '#title' => t('Upload product image'), '#size' => 48, '#description' => t('Maximum dimensions are %dimensions and the maximum size is %size kB.', array('%dimensions' => '250x250', '%size' => '30'))); $form['#validate'][] = 'product_validate_picture'; $form['#redirect'] = 'product_list'; $form['#attributes']['enctype'] = 'multipart/form-data'; return $form; }
STEP 5.
Then we will update the product detail. If user want to delete product image then we will delete the current image of product.
function edit_product_form_submit($form_id, $form) { $form_values = $form['values']; $product_name = trim($form_values['product_name']); $image_path = $form_values['_product']->image_path; // Delete picture if requested, and if no replacement picture was given. if (!empty($form_values['picture_delete']) && $form_values['picture'] == '') { if ($form_values['_product']->image_path && file_exists($form_values['_product']->image_path)) { file_delete($form_values['_product']->image_path); } $image_path = ''; } if (isset($form_values['picture']) && $form_values['picture'] != '') { $image_path = $form_values['picture']; } $query = "UPDATE {product} SET product_name = '%s', image_path = '%s' WHERE productid = '%d'"; db_query($query, $product_name, $image_path, $productid); drupal_set_message(t('Product has been updated successfully.')); }
STEP 6.
We are also theming our product image. For this, we are using a product-picture.tpl.php file. First we register our theme in theme registry using hook_theme() callback function.
function product_theme() { return array( 'product_picture' => array( 'arguments' => array('product' => NULL), 'template' => 'product-picture', ), ); } function template_preprocess_product_picture(&$variables) { $variables['picture'] = ''; $product = $variables['product']; if (!empty($product->image_path) && file_exists($product->image_path)) { $picture = file_create_url($product->image_path); } else if (variable_get('product_picture_default', '')) { $picture = variable_get('product_picture_default', ''); } if (isset($picture)) { $alt = t("@product's picture", array('@product' => $product->product_name )); $variables['picture'] = theme('image', $picture, $alt, $alt, '', FALSE); } }
<?php //product-picture.tpl.php file ?> <div class="picture"> <?php print $picture; ?> </div>
Create Drupal Form elements like ‘DATE’ element
Posted by admin in Drupal, Open Source on March 30th, 2009

Drupal Form Elements
Sometime we require from elements for filling “Height” or “Credit card expiry date” or “Phone number(in any specific format)”. We can easily create these type of form elements in Drupal. I am showing here how we can create such type of form elements.
Here I am taking the example of “Height” element. For height element we require two select boxes. One for showing feet and other for inches.
Step 1: First create “hook_elements()” in your module. This hook allows modules to declare their own Forms API element types and specify their default values.
function module_name_elements() { return array( 'height' => array( '#input' => TRUE, '#process' => array('expand_height'), '#element_validate' => array('height_validate'), ), }
Step 2: Define the “#process” callback function “expand_height()”. This will take $element as argument and process it.
function expand_height($element) { // Assigning default values to element. You can override these values by passing your own values using "#default_value". if (empty($element['#value'])) { $element['#value'] = array( 'feet' => '-', 'inch' => '-', ); } $element['#tree'] = TRUE; $parents = $element['#parents']; $parents[] = 'feet'; $element['feet'] = array( '#type' => 'select', '#default_value' => $element['#value']['feet'], '#options' => array('-' => '-', '2' => '2', '3' => '3', '4' => '4', '5' => '5', '6' => '6', '7' => '7', '8' => '8',), '#suffix' => t('ft.'), ); $parents = $element['#parents']; $parents[] = 'inch'; $element['inch'] = array( '#type' => 'select', '#default_value' => $element['#value']['inch'], '#options' => array('-' => '-', '0' => '0', '1' => '1', '2' => '2', '3' => '3', '4' => '4', '5' => '5', '6' => '6', '7' => '7', '8' => '8', '9' => '9', '10' => '10', '11' => '11'), '#suffix' => t('in.'), ); return $element; }
Step 3: Theme your custom element by defining “theme_hook()” for your element.
function theme_height($element) { return theme('form_element', $element, '<div class="container-inline">'. $element['#children'] .'</div>'); }
step 4: We need to register our above theme function “theme_height()” in theme registry. We can do that using “hook_theme()” hook. To see effect of your code, clear your cache because all theme hook is cached by Drupal.
function module_name_theme() { return array( 'height' => array( 'arguments' => array('element' => NULL)) ); }
Step 5: Define “hook_validate()” for your element. This function is automatically called by FORM API and it validate the given value of element. We verify the values of element and if value is not valid we will set error message using form_error() function. This function take $element as argument.
function height_validate($element) { if (trim($element['#value']['feet']) == '-' || trim($element['#value']['inch']) == '-') { form_error($element, t('The specified height is invalid.')); } }
How you use it:
$user->height = array('feet' => '5', 'inch' => '8'); // Create element of 'type' height $form['height'] = array('#type' => 'height', '#title' => t('Height'), '#required' => TRUE, '#default_value' => isset($user->height) ? $user->height : array('feet' => '-', 'inch' => '-'), ); //Access it in form_submit() like this $height = $edit['height']['feet']. '-' . $edit['height']['inch'];
Code for Phone Number Element
<php function mymodule_theme() { return array( 'phone' => array( 'arguments' => array('element' => NULL)) ); } function mymodule_elements() { return array( 'phone' => array( '#input' => TRUE, '#process' => array('expand_phone'), '#element_validate' => array('phone_validate'), ), ); } function expand_phone($element) { if (empty($element['#value'])) { $element['#value'] = array( 'part1' => '', 'part2' => '', 'part3' => '', ); } $element['#tree'] = TRUE; $parents = $element['#parents']; $parents[] = 'part1'; $element['part1'] = array( '#type' => 'textfield', '#size' => 4, '#maxlength' => 3, '#default_value' => $element['#value']['part1'], '#prefix' => '(', '#suffix' => ')', ); $parents = $element['#parents']; $parents[] = 'part2'; $element['part2'] = array( '#type' => 'textfield', '#size' => 4, '#maxlength' => 3, '#default_value' => $element['#value']['part2'], '#suffix' => '-', ); $parents = $element['#parents']; $parents[] = 'part3'; $element['part3'] = array( '#type' => 'textfield', '#size' => 6, '#maxlength' => 4, '#default_value' => $element['#value']['part3'], ); return $element; } function theme_phone($element) { return theme('form_element', $element, '<div class="container-inline">'. $element['#children'] .'</div>'); } function phone_validate($element) { if (trim($element['#value']['part1']) == '' || trim($element['#value']['part2']) == '' || trim($element['#value']['part3']) == '' || !is_numeric($element['#value']['part1']) || !is_numeric($element['#value']['part2']) || !is_numeric($element['#value']['part3'])) { form_error($element, t('The specified phone number is invalid.')); } } ?>
Code for Credit card expiry date Element
<?php function mymodule_theme() { return array( 'cardexpdate' => array( 'arguments' => array('element' => NULL)) ); } function mymodule_elements() { return array( 'cardexpdate' => array( '#input' => TRUE, '#process' => array('expand_cardexpdate'), '#element_validate' => array('cardexpdate_validate'), ), ); } function expand_cardexpdate($element) { $next_month_date = strtotime("+1 month"); $month = date('n', $next_month_date); $year = date('Y', $next_month_date); if (empty($element['#value'])) { $element['#value'] = array( 'month' => $month, 'year' => $year, ); } $element['#tree'] = TRUE; $parents = $element['#parents']; $parents[] = 'month'; $element['month'] = array( '#type' => 'select', '#default_value' => $element['#value']['month'], '#options' => drupal_map_assoc(range(1, 12), 'cardexpdate_map_month'), ); $parents = $element['#parents']; $parents[] = 'year'; $element['year'] = array( '#type' => 'select', '#default_value' => $element['#value']['year'], '#options' => drupal_map_assoc(range($year, 2020)), ); return $element; } function theme_cardexpdate($element) { return theme('form_element', $element, '<div class="container-inline">'. $element['#children'] .'</div>'); } function cardexpdate_validate($element) { $month = $element['#value']['month']; $year = $element['#value']['year']; $exp_date = mktime(0, 0, 0, $month, date('j'), $year); if ($exp_date < time()) { form_error($element, t('The specified expiry date is invalid.')); } } function cardexpdate_map_month($month) { return format_date(gmmktime(0, 0, 0, $month, 2, 1970), 'custom', 'm', 0); } ?>
Assign attributes to options using FormAPI in Drupal
Posted by admin in Drupal, Open Source, PHP on March 25th, 2009
Drupal FAPI does not support this feature but we can add attributes to options by slightly modifying form.inc file.
You only need to change following line in form.inc file.
<?php $options .= '<option value="'. check_plain($key) .'"'. $selected .'>'. check_plain($choice) .'</option>'; //line no. 1437 of form_select_options() in drupal 6.10 ?>
with this
<?php $options .= '<option value="'. check_plain($key) .'"'. $selected . drupal_attributes($element['#options_attribute'][$key]) . '>'. check_plain($choice) .'</option>'; ?>
You can use like it.
<?php $shipping_methods = array('1' => 'Standard Two-Day ($17.00 Total Shipping Price)', '2' => 'Priority Overnight ($26.00 Total Shipping Price)', '3' => 'FedEx Next Day Delivery $23.95', '4' => 'FedEx Next Day Delivery $23.95'); $shipping_method_options = array( '1' => array('price' => '17.00', 'class' => 'one'), '2' => array('price' => '26.00', 'class' => 'two'), '3' => array('price' => '23.95', 'class' => 'three'), '4' => array('price' => '23.95', 'class' => 'four')); $form['default_shipping_method'] = array( '#title' => t('Shipping Method'), '#type' => 'select', '#options' => $shipping_methods, '#options_attribute' => $shipping_method_options, '#default_value' => variable_get('default_shipping_method', 0), '#description' => t('You can change your shipping method from here.'), ); ?>
Create Drupal form using theme_table() like module list form
Posted by admin in Drupal, Open Source, PHP on March 22nd, 2009
I am showing here an example for Drupal form like you saw in Module list pages.

Drupal Form using theme_table() function
I am considering here an example of Featured Product Management form so we can easily understand it.
Step 1:- Create a menu hook for registering your page path. This registered path will display the featured product management page.
<?php function product_menu() { $items = array(); $items['featured_product_mgmt'] = array( ‘title’ => ‘Manage featured product’, ‘page callback’ => ‘drupal_get_form’, ‘page arguments’ => array(’featured_product_form’), ‘access arguments’ => array(’access product’), ‘type’ => MENU_CALLBACK, ); return $items; } ?>
Step 2:- Create form object for your form. Here the important part is “featured” form element of type “checkboxes”. Checkboxes type is a group of checkbox and format a set of checkboxes. #options is an associative array, where the key is the #return_value of the checkbox and the value is displayed. We will store other information like name, category, discount etc. of product in form array so we can display it later.
<?php function featured_product_form() { $query = “SELECT p.*, UNIX_TIMESTAMP(p.create_date) AS create_date, c.name FROM {product} p LEFT JOIN {category} c ON p.cid = c.cid WHERE p.status = 1 ORDER BY p.product_name”; $rs = db_query($query); $featured_products = featured_product(); $status = array(); if ($rs) { while ($data = db_fetch_object($rs)) { $options[$data->productid] = ”; $form[$data->productid]['name'] = array(’#value’ => stripslashes($data->product_name)); $form[$data->productid]['category'] = array(’#value’ => stripslashes($data->name)); $form[$data->productid]['discount'] = array(’#value’ => $data->discount . ‘%’); $form[$data->productid]['createdon'] = array(’#value’ => date(’m-d-Y’, $data->create_date)); if (in_array($data->productid, $featured_products)) { $status[] = $data->productid; } } } $form['featured'] = array( ‘#type’ => ‘checkboxes’, ‘#options’ => $options, ‘#default_value’ => $status, ); $form['submit'] = array( ‘#type’ => ’submit’, ‘#value’ => t(’Submit’), ); $form['cancel'] = array( ‘#type’ => ‘markup’, ‘#value’ => l(t(’Cancel’), ‘dashboard’), ); $form['#redirect'] = ‘featured_product_mgmt’; return $form; } ?>
Step 3:- Create a form submit function. Any form submitted using the “submit” button will pass to their corresponding function if it is available. We will extract the user input from the form array and update our database accordingly.
<?php function featured_product_form_submit($form_id, $form) { $form_values = $form['values']; $featured = $form_values['featured']; $selected_products = array(); foreach($featured as $key => $value) { if ($value) { $selected_products[] = “(’$value’)”; } } $value_string = @implode(’,', $selected_products); // Delete all previous featured products $query = “DELETE FROM {product_featured}”; db_query($query); // Insert new featured products if (count($selected_products)) { $query = “INSERT INTO {product_featured}(productid) VALUES $value_string”; db_query($query); } drupal_set_message(t(’Featured product list has been updated successfully.’)); } ?>
Step 4:- Register your modules theme implementation using “hook_theme()” function.
<?php function product_theme() { return array( ‘featured_product_form’ => array(‘arguments’ => array(’form’ => NULL),), ); } ?>
Step 5:- Then finally define your theme function which actually format your form layout in list using the $form array. Using “foreach” loop we navigate through each item of $form array and create $rows array which we pass to “theme_table()” function later. Create the header of your table and call “table_theme()” function with $header and $rows parameters. Call the “drupal_render()” function for render the submit and cancel buttons.
<?php function theme_featured_product_form($form) { $rows = array(); foreach (element_children($form) as $key) { $row = array(); if (isset($form[$key]['name'])) { $status = drupal_render($form['featured'][$key]); $row[] = array(’data’ => $status, ‘class’ => ‘checkbox’); $row[] = ‘‘. drupal_render($form[$key]['name']) .’‘; $row[] = array(’data’ => drupal_render($form[$key]['category'])); $row[] = array(’data’ => drupal_render($form[$key]['discount'])); $row[] = array(’data’ => drupal_render($form[$key]['createdon'])); $rows[] = $row; } } // Individual table headers. $header = array(); $header[] = array(’data’ => t(’Featured’), ‘class’ => ‘checkbox’); $header[] = t(’Name’); $header[] = t(’Category’); $header[] = t(’Discount’); $header[] = t(’Created on’); $output = theme(’table’, $header, $rows); $output .= drupal_render($form); return $output; } ?>