Tag Archives: facebook

2015년에 변경된 페이스북 메신저 API 개요

Facebook-Developers-Logo

예전부터 페이스북 API중에 Chat API가 있었습니다. 카카오톡의 게임 홍보 메시지처럼 페이스북 내에 있는 메신져 플랫폼에 메시지를 발송할 수 있는 API였는데요. 기존에 사용 빈도가 적었던것인지 사용에 특별한 제약 사항이 없는 자유롭게 사용할 수 있는 API였습니다. 심지어 Graph API를 이용해서 서버상에서 호출하는것도 가능했었습니다. 하지만 언제부터인가 기존의 Chat API가 Deprecated 되었습니다.

facebook_messenser_api_overview_01

2014년 4월 30일에 기존의 XMPP기반의 채팅 API가 Deprecate 될것임을 공지하였었고 2015년 4월 30일에 더이상 이 API들을 사용할 수 없게 되었습니다. 기존에 제공되었던 API를 사용하여 활용할 수 있는 기능은 다음과 같았습니다.

  • addUserToGroup : 특정 유저를 그룹 채팅에 추가합니다.
  • changeArchivedStatus : 채팅을 보관 상태로 변경. 채팅 목록에서 즉시 사라지며 보관함에 들어갑니다.
  • deleteMessage : 이미 발송된 메시지를 삭제합니다.
  • deleteThread : 채팅을 삭제합니다.
  • getCurrentUserID : 현재 로그인되어있는 유저의 페이스북 ID를 가져옵니다.
  • getFriendsList : 친구의 페이스북 계정 정보를 가져옵니다. 이름, 성별, 프로필사진, 생일여부등을 알 수 있습니다.
  • getOnlineUsers : 친구들의 온라인 상태 정보를 가져옵니다. 오프라인, 유휴상태, 온라인, 모바일사용중으로 분류됩니다.
  • getThreadHistory : 현재 채팅의 히소토리를 가져옵니다.
  • getThreadList : 채팅 목록을 가져옵니다.
  • getUserID : 페이스북상의 이름으로 ID를 가져옵니다.
  • getUserInfo : 페이스북 ID로 계정의 정보를 가져옵니다.
  • markAsRead : 특정 채팅의 내용을 모두 읽은것으로 지정합니다.
  • removeUserFromGroup : 그룹에서 유저를 제거합니다.
  • searchForThread : 채팅방의 이름을 검색합니다.
  • sendMessage : 메시지를 발송합니다. 텍스트, 스티커ID, 파일/이미지, URL을 포함할 수 있습니다.
  • sendTypingIndicator : “유저가 글을 작성중입니다”라는 상태 메시지를 상대측에 보여줍니다.
  • setTitle : 그룹 채팅의 방 이름을 설정합니다.

그리고 2015년 3월 25일에 새로운 메신저 플랫폼이 발표되었습니다. 결과적으로 기존의 XMPP 버전의 API들과는 차원이 다르게 기능이 축소되었습니다. 아마도 친구들의 정보라던지 온라인 상태를 가져다 쓸 수 있다는 점을 문제로 생각한게 아닐까 생각됩니다. 페이스북 자체의 새로운 메신저 플랫폼에 대해서 밀어붙이던 시기이기도 했고요. 아무래도 기존에는 메신저 기능을 페이스북의 +@ 기능정도로 생각하다가 좀 더 적극적인 방향으로 생각해보기로 하면서 방향을 폐쇄적으로 변경한것인지도 모르겠습니다. 다음은 페이스북의 메신져 플랫폼에 대한 소개글을 번역한 내용입니다.

Messenger Platform

메신저 플랫폼은 개발자들이 개발중인 앱에 메신저를 통합하는 과정을 좀 더 쉽게 해줍니다. 그렇게 함으로 써 메신저를 사용하는 6억이 넘는 사용자들이 GIF, 사진, 비디오, 음성메시지 등등을 활용하여 새롭고 재미있게 그들을 표현할 수 있는 방법을 찾을 수 있을 것입니다. 이 메신저 플랫폼을 사용하면 앱의 컨텐츠들은 개인 혹은 그룹 메시지들이 창의적이고 표현적인 대화가 이루어질 수 있도록 해줄것입니다. 또한 개발자들에게 성장과 재참여를 할 갖게할 기회를 제공할 것입니다.

메신저 플랫폼의 앱은 메신저를 통해 유저로 하여금 앱을 설치하도록 유도하거나 앱의 컨텐츠를 사용하여 답장을 할 수 있게 해줍니다. 만약 메시지를 받은 유저가 앱이 설치되어있지 않아서 바로 답장을 할 수 없는 상황이라면 메신저를 앱스토어로 이동시켜 바로 설치를 할 수 있도록 유도할 것입니다. 이 과정은 유저가 친구들과의 대화를 통해서 자연스럽게 새로운 앱을 추천받게 되는 과정이 될 것입니다.

facebook_messenser_api_overview_02

메신저 플랫폼을 이용하여 개발자는 앱의 사용율을 높일 수 있습니다. 만약 메시지를 받은 유저가 이미 그 앱을 설치한 상태라면 그들은 바로 메시지의 이미지에 함께 표시되는 답장(Reply) 버튼을 누를 수 있습니다. 그러면 바로 해당 앱을 실행시켜 결과를 가지고 메신저에 바로 공유할 수 있습니다.

facebook_messenser_api_overview_03

기본적인 이미지, 동영상, 텍스트를 전송하는것을 Basic 메시지라고 부르고 이렇게 설치나, 답장을 유도할 수 있는 메시지를 Optimized 메시지라고 부릅니다. 이 Optimized 메시지를 사용하기 위해서는 몇가지 제약 사항이 있습니다.

  • 컨텐츠 공유하기를 하였을 때 선택가능한 공유 선택지중에 페이스북 메신저가 가장 첫번째로 떠야 합니다. 유저 경험 차원이라는데 그냥 페이스북의 욕심이겠죠^^;
  • 페이스북이 제공하는 통계분석툴(Analytics)를 반드시 붙여야 합니다. 이것을 붙임으로써 각종 통계를 볼 수 있게 된다고 합니다. 근데 강제 하는 이유는 페이스북도 궁금해서겠지요^^?
  • 이러한 기능을 사용하기 위해서는 페이스북에 앱 심사를 거쳐야 합니다. 이때에 당신의 앱은 앱스토어에 이미 출시되어있어야 합니다.

facebook_messenser_api_overview_06

몇몇 앱의 메신저 통합된 사례는 메신저의 대화상의 새로운 탭을 통해서 바로 연결되는 홍보 기회를 가질 수 있습니다. 이 메뉴를 통해 이미 설치된 앱을 바로 실행할 수 있는 바로가기로 사용될 수 있고 새로운 유저가 시도해 볼 수 있는 기회를 얻을 수 도 있습니다.

Business on Messenger

메신저 플랫폼 이외에는 어떻게 사람들과 기업간의 커뮤니케이션을 개선시킬 수 있을까를 목적으로 만들어진 메신저 비지니스 플랫폼도 있습니다. 이 메신저 비지니스 플랫폼은 다음과 같은것을 가능케 합니다. 유저가 어떤 기업의 사이트에서 물건을 구매하는 과정에서 유저는 기업과 대화를 시작하게 됩니다. 구매가 확정되었는지 배송 상태는 어떻게 되는지 메신저를 통해 받을 수 있으며, 특정 폼 없이 자유로운 양식으로 기업에게 질문을 하고 빠른 답장을 받을 수 있습니다.

facebook_messenser_api_overview_07

facebook_messenser_api_overview_08

현재 Everlane, Zulily, Zendesk 등의 기업에서 적용하여 경험을 실시간 채팅 경험을 만들어 나가고 있습니다. 더 자세한 내용은 [이곳]을 참고하시기 바랍니다.

참고 : https://developers.facebook.com/docs/messenger/overview

대표적인 미국의 SNS : 페이스북(facebook) 소스코드 유출!

사용자 삽입 이미지미국의 SNS 시장을 이끌고 있는 서비스로는 Myspace가 있습니다. 그 뒤로 Facebook이 있지요.

Facebook은 대학교의 인맥형성을 주제로 생겨났다가 가입자체를 공개하면서 엄청난 성장세를 보여왔습니다.

위키피디아 의 자료를 보시면 알수 있겠지만 현재 Facebook의 가입자수는 3500만명에 달합니다.

Facebook의 경우 가입자수는 한달이내로 로그인한적이 있는 Active 유저만을 계산한다고 하는군요.

위에 싸이월드도 있군요. 1500만명입니다.

저도 Facebook을 사용하고 있습니다만 친구들의 세세한 업데이트 현황까지 보여주며 또한 네트워크라는것을 통해 인맥을 형성하는데 강력한 기능을 제공한다는 점이 대단하다고 생각되어집니다.

또한 저같은 개발자들이 눈에 띄게 보게 되는것이 강력한 위젯 개발 기능을 제공하는데요.

이런 대형 서비스에서 서비스의 핵심코드까지 접근할 수 있는 강력한 기능을 일반 개발자들에게 준다는것이 저로서는 조금 이해하기 힘든 부분입니다만, 개발자의 입장에서는 정말 멋진 서비스임에 틀림이 없습니다.

다만 문제가, 사용자가 원하는 쿼리문까지 작성하여 디비에 직접 엑세스 할수 있고, 모든 자원에 접근 할수 있다는점이 문제가 아닐까 생각되는데요.

Developers에 등록을 하고, API Key를 받아야만 개발을 할수가 있습니다. 하지만 악의적인 마음을 품고 있는 사람에게 그것도 무용지물인 제제안이 될것입니다.

아니나 다를까, 해킹을 당했다고 합니다. TechCrunch에는 Facebook Source Code Leaked 라는 제목의 글이 올라왔습니다.

공식 확인된것은 아니지만 유출된 코드가 Facebook의 코드가 맞는듯 합니다.

코드를 유출한 사람은 blogspot에 facebooksecrets라는 계정을 만들고 index.php 소스내용을 올려둔 상태입니다.

과연 어떤 경로로 소스가 유출되었을까요? 솔직히 위젯 개발에서 PHP의 왠만한 함수를 접근할수 있다면 얼마든지 소스를 유출할수 있다고 생각되어집니다만, 확인해볼만큼 간이 크진 않습니다^^

확실히 생각해 볼수록 PHP로 제작된 서비스에서는 사용자가 PHP코드에 접근할수 있어서는 안되는 것이 맞다고 생각되어집니다.

[#M_ more.. | less.. |Index.php

[code type=php]include_once $_SERVER[‘PHP_ROOT’].’/html/init.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/home.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/requests.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/feed/newsfeed.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/poke.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/share.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/orientation.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/feed/newsfeed.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/mobile/register.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/forms_lib.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/contact_importer/contact_importer.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/feed/util.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/hiding_prefs.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/abtesting.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/friends.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/statusupdates.php’;

// lib/display/feed.php has to be declared here for scope issues.
// This keeps display/feed.php cleaner and easier to understand.
include_once $_SERVER[‘PHP_ROOT’].’/lib/display/feed.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/monetization_box.php’;

// require login
$user = require_login();
print_time(‘require_login’);
param_request(array( ‘react’ => $PARAM_EXISTS));

// Check and fix broken emails
// LN – disabling due to excessive can_see dirties and sets when enabled.
//check_and_fix_broken_emails($user);

// migrate AIM screenname from profile to screenname table if needed
migrate_screenname ($user);

// homepage announcement variables
$HIDE_ANNOUNCEMENT_BIT = get_site_variable(‘HIDE_ANNOUNCEMENT_BIT’);
$HIDE_INTRO_BITMASK = get_site_variable(‘HIDE_INTRO_BITMASK’);

// redirects
if (is_sponsor_user()) {
redirect(‘bizhome.php’, ‘www’);
}

include_once $_SERVER[‘PHP_ROOT’].’/lib/mesg.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/invitetool.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/grammar.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/securityq.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/events.php’;
include_once $_SERVER[‘PHP_ROOT’].’/lib/rooster/stories.php’;

// todo: password confirmation redirects here (from html/reset.php),
// do we want a confirmation message?

param_get_slashed(array(
‘feeduser’ => $PARAM_INT, //debug: gets feed for user here
‘err’ => $PARAM_STRING, // returning from a failed entry on an orientation form
‘error’ => $PARAM_STRING, // an error can also be here because the profile photo upload code is crazy
‘ret’ => $PARAM_INT,
‘success’ => $PARAM_INT, // successful profile picture upload
‘jn’ => $PARAM_INT, // joined a network for orientation
‘np’ => $PARAM_INT, // network pending (for work/address network)
‘me’ => $PARAM_STRING, // mobile error
‘mr’ => $PARAM_EXISTS, // force mobile reg view
‘mobile’ => $PARAM_EXISTS, // mobile confirmation code sent
‘jif’ => $PARAM_EXISTS, // just imported friends
‘ied’ => $PARAM_STRING, // import email domain
‘o’ => $PARAM_EXISTS, // first time orientation, passed on confirm
‘verified’ => $PARAM_EXISTS)); // verified mobile phone

param_post(array(
‘leave_orientation’ => $PARAM_EXISTS,
‘show_orientation’ => $PARAM_INT, // show an orientation step
‘hide_orientation’ => $PARAM_INT)); // skip an orientation step

// homepage actions
if ($req_react && validate_expiring_hash($req_react, $GLOBALS[‘url_md5key’])) {
$show_reactivated_message = true;
} else {
$show_reactivated_message = false;
}
tpl_set(‘show_reactivated_message’, $show_reactivated_message);

// upcoming events
events_check_future_events($user); // make sure big tunas haven’t moved around
$upcoming_events = events_get_imminent_for_user($user);

// this is all stuff that can be fetched together!
$upcoming_events_short = array();
obj_multiget_short(array_keys($upcoming_events), true, $upcoming_events_short);
$new_pokes = 0;
//only get the next N pokes for display
//where N is set in the dbget to avoid caching issues
$poke_stats = get_num_pokes($user);
get_next_pokes($user, true, $new_pokes);
$poke_count = $poke_stats[‘unseen’];

$targeted_data = array();
home_get_cache_targeted_data($user, true, $targeted_data);
$announcement_data = array();
home_get_cache_announcement_data($user, true, $announcement_data);
$orientation = 0;
orientation_get_status($user, true, $orientation);
$short_profile = array();
profile_get_short($user, true, $short_profile);
// pure priming stuff
privacy_get_network_settings($user, true);
$presence = array();
mobile_get_presence_data($user, true, $presence);
feedback_get_event_weights($user, true);
// Determine if we want to display the feed intro message
$intro_settings = 0;
user_get_hide_intro_bitmask($user, true, $intro_settings);
$user_friend_finder = true;
contact_importer_get_used_friend_finder($user, true, $used_friend_finder);
$all_requests = requests_get_cache_data($user);
// FIXME?: is it sub-optimal to call this both in requests_get_cache_data and here?
$friends_status = statusupdates_get_recent($user, null, 3);
memcache_dispatch(); // populate cache data

// Merman’s Admin profile always links to the Merman’s home
if (user_has_obj_attached($user)) {
redirect(‘mhome.php’, ‘www’);
}

if (is_array($upcoming_events)) {
foreach ($upcoming_events as $event_id => $data) {
$upcoming_events[$event_id][‘name’] = txt_set($upcoming_events_short[$event_id][‘name’]);
}
}

tpl_set(‘upcoming_events’ , $upcoming_events);

// disabled account actions
$disabled_warning = ((IS_DEV_SITE || IS_QA_SITE) && is_disabled_user($user));
tpl_set(‘disabled_warning’, $disabled_warning);

// new pokes (no more messages here, they are in the top nav!)
if (!user_is_guest($user)) {
tpl_set(‘poke_count’ , $poke_count);
tpl_set(‘pokes’ , $new_pokes);
}

// get announcement computations
tpl_set(‘targeted_data’ , $targeted_data);
tpl_set(‘announcement_data’ , $announcement_data);

// birthday notifications
tpl_set(‘birthdays’ , $birthdays = user_get_birthday_notifications($user, $short_profile));
tpl_set(‘show_birthdays’ , $show_birthdays = (count($birthdays) || !$orientation));

// user info
tpl_set(‘first_name’ , user_get_first_name(txt_set($short_profile[‘id’])));
tpl_set(‘user’ , $user);

// decide if there are now any requests to show
$show_requests = false;
foreach ($all_requests as $request_category) {
if ($request_category) {
$show_requests = true;
break;
}
}
tpl_set(‘all_requests’, $show_requests ? $all_requests : null);

$permissions = privacy_get_reduced_network_permissions($user, $user);

// status
$user_info = array(‘user’ => $user,
‘firstname’ => user_get_first_name($user),
‘see_all’ => ‘/statusupdates/?ref=hp’,
‘profile_pic’ => make_profile_image_src_direct($user, ‘thumb’),
‘square_pic’ => make_profile_image_src_direct($user, ‘square’));

if (!empty($presence) && $presence[‘status_time’] > (time() – 60*60*24*7)) {
$status = array(‘message’ => txt_set($presence[‘status’]),
‘time’ => $presence[‘status_time’],
‘source’ => $presence[‘status_source’]);
} else {
$status = array(‘message’ => null, ‘time’ => null, ‘source’ => null);
}
tpl_set(‘user_info’, $user_info);

tpl_set(‘show_status’, $show_status = !$orientation);
tpl_set(‘status’, $status);
tpl_set(‘status_custom’, $status_custom = mobile_get_status_custom($user));
tpl_set(‘friends_status’, $friends_status);

// orientation
if ($orientation) {
if ($post_leave_orientation) {
orientation_update_status($user, $orientation, 2);
notification_notify_exit_orientation($user);
dirty_user($user);
redirect(‘home.php’);
} else if (orientation_eligible_exit(array(‘uid’=>$user)) == 2) {
orientation_update_status($user, $orientation, 1);
notification_notify_exit_orientation($user);
dirty_user($user);
redirect(‘home.php’);
}
}

// timezone – outside of stealth, update user’s timezone if necessary
$set_time = !user_is_alpha($user, ‘stealth’);
tpl_set(‘timezone_autoset’, $set_time );
if ($set_time) {
$daylight_savings = get_site_variable(‘DAYLIGHT_SAVINGS_ON’);
tpl_set(‘timezone’, $short_profile[‘timezone’] – ($daylight_savings ? 4 : 5) );
}

// set next step if we can
if (!$orientation) {
user_set_next_step($user, $short_profile);
}

// note: don’t make this an else with the above statement, because then no news feed stories will be fetched if they’re exiting orientation
if ($orientation) {
extract(orientation_get_const());

require_js(‘js/dynamic_dialog.js’);
require_js(‘js/suggest.js’);
require_js(‘js/typeahead_ns.js’);
require_js(‘js/suggest.js’);
require_js(‘js/editregion.js’);
require_js(‘js/orientation.js’);
require_css(‘css/typeahead.css’);
require_css(‘css/editor.css’);

if ($post_hide_orientation && $post_hide_orientation <= $ORIENTATION_MAX) {
$orientation[‘orientation_bitmask’] |= ($post_hide_orientation * $ORIENTATION_SKIPPED_MODIFIER);
orientation_update_status($user, $orientation);
} else if ($post_show_orientation && $post_show_orientation <= $ORIENTATION_MAX) {
$orientation[‘orientation_bitmask’] &= ~($post_show_orientation * $ORIENTATION_SKIPPED_MODIFIER);
orientation_update_status($user, $orientation);
}

$stories = orientation_get_stories($user, $orientation);
switch ($get_err) {
case $ORIENTATION_ERR_COLLEGE:
$temp = array(); // the affil_retval_msg needs some parameters won’t be used
$stories[$ORIENTATION_NETWORK][‘failed_college’]=affil_retval_msg($get_ret, $temp, $temp);
break;
case $ORIENTATION_ERR_CORP:
$temp = array();
// We special case the network not recognized error here, because affil_retval_msg is retarded.
$stories[$ORIENTATION_NETWORK][‘failed_corp’] = ($get_ret == 70) ? ‘The email you entered did not match any of our supported networks. ‘ .
‘Click here to see our supported list. ‘ .
‘Go here to suggest your network for the future.’
: affil_retval_msg($get_ret, $temp, $temp);
break;
}

// photo upload error
if ($get_error) {
$stories[$ORIENTATION_ORDER[$ORIENTATION_PROFILE]][‘upload_error’] = pic_get_error_text($get_error);
}
// photo upload success
else if ($get_success == 1) {
$stories[$ORIENTATION_ORDER[$ORIENTATION_PROFILE]][‘uploaded_pic’] = true;
// join network success
} else if ($get_jn) {
$stories[$ORIENTATION_ORDER[$ORIENTATION_NETWORK]][‘joined’] = array(
‘id’ => $get_jn,
‘name’ => network_get_name($get_jn));
// network join pending
} else if ($get_np) {

$stories[$ORIENTATION_ORDER[$ORIENTATION_NETWORK]][‘join_pending’] = array(
‘id’ => $get_np,
’email’ => get_affil_email_conf($user, $get_np),
‘network’ => network_get_name($get_np));
// just imported friend confirmation
} else if ($get_jif) {
$stories[$ORIENTATION_ORDER[$ORIENTATION_NETWORK]][‘just_imported_friends’] = true;
$stories[$ORIENTATION_ORDER[$ORIENTATION_NETWORK]][‘domain’] = $get_ied;
}

// Mobile web API params
if ($get_mobile) {
$stories[$ORIENTATION_ORDER[$ORIENTATION_MOBILE]][‘sent_code’] = true;
$stories[$ORIENTATION_ORDER[$ORIENTATION_MOBILE]][‘view’] = ‘confirm’;
}
if ($get_verified) {
$stories[$ORIENTATION_ORDER[$ORIENTATION_MOBILE]][‘verified’] = true;
}
if ($get_me) {
$stories[$ORIENTATION_ORDER[$ORIENTATION_MOBILE]][‘error’] = $get_me;
}
if ($get_mr) {
$stories[$ORIENTATION_ORDER[$ORIENTATION_MOBILE]][‘view’] = ‘register’;
}

if (orientation_eligible_exit($orientation)) {
tpl_set(‘orientation_show_exit’, true);
}
tpl_set(‘orientation_stories’, $stories);

//if in orientation, we hide all feed intros (all 1’s in bitmask)
$intro_settings = -1;

}
tpl_set(‘orientation’, $orientation);

// Rooster Stories
if (!$orientation &&
((get_site_variable(‘ROOSTER_ENABLED’) == 2) ||
(get_site_variable(‘ROOSTER_DEV_ENABLED’) == 2))) {
$rooster_story_count = get_site_variable(‘ROOSTER_STORY_COUNT’);
if (!isset($rooster_story_count)) {
// Set default if something is wrong with the sitevar
$rooster_story_count = 2;
}
$rooster_stories = rooster_get_stories($user, $rooster_story_count, $log_omissions = true);
if (!empty($rooster_stories) && !empty($rooster_stories[‘stories’])) {
// Do page-view level logging here
foreach($rooster_stories[‘stories’] as $story) {
rooster_log_action($user, $story, ROOSTER_LOG_ACTION_VIEW);
}
tpl_set(‘rooster_stories’, $rooster_stories);
}
}

// set the variables for the home announcement code
$hide_announcement_tpl = ($intro_settings | $HIDE_INTRO_BITMASK) & $HIDE_ANNOUNCEMENT_BIT;
// if on qa/dev site, special rules
$HIDE_INTRO_ON_DEV = get_site_variable(‘HIDE_INTRO_ON_DEV’);
if ((IS_QA_SITE || IS_DEV_SITE) && !$HIDE_INTRO_ON_DEV) {
$hide_announcement_tpl = 0;
}

tpl_set(‘hide_announcement’, $hide_announcement_tpl);
if($is_candidate = is_candidate_user($user)) {
tpl_set(‘hide_announcement’, false);
}
$home_announcement_tpl = !$hide_announcement_tpl || $is_candidate ? home_get_announcement_info($user) : 0;
tpl_set(‘home_announcement’, $home_announcement_tpl);
tpl_set(‘hide_announcement_bit’, $HIDE_ANNOUNCEMENT_BIT);

$show_friend_finder = !$orientation && contact_importer_enabled($user) && !user_get_hiding_pref($user, ‘home_friend_finder’);
tpl_set(‘show_friend_finder’, $show_friend_finder);
if ($show_friend_finder && (user_get_friend_count($user) > 20)) {
tpl_set(‘friend_finder_hide_options’, array(‘text’=>’close’,
‘onclick’=>”return clearFriendFinder()”));
} else {
tpl_set(‘friend_finder_hide_options’, null);
}

$account_info = user_get_account_info($user);
$account_create_time = $account_info[‘time’];

tpl_set(‘show_friend_finder_top’,
!$used_friend_finder);

tpl_set(‘user’, $user);

// MONETIZATION BOX
$minimize_monetization_box = user_get_hiding_pref($user, ‘home_monetization’);
$show_monetization_box = (!$orientation &&
get_site_variable(‘HOMEPAGE_MONETIZATION_BOX’));
tpl_set(‘show_monetization_box’, $show_monetization_box);
tpl_set(‘minimize_monetization_box’, $minimize_monetization_box);

if ($show_monetization_box) {
$monetization_box_data = monetization_box_user_get_data($user);
txt_set(‘monetization_box_data’, $monetization_box_data);
}

// ORIENTATION
if ($orientation) {
$network_ids = id_get_networks($user);
$network_names = multiget_network_name($network_ids);
$in_corp_network = in_array($GLOBALS[‘TYPE_CORP’], array_map(‘extract_network_type’, $network_ids));
$show_corp_search = $in_corp_network ||
get_age(user_get_basic_info_attr($user, ‘birthday’)) >= 21;
$pending_hs = is_hs_pending_user($user);
$hs_id = null;
$hs_name = null;
if ($pending_hs) {
foreach (id_get_pending_networks($user) as $network) {
if (extract_network_type($network[‘network_key’]) == $GLOBALS[‘TYPE_HS’]) {
$hs_id = $network[‘network_key’];
$hs_name = network_get_name($hs_id);
break;
}
}
}
//$orientation_people = orientation_get_friend_and_inviter_ids($user);
$orientation_people = array(‘friends’ => user_get_all_friends($user),
‘pending’ => array_keys(user_get_friend_requests($user)),
‘inviters’=> array(), // wc: don’t show inviters for now
);
$orientation_info = array_merge($orientation_people,
array(‘network_names’ => $network_names,
‘show_corp_search’ => $show_corp_search,
‘pending_hs’ => array(‘hs_id’=>$hs_id,
‘hs_name’=>$hs_name),
‘user’ => $user,
));
tpl_set(‘orientation_info’, $orientation_info);

tpl_set(‘simple_orientation_first_login’, $get_o); // unused right now
}

// Roughly determine page length for ads
// first, try page length using right-hand panel
$ads_page_length_data = 3 + // 3 for profile pic + next step
($show_friend_finder ? 1 : 0) +
($show_status ? ($status_custom ? count($friends_status) : 0) : 0) +
($show_monetization_box ? 1 : 0) +
($show_birthdays ? count($birthdays) : 0) +
count($new_pokes);

// page length using feed stories
if ($orientation) {
$ads_page_length_data = max($ads_page_length_data, count($stories) * 5);
}
tpl_set(‘ads_page_length_data’, $ads_page_length_data);

$feed_stories = null;
if (!$orientation) { // if they’re not in orientation they get other cool stuff
// ad_insert: the ad type to try to insert for the user
// (0 if we don’t want to try an insert)
$ad_insert = get_site_variable(‘FEED_ADS_ENABLE_INSERTS’);

$feed_off = false;

if (check_super($user) && $get_feeduser){
$feed_stories = user_get_displayable_stories($get_feeduser, 0, null, $ad_insert);
} else if (can_see($user, $user, ‘feed’)) {
$feed_stories = user_get_displayable_stories($user, 0, null, $ad_insert);
} else {
$feed_off = true;
}

// Friend’s Feed Selector – Requires dev.php constant
if (is_friendfeed_user($user)) {
$friendfeed = array();
$friendfeed[‘feeduser’] = $get_feeduser;
$friendfeed[‘feeduser_name’] = user_get_name($get_feeduser);
$friendfeed[‘friends’] = user_get_all_friends($user);
tpl_set(‘friendfeed’, $friendfeed);
}

$feed_stories = feed_adjust_timezone($user, $feed_stories);

tpl_set(‘feed_off’, $feed_off ? redirect(‘privacy.php?view=feeds’, null, false) : false);
}
tpl_set(‘feed_stories’, $feed_stories);

render_template($_SERVER[‘PHP_ROOT’].’/html/home.phpt’);[/code]
_M#]