| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							| 
									
										
										
										
											2009-08-25 18:29:56 -04:00
										 |  |  |  * StatusNet, the distributed open-source microblogging tool | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |  * Superclass for plugins that do "real time" updates of timelines using Ajax | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |  * | 
					
						
							|  |  |  |  * PHP version 5 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * LICENCE: This program is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU Affero General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU Affero General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Affero General Public License | 
					
						
							|  |  |  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @category  Plugin | 
					
						
							| 
									
										
										
										
											2009-08-25 18:29:56 -04:00
										 |  |  |  * @package   StatusNet | 
					
						
							|  |  |  |  * @author    Evan Prodromou <evan@status.net> | 
					
						
							|  |  |  |  * @copyright 2009 StatusNet, Inc. | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0 | 
					
						
							| 
									
										
										
										
											2009-08-25 18:29:56 -04:00
										 |  |  |  * @link      http://status.net/ | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-26 10:41:36 -04:00
										 |  |  | if (!defined('STATUSNET') && !defined('LACONICA')) { | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |     exit(1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |  * 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 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @category Plugin | 
					
						
							| 
									
										
										
										
											2009-08-25 18:29:56 -04:00
										 |  |  |  * @package  StatusNet | 
					
						
							|  |  |  |  * @author   Evan Prodromou <evan@status.net> | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |  * @license  http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0 | 
					
						
							| 
									
										
										
										
											2009-08-25 18:29:56 -04:00
										 |  |  |  * @link     http://status.net/ | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  | class RealtimePlugin extends Plugin | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |     protected $replyurl = null; | 
					
						
							|  |  |  |     protected $favorurl = null; | 
					
						
							|  |  |  |     protected $deleteurl = null; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-23 17:07:20 -04:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * When it's time to initialize the plugin, calculate and | 
					
						
							|  |  |  |      * pass the URLs we need. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |     function onInitializePlugin() | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |         $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')); | 
					
						
							| 
									
										
										
										
											2009-09-23 17:07:20 -04:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function onEndShowScripts($action) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-09-23 17:07:20 -04:00
										 |  |  |         $timeline = $this->_getTimeline($action); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-23 17:07:20 -04:00
										 |  |  |         // If there's not a timeline on this page,
 | 
					
						
							|  |  |  |         // just return true
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (empty($timeline)) { | 
					
						
							| 
									
										
										
										
											2009-09-23 11:08:35 -04:00
										 |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |         $scripts = $this->_getScripts(); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         foreach ($scripts as $script) { | 
					
						
							| 
									
										
										
										
											2009-08-05 20:15:00 -04:00
										 |  |  |             $action->script($script); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $user = common_current_user(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($user->id)) { | 
					
						
							|  |  |  |             $user_id = $user->id; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $user_id = 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $action->elementStart('script', array('type' => 'text/javascript')); | 
					
						
							| 
									
										
										
										
											2009-09-23 17:07:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $script = ' $(document).ready(function() { '. | 
					
						
							|  |  |  |           $this->_updateInitialize($timeline, $user_id). | 
					
						
							|  |  |  |           '}); '; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $action->raw($script); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |         $action->elementEnd('script'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function onEndNoticeSave($notice) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-07-15 15:30:33 -04:00
										 |  |  |         $paths = array(); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-23 11:08:35 -04:00
										 |  |  |         // XXX: Add other timelines; this is just for the public one
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ($notice->is_local || | 
					
						
							|  |  |  |             ($notice->is_local == 0 && !common_config('public', 'localonly'))) { | 
					
						
							| 
									
										
										
										
											2009-09-23 11:08:35 -04:00
										 |  |  |             $paths[] = array('public'); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $tags = $this->getNoticeTags($notice); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($tags)) { | 
					
						
							|  |  |  |             foreach ($tags as $tag) { | 
					
						
							| 
									
										
										
										
											2009-07-15 15:30:33 -04:00
										 |  |  |                 $paths[] = array('tag', $tag); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-15 15:30:33 -04:00
										 |  |  |         if (count($paths) > 0) { | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |             $json = $this->noticeAsJson($notice); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |             $this->_connect(); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-15 15:30:33 -04:00
										 |  |  |             foreach ($paths as $path) { | 
					
						
							|  |  |  |                 $timeline = $this->_pathToChannel($path); | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |                 $this->_publish($timeline, $json); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |             $this->_disconnect(); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-23 17:07:20 -04:00
										 |  |  |     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
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  |     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); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     function _getScripts() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-08-05 20:15:00 -04:00
										 |  |  |         return array('plugins/Realtime/realtimeupdate.js', | 
					
						
							|  |  |  |                      'plugins/Realtime/json2.js'); | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-14 15:39:02 -04:00
										 |  |  |     function _updateInitialize($timeline, $user_id) | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-07-14 17:06:29 -04:00
										 |  |  |         return "RealtimeUpdate.init($user_id, \"$this->replyurl\", \"$this->favorurl\", \"$this->deleteurl\"); ";
 | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function _connect() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-14 15:39:02 -04:00
										 |  |  |     function _publish($timeline, $json) | 
					
						
							| 
									
										
										
										
											2009-07-14 15:26:39 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function _disconnect() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-15 15:30:33 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     function _pathToChannel($path) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return ''; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-09-23 17:07:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     function _getTimeline($action) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $path = null; | 
					
						
							|  |  |  |         $timeline = null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         switch ($action->trimmed('action')) { | 
					
						
							|  |  |  |          case 'public': | 
					
						
							|  |  |  |             $path = array('public'); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |          case 'tag': | 
					
						
							|  |  |  |             $tag = $action->trimmed('tag'); | 
					
						
							|  |  |  |             if (!empty($tag)) { | 
					
						
							|  |  |  |                 $path = array('tag', $tag); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |          default: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!is_null($path)) { | 
					
						
							|  |  |  |             $timeline = $this->_pathToChannel($path); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $timeline; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-13 09:58:52 -04:00
										 |  |  | } |