* @category Personal
* @package StatusNet
* @author Evan Prodromou <evan@status.net>
- * @copyright 2008-2009 StatusNet, Inc.
+ * @copyright 2008-2011 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/
*/
* @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
* @link http://status.net/
*/
-
-class ShownoticeAction extends OwnerDesignAction
+class ShownoticeAction extends Action
{
/**
* Notice object to show
*/
-
var $notice = null;
/**
* Profile of the notice object
*/
-
var $profile = null;
/**
* Avatar of the profile of the notice object
*/
-
var $avatar = null;
/**
*
* @return success flag
*/
-
- function prepare($args)
+ protected function prepare(array $args=array())
{
parent::prepare($args);
+ if ($this->boolean('ajax')) {
+ StatusNet::setApi(true);
+ }
- $id = $this->arg('notice');
+ $this->notice = $this->getNotice();
- $this->notice = Notice::staticGet($id);
+ $cur = common_current_user();
- if (empty($this->notice)) {
- // Did we used to have it, and it got deleted?
- $deleted = Deleted_notice::staticGet($id);
- if (!empty($deleted)) {
- $this->clientError(_('Notice deleted.'), 410);
- } else {
- $this->clientError(_('No such notice.'), 404);
- }
- return false;
+ if (!empty($cur)) {
+ $curProfile = $cur->getProfile();
+ } else {
+ $curProfile = null;
+ }
+
+ if (!$this->notice->inScope($curProfile)) {
+ // TRANS: Client exception thrown when trying a view a notice the user has no access to.
+ throw new ClientException(_('Not available.'), 403);
}
$this->profile = $this->notice->getProfile();
if (empty($this->profile)) {
+ // TRANS: Server error displayed trying to show a notice without a connected profile.
$this->serverError(_('Notice has no profile.'), 500);
- return false;
}
- $this->user = User::staticGet('id', $this->profile->id);
+ $this->user = User::getKV('id', $this->profile->id);
- $this->avatar = $this->profile->getAvatar(AVATAR_PROFILE_SIZE);
+ try {
+ $this->avatar = $this->profile->getAvatar(AVATAR_PROFILE_SIZE);
+ } catch (Exception $e) {
+ $this->avatar = null;
+ }
return true;
}
+ /**
+ * Fetch the notice to show. This may be overridden by child classes to
+ * customize what we fetch without duplicating all of the prepare() method.
+ *
+ * @return Notice
+ */
+ protected function getNotice()
+ {
+ $id = $this->arg('notice');
+
+ $notice = Notice::getKV('id', $id);
+
+ if (!$notice instanceof Notice) {
+ // Did we used to have it, and it got deleted?
+ $deleted = Deleted_notice::getKV($id);
+ if ($deleted instanceof Deleted_notice) {
+ // TRANS: Client error displayed trying to show a deleted notice.
+ $this->clientError(_('Notice deleted.'), 410);
+ } else {
+ // TRANS: Client error displayed trying to show a non-existing notice.
+ $this->clientError(_('No such notice.'), 404);
+ }
+ return false;
+ }
+ return $notice;
+ }
+
/**
* Is this action read-only?
*
* @return boolean true
*/
-
function isReadOnly($args)
{
return true;
*
* @return int last-modified date as unix timestamp
*/
-
function lastModified()
{
return max(strtotime($this->notice->modified),
*
* @return string etag
*/
-
function etag()
{
$avtime = ($this->avatar) ?
strtotime($this->avatar->modified) : 0;
return 'W/"' . implode(':', array($this->arg('action'),
+ common_user_cache_hash(),
common_language(),
$this->notice->id,
strtotime($this->notice->created),
*
* @return string title of the page
*/
-
function title()
{
- if (!empty($this->profile->fullname)) {
- $base = $this->profile->fullname . ' (' . $this->profile->nickname . ')';
- } else {
- $base = $this->profile->nickname;
- }
+ $base = $this->profile->getFancyName();
+ // TRANS: Title of the page that shows a notice.
+ // TRANS: %1$s is a user name, %2$s is the notice creation date/time.
return sprintf(_('%1$s\'s status on %2$s'),
$base,
common_exact_date($this->notice->created));
*
* @return void
*/
-
- function handle($args)
+ protected function handle()
{
- parent::handle($args);
-
- if ($this->notice->is_local == Notice::REMOTE_OMB) {
- if (!empty($this->notice->url)) {
- $target = $this->notice->url;
- } else if (!empty($this->notice->uri) && preg_match('/^https?:/', $this->notice->uri)) {
- // Old OMB posts saved the remote URL only into the URI field.
- $target = $this->notice->uri;
- } else {
- // Shouldn't happen.
- $target = false;
- }
- if ($target && $target != $this->selfUrl()) {
- common_redirect($target, 301);
- return false;
+ parent::handle();
+
+ if ($this->boolean('ajax')) {
+ $this->showAjax();
+ } else {
+ if ($this->notice->is_local == Notice::REMOTE) {
+ try {
+ $target = $this->notice->getUrl()
+ if ($target != $this->selfUrl()) {
+ common_redirect($target, 301);
+ }
+ } catch (InvalidUrlException $e) {
+ common_debug('ShownoticeAction could not redirect to remote notice with id='.$this->notice->id . '. Falling back to showPage().');
+ }
}
+ $this->showPage();
}
- $this->showPage();
- }
-
- /**
- * Don't show local navigation
- *
- * @return void
- */
-
- function showLocalNavBlock()
- {
}
/**
*
* @return void
*/
-
function showContent()
{
$this->elementStart('ol', array('class' => 'notices xoxo'));
$this->elementEnd('ol');
}
+ function showAjax()
+ {
+ $this->startHTML('text/xml;charset=utf-8');
+ $this->elementStart('head');
+ // TRANS: Title for page that shows a notice.
+ $this->element('title', null, _m('TITLE','Notice'));
+ $this->elementEnd('head');
+ $this->elementStart('body');
+ $nli = new NoticeListItem($this->notice, $this);
+ $nli->show();
+ $this->elementEnd('body');
+ $this->endHTML();
+ }
+
/**
* Don't show page notice
*
* @return void
*/
-
function showPageNoticeBlock()
{
}
*
* @return void
*/
-
function showAside() {
}
*
* @return void
*/
-
function extraHead()
{
- $user = User::staticGet($this->profile->id);
+ $user = User::getKV($this->profile->id);
if (!$user) {
return;
array(),
array('format'=>'xml','url'=>$this->notice->uri)),
'title'=>'oEmbed'),null);
+
+ // Extras to aid in sharing notices to Facebook
+ $avatarUrl = $this->profile->avatarUrl(AVATAR_PROFILE_SIZE);
+ $this->element('meta', array('property' => 'og:image',
+ 'content' => $avatarUrl));
+ $this->element('meta', array('property' => 'og:description',
+ 'content' => $this->notice->content));
}
}
-class SingleNoticeItem extends NoticeListItem
+// @todo FIXME: Class documentation missing.
+class SingleNoticeItem extends DoFollowListItem
{
- /**
- * recipe function for displaying a single notice.
- *
- * We overload to show attachments.
- *
- * @return void
- */
-
- function show()
+ function avatarSize()
{
- $this->showStart();
- $this->showNotice();
- $this->showNoticeAttachments();
- $this->showNoticeInfo();
- $this->showNoticeOptions();
- $this->showEnd();
- }
-
- function showNoticeAttachments() {
- $al = new AttachmentList($this->notice, $this->out);
- $al->show();
+ return AVATAR_STREAM_SIZE;
}
}