| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463 | <?php/** * Implements hook_field_widget_info(). */function geocoder_field_widget_info() {  return array(    'geocoder' => array(      'label' => t('Geocode from another field'),      'field types' => array('geofield', 'geolocation_latlng', 'location'),      'behaviors' => array(        'multiple values' => FIELD_BEHAVIOR_CUSTOM,        'default value' => FIELD_BEHAVIOR_NONE,      ),    ),  );}/** * Implements field_widget_settings_form(). */function geocoder_field_widget_settings_form($this_field, $instance) {  $settings = $instance['widget']['settings'];    $entity_fields = field_info_instances($instance['entity_type'], $instance['bundle']);  $all_fields = field_info_fields();  $supported_field_types = geocoder_supported_field_types();  $processors = geocoder_handler_info();  $handlers_by_type = array();  $field_types = array();  $valid_fields = array();  $available_handlers = array();    // Add in the title/name  //@@TODO Do this programatically by getting entity_info  switch ($instance['entity_type']) {    case 'node':      $all_fields['title'] = array(        'field_name' => 'title',        'type' => 'text',      );      $entity_fields['title']['label'] = t('Title');      break;    case 'taxonomy_term':      $all_fields['name'] = array(        'field_name' => 'name',        'type' => 'text',      );      $entity_fields['name']['label'] = t('Name');      break;  }    // Get a list of all valid fields that we both support and are part of this entity  foreach ($all_fields as $field) {    if (array_key_exists($field['field_name'], $entity_fields)) {      if (in_array($field['type'], array_keys($supported_field_types)) && ($field['field_name'] != $this_field['field_name'])) {        $valid_fields[$field['field_name']] = $entity_fields[$field['field_name']]['label'];        foreach ($supported_field_types[$field['type']] as $handler) {          $available_handlers[$handler] = $processors[$handler]['title'];          $handlers_by_type[$field['type']][] = $handler;          $field_types[$field['field_name']] = $field['type'];        }      }    }  }    $form['geocoder_field'] = array(    '#type' => 'select',    '#title' => t('Geocode from field'),    '#default_value' => isset($settings['geocoder_field']) ? $settings['geocoder_field']: '',    '#options' => $valid_fields,    '#description' => t('Select which field you would like to geocode from.'),    '#required' => TRUE,  );    $form['geocoder_handler'] = array(    '#type' => 'select',    '#title' => t('Geocoder'),    '#prefix' => '<div id="geocoder-handler-div">',    '#suffix' => '</div>',    '#default_value' => isset($settings['geocoder_handler']) ? $settings['geocoder_handler']: '',    '#options' => $available_handlers,    '#description' => t('Select which type of geocoding handler you would like to use'),    '#required' => TRUE,  );    $form['handler_settings'] = array(    '#tree' => TRUE,  );    // Add the handler settings forms  foreach ($processors as $handler_id => $handler) {    if (isset($handler['settings_callback']) || isset($handler['terms_of_service'])) {      $default_values = isset($settings['handler_settings'][$handler_id]) ? $settings['handler_settings'][$handler_id] : array();      $form['handler_settings'][$handler_id] = array();      $form['handler_settings'][$handler_id]['#type'] = 'fieldset';      $form['handler_settings'][$handler_id]['#attributes'] = array('class' => array('geocoder-handler-setting', 'geocoder-handler-setting-' . $handler_id));      $form['handler_settings'][$handler_id]['#title'] = $handler['title'] . ' Settings';      $form['handler_settings'][$handler_id]['#states'] = array(        'visible' => array(          ':input[id="edit-instance-widget-settings-geocoder-handler"]' => array('value' => $handler_id),        ),      );      if (isset($handler['terms_of_service'])) {        $form['handler_settings'][$handler_id]['tos'] = array(          '#type' => 'item',          '#markup' => t('This handler has terms of service. Click the following link to learn more.') . ' ' . l($handler['terms_of_service'], $handler['terms_of_service']),        );      }      if (isset($handler['settings_callback'])) {        $settings_callback = $handler['settings_callback'];        $form['handler_settings'][$handler_id] = array_merge($form['handler_settings'][$handler_id], $settings_callback($default_values));      }    }  }    $form['delta_handling'] = array(    '#type' => 'select',    '#title' => t('Multi-value input handling'),    '#description' => t('Should geometries from multiple inputs be: <ul><li>Matched with each input (e.g. One POINT for each address field)</li><li>Aggregated into a single MULTIPOINT geofield (e.g. One MULTIPOINT polygon from multiple address fields)</li><li>Broken up into multiple geometries (e.g. One MULTIPOINT to multiple POINTs.)</li></ul>'),    '#default_value' => isset($settings['delta_handling']) ? $settings['delta_handling']: 'default',    '#options' => array(      'default' => 'Match Multiples (default)',      'm_to_s' =>  'Multiple to Single',      's_to_m' =>  'Single to Multiple',      'c_to_s' =>  'Concatenate to Single',      'c_to_m' =>  'Concatenate to Multiple',    ),    '#required' => TRUE,  );    // Add javascript to sync allowed values. Note that we are not using AJAX because we do not have access to the raw form_state here  drupal_add_js(array('geocoder_widget_settings' => array('handlers' => $handlers_by_type, 'types' => $field_types)), 'setting');  drupal_add_js(drupal_get_path('module', 'geocoder') . '/geocoder.admin.js', 'file');    return $form;}/** * Implements hook_field_attach_presave(). * * Geocoding for the geocoder widget is done here to ensure that only validated * and fully processed fields values are accessed. */function geocoder_field_attach_presave($entity_type, $entity) {  // Loop over any geofield using our geocode widget  $entity_info = entity_get_info($entity_type);  $bundle_name = empty($entity_info['entity keys']['bundle']) ? $entity_type : $entity->{$entity_info['entity keys']['bundle']};  foreach (field_info_instances($entity_type, $bundle_name) as $field_instance) {    if ($field_instance['widget']['type'] === 'geocoder') {      if (($field_value = geocoder_widget_get_field_value($entity_type, $field_instance, $entity)) !== FALSE) {        $entity->{$field_instance['field_name']} = $field_value;      }      else {        $entity->{$field_instance['field_name']} = array();      }    }  }}/** * Get a field's value based on geocoded data. *  * @param $entity_type *   Type of entity * @para field_instance *   Field instance definition array * @param $entity *  Optionally, the entity. You must pass either the entity or $source_field_values * @param $source_field_values *  Array of deltas / source field values. You must pass either this or $entity. *  * @return *  Three possibilities could be returned by this function: *    - FALSE: do nothing. *    - An empty array: use it to unset the existing field value. *    - A populated array: assign a new field value. */function geocoder_widget_get_field_value($entity_type, $field_instance, $entity = NULL, $source_field_values = NULL) {  if (!$source_field_values && !$entity) {    trigger_error('geocoder_widget_get_field_value: You must pass either $source_field_values OR $entity', E_USER_ERROR);    return FALSE;  }  $entity_info = entity_get_info($entity_type);  // Required settings  if (isset($field_instance['widget']['settings']['geocoder_handler']) && isset($field_instance['widget']['settings']['geocoder_field'])) {    $handler = geocoder_get_handler($field_instance['widget']['settings']['geocoder_handler']);    $field_name = is_array($field_instance['widget']['settings']['geocoder_field']) ? reset($field_instance['widget']['settings']['geocoder_field']) : $field_instance['widget']['settings']['geocoder_field'];    $target_info = field_info_field($field_instance['field_name']);    $target_type = $target_info['type'];        // Determine the source type, if it's a entity-key, we mock it as a "text" field    if (in_array($field_name, $entity_info['entity keys']) && $entity) {      $field_info = array('type' => 'text', 'entity_key' => TRUE);    }    else {      $field_info = field_info_field($field_name);      $field_info['entity_key'] = FALSE;    }        // Get the source values    if (!$source_field_values) {      if ($field_info['entity_key'] && $entity) {        $source_field_values = array(array('value' => $entity->$field_name));      }      else if ($entity) {        $source_field_values = field_get_items($entity_type, $entity, $field_name, isset($entity->language) ? $entity->language : NULL);      }      else {        // We can't find the source values        return FALSE;      }    }        // Remove source values that are not valid.    if ($source_field_values) {      foreach ($source_field_values as $delta => $item) {        if (!is_numeric($delta)) {          unset($source_field_values[$delta]);        }      }    }    // If no valid source values were passed.    if (empty($source_field_values)) {      return FALSE;    }    // For entities being updated, determine if another geocode is necessary    if ($entity) {      if (!empty($entity->original)) {        //@@TODO: Deal with entity-properties (non-fields)        //@@TODO: This isn't working with file fields. Should use some kind of lookup / map        $field_original = field_get_items($entity_type, $entity->original, $field_name, isset($entity->original->language) ? $entity->original->language : NULL);        if (!empty($field_original)) {          $diff = geocoder_widget_array_recursive_diff($field_original, $source_field_values);          if (empty($diff)) {            return FALSE;          }        }      }    }        // Get the handler-specific-settings    if (isset($field_instance['widget']['settings']['handler_settings'][$handler['name']])) {      $handler_settings = $field_instance['widget']['settings']['handler_settings'][$handler['name']];    }        // Determine how we deal with deltas (multi-value fields)    if (empty($field_instance['widget']['settings']['delta_handling'])) {      $delta_handling = 'default';    }    else {      $delta_handling = $field_instance['widget']['settings']['delta_handling'];    }        // Check to see if we should be concatenating    if ($delta_handling == 'c_to_s' || $delta_handling == 'c_to_m') {      $source_field_values = geocoder_widget_get_field_concat($source_field_values);    }        if (is_array($source_field_values) && count($source_field_values)) {      $values = array();            // Geocode geometries      $geometries = array();      foreach ($source_field_values as $delta => $item) {        // Geocode any value from our source field.        try {          if (isset($handler_settings)) {            $geometry = call_user_func($handler['field_callback'], $field_info, $item, $handler_settings);          }          else {            $geometry = call_user_func($handler['field_callback'], $field_info, $item);          }          if ($geometry instanceof Geometry) {            $geometries[] = $geometry;          }          else {            // An error occured            return FALSE;          }        }        catch (Exception $e) {          watchdog_exception('geocoder', $e);          return FALSE;        }      }            if (empty($geometries)) {        // This field has no data, so set the field to an empty array in        // order to delete its saved data.        $values = array(NULL);        return array();      }      else {        // Resolve multiple-values - get back values from our delta-resolver        $values = geocoder_widget_resolve_deltas($geometries, $delta_handling, $target_type);                // Set the values - geofields do not support languages        return array(LANGUAGE_NONE => $values);      }    }  }}/** * Get field items and info *  * We always pass the full field-item array (with all columns) to the handler, but there is some preprocessing * that we need to do for the special case of entity-labels and multi-field contactenation * For these two special cases we "mock-up" a text-field and pass it back for geocoding */function geocoder_widget_get_field_concat($items) {  // Check if we should concatenate  $concat = '';  foreach ($items as $item) {    $concat .= trim($item['value']) . ', ';  }  $concat = trim($concat, ', ');  $items = array(array('value' => $concat));  return $items;}/** * Geocder Widget - Resolve Deltas *  * Given a list of geometries, and user configuration on how to handle deltas,  * we created a list of items to be inserted into the fields. */function geocoder_widget_resolve_deltas($geometries, $delta_handling = 'default', $field_type) {  $values = array();    // Default delta handling: just pass one delta to the next  if ($delta_handling == 'default') {    foreach ($geometries as $geometry) {      $values[] = geocoder_widget_values_from_geometry($geometry, $field_type);    }  }    // Single-to-multiple handling - if we can, explode out the component geometries  if ($delta_handling == 's_to_m' || $delta_handling == 'c_to_m') {    $type = $geometries[0]->geometryType();    if (in_array($type, array('MultiPoint', 'MultiLineString', 'MultiPolygon', 'GeometryCollection'))) {      $components = $geometries[0]->getComponents();      foreach ($components as $component) {        $values[] = geocoder_widget_values_from_geometry($component, $field_type);      }    }    else {      $values[] = geocoder_widget_values_from_geometry($geometries[0], $field_type);    }  }    // For multiple-to-single handling, run it though geometryReduce   if ($delta_handling == 'm_to_s' || $delta_handling == 'c_to_s') {    $reduced_geom = geoPHP::geometryReduce($geometries);    $values[] = geocoder_widget_values_from_geometry($reduced_geom, $field_type);  }    return $values;}/** * Geocder Widget - Field values from geometry *  * Given a geometry and the field type, return back a values array for that field. * The passed back array represents a single delta.  */function geocoder_widget_values_from_geometry($geometry, $field_type) {  if ($field_type == 'geofield') return geofield_get_values_from_geometry($geometry);  if ($field_type == 'geolocation_latlng') {    $centroid = $geometry->centroid();    $lat = $centroid->y();    $lng = $centroid->x();        return array(      'lat' => $lat,      'lng' => $lng,      'lat_sin' => sin(deg2rad($lat)),      'lat_cos' => cos(deg2rad($lat)),      'lng_rad' => deg2rad($lng),    );  }  if ($field_type == 'location') {    $centroid = $geometry->centroid();    return array(      'latitude' => $centroid->y(),      'longitude' => $centroid->x(),      'source' => 2,    );  }}/** * Geocoder Widget - Parse an address field */function geocoder_widget_parse_addressfield($field_item) {  $address = '';  if (!empty($field_item['premise']))                 $address .= $field_item['premise'] . ',';  if (!empty($field_item['thoroughfare']))            $address .= $field_item['thoroughfare'] . ',';  if (!empty($field_item['locality']))                $address .= $field_item['locality'] . ',';  if (!empty($field_item['administrative_area']))     $address .= $field_item['administrative_area'] . ',';  if (!empty($field_item['sub_administrative_area'])) $address .= $field_item['sub_administrative_area'] . ',';  if (!empty($field_item['country']))                 $address .= $field_item['country'] . ',';  if (!empty($field_item['postal_code']))             $address .= $field_item['postal_code'] . ',';      $address = rtrim($address, ', ');    return $address;}/** * Geocoder Widget - Parse a location field */function geocoder_widget_parse_locationfield($field_item) {  $address = '';  if (!empty($field_item['name']))        $address .= $field_item['name'] . ',';  if (!empty($field_item['street']))      $address .= $field_item['street'] . ',';  if (!empty($field_item['additional']))  $address .= $field_item['additional'] . ',';  if (!empty($field_item['city']))        $address .= $field_item['city'] . ',';  if (!empty($field_item['province']) && function_exists('location_province_name')) {    $province_fullname = location_province_name($field_item['country'], $field_item['province']);    $address .=  $province_fullname . ',';  }  if (!empty($field_item['country']))     $address .= $field_item['country'] . ',';  if (!empty($field_item['postal_code'])) $address .= $field_item['postal_code'] . ',';      $address = rtrim($address, ', ');  return $address;}function geocoder_widget_array_recursive_diff($aArray1, $aArray2) {  $aReturn = array();  if (empty($aArray1)) {    return $aReturn;  }  foreach ($aArray1 as $mKey => $mValue) {    if (array_key_exists($mKey, $aArray2)) {      if (is_array($mValue)) {        $aRecursiveDiff = geocoder_widget_array_recursive_diff($mValue, $aArray2[$mKey]);        if (count($aRecursiveDiff)) { $aReturn[$mKey] = $aRecursiveDiff; }      } else {        if ($mValue != $aArray2[$mKey]) {          $aReturn[$mKey] = $mValue;        }      }    } else {      $aReturn[$mKey] = $mValue;    }  }  return $aReturn;} 
 |