'fb_form_invite_page', 'access callback' => TRUE, 'type' => MENU_CALLBACK, ); $items['fb_form_friend_selector_autocomplete'] = array( 'page callback' => 'fb_form_friend_selector_autocomplete', 'access callback' => TRUE, 'type' => MENU_CALLBACK, ); return $items; } /** * Implementation of hook_form_alter(). */ function fb_form_form_alter(&$form, &$form_state, $form_id) { /* Drupal allows no clean way to set $form['#type'], so we hack... */ if (isset($form['#fb_form_type_hack']) && ($type = $form['#fb_form_type_hack'])) { $form['#type'] = $type; unset($form['#fb_form_type_hack']); } // Support for ahah (see ahah_forms.module) if (fb_is_fbml_canvas()) { $form['#after_build'][] = 'fb_ahah_bind_form'; } } // This ahah code was written for D5 and still needs updating. /** * Support for AHAH in forms. * * This is intended to be compatible with the syntax of ahah_forms.module. * However, much of what that module does requires jquery. In FBML we have to * jquery and are therefor limited in what we can support. For example, we * only support the id selectors, you can't specify selector in your * ahah_bindings. */ function fb_ahah_bind_form($form) { static $one_time_only; // Facebook javascript appears to work only when the user is logged in. Not // sure whether this is on purpose or not. So we only activate AHAH when // user is logged in. global $_fb; if ($_fb && $_fb->api_client->added && function_exists('ahah_forms_scan_form_children')) { $bindings = array(); ahah_forms_scan_form_children( $form, $form['#id'], $bindings ); //drupal_set_message( "After Scan: Wrapper Bindings = " . print_r( $bindings, TRUE ) ); if (count($bindings) > 0) { if (!$one_time_only) { //add in required javascript files $module_path = drupal_get_path('module', 'ahah_forms'); fb_add_js(drupal_get_path('module', 'fb_form') . '/fb_ahah_forms.js', 'module'); drupal_add_js(array( 'ahah' => array( 'basePaths' => array( 'base' => base_path(), 'module' => $module_path ), 'bindings' => array( $bindings ), ), ), 'setting', 'fbml' ); $one_time_only = TRUE; } else { drupal_add_js(array( 'ahah' => array( 'bindings' => array( $bindings ), ), ), 'setting', 'fbml' ); } } } return $form; } /** * Create a page to invite friends to add an app. * * This page will succeed only if: * - shown on a canvas page * - it is an FBML canvas page, not an iframe * - the current user has added the application (not sure about this) */ function fb_form_invite_page() { global $_fb, $_fb_app; if (!fb_is_fbml_canvas()) { drupal_set_message(t('Unable to display page. FBML required.'), 'error'); drupal_not_found(); exit(); } if ($_fb_app) { drupal_set_title(t('Invite friends to use %application', array('%application' => $_fb_app->label))); } $output = drupal_get_form('fb_form_multi_add_invite_form'); return $output; } /** * Create a form allowing the user to invite friends to add the app. * * Facebook provides a very specific way to build this form using FBML. This will only display properly on FBML canvas pages. * The FBML for this form requires the tag where the
tag would normally be. Also the form provides its own buttons. These two things make it difficult to use Drupal's Form API to build the form. Still, we use FAPI, because we want modules to be able to alter the form (i.e. to add descriptive text). Alteration that rely on #submit or even additional input fields will probably not work properly, however. */ function fb_form_multi_add_invite_form() { global $_fb, $_fb_app; // TODO: confirm that we're displaying an FBML canvas page. if ($fbu = fb_facebook_user($_fb)) { // Exclude friends who have already installed app. // http://wiki.developers.facebook.com/index.php/Fb:request-form $rs = $_fb->api_client->fql_query("SELECT uid FROM user WHERE has_added_app=1 and uid IN (SELECT uid2 FROM friend WHERE uid1 = $fbu)"); // FQL, no {curly_brackets}! $arFriends = ""; // Build an delimited list of users... if ($rs) { $arFriends .= $rs[0]["uid"]; for ( $i = 1; $i < count($rs); $i++ ) { if ( $arFriends != "" ) $arFriends .= ","; $arFriends .= $rs[$i]["uid"]; } } } // Use node body in invite message. $node = node_load($_fb_app->nid); $node = node_prepare($node); $content = $node->teaser; // Do we need to append &next=[someURL] to the url here? $content .= "apikey}\" label=\"" . t('Add !title application.', array('!title' => $_fb_app->label)) . "\" />"; // form type fb:request-form $form = array('#fb_form_type_hack' => 'fb_form_request', /* becomes #type during form_alter */ '#attributes' => array('type' => $_fb_app->label, 'content' => htmlentities($content), 'invite' => 'true', ), '#action' => 'http://apps.facebook.com/' . $_fb_app->canvas, ); $form['friends'] = array('#type' => 'fb_form_request_selector', '#title' => t('Select the friends to invite.'), '#attributes' => array('exclude_ids' => $arFriends), ); return $form; } /** * Helper function to produce a request or invite form. Note that this does * not produce a full form (i.e. never use * drupal_get_form('fb_form_request_form')). The caller is expected to fill * out the rest of the form before returning it for use with drupal_get_form. * * DEPRECATED */ function fb_form_request_form($config = array()) { global $_fb, $_fb_app; // only works on canvas pages. // Default config $config = array_merge(array('type' => $_fb_app->label, 'content' => 'INVITE CONTENT XXX', 'action' => 'http://apps.facebook.com/' . $_fb_app->canvas, 'invite' => 'true', 'method' => 'POST', ), $config); // form type fb:request-form $form = array('#fb_form_type_hack' => 'fb_form_request', /* becomes #type during form_alter */ '#attributes' => array('type' => $config['type'], 'content' => htmlentities($config['content']), 'invite' => $config['invite'], ), '#action' => $config['action'], ); // Caller must add fb:multi-friend-selector or some other selector. return $form; } /** * Based on theme_form, this renders an fb:request-form. */ function theme_fb_form_request($element) { // TODO: verify attributes required by facebook are found. // Anonymous div to satisfy XHTML compliance. $action = $element['#action'] ? 'action="' . check_url($element['#action']) . '" ' : ''; $output = '\n
". $element['#children'] ."\n
\n"; return $output; } // Because the fb:request-form includes its own buttons, including the particularly annoying skip button, this submit callback is not used. Will probably delete it. If I can't make it work properly. function fb_form_multi_add_invite_form_submit() { //watchdog('fb_debug', 'fb_form_multi_add_invite_form_submit' . print_r(func_get_args(), 1)); //return "foo/bar"; } function fb_form_group_options($fbu) { $groups = fb_get_groups_data($fbu); $items = array(); if ($groups && count($groups)) foreach ($groups as $data) { $items[$data['gid']] = $data['name']; } // TODO: alphabetize list return $items; } // TODO: make this work whether in canvas page or not. function fb_form_friend_options($fbu) { global $_fb; $items = array(); if ($_fb) { $query = "SELECT last_name, first_name, uid, pic_square FROM user WHERE uid IN (SELECT uid2 FROM friend WHERE uid1=$fbu)"; // FQL not SQL, no {curly_brackets}! $result = $_fb->api_client->fql_query($query); // TODO: sort results by name foreach ($result as $data) { $items[$data['uid']] = $data['first_name'] . ' ' . $data['last_name']; } } return $items; } function fb_form_group_member_options($fbg, $fbu) { global $_fb; $query = "SELECT uid FROM group_member WHERE gid=$fbg"; // FQL not SQL, no {curly_brackets}! $result = $_fb->api_client->fql_query($query); $query = "SELECT uid, first_name, last_name FROM user WHERE uid IN (SELECT uid FROM group_member WHERE gid=$fbg)"; // FQL not SQL, no {curly_brackets} $result = $_fb->api_client->fql_query($query); // TODO: sort results by name $options = array(); foreach ($result as $data) { if ($data['uid'] != $fbu) $options[$data['uid']] = $data['first_name'] . ' ' . $data['last_name']; } return $options; } function fb_form_elements() { $items = array(); $items['fb_form_request_selector'] = array( '#input' => TRUE, '#tree' => TRUE, /* not sure what this does */ '#process' => array('fb_form_process_request_selector' => array()), // The submit callback does not work properly in '#executes_submit_callback' => TRUE, ); $items['fb_form_friend_selector'] = array( '#input' => TRUE, '#tree' => TRUE, '#process' => array('fb_form_friend_selector_process' => array()), ); return $items; } /** * Build a friend selector for use in . * * Use this to select friends when sending an invite or request. */ function fb_form_process_request_selector($orig) { global $_fb; // replace with FBML markup $element = array('#type' => 'markup', '#value' => 'api_client->fql_query($query); // Store list of friends in SESSION, so our autocomplete function will not // have to query it every time. $_SESSION['fb_form_friend_selector_result'] = $result; } $element = array('#validate' => array('fb_form_friend_selector_validate' => array($orig))); foreach (array('#title', '#parents', '#description', '#default_value', '#weight', '#multiple', '#required', '#name', '#value', '#id', '#size', '#rows', '#validate') as $key) if (isset($orig[$key])) $element[$key] = $orig[$key]; // Allow use of textarea instead of textfield, autocomplete will not work. if (isset($orig['#rows']) && $orig['#rows'] > 0) $element['#type'] = 'textarea'; else $element['#type'] = 'textfield'; $element['#autocomplete_path'] = url('fb_form_friend_selector_autocomplete', array('absolute' => TRUE)); return $element; } /** * Autocomplete friend names */ function fb_form_friend_selector_autocomplete($string) { // Regexp copied from taxonomy_autocomplete $regexp = '%(?:^|,\ *)("(?>[^"]*)(?>""[^"]* )*"|(?: [^",]*))%x'; preg_match_all($regexp, $string, $preg_matches); $typed_names = $preg_matches[1]; $last_string = trim(array_pop($typed_names)); $prefix = count($typed_names) ? implode(', ', $typed_names) .', ' : ''; // Get list of friends from session $result = $_SESSION['fb_form_friend_selector_result']; $matches = array(); if (count($result) && $last_string) { foreach ($result as $data) { $name = strtolower($data['name']); if (strpos($name, strtolower($last_string)) !== FALSE && !in_array($data['name'], $typed_names)) { $markup = "{$data[name]}"; $matches[$prefix . $data['name']] = $markup; } } } if (count($matches)) print drupal_to_js($matches); exit(); } /** * Convert #default_value into a value for the form field */ function fb_form_friend_selector_value(&$element) { //drupal_set_message("friend_selector_value" . dpr($form, 1)); // default value passed in will be an array of ids, we need to display a comma seperated list of names. $info = fb_users_getInfo($element['#default_value']); $items = array(); foreach ($info as $data) { $items[] = $data['name']; } if (count($items)) $element['#value'] = implode(', ', $items); } function fb_form_friend_selector_validate($element, $set_errors = TRUE) { //dpm(func_get_args(), "fb_form_friend_selector_validate"); if (!trim($element['#value'])) return; $names = explode(',', $element['#value']); $items = array(); // Facebook user ids $result = $_SESSION['fb_form_friend_selector_result']; foreach ($names as $name) { $found = FALSE; foreach ($result as $data) { $fb_name = strtolower($data['name']); if (strtolower(trim($name)) == $fb_name) { if ($found) { // TODO: handle name collisions more gracefully! if ($set_errors) form_set_error(implode('][', $element['#parents']), t('\'%name\' matched more than one friend.', array('%name' => $name))); $found = -1; } else $found = $data['uid']; } } if ($found > 0) { // Running list of ids $items[] = $found; } elseif ($found == 0) if ($set_errors) form_set_error(implode('][', $element['#parents']), t('Could not find a friend named \'%name\'.', array('%name' => $name))); } // Make the submitted value a list of ids, not a comma-seperated list of names. form_set_value($element, $items); _form_set_value($_POST, $element, $element['#parents'], $items); } function fb_form_theme() { return array( 'fb_form_multi_friend_selector' => array( 'arguments' => array('elements' => NULL), ), 'fb_form_requestform' => array( 'arguments' => array('elements' => NULL), ), 'fb_form_req_choice' => array( 'arguments' => array('elements' => NULL), ), 'fb_form_serverfbml' => array( 'arguments' => array('elements' => NULL), ), ); } function theme_fb_form_multi_friend_selector($elements) { $output = "" . (isset($elements['#children']) ? $elements['#children'] : '') . "\n"; return $output; } function theme_fb_form_requestform($elements) { // content attribute is special. if (is_array($elements['#attributes']['content'])) { $elements['#attributes']['content'] = drupal_render($elements['#attributes']['content']); } $output = "". (isset($elements['#children']) ? $elements['#children'] : '') . "\n"; return $output; } function theme_fb_form_req_choice($elements) { // This special tag has no children $output = "\n"; return $output; } function theme_fb_form_serverfbml($elements) { $output = '\n"; return $output; } function fb_form_multi_selector($attrs = array()) { $element = array( '#type' => 'fb_form_multi_friend_selector', '#attributes' => $attrs, ); return $element; } /** * Helper function to build fb_form_request_form element. */ function fb_form_requestform($attrs = array()) { $element = array( '#type' => 'fb_form_requestform', '#attributes' => $attrs, ); return $element; }