. * * @category Plugin * @package StatusNet * @author Evan Prodromou * @copyright 2009 StatusNet, Inc. * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0 * @link http://status.net/ */ if (!defined('STATUSNET') && !defined('LACONICA')) { exit(1); } /** * Superclass for plugin to do realtime updates * * Based on experience with the Comet and Meteor plugins, * this superclass extracts out some of the common functionality * * @category Plugin * @package StatusNet * @author Evan Prodromou * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0 * @link http://status.net/ */ class RealtimePlugin extends Plugin { protected $replyurl = null; protected $favorurl = null; protected $deleteurl = null; /** * When it's time to initialize the plugin, calculate and * pass the URLs we need. */ function onInitializePlugin() { $this->replyurl = common_local_url('newnotice'); $this->favorurl = common_local_url('favor'); // FIXME: need to find a better way to pass this pattern in $this->deleteurl = common_local_url('deletenotice', array('notice' => '0000000000')); return true; } function onEndShowScripts($action) { $timeline = $this->_getTimeline($action); // If there's not a timeline on this page, // just return true if (empty($timeline)) { return true; } $scripts = $this->_getScripts(); foreach ($scripts as $script) { $action->script($script); } $user = common_current_user(); if (!empty($user->id)) { $user_id = $user->id; } else { $user_id = 0; } $action->elementStart('script', array('type' => 'text/javascript')); $script = ' $(document).ready(function() { '. $this->_updateInitialize($timeline, $user_id). '}); '; $action->raw($script); $action->elementEnd('script'); return true; } function onEndNoticeSave($notice) { $paths = array(); // Add to the author's timeline $user = User::staticGet('id', $notice->profile_id); if (!empty($user)) { $paths[] = array('showstream', $user->nickname); } // Add to the public timeline if ($notice->is_local || ($notice->is_local == 0 && !common_config('public', 'localonly'))) { $paths[] = array('public'); } // Add to the tags timeline $tags = $this->getNoticeTags($notice); if (!empty($tags)) { foreach ($tags as $tag) { $paths[] = array('tag', $tag); } } // Add to inbox timelines // XXX: do a join $inbox = new Notice_inbox(); $inbox->notice_id = $notice->id; if ($inbox->find()) { while ($inbox->fetch()) { $user = User::staticGet('id', $inbox->user_id); $paths[] = array('all', $user->nickname); } } // Add to the replies timeline $reply = new Reply(); $reply->notice_id = $notice->id; if ($reply->find()) { while ($reply->fetch()) { $user = User::staticGet('id', $reply->profile_id); if (!empty($user)) { $paths[] = array('replies', $user->nickname); } } } // Add to the group timeline // XXX: join $gi = new Group_inbox(); $gi->notice_id = $notice->id; if ($gi->find()) { while ($gi->fetch()) { $ug = User_group::staticGet('id', $gi->group_id); $paths[] = array('showgroup', $ug->nickname); } } if (count($paths) > 0) { $json = $this->noticeAsJson($notice); $this->_connect(); foreach ($paths as $path) { $timeline = $this->_pathToChannel($path); $this->_publish($timeline, $json); } $this->_disconnect(); } return true; } function onStartShowPageNotice($action) { $timeline = $this->_getTimeline($action); if (!empty($timeline)) { $base = $action->selfUrl(); if (mb_strstr($url, '?')) { $url = $base . '&realtime=1'; } else { $url = $base . '?realtime=1'; } $title = $action->title(); $code = "window.open('$url', '$title', 'toolbar=no,resizable=yes,scrollbars=yes,status=yes,height=640,width=575');"; $action->element('a', array('href' => $base, 'onclick' => $code, 'id' => 'realtime_timeline', 'title' => _('Pop up')), 'Pop up'); } return true; } function onStartShowBody($action) { $realtime = $action->boolean('realtime'); if (!$realtime) { return true; } $action->elementStart('body', (common_current_user()) ? array('id' => $action->trimmed('action'), 'class' => 'user_in') : array('id' => $action->trimmed('action'))); if (common_logged_in()) { $action->showNoticeForm(); } $action->showContent(); $action->elementEnd('body'); return false; // No default processing } function noticeAsJson($notice) { // FIXME: this code should be abstracted to a neutral third // party, like Notice::asJson(). I'm not sure of the ethics // of refactoring from within a plugin, so I'm just abusing // the TwitterApiAction method. Don't do this unless you're me! require_once(INSTALLDIR.'/lib/twitterapi.php'); $act = new TwitterApiAction('/dev/null'); $arr = $act->twitter_status_array($notice, true); $arr['url'] = $notice->bestUrl(); $arr['html'] = htmlspecialchars($notice->rendered); $arr['source'] = htmlspecialchars($arr['source']); if (!empty($notice->reply_to)) { $reply_to = Notice::staticGet('id', $notice->reply_to); if (!empty($reply_to)) { $arr['in_reply_to_status_url'] = $reply_to->bestUrl(); } $reply_to = null; } $profile = $notice->getProfile(); $arr['user']['profile_url'] = $profile->profileurl; return $arr; } function getNoticeTags($notice) { $tags = null; $nt = new Notice_tag(); $nt->notice_id = $notice->id; if ($nt->find()) { $tags = array(); while ($nt->fetch()) { $tags[] = $nt->tag; } } $nt->free(); $nt = null; return $tags; } // Push this up to Plugin function log($level, $msg) { common_log($level, get_class($this) . ': '.$msg); } function _getScripts() { return array('plugins/Realtime/realtimeupdate.js', 'plugins/Realtime/json2.js'); } function _updateInitialize($timeline, $user_id) { return "RealtimeUpdate.init($user_id, \"$this->replyurl\", \"$this->favorurl\", \"$this->deleteurl\"); "; } function _connect() { } function _publish($timeline, $json) { } function _disconnect() { } function _pathToChannel($path) { return ''; } function _getTimeline($action) { $path = null; $timeline = null; $action_name = $action->trimmed('action'); switch ($action_name) { case 'public': $path = array('public'); break; case 'tag': $tag = $action->trimmed('tag'); if (!empty($tag)) { $path = array('tag', $tag); } break; case 'showstream': case 'all': case 'replies': case 'showgroup': $nickname = common_canonical_nickname($action->trimmed('nickname')); if (!empty($nickname)) { $path = array($action_name, $nickname); } break; default: break; } if (!empty($path)) { $timeline = $this->_pathToChannel($path); } return $timeline; } }