'admin/store/orders/'. arg(3) .'/packages', 'access' => user_access('fulfill orders'), 'title' => t('Packages'), 'callback' => 'uc_shipping_order_packages', 'callback arguments' => array(arg(3)), 'weight' => 6, 'type' => MENU_LOCAL_TASK, ); $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/packages/new', 'access' => user_access('fulfill orders'), 'title' => t('New packages'), 'callback' => 'drupal_get_form', 'callback arguments' => array('uc_shipping_new_package', arg(3)), 'type' => MENU_CALLBACK, ); if (is_numeric(arg(5))) { $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/packages/'. arg(5) .'/edit', 'access' => user_access('fulfill orders'), 'title' => t('Edit package'), 'callback' => 'drupal_get_form', 'callback arguments' => array('uc_shipping_package_edit', arg(3), arg(5)), 'type' => MENU_CALLBACK, ); $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/packages/'. arg(5) .'/cancel', 'access' => user_access('fulfill orders'), 'title' => t('Cancel package shipment'), 'callback' => 'drupal_get_form', 'callback arguments' => array('uc_shipping_package_cancel_confirm', arg(3), arg(5)), 'type' => MENU_CALLBACK, ); $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/packages/'. arg(5) .'/delete', 'access' => user_access('fulfill orders'), 'title' => t('Delete package'), 'callback' => 'drupal_get_form', 'callback arguments' => array('uc_shipping_package_delete_confirm', arg(3), arg(5)), 'type' => MENU_CALLBACK, ); } $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/shipments', 'access' => user_access('fulfill orders'), 'title' => t('Shipments'), 'callback' => 'uc_shipping_order_shipments', 'callback arguments' => array(arg(3)), 'weight' => 7, 'type' => MENU_LOCAL_TASK, ); $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/shipments/new', 'access' => user_access('fulfill orders'), 'title' => t('New shipment'), 'callback' => 'drupal_get_form', 'callback arguments' => array('uc_shipping_new_shipment', arg(3)), 'type' => MENU_CALLBACK, ); if (is_numeric(arg(5))) { $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/shipments/'. arg(5), 'title' => t('Shipment !id', array('!id' => arg(5))), 'callback' => 'uc_shipping_shipment_view', 'callback arguments' => array(arg(3), arg(5)), 'type' => MENU_CALLBACK, ); $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/shipments/'. arg(5) .'/view', 'title' => t('View'), 'weight' => -5, 'type' => MENU_DEFAULT_LOCAL_TASK, ); $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/shipments/'. arg(5) .'/edit', 'access' => user_access('fulfill orders'), 'title' => t('Edit'), 'callback' => 'drupal_get_form', 'callback arguments' => array('uc_shipping_shipment_edit', arg(3), arg(5)), 'type' => MENU_LOCAL_TASK, ); $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/shipments/'. arg(5) .'/cancel', 'access' => user_access('fulfill orders'), 'title' => t('Cancel shipment'), 'callback' => 'drupal_get_form', 'callback arguments' => array('uc_shipping_shipment_cancel_confirm', arg(3), arg(5)), 'type' => MENU_CALLBACK, ); $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/shipments/'. arg(5) .'/delete', 'access' => user_access('fulfill orders'), 'title' => t('Delete shipment'), 'callback' => 'drupal_get_form', 'callback arguments' => array('uc_shipping_shipment_delete_confirm', arg(3), arg(5)), 'type' => MENU_CALLBACK, ); } $items[] = array('path' => 'admin/store/orders/'. arg(3) .'/ship', 'access' => user_access('fulfill orders'), 'title' => t('Ship packages'), 'callback' => 'uc_shipping_make_shipment', 'callback arguments' => array(arg(3)), 'type' => MENU_CALLBACK, ); } } return $items; } /** * Implementation of hook_perm(). */ function uc_shipping_perm() { return array('fulfill orders'); } /****************************************************************************** * Übercart hooks * ******************************************************************************/ function uc_shipping_order_pane() { $panes[] = array( 'id' => 'packages', 'callback' => 'uc_shipping_order_pane_packages', 'title' => t('Tracking numbers'), 'desc' => t('Display tracking numbers of shipped packages.'), 'class' => 'pos-left', 'weight' => 7, 'show' => array('view', 'invoice', 'customer'), ); return $panes; } function uc_shipping_order_actions($order) { $actions = array(); $module_path = base_path() . drupal_get_path('module', 'uc_shipping'); if (user_access('fulfill orders')) { $result = db_query("SELECT nid FROM {uc_order_products} WHERE order_id = %d AND data LIKE '%%s:9:\"shippable\";s:1:\"1\";%%'", $order->order_id); if (db_num_rows($result)) { $title = t('Package order !order_id products.', array('!order_id' => $order->order_id)); $actions[] = array( 'name' => t('Package'), 'url' => 'admin/store/orders/'. $order->order_id .'/packages', 'icon' => ''. $title .'', 'title' => $title, ); $result = db_query("SELECT package_id FROM {uc_packages} WHERE order_id = %d", $order->order_id); if (db_num_rows($result)) { $title = t('Ship order !order_id packages.', array('!order_id' => $order->order_id)); $actions[] = array( 'name' => t('Ship'), 'url' => 'admin/store/orders/'. $order->order_id .'/shipments', 'icon' => ''. $title .'', 'title' => $title, ); } } } return $actions; } /****************************************************************************** * Menu callbacks * ******************************************************************************/ /** * Display a list of an order's packaged products. */ function uc_shipping_order_packages($order_id) { $header = array(t('Package ID'), t('Products'), t('Shipping type'), t('Package type'), t('Shipment ID'), t('Tracking number'), t('Labels'), array('data' => t('Actions'), 'colspan' => 3)); $rows = array(); $result = db_query("SELECT * FROM {uc_packages} WHERE order_id = %d", $order_id); while ($package = db_fetch_object($result)) { $row = array(); $row[] = $package->package_id; $product_list = array(); $result2 = db_query("SELECT op.order_product_id, pp.qty, op.title, op.model FROM {uc_packaged_products} AS pp LEFT JOIN {uc_order_products} AS op ON op.order_product_id = pp.order_product_id WHERE pp.package_id = %d", $package->package_id); while ($product = db_fetch_object($result2)) { $product_list[] = $product->qty .' x '. check_plain($product->model); } $row[] = ''; $row[] = strtr($package->shipping_type, '_', ' '); $row[] = check_plain($package->pkg_type); $row[] = isset($package->sid) ? l($package->sid, 'admin/store/orders/'. $order_id .'/shipments/'. $package->sid .'/view') : ''; $row[] = isset($package->tracking_number) ? check_plain($package->tracking_number) : ''; if ($package->sid && $package->label_image) { $method = db_result(db_query("SELECT shipping_method FROM {uc_shipments} WHERE sid = %d", $package->sid)); } $row[] = isset($package->label_image) ? l(theme('imagecache', 'uc_thumbnail', $package->label_image, t('Shipping Label'), t('Shipping Label')), 'admin/store/orders/'. $order_id .'/shipments/labels/'. $method .'/'. $package->label_image, array(), null, null, false, true) : ''; $row[] = l(t('edit'), 'admin/store/orders/'. $order_id .'/packages/'. $package->package_id .'/edit'); $row[] = l(t('delete'), 'admin/store/orders/'. $order_id .'/packages/'. $package->package_id .'/delete'); if ($package->sid) { $row[] = l(t('cancel shipment'), 'admin/store/orders/'. $order_id .'/packages/'. $package->package_id .'/cancel'); } else { $row[] = ''; } $rows[] = $row; } if (empty($rows)) { $rows[][] = array('data' => t("This order's products have not been organized into packages."), 'colspan' => 10); } $output = theme('table', $header, $rows); $result = db_query("SELECT op.order_product_id, CAST(SUM(op.qty) / COUNT(pp.qty) AS UNSIGNED) AS total, SUM(pp.qty) AS packaged FROM {uc_order_products} AS op LEFT JOIN {uc_packaged_products} AS pp ON op.order_product_id = pp.order_product_id WHERE op.order_id = %d AND op.data LIKE '%%%s%%' GROUP BY op.order_product_id HAVING SUM(pp.qty) IS NULL OR CAST(SUM(op.qty) / COUNT(pp.qty) AS UNSIGNED) > SUM(pp.qty)", $order_id, 's:9:"shippable";s:1:"1";'); if (db_num_rows($result)) { $output .= l(t('Create packages.'), 'admin/store/orders/'. $order_id .'/packages/new'); } return $output; } /** * Put ordered products into a package. * * @ingroup forms * @see theme_uc_shipping_new_package_fieldset * @see uc_shipping_new_package_validate * @see uc_shipping_new_package_submit */ function uc_shipping_new_package($order_id) { $breadcrumb = drupal_get_breadcrumb(); $breadcrumb[] = l(t('Packages'), 'admin/store/orders/'. $order_id .'/packages'); drupal_set_breadcrumb($breadcrumb); $form = array('#tree' => true); $form['instructions'] = array('#value' => t('Organize products into packages. Package numbers in multiple shipping types are of the first shipping type they appear in. All packages are given a unique ID when they are saved. Choose the default package "Sep." to automatically create a package for each of the selected quantity of products in that row.')); $order = uc_order_load($order_id); $shipping_types_products = array(); foreach ($order->products as $product) { if ($product->data['shippable']) { $product->shipping_type = uc_product_get_shipping_type($product); $shipping_types_products[$product->shipping_type][] = $product; } } $shipping_type_weights = variable_get('uc_quote_type_weight', array()); $packaged_products = array(); $result = db_query("SELECT op.order_product_id, SUM(pp.qty) AS quantity FROM {uc_packaged_products} AS pp LEFT JOIN {uc_packages} AS p ON pp.package_id = p.package_id LEFT JOIN {uc_order_products} AS op ON op.order_product_id = pp.order_product_id WHERE p.order_id = %d GROUP BY op.order_product_id", $order_id); while ($boxed_product = db_fetch_object($result)) { $packaged_products[$boxed_product->order_product_id] = $boxed_product->quantity; } $form['shipping_types'] = array(); foreach ($shipping_types_products as $shipping_type => $products) { $form['shipping_types'][$shipping_type] = array('#type' => 'fieldset', '#title' => ucwords(str_replace('_', ' ', $shipping_type)), '#collapsible' => true, '#collapsed' => false, '#weight' => $shipping_type_weights[$shipping_type], ); foreach ($products as $product) { $unboxed_qty = $product->qty - $packaged_products[$product->order_product_id]; if ($unboxed_qty > 0) { $product_row = array(); $product_row['checked'] = array('#type' => 'checkbox', '#default_value' => 0); $product_row['model'] = array('#type' => 'markup', '#value' => check_plain($product->model)); $product_row['name'] = array('#type' => 'markup', '#value' => filter_xss_admin($product->title)); $product_row['qty'] = array('#type' => 'select', '#options' => drupal_map_assoc(uc_range(1, $unboxed_qty)), '#default_value' => $unboxed_qty, ); $options = drupal_map_assoc(uc_range(0, count($order->products))); $options[0] = t('Sep.'); $product_row['package'] = array('#type' => 'select', '#options' => $options, '#default_value' => 0, ); $form['shipping_types'][$shipping_type][$product->order_product_id] = $product_row; } } $form['shipping_types'][$shipping_type]['#theme'] = 'uc_shipping_new_package_fieldset'; } $form['order_id'] = array('#type' => 'hidden', '#value' => $order_id); $form['create'] = array('#type' => 'submit', '#value' => t('Make packages')); $form['combine'] = array('#type' => 'submit', '#value' => t('Create one package')); $form['cancel'] = array('#type' => 'submit', '#value' => t('Cancel')); return $form; } /** * Format and display the products in a shipping type fieldset. * * @ingroup themeable */ function theme_uc_shipping_new_package_fieldset($fieldset) { $output = ''; $header = array(theme('table_select_header_cell'), t('Model'), t('Title'), t('Qty'), t('Package')); $rows = array(); foreach (element_children($fieldset) as $op_id) { $row = array(); $row[] = drupal_render($fieldset[$op_id]['checked']); $row[] = drupal_render($fieldset[$op_id]['model']); $row[] = drupal_render($fieldset[$op_id]['name']); $row[] = drupal_render($fieldset[$op_id]['qty']); $row[] = drupal_render($fieldset[$op_id]['package']); $rows[] = $row; } $output .= theme('table', $header, $rows); $output .= drupal_render($fieldset); return $output; } /** * Validation handler for uc_shipping_new_package(). * * Do not allow empty packages. */ function uc_shipping_new_package_validate($form_id, $form_values) { if ($form_values['op'] != t('Cancel')) { $empty = true; foreach ($form_values['shipping_types'] as $shipping_type => $products) { foreach ($products as $product) { if ($product['checked'] != 0) { $empty = false; break 2; } } } if ($empty) { form_set_error($shipping_type, t('Packages should have at least one product in them.')); } } } /** * Submit handler for uc_shipping_new_package(). */ function uc_shipping_new_package_submit($form_id, $form_values) { if ($form_values['op'] != t('Cancel')) { $packages = array(0 => array()); foreach ($form_values['shipping_types'] as $shipping_type => $products) { foreach ($products as $id => $product) { if ($product['checked']) { if ($form_values['op'] == t('Create one package')) { $product['package'] = 1; } if ($product['package'] != 0) { $packages[$product['package']]['products'][$id] = (object)$product; if (!isset($packages[$product['package']]['shipping_type'])) { $packages[$product['package']]['shipping_type'] = $shipping_type; } } else { $packages[0][$shipping_type][$id] = (object)$product; } } } if (is_array($packages[0][$shipping_type])) { foreach ($packages[0][$shipping_type] as $id => $product) { $qty = $product->qty; $product->qty = 1; for ($i = 0; $i < $qty; $i++) { $packages[] = array('products' => array($id => $product), 'shipping_type' => $shipping_type); } } } unset($packages[0][$shipping_type]); } if (empty($packages[0])) { unset($packages[0]); } foreach ($packages as $package) { $package['order_id'] = $form_values['order_id']; uc_shipping_package_save($package); } } return 'admin/store/orders/'. $form_values['order_id'] .'/packages'; } /** * Display the details of a package. */ function uc_shipping_package_view($package_id) { $package = uc_shipping_package_load($package_id); $shipment = uc_shipping_shipment_load($package->sid); $output = ''; $rows = array(); $output .= '
'. t('Package %id:', array('%id' => $package_id)) .'
'; $rows[] = array(t('Contents:'), filter_xss_admin($package->description)); if ($shipment) { $methods = module_invoke_all('shipping_method'); $method = $methods[$shipment->shipping_method]; $pkg_type = $method['ship']['pkg_types'][$package->pkg_type]; } $rows[] = array(t('Package type:'), strlen($pkg_type) ? $pkg_type : check_plain($package->pkg_type)); if ($package->length && $package->width && $package->height) { $rows[] = array(t('Dimensions:'), t('!l x !w x !h', array('!l' => uc_length_format($package->length), '!w' => uc_length_format($package->width), '!h' => uc_length_format($package->height)))); } $rows[] = array(t('Insured value:'), uc_currency_format($package->value)); if ($package->tracking_number) { $rows[] = array(t('Tracking number:'), check_plain($package->tracking_number)); } if ($shipment && $package->label_image && file_exists($package->label_image)) { $rows[] = array(t('Label:'), l(t('Click to view.'), 'admin/store/orders/'. $package->order_id .'/shipments/labels/'. $shipment->shipping_method .'/'. $package->label_image)); } else { $rows[] = array(t('Label:'), t('n/a')); } $output .= theme('table', array(), $rows, array('style' => 'width:auto;')); $output .= '
'; return $output; } /** * Rearrange the products in or out of a package. * * @ingroup forms * @see theme_uc_shipping_edit_package_fieldset * @see uc_shipping_package_edit_submit */ function uc_shipping_package_edit($order_id, $package_id) { $package = uc_shipping_package_load($package_id); $products = array(); $order = uc_order_load($order_id); $shipping_types_products = array(); foreach ($order->products as $product) { if ($product->data['shippable']) { $product->shipping_type = uc_product_get_shipping_type($product); $shipping_types_products[$product->shipping_type][$product->order_product_id] = $product; $products[$product->order_product_id] = $product; } } $result = db_query("SELECT order_product_id, SUM(qty) AS quantity FROM {uc_packaged_products} AS pp LEFT JOIN {uc_packages} AS p ON pp.package_id = p.package_id WHERE p.order_id = %d GROUP BY order_product_id", $order_id); while ($packaged_product = db_fetch_object($result)) { //Make already packaged products unavailable, except those in this package. $products[$packaged_product->order_product_id]->qty = $products[$packaged_product->order_product_id]->qty - $packaged_product->quantity + $package->products[$packaged_product->order_product_id]->qty; } $form = array(); $form['#tree'] = true; $form['package_id'] = array('#type' => 'hidden', '#value' => $package_id); $form['products'] = array(); foreach ($products as $product) { if ($product->qty > 0) { $product_row = array(); $product_row['checked'] = array('#type' => 'checkbox', '#default_value' => isset($package->products[$product->order_product_id])); $product_row['model'] = array('#type' => 'markup', '#value' => check_plain($product->model)); $product_row['name'] = array('#type' => 'markup', '#value' => filter_xss_admin($product->title)); $product_row['qty'] = array('#type' => 'select', '#options' => drupal_map_assoc(uc_range(1, $product->qty)), '#default_value' => $package->products[$product->order_product_id]->qty, ); $form['products'][$product->order_product_id] = $product_row; } } $form['products']['#theme'] = 'uc_shipping_edit_package_fieldset'; $options = array(); foreach (array_keys($shipping_types_products) as $type) { $options[$type] = ucwords(str_replace('_', ' ', $type)); } $form['shipping_type'] = array('#type' => 'select', '#title' => t('Shipping type'), '#options' => $options, '#default_value' => $package->shipping_type, ); $form['submit'] = array('#type' => 'submit', '#value' => t('Submit')); $form['cancel'] = array('#type' => 'submit', '#value' => t('Cancel')); return $form; } /** * Display a formatted shipping type fieldset. * * @ingroup themeable */ function theme_uc_shipping_edit_package_fieldset($fieldset) { $output = ''; $header = array(theme('table_select_header_cell'), t('Model'), t('Title'), t('Qty')); $rows = array(); foreach (element_children($fieldset) as $op_id) { $row = array(); $row[] = drupal_render($fieldset[$op_id]['checked']); $row[] = drupal_render($fieldset[$op_id]['model']); $row[] = drupal_render($fieldset[$op_id]['name']); $row[] = drupal_render($fieldset[$op_id]['qty']); $rows[] = $row; } $output .= theme('table', $header, $rows); $output .= drupal_render($fieldset); return $output; } /** * Submit handler for uc_shipping_package_edit(). */ function uc_shipping_package_edit_submit($form_id, $form_values) { $package = uc_shipping_package_load($form_values['package_id']); if ($form_values['op'] != t('Cancel')) { foreach ($form_values['products'] as $id => $product) { if ($product['checked']) { $package->products[$id] = (object)$product; } else { unset($package->products[$id]); } } $package->shipping_type = $form_values['shipping_type']; uc_shipping_package_save($package); } return 'admin/store/orders/'. $package->order_id .'/packages'; } function uc_shipping_package_cancel_confirm($order_id, $package_id) { $form = array(); $form['order_id'] = array('#type' => 'value', '#value' => $order_id); $form['package_id'] = array('#type' => 'value', '#value' => $package_id); $output = confirm_form($form, t('Are you sure you want to cancel the shipment of this package?'), 'admin/store/orders/'. $order_id .'/packages', t('It will be available for reshipment.'), t('Cancel shipment'), t('Nevermind')); return $output; } function uc_shipping_package_cancel_confirm_submit($form_id, $form_values) { $package = uc_shipping_package_load($form_values['package_id']); $shipment = uc_shipping_shipment_load($package->sid); $methods = module_invoke_all('shipping_method'); if (function_exists($methods[$shipment->shipping_method]['cancel'])) { $result = call_user_func($methods[$shipment->shipping_method]['cancel'], $shipment->tracking_number, array($package->tracking_number)); if ($result) { db_query("UPDATE {uc_packages} SET sid = NULL, label_image = NULL, tracking_number = NULL WHERE package_id = %d", $package->package_id); unset($shipment->packages[$package->package_id]); if (!count($shipment->packages)) { uc_shipping_shipment_delete($shipment->sid); } } } return 'admin/store/orders/'. $form_values['order_id'] .'/packages'; } /** * Decide to unpackage products. * * @ingroup forms * @see uc_shipping_package_delete_confirm_submit */ function uc_shipping_package_delete_confirm($order_id, $package_id) { $form = array(); $form['order_id'] = array('#type' => 'value', '#value' => $order_id); $form['package_id'] = array('#type' => 'value', '#value' => $package_id); $output = confirm_form($form, t('Are you sure you want to delete this package?'), 'admin/store/orders/'. $order_id .'/packages', t('The products it contains will be available for repackaging.'), t('Delete'), t('Cancel')); return $output; } /** * Submit handler for uc_shipping_package_delete_confirm(). */ function uc_shipping_package_delete_confirm_submit($form_id, $form_values) { uc_shipping_package_delete($form_values['package_id']); return 'admin/store/orders/'. $form_values['order_id'] .'/packages'; } /** * Display a list of shipments for an order. * * @param $order_id * The order's id. */ function uc_shipping_order_shipments($order_id) { $result = db_query("SELECT * FROM {uc_shipments} WHERE order_id = %d", $order_id); $header = array(t('Shipment ID'), t('Name'), t('Company'), t('Destination'), t('Ship date'), t('Estimated delivery'), t('Tracking number'),array('data' => t('Actions'), 'colspan' => 3)); $rows = array(); while ($shipment = db_fetch_object($result)) { $row = array(); $row[] = $shipment->sid; $row[] = check_plain($shipment->d_first_name) .' '. check_plain($shipment->d_last_name); $row[] = check_plain($shipment->d_company); $row[] = check_plain($shipment->d_city) .', '. uc_get_zone_code($shipment->d_zone) .' '. check_plain($shipment->d_postal_code); $row[] = format_date($shipment->ship_date, 'custom', variable_get('uc_date_format_default', 'm/d/Y')); $row[] = format_date($shipment->expected_delivery, 'custom', variable_get('uc_date_format_default', 'm/d/Y')); $row[] = is_null($shipment->tracking_number) ? t('n/a') : check_plain($shipment->tracking_number); $row[] = l(t('view'), 'admin/store/orders/'. $order_id .'/shipments/'. $shipment->sid .'/view'); $row[] = l(t('edit'), 'admin/store/orders/'. $order_id .'/shipments/'. $shipment->sid .'/edit'); $row[] = l(t('delete'), 'admin/store/orders/'. $order_id .'/shipments/'. $shipment->sid .'/delete'); $rows[] = $row; } if (empty($rows)) { $rows[] = array(array('data' => t('No shipments have been made for this order.'), 'colspan' => 10)); } $output = theme('table', $header, $rows); $packages = db_num_rows(db_query("SELECT * FROM {uc_packages} WHERE order_id = %d AND sid IS NULL", $order_id)); if ($packages) { $output .= l(t('Make a new shipment'), 'admin/store/orders/'. $order_id .'/shipments/new'); } else { $result = db_query("SELECT op.order_product_id, CAST(SUM(op.qty) / COUNT(pp.qty) AS UNSIGNED) AS total, SUM(pp.qty) AS packaged FROM {uc_order_products} AS op LEFT JOIN {uc_packaged_products} AS pp ON op.order_product_id = pp.order_product_id WHERE op.order_id = %d AND op.data LIKE '%%%s%%' GROUP BY op.order_product_id HAVING SUM(pp.qty) IS NULL OR CAST(SUM(op.qty) / COUNT(pp.qty) AS UNSIGNED) > SUM(pp.qty)", $order_id, 's:9:"shippable";s:1:"1";'); if (db_num_rows($result)) { $output .= l(t('Put products into packages to make shipments.'), 'admin/store/orders/'. $order_id .'/packages/new'); } } return $output; } /** * Set up a new shipment with the chosen packages. * * @ingroup forms * @see theme_uc_shipping_new_shipment * @see uc_shipping_new_shipment_submit */ function uc_shipping_new_shipment($order_id) { $breadcrumb = drupal_get_breadcrumb(); $breadcrumb[] = l(t('Shipments'), 'admin/store/orders/'. $order_id .'/shipments'); drupal_set_breadcrumb($breadcrumb); $form = array('#tree' => true); $form['order_id'] = array('#type' => 'hidden', '#value' => $order_id); $packages_by_type = array(); $result = db_query("SELECT * FROM {uc_packages} WHERE order_id = %d AND sid IS NULL", $order_id); while ($package = db_fetch_object($result)) { $products = array(); $weight = 0; $result2 = db_query("SELECT pp.order_product_id, pp.qty, pp.qty * op.weight AS weight, op.title, op.model FROM {uc_packaged_products} AS pp LEFT JOIN {uc_order_products} AS op ON op.order_product_id = pp.order_product_id WHERE pp.package_id = %d", $package->package_id); while ($product = db_fetch_object($result2)) { $weight += $product->weight; $products[$product->order_product_id] = $product; } $package->weight = $weight; $package->products = $products; $packages_by_type[$package->shipping_type][$package->package_id] = $package; } $option_methods = array(); $shipping_types = module_invoke_all('shipping_type'); $shipping_methods = module_invoke_all('shipping_method'); $shipping_methods_by_type = array(); foreach ($shipping_methods as $method) { if (isset($method['ship'])) { $shipping_methods_by_type[$method['ship']['type']][] = $method; } } foreach ($packages_by_type as $shipping_type => $packages) { $form[$shipping_type] = array('#type' => 'fieldset', '#title' => $shipping_types[$shipping_type]['title'], ); $form[$shipping_type]['packages'] = array(); foreach ($packages as $package) { $pkgs_exist = true; $package_row = array(); $package_row['checked'] = array('#type' => 'checkbox', '#default_value' => 0); $package_row['package_id'] = array('#value' => $package->package_id); $product_list = array(); foreach ($package->products as $product) { $product_list[] = $product->qty .' x '. check_plain($product->model); } $package_row['products'] = array('#value' => ''); $package_row['weight'] = array('#value' => $package->weight); $form[$shipping_type]['packages'][$package->package_id] = $package_row; } if (is_array($shipping_methods_by_type[$shipping_type])) { foreach ($shipping_methods_by_type[$shipping_type] as $method) { $option_methods += array($method['id'] => $method['title']); } } } if ($pkgs_exist) { $option_methods = array('all' => t('Ship Manually')) + $option_methods; $form['method'] = array('#type' => 'select', '#title' => t('Shipping method'), '#options' => $option_methods, '#default_value' => 'all', ); $form['ship'] = array('#type' => 'submit', '#value' => t('Ship packages'), ); } //drupal_set_message('
'. print_r($form, true) .'
'); return $form; } /** * Format and display the new shipment form. * * @ingroup themeable */ function theme_uc_shipping_new_shipment($form) { $output = ''; $header = array(theme('table_select_header_cell'), t('Package'), t('Products'), t('Weight')); foreach (element_children($form) as $shipping_type) { $rows = array(); foreach (element_children($form[$shipping_type]['packages']) as $package_id) { $row = array(); $row[] = drupal_render($form[$shipping_type]['packages'][$package_id]['checked']); $row[] = drupal_render($form[$shipping_type]['packages'][$package_id]['package_id']); $row[] = drupal_render($form[$shipping_type]['packages'][$package_id]['products']); $row[] = drupal_render($form[$shipping_type]['packages'][$package_id]['weight']); $rows[] = $row; } if (count($rows)) { $form[$shipping_type]['packages']['table'] = array('#value' => theme('table', $header, $rows)); } //$output .= drupal_render($form[$shipping_type]['methods']); } $output .= drupal_render($form); return $output; } /** * Submit handler for uc_shipping_new_shipment(). * * Send package information to the chosen method. * * @see uc_shipping_make_shipment */ function uc_shipping_new_shipment_submit($form_id, $form_values) { $packages = array(); foreach ($form_values as $shipping_type) { if (is_array($shipping_type['packages'])) { foreach ($shipping_type['packages'] as $id => $input) { if ($input['checked']) { $packages[] = $id; } } } } return 'admin/store/orders/'. $form_values['order_id'] .'/ship/'. $form_values['method'] .'/'. implode('/', $packages); } /** * Default method to send packages on a shipment. */ function uc_shipping_make_shipment() { $args = func_get_args(); //print_r($args, true)); if (count($args) > 2) { $order_id = array_shift($args); $method_id = array_shift($args); $package_ids = $args; $methods = module_invoke_all('shipping_method'); $method = $methods[$method_id]; if (isset($method)) { return drupal_get_form($method['ship']['callback'], $order_id, $package_ids); } else { $shipment = new stdClass(); $shipment->order_id = $order_id; $shipment->packages = array(); foreach ($package_ids as $id) { $package = uc_shipping_package_load($id); $shipment->packages[$id] = $package; } return drupal_get_form('uc_shipping_shipment_edit', $order_id, $shipment); } } else { drupal_set_message(t('There is no sense in making a shipment with no packages on it, right?')); drupal_goto('admin/store/orders/'. $args[0]); } } /** * Display shipment details. */ function uc_shipping_shipment_view($order_id, $sid) { $breadcrumb = drupal_get_breadcrumb(); $breadcrumb[] = l(t('Shipments'), 'admin/store/orders/'. $order_id .'/shipments'); drupal_set_breadcrumb($breadcrumb); $shipment = uc_shipping_shipment_load($sid); $output = ''; $origin = uc_order_address($shipment, 'o'); $destination = uc_order_address($shipment, 'd'); $output .= '
'. t('Pickup Address:') .'
'. $origin .'
'; $output .= '
'. t('Delivery Address:') .'
'. $destination .'
'; $output .= '
'. t('Schedule:') .'
'; $rows = array(); $rows[] = array(t('Ship date:'), format_date($shipment->ship_date, 'custom', 'D, '. variable_get('uc_date_format_default', 'm/d/Y'))); $rows[] = array(t('Expected delivery:'), format_date($shipment->expected_delivery, 'custom', 'D, '. variable_get('uc_date_format_default', 'm/d/Y'))); $output .= theme('table', array(), $rows, array('style' => 'width:auto')); $output .= '
'; $output .= '
'. t('Shipment Details:') .'
'; $rows = array(); $rows[] = array(t('Carrier:'), check_plain($shipment->carrier)); if ($shipment->transaction_id) { $rows[] = array(t('Transaction ID:'), check_plain($shipment->transaction_id)); } if ($shipment->tracking_number) { $rows[] = array(t('Tracking Number:'), check_plain($shipment->tracking_number)); } $methods = module_invoke_all('shipping_method'); $method = $methods[$shipment->shipping_method]; if (isset($method['quote']['accessorials'][$shipment->accessorials])) { $rows[] = array(t('Services:'), $method['quote']['accessorials'][$shipment->accessorials]); } else { $rows[] = array(t('Services:'), $shipment->accessorials); } $rows[] = array(t('Cost:'), uc_currency_format($shipment->cost)); $output .= theme('table', array(), $rows, array('style' => 'width:auto')); $output .= '
'; foreach ($shipment->packages as $package) { $output .= uc_shipping_package_view($package->package_id); } return $output; } /** * Create or edit a shipment. * * @ingroup forms * @see theme_uc_shipping_package_dimensions * @see theme_uc_shipping_address * @see uc_shipping_shipment_edit_submit */ function uc_shipping_shipment_edit($order_id, $shipment) { drupal_add_css(drupal_get_path('module', 'uc_shipping') .'/uc_shipping.css'); $order = uc_order_load($order_id); if (is_numeric($shipment)) { $shipment = uc_shipping_shipment_load($shipment); } $form = array(); $form['order_id'] = array('#type' => 'value', '#value' => $order_id); if (isset($shipment->sid)) { $form['sid'] = array('#type' => 'value', '#value' => $shipment->sid); $methods = module_invoke_all('shipping_method'); $method = $methods[$shipment->shipping_method]; } $addresses = array(); $form['packages'] = array( '#tree' => true, '#weight' => 1, ); if ($shipment->o_street1) { $o_address = new stdClass(); foreach ($shipment as $field => $value) { if (substr($field, 0, 2) == 'o_') { $o_address->{substr($field, 2)} = $value; } } $addresses[] = $o_address; } foreach ($shipment->packages as $id => $package) { foreach ($package->addresses as $address) { if (!in_array($address, $addresses)) { $addresses[] = $address; } } $declared_value = 0; $product_list = array(); foreach ($package->products as $product) { $product_list[] = $product->qty .' x '. check_plain($product->model); $declared_value += $product->qty * $product->price; } $form['packages'][$id] = array('#type' => 'fieldset', '#title' => t('Package @id', array('@id' => $id)), '#collapsible' => true, ); $form['packages'][$id]['products'] = array('#value' => theme('item_list', $product_list)); $form['packages'][$id]['pkg_type'] = array('#type' => 'textfield', '#title' => t('Package type'), '#default_value' => $package->pkg_type, '#description' => t('E.g.: Box, pallet, tube, treasure chest, cocoon, etc.'), ); if (isset($method) && is_array($method['ship']['pkg_types'])) { $form['packages'][$id]['pkg_type']['#type'] = 'select'; $form['packages'][$id]['pkg_type']['#options'] = $method['ship']['pkg_types']; $form['packages'][$id]['pkg_type']['#description'] = ''; } $form['packages'][$id]['dimensions'] = array('#type' => 'fieldset', '#title' => t('Dimensions'), '#description' => t('Physical dimensions of the packaged product.'), '#theme' => 'uc_shipping_package_dimensions', ); $form['packages'][$id]['dimensions']['units'] = array('#type' => 'select', '#title' => t('Units of measurement'), '#options' => array( 'in' => t('Inches'), 'ft' => t('Feet'), 'cm' => t('Centimeters'), 'mm' => t('Millimeters'), ), '#default_value' => $package->length_units ? $package->length_units : variable_get('uc_length_unit', 'in'), ); $form['packages'][$id]['dimensions']['length'] = array('#type' => 'textfield', '#title' => t('Length'), '#default_value' => $package->length, '#size' => 8, ); $form['packages'][$id]['dimensions']['width'] = array('#type' => 'textfield', '#title' => t('Width'), '#default_value' => $package->width, '#size' => 8, ); $form['packages'][$id]['dimensions']['height'] = array('#type' => 'textfield', '#title' => t('Height'), '#default_value' => $package->height, '#size' => 8, ); $form['packages'][$id]['declared_value'] = array('#type' => 'textfield', '#title' => t('Declared value'), '#field_prefix' => variable_get('uc_sign_after_amount', FALSE) ? '' : variable_get('uc_currency_sign', '$'), '#field_suffix' => variable_get('uc_sign_after_amount', FALSE) ? variable_get('uc_currency_sign', '$') : '', '#default_value' => isset($package->value) ? $package->value : $declared_value, ); $form['packages'][$id]['tracking_number'] = array('#type' => 'textfield', '#title' => t('Tracking number'), '#default_value' => $package->tracking_number, ); } $form = array_merge($form, uc_shipping_address_form($addresses, $order)); $form['shipment'] = array('#type' => 'fieldset', '#title' => t('Shipment data'), '#collapsible' => true, '#weight' => 0, ); $form['shipment']['shipping_method'] = array('#type' => 'hidden', '#value' => isset($shipment->shipping_method) ? $shipment->shipping_method : 'manual', ); $form['shipment']['carrier'] = array('#type' => 'textfield', '#title' => t('Carrier'), '#default_value' => $shipment->carrier, ); $form['shipment']['accessorials'] = array('#type' => 'textfield', '#title' => t('Shipment options'), '#default_value' => $shipment->accessorials, '#description' => t('Short notes about the shipment, e.g. residential, overnight, etc.'), ); $form['shipment']['transaction_id'] = array('#type' => 'textfield', '#title' => t('Transaction ID'), '#default_value' => $shipment->transaction_id, ); $form['shipment']['tracking_number'] = array('#type' => 'textfield', '#title' => t('Tracking number'), '#default_value' => $shipment->tracking_number, ); if (isset($shipment->ship_date)) { $ship_date = getdate($shipment->ship_date); } else { $ship_date = getdate(); } if (isset($shipment->expected_delivery)) { $exp_delivery = getdate($shipment->expected_delivery); } else { $exp_delivery = getdate(); } $form['shipment']['ship_date'] = array('#type' => 'date', '#title' => t('Ship date'), '#default_value' => array('year' => $ship_date['year'], 'month' => $ship_date['mon'], 'day' => $ship_date['mday']), ); $form['shipment']['expected_delivery'] = array('#type' => 'date', '#title' => t('Expected delivery'), '#default_value' => array('year' => $exp_delivery['year'], 'month' => $exp_delivery['mon'], 'day' => $exp_delivery['mday']), ); $form['shipment']['cost'] = array('#type' => 'textfield', '#title' => t('Shipping cost'), '#default_value' => $shipment->cost, '#field_prefix' => variable_get('uc_sign_after_amount', FALSE) ? '' : variable_get('uc_currency_sign', '$'), '#field_suffix' => variable_get('uc_sign_after_amount', FALSE) ? variable_get('uc_currency_sign', '$') : '', ); $form['submit'] = array('#type' => 'submit', '#value' => t('Save shipment'), '#weight' => 10); return $form; } /** * Display length, width, and height fields on one line. * * @ingroup themeable */ function theme_uc_shipping_package_dimensions($form) { $output = ''; $row = array(); foreach (element_children($form) as $dimension) { $row[] = drupal_render($form[$dimension]); } $output .= theme('table', array(), array($row)); return $output; } /** * Compact the address into a table. * * @ingroup themeable */ function theme_uc_shipping_address($address) { drupal_add_css(drupal_get_path('module', 'uc_cart') .'/uc_cart.css'); if ($address['#collapsed']) { $collapsed = ' collapsed'; } $output = ''; $req = '*'; foreach (element_children($address) as $field) { list($type, $name) = explode('_', $field, 2); if ($address !== NULL) { $title = $address[$field]['#title'] .':'; unset($address[$field]['#title']); if ($name == 'street1') { $title = uc_get_field_name('street') .':'; } elseif ($name == 'street2') { $title = ' '; } $output .= ''; } } $output .= '
'; if ($address[$field]['#required']) { $output .= $req; } $output .= $title .'' . drupal_render($address[$field]) .'
'; foreach (element_children($address) as $element) { $output .= drupal_render($address[$element]); } return $output; } /** * Submit handler for uc_shipping_shipment_edit(). */ function uc_shipping_shipment_edit_submit($form_id, $form_values) { if ($form_values['op'] != t('Cancel')) { $shipment = new stdClass(); $shipment->order_id = $form_values['order_id']; if (isset($form_values['sid'])) { $shipment->sid = $form_values['sid']; } $shipment->origin = new stdClass(); $shipment->destination = new stdClass(); foreach ($form_values as $key => $value) { if (substr($key, 0, 7) == 'pickup_') { $field = substr($key, 7); $shipment->origin->$field = $value; } else if (substr($key, 0, 9) == 'delivery_') { $field = substr($key, 9); $shipment->destination->$field = $value; } } $shipment->packages = array(); foreach ($form_values['packages'] as $id => $pkg_form) { $package = uc_shipping_package_load($id); $package->pkg_type = $pkg_form['pkg_type']; $package->value = $pkg_form['declared_value']; $package->length = $pkg_form['dimensions']['length']; $package->width = $pkg_form['dimensions']['width']; $package->height = $pkg_form['dimensions']['height']; $package->length_units = $pkg_form['dimensions']['units']; $package->tracking_number = $pkg_form['tracking_number']; $package->qty = 1; $shipment->packages[$id] = $package; } $shipment->shipping_method = $form_values['shipping_method']; $shipment->accessorials = $form_values['accessorials']; $shipment->carrier = $form_values['carrier']; $shipment->transaction_id = $form_values['transaction_id']; $shipment->tracking_number = $form_values['tracking_number']; $shipment->ship_date = gmmktime(12, 0, 0, $form_values['ship_date']['month'], $form_values['ship_date']['day'], $form_values['ship_date']['year']); $shipment->expected_delivery = gmmktime(12, 0, 0, $form_values['expected_delivery']['month'], $form_values['expected_delivery']['day'], $form_values['expected_delivery']['year']); $shipment->cost = $form_values['cost']; uc_shipping_shipment_save($shipment); } return 'admin/store/orders/'. $form_values['order_id'] .'/shipments'; } /** * Decide to release packages to be put on another shipment. * * @ingroup forms * @see uc_shipping_shipment_delete_confirm_submit */ function uc_shipping_shipment_delete_confirm($order_id, $sid) { $form = array(); $form['order_id'] = array('#type' => 'value', '#value' => $order_id); $form['sid'] = array('#type' => 'value', '#value' => $sid); $output = confirm_form($form, t('Are you sure you want to delete this shipment?'), 'admin/store/orders/'. $order_id .'/shipments', t('The shipment will be canceled and the packages it contains will be available for reshipment.'), t('Delete'), t('Cancel')); return $output; } /** * Submit handler for uc_shipping_shipment_delete_confirm(). */ function uc_shipping_shipment_delete_confirm_submit($form_id, $form_values) { $shipment = uc_shipping_shipment_load($form_values['sid']); $methods = module_invoke_all('shipping_method'); if ($shipment->tracking_number && function_exists($methods[$shipment->shipping_method]['cancel'])) { $result = call_user_func($methods[$shipment->shipping_method]['cancel'], $shipment->tracking_number); if ($result) { uc_shipping_shipment_delete($form_values['sid']); } else { drupal_set_message(t('The shipment %tracking could not be canceled with %carrier. To delete it anyway, remove the tracking number and try again.', array('%tracking' => $shipment->tracking_number, '%carrier' => $shipment->carrier))); } } else { uc_shipping_shipment_delete($form_values['sid']); } return 'admin/store/orders/'. $form_values['order_id'] .'/shipments'; } /****************************************************************************** * Module and helper functions * ******************************************************************************/ /** * Load a package and its products. */ function uc_shipping_package_load($package_id) { static $packages = array(); if (!isset($packages[$package_id])) { $products = array(); $descripion = ''; $weight = 0; $units = variable_get('uc_weight_unit', 'lb'); $addresses = array(); $result = db_query("SELECT op.order_product_id, pp.qty, pp.qty * op.weight AS weight, p.weight_units, op.nid, op.title, op.model, op.price FROM {uc_packaged_products} AS pp LEFT JOIN {uc_order_products} AS op ON op.order_product_id = pp.order_product_id LEFT JOIN {uc_products} AS p ON op.nid = p.nid WHERE pp.package_id = %d", $package_id); while ($product = db_fetch_object($result)) { $address = uc_quote_get_default_shipping_address($product->nid); // TODO: Lodge complaint that array_unique() compares as strings. if (!in_array($address, $addresses)) { $addresses[] = $address; } $description .= ', '. $product->qty .' x '. $product->model; // Normalize all weights to default units. $weight += $product->weight * uc_weight_conversion($product->weight_units, $units); $products[$product->order_product_id] = $product; } $result = db_query("SELECT * FROM {uc_packages} WHERE package_id = %d", $package_id); $packages[$package_id] = db_fetch_object($result); $packages[$package_id]->addresses = $addresses; $packages[$package_id]->description = substr($description, 2); $packages[$package_id]->weight = $weight; $packages[$package_id]->weight_units = $units; $packages[$package_id]->products = $products; } return $packages[$package_id]; } /** * Save a package. */ function uc_shipping_package_save($package) { $package = (object)$package; if (!isset($package->package_id)) { $package->package_id = db_next_id('{uc_packages}_package_id'); db_query("INSERT INTO {uc_packages} (package_id, order_id) VALUES (%d, %d)", $package->package_id, $package->order_id); } if (count($package->products)) { $types = array(); $values = array(); foreach ($package->products as $id => $product) { $types[] = '(%d, %d, %d)'; $values[] = $package->package_id; $values[] = $id; $values[] = $product->qty; $result = db_query("SELECT data FROM {uc_order_products} WHERE order_product_id = %d", $id); if ($order_product = db_fetch_object($result)) { $order_product->data = unserialize($order_product->data); $order_product->data['package_id'] = intval($package->package_id); db_query("UPDATE {uc_order_products} SET data = '%s' WHERE order_product_id = %d", serialize($order_product->data), $id); } } db_query("DELETE FROM {uc_packaged_products} WHERE package_id = %d", $package->package_id); db_query("INSERT INTO {uc_packaged_products} (package_id, order_product_id, qty) VALUES ". implode(',', $types), $values); } $types = array("shipping_type = '%s'"); $values = array($package->shipping_type); if (isset($package->pkg_type)) { $types[] = "pkg_type = '%s'"; $values[] = $package->pkg_type; } if (isset($package->length) && isset($package->width) && isset($package->height) && isset($package->length_units)) { array_push($types, 'length = %f', 'width = %f', 'height = %f', "length_units = '%s'"); array_push($values, $package->length, $package->width, $package->height, $package->length_units); } if (isset($package->value)) { $types[] = 'value = %f'; $values[] = $package->value; } if (isset($package->sid)) { $types[] = 'sid = %d'; $values[] = $package->sid; } if (isset($package->tracking_number)) { $types[] = "tracking_number = '%s'"; $values[] = $package->tracking_number; } if (isset($package->label_image)) { $types[] = "label_image = '%s'"; $values[] = $package->label_image; } $values[] = $package->package_id; if (count($types)) { // Let it be known that I think it's ridiculous that Drupal doesn't put NULL into its database. --JLM db_query("UPDATE {uc_packages} SET ". implode(',', $types) ." WHERE package_id = %d", $values); } } /** * Delete a package. */ function uc_shipping_package_delete($package_id) { db_query("DELETE FROM {uc_packages} WHERE package_id = %d", $package_id); db_query("DELETE FROM {uc_packaged_products} WHERE package_id = %d", $package_id); drupal_set_message(t('Package @id has been deleted.', array('@id' => $package_id))); } /** * Load a shipment and it's packages. */ function uc_shipping_shipment_load($shipment_id) { $shipment = db_fetch_object(db_query("SELECT * FROM {uc_shipments} WHERE sid = %d", $shipment_id)); if ($shipment) { $result = db_query("SELECT package_id FROM {uc_packages} WHERE sid = %d", $shipment_id); $packages = array(); while ($package = db_fetch_object($result)) { $packages[$package->package_id] = uc_shipping_package_load($package->package_id); } $shipment->packages = $packages; $extra = module_invoke_all('shipment', 'load', $shipment); if (is_array($extra)) { foreach ($extra as $key => $value) { $shipment->$key = $value; } } } return $shipment; } /** * Save a shipment. */ function uc_shipping_shipment_save($shipment) { if (!$shipment->sid) { $shipment->sid = db_next_id('{uc_shipments}_sid'); db_query("INSERT INTO {uc_shipments} (sid, order_id) VALUES (%d, %d)", $shipment->sid, $shipment->order_id); $shipment->is_new = TRUE; } else { $shipment->is_new = FALSE; } if (is_array($shipment->packages)) { foreach ($shipment->packages as $package) { $package->sid = $shipment->sid; // Since the products haven't changed, we take them out of the object so that they are not deleted and re-inserted. $products = $package->products; unset($package->products); uc_shipping_package_save($package); // But they're still necessary for hook_shipment(), so they're added back in. $package->products = $products; } } if (isset($shipment->origin)) { foreach ($shipment->origin as $field => $value) { $field = 'o_'. $field; $shipment->$field = $value; } } if (isset($shipment->destination)) { foreach ($shipment->destination as $field => $value) { $field = 'd_'. $field; $shipment->$field = $value; } } db_query("UPDATE {uc_shipments} SET order_id = %d, o_first_name = '%s', o_last_name = '%s', o_company = '%s', o_street1 = '%s', o_street2 = '%s', o_city = '%s', o_zone = %d, o_postal_code = '%s', o_country = %d, d_first_name = '%s', d_last_name = '%s', d_company = '%s', d_street1 = '%s', d_street2 = '%s', d_city = '%s', d_zone = %d, d_postal_code = '%s', d_country = %d, shipping_method = '%s', accessorials = '%s', carrier = '%s', transaction_id = '%s', tracking_number = '%s', ship_date = %d, expected_delivery = %d, cost = %f WHERE sid = %d", $shipment->order_id, $shipment->o_first_name, $shipment->o_last_name, $shipment->o_company, $shipment->o_street1, $shipment->o_street2, $shipment->o_city, $shipment->o_zone, $shipment->o_postal_code, $shipment->o_country, $shipment->d_first_name, $shipment->d_last_name, $shipment->d_company, $shipment->d_street1, $shipment->d_street2, $shipment->d_city, $shipment->d_zone, $shipment->d_postal_code, $shipment->d_country, $shipment->shipping_method, $shipment->accessorials, $shipment->carrier, $shipment->transaction_id, $shipment->tracking_number, $shipment->ship_date, $shipment->expected_delivery, $shipment->cost, $shipment->sid ); module_invoke_all('shipment', 'save', $shipment); } /** * Delete a shipment. */ function uc_shipping_shipment_delete($shipment_id) { $shipment = uc_shipping_shipment_load($shipment_id); foreach ($shipment->packages as $package) { if (file_exists($package->label_image)) { file_delete($package->label_image); } } db_query("UPDATE {uc_packages} SET sid = NULL, tracking_number = NULL, label_image = NULL WHERE sid = %d", $shipment_id); db_query("DELETE FROM {uc_shipments} WHERE sid = %d", $shipment_id); module_invoke_all('shipment', 'delete', $shipment); } function uc_shipping_order_pane_packages($op, $arg1) { switch ($op) { case 'view': case 'customer': $tracking = array(); $result = db_query("SELECT sid FROM {uc_shipments} WHERE order_id = %d", $arg1->order_id); while ($shipment = db_fetch_object($result)) { $shipment = uc_shipping_shipment_load($shipment->sid); if ($shipment->tracking_number) { $tracking[$shipment->carrier]['children'][] = check_plain($shipment->tracking_number); } else { foreach ($shipment->packages as $package) { if ($package->tracking_number) { $tracking[$shipment->carrier]['children'][] = check_plain($package->tracking_number); } } } } $output = ''; foreach ($tracking as $carrier => $item) { $output .= ''. $carrier .':'. theme('item_list', $item); } return $output; break; } } /** * Choose an address to fill out a form. */ function uc_shipping_select_address($addresses, $onchange = '', $title = NULL, $icon_suffix = FALSE) { if (!is_array($addresses) || count($addresses) == 0) { $addresses = array(); } $store_address = variable_get('uc_quote_store_default_address', new stdClass()); if (!in_array($store_address, $addresses)) { $addresses[] = $store_address; } $blank = array('first_name' => '', 'last_name' => '', 'phone' => '', 'company' => '', 'street1' => '', 'street2' => '', 'city' => '', 'postal_code' => '', 'country' => 0, 'zone' => 0, ); $options = array(drupal_to_js($blank) => t('')); foreach ($addresses as $address) { $options[drupal_to_js($address)] = $address->company .' '. $address->street1 .' '. $address->city; } $select = array( '#type' => 'select', '#title' => is_null($title) ? t('Address book') : $title, '#options' => $options, '#default_value' => drupal_to_js($addresses[0]), '#attributes' => array('onchange' => $onchange), '#suffix' => $icon_suffix ? uc_store_get_icon('file:address_book', FALSE, 'address-book-icon') : NULL, ); return $select; } function uc_shipping_address_form($addresses, $order) { uc_add_js(drupal_get_path('module', 'uc_shipping') .'/uc_shipping.js'); $form = array(); $form['origin'] = array('#type' => 'fieldset', '#title' => t('Origin address'), '#collapsible' => true, '#collapsed' => false, '#weight' => -2, '#theme' => 'uc_shipping_address', ); $address = reset($addresses); $form['origin']['pickup_address_select'] = uc_shipping_select_address($addresses, 'apply_address(\'pickup\', this.value);', t('Saved Addresses'), TRUE); $form['origin']['pickup_address_select']['#weight'] = -2; $form['origin']['pickup_email'] = uc_textfield(uc_get_field_name('email'), variable_get('uc_store_email', null), FALSE); $form['origin']['pickup_email']['#weight'] = -1; $form['origin']['pickup_first_name'] = uc_textfield(uc_get_field_name('first_name'), $address->first_name, FALSE); $form['origin']['pickup_last_name'] = uc_textfield(uc_get_field_name('last_name'), $address->last_name, FALSE); $form['origin']['pickup_phone'] = uc_textfield(uc_get_field_name('phone'), variable_get('uc_store_phone', null), FALSE, NULL, 32, 16); $form['origin']['pickup_company'] = uc_textfield(uc_get_field_name('company'), $address->company, FALSE); $form['origin']['pickup_street1'] = uc_textfield(uc_get_field_name('street1'), $address->street1, FALSE, NULL, 64); $form['origin']['pickup_street2'] = uc_textfield(uc_get_field_name('street2'), $address->street2, FALSE, NULL, 64); $form['origin']['pickup_city'] = uc_textfield(uc_get_field_name('city'), $address->city, FALSE); $form['origin']['pickup_country'] = uc_country_select(uc_get_field_name('country'), $address->country); if (isset($_POST['pickup_country'])) { $country = $_POST['pickup_country']; } else { $country = $address->country; } $form['origin']['pickup_zone'] = uc_zone_select(uc_get_field_name('zone'), $address->zone, null, $country); $form['origin']['pickup_postal_code'] = uc_textfield(uc_get_field_name('postal_code'), $address->postal_code, FALSE, NULL, 10, 10); $order_form = uc_order_pane_ship_to('edit-form', $order); $form['destination'] = $order_form['ship_to']; $form['destination']['delivery_email'] = uc_textfield(uc_get_field_name('email'), $order->primary_email, FALSE); $form['destination']['delivery_email']['#weight'] = -1; $form['destination']['#title'] = t('Destination Address'); $form['destination']['#collapsible'] = true; $form['destination']['#weight'] = -1; $form['destination']['#theme'] = 'uc_shipping_address'; return $form; }