summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRobin Millette <millette@plantard.controlezvous.ca>2009-01-22 20:33:34 +0000
committerRobin Millette <millette@plantard.controlezvous.ca>2009-01-22 20:33:34 +0000
commit9e9b8349a2d32ae664a5b000ef34a7bb13a039f5 (patch)
tree029c69c1c9acf04fbe91994896d971cd25e7a48e
parent5a84d7f975c2ae356d1fbd88757414ecb385ff5e (diff)
parent248fd763bde70537e5e9a14db435c27a52849c9a (diff)
Merge branch 'master' of /var/www/trunk
-rw-r--r--actions/favorited.php7
-rw-r--r--actions/public.php10
-rw-r--r--actions/subscribers.php92
-rw-r--r--actions/subscriptions.php113
-rw-r--r--actions/usergroups.php3
-rw-r--r--classes/Memcached_DataObject.php49
-rw-r--r--classes/Notice.php6
-rw-r--r--classes/NoticeWrapper.php62
-rw-r--r--lib/arraywrapper.php79
-rw-r--r--lib/gallery.php346
-rw-r--r--lib/galleryaction.php168
-rw-r--r--lib/grouplist.php22
-rw-r--r--lib/groupsbypostssection.php78
-rw-r--r--lib/groupsection.php103
-rw-r--r--lib/noticesection.php107
-rw-r--r--lib/popularnoticesection.php83
-rw-r--r--lib/profilelist.php27
-rw-r--r--lib/profilesection.php101
-rw-r--r--lib/section.php108
-rw-r--r--lib/subgroupnav.php111
-rw-r--r--lib/topposterssection.php106
-rw-r--r--theme/base/css/display.css115
-rw-r--r--theme/identica/css/display.css5
23 files changed, 1304 insertions, 597 deletions
diff --git a/actions/favorited.php b/actions/favorited.php
index 0223564f3..4155b3a23 100644
--- a/actions/favorited.php
+++ b/actions/favorited.php
@@ -181,10 +181,9 @@ class FavoritedAction extends Action
$qry .= ' LIMIT ' . $offset . ', ' . $limit;
}
- // XXX: Figure out how to cache this query
-
- $notice = new Notice;
- $notice->query(sprintf($qry, common_config('popular', 'dropoff')));
+ $notice = Memcached_DataObject::cachedQuery('Notice',
+ sprintf($qry, common_config('popular', 'dropoff')),
+ 600);
$nl = new NoticeList($notice, $this);
diff --git a/actions/public.php b/actions/public.php
index 0ceeef98e..b51a95f24 100644
--- a/actions/public.php
+++ b/actions/public.php
@@ -197,4 +197,14 @@ class PublicAction extends Action
'version' => 'Atom 1.0',
'item' => 'publicatom')));
}
+
+ function showSections()
+ {
+ $top = new TopPostersSection($this);
+ $top->show();
+ $pop = new PopularNoticeSection($this);
+ $pop->show();
+ $gbp = new GroupsByPostsSection($this);
+ $gbp->show();
+ }
}
diff --git a/actions/subscribers.php b/actions/subscribers.php
index 31d0468d9..408829b54 100644
--- a/actions/subscribers.php
+++ b/actions/subscribers.php
@@ -1,9 +1,12 @@
<?php
-/*
- * Laconica - a distributed open-source microblogging tool
- * Copyright (C) 2008, Controlez-Vous, Inc.
+/**
+ * Laconica, the distributed open-source microblogging tool
*
- * This program is free software: you can redistribute it and/or modify
+ * List a user's subscribers
+ *
+ * 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.
@@ -15,56 +18,85 @@
*
* 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 Social
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @author Sarven Capadisli <csarven@controlyourself.ca>
+ * @copyright 2008-2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
*/
-if (!defined('LACONICA')) { exit(1); }
+if (!defined('LACONICA')) {
+ exit(1);
+}
-require_once(INSTALLDIR.'/lib/gallery.php');
+/**
+ * List a user's subscribers
+ *
+ * @category Social
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
class SubscribersAction extends GalleryAction
{
-
- function gallery_type()
+ function title()
{
- return _('Subscribers');
+ if ($this->page == 1) {
+ return sprintf(_('%s subscribers'), $this->user->nickname);
+ } else {
+ return sprintf(_('%s subscribers, page %d'),
+ $this->user->nickname,
+ $this->page);
+ }
}
- function get_instructions(&$profile)
+ function showPageNotice()
{
$user =& common_current_user();
- if ($user && ($user->id == $profile->id)) {
- return _('These are the people who listen to your notices.');
+ if ($user && ($user->id == $this->profile->id)) {
+ $this->element('p', null,
+ _('These are the people who listen to '.
+ 'your notices.'));
} else {
- return sprintf(_('These are the people who listen to %s\'s notices.'), $profile->nickname);
+ $this->element('p', null,
+ sprintf(_('These are the people who '.
+ 'listen to %s\'s notices.'),
+ $this->profile->nickname));
}
}
- function fields()
+ function showContent()
{
- return array('subscriber', 'subscribed');
- }
+ $offset = ($this->page-1) * PROFILES_PER_PAGE;
+ $limit = PROFILES_PER_PAGE + 1;
- function div_class()
- {
- return 'subscribers';
- }
+ $subscribers = $this->user->getSubscribers($offset, $limit);
- function get_other(&$subs)
- {
- return $subs->subscriber;
- }
+ if ($subscribers) {
+ $subscribers_list = new SubscribersList($subscribers, $this->user, $this);
+ $subscribers_list->show();
+ }
- function profile_list_class()
- {
- return 'SubscribersList';
+ $subscribers->free();
+
+ $this->pagination($this->page > 1, $cnt > PROFILES_PER_PAGE,
+ $this->page, 'subscribers',
+ array('nickname' => $this->user->nickname));
}
}
class SubscribersList extends ProfileList
{
- function show_owner_controls($profile)
+ function showOwnerControls($profile)
{
- common_block_form($profile, array('action' => 'subscribers',
- 'nickname' => $this->owner->nickname));
+ $bf = new BlockForm($this->out, $profile,
+ array('action' => 'subscribers',
+ 'nickname' => $this->owner->nickname));
+ $bf->show();
}
}
diff --git a/actions/subscriptions.php b/actions/subscriptions.php
index 7a87a144f..bcc557891 100644
--- a/actions/subscriptions.php
+++ b/actions/subscriptions.php
@@ -1,9 +1,12 @@
<?php
-/*
- * Laconica - a distributed open-source microblogging tool
- * Copyright (C) 2008, Controlez-Vous, Inc.
+/**
+ * Laconica, the distributed open-source microblogging tool
*
- * This program is free software: you can redistribute it and/or modify
+ * List of a user's subscriptions
+ *
+ * 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.
@@ -15,73 +18,107 @@
*
* 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 Social
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @author Sarven Capadisli <csarven@controlyourself.ca>
+ * @copyright 2008-2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
*/
-if (!defined('LACONICA')) { exit(1); }
+if (!defined('LACONICA')) {
+ exit(1);
+}
-require_once(INSTALLDIR.'/lib/gallery.php');
+/**
+ * A list of the user's subscriptions
+ *
+ * @category Social
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+if (!defined('LACONICA')) { exit(1); }
class SubscriptionsAction extends GalleryAction
{
-
- function gallery_type()
+ function title()
{
- return _('Subscriptions');
+ if ($this->page == 1) {
+ return sprintf(_('%s subscriptions'), $this->user->nickname);
+ } else {
+ return sprintf(_('%s subscriptions, page %d'),
+ $this->user->nickname,
+ $this->page);
+ }
}
- function get_instructions(&$profile)
+ function showPageNotice()
{
$user =& common_current_user();
- if ($user && ($user->id == $profile->id)) {
- return _('These are the people whose notices you listen to.');
+ if ($user && ($user->id == $this->profile->id)) {
+ $this->element('p', null,
+ _('These are the people whose notices '.
+ 'you listen to.'));
} else {
- return sprintf(_('These are the people whose notices %s listens to.'), $profile->nickname);
+ $this->element('p', null,
+ sprintf(_('These are the people whose '.
+ 'notices %s listens to.'),
+ $this->profile->nickname));
}
}
- function fields()
+ function getAllTags()
{
- return array('subscribed', 'subscriber');
+ return $this->getTags('subscribed', 'subscriber');
}
- function div_class()
+ function showContent()
{
- return 'subscriptions';
- }
+ parent::showContent();
- function get_other(&$subs)
- {
- return $subs->subscribed;
- }
+ $offset = ($this->page-1) * PROFILES_PER_PAGE;
+ $limit = PROFILES_PER_PAGE + 1;
- function profile_list_class()
- {
- return 'SubscriptionsList';
+ $subscriptions = $this->user->getSubscriptions($offset, $limit);
+
+ if ($subscriptions) {
+ $subscriptions_list = new SubscriptionsList($subscriptions, $this->user, $this);
+ $subscriptions_list->show();
+ }
+
+ $subscriptions->free();
+
+ $this->pagination($this->page > 1, $cnt > PROFILES_PER_PAGE,
+ $this->page, 'subscriptions',
+ array('nickname' => $this->user->nickname));
}
}
class SubscriptionsList extends ProfileList
{
-
- function show_owner_controls($profile)
+ function showOwnerControls($profile)
{
-
$sub = Subscription::pkeyGet(array('subscriber' => $this->owner->id,
'subscribed' => $profile->id));
if (!$sub) {
return;
}
- $this->elementStart('form', array('id' => 'subedit-' . $profile->id,
- 'method' => 'post',
- 'class' => 'subedit',
- 'action' => common_local_url('subedit')));
- $this->hidden('token', common_session_token());
- $this->hidden('profile', $profile->id);
- $this->checkbox('jabber', _('Jabber'), $sub->jabber);
- $this->checkbox('sms', _('SMS'), $sub->sms);
- $this->submit('save', _('Save'));
- $this->elementEnd('form');
+ $this->out->elementStart('form', array('id' => 'subedit-' . $profile->id,
+ 'method' => 'post',
+ 'class' => 'form_subcription_edit',
+ 'action' => common_local_url('subedit')));
+ $this->out->hidden('token', common_session_token());
+ $this->out->hidden('profile', $profile->id);
+ $this->out->checkbox('jabber', _('Jabber'), $sub->jabber);
+ $this->out->checkbox('sms', _('SMS'), $sub->sms);
+ $this->out->submit('save', _('Save'));
+ $this->out->elementEnd('form');
return;
}
}
diff --git a/actions/usergroups.php b/actions/usergroups.php
index 62ad3b3a2..f56f9c6b2 100644
--- a/actions/usergroups.php
+++ b/actions/usergroups.php
@@ -108,7 +108,8 @@ class UsergroupsAction extends Action
function showLocalNav()
{
- // XXX: Add to the subscriptions tabset
+ $nav = new SubGroupNav($this, $this->user);
+ $nav->show();
}
function showContent()
diff --git a/classes/Memcached_DataObject.php b/classes/Memcached_DataObject.php
index b9f599dbc..97e1ed736 100644
--- a/classes/Memcached_DataObject.php
+++ b/classes/Memcached_DataObject.php
@@ -21,7 +21,7 @@ if (!defined('LACONICA')) { exit(1); }
require_once INSTALLDIR.'/classes/Memcached_DataObject.php';
-class Memcached_DataObject extends DB_DataObject
+class Memcached_DataObject extends DB_DataObject
{
function &staticGet($cls, $k, $v=null)
{
@@ -69,7 +69,7 @@ class Memcached_DataObject extends DB_DataObject
$result = parent::insert();
return $result;
}
-
+
function update($orig=null)
{
if (is_object($orig) && $orig instanceof Memcached_DataObject) {
@@ -81,21 +81,21 @@ class Memcached_DataObject extends DB_DataObject
}
return $result;
}
-
+
function delete()
{
$this->decache(); # while we still have the values!
return parent::delete();
}
-
+
static function memcache() {
return common_memcache();
}
-
+
static function cacheKey($cls, $k, $v) {
return common_cache_key(strtolower($cls).':'.$k.':'.$v);
}
-
+
static function getcached($cls, $k, $v) {
$c = Memcached_DataObject::memcache();
if (!$c) {
@@ -114,7 +114,7 @@ class Memcached_DataObject extends DB_DataObject
}
return $_DB_DATAOBJECT['INI'][$this->_database][$this->__table."__keys"];
}
-
+
function encache()
{
$c = $this->memcache();
@@ -122,7 +122,7 @@ class Memcached_DataObject extends DB_DataObject
return false;
} else {
$pkey = array();
- $pval = array();
+ $pval = array();
$types = $this->keyTypes();
ksort($types);
foreach ($types as $key => $type) {
@@ -139,7 +139,7 @@ class Memcached_DataObject extends DB_DataObject
$c->set($this->cacheKey($this->tableName(), $pkeys, $pvals), $this);
}
}
-
+
function decache()
{
$c = $this->memcache();
@@ -147,7 +147,7 @@ class Memcached_DataObject extends DB_DataObject
return false;
} else {
$pkey = array();
- $pval = array();
+ $pval = array();
$types = $this->keyTypes();
ksort($types);
foreach ($types as $key => $type) {
@@ -201,4 +201,33 @@ class Memcached_DataObject extends DB_DataObject
}
return $search_engine;
}
+
+ static function cachedQuery($cls, $qry, $expiry=3600)
+ {
+ $c = Memcached_DataObject::memcache();
+ if (!$c) {
+ $inst = new $cls();
+ $inst->query($qry);
+ return $inst;
+ }
+ $key_part = common_keyize($cls).':'.md5($qry);
+ $ckey = common_cache_key($key_part);
+ $stored = $c->get($ckey);
+ if ($stored) {
+ return new ArrayWrapper($stored);
+ }
+
+ $inst = new $cls();
+ $result = $inst->query($qry);
+ if (!$result) {
+ return $inst;
+ }
+ $cached = array();
+ while ($inst->fetch()) {
+ $cached[] = clone($inst);
+ }
+ $inst->free();
+ $c->set($ckey, $cached, MEMCACHE_COMPRESSED, $expiry);
+ return new ArrayWrapper($cached);
+ }
}
diff --git a/classes/Notice.php b/classes/Notice.php
index 5fa078205..de7540705 100644
--- a/classes/Notice.php
+++ b/classes/Notice.php
@@ -443,7 +443,7 @@ class Notice extends Memcached_DataObject
# On a cache hit, return a DB-object-like wrapper
if ($notices !== false) {
- $wrapper = new NoticeWrapper(array_slice($notices, $offset, $limit));
+ $wrapper = new ArrayWrapper(array_slice($notices, $offset, $limit));
return $wrapper;
}
@@ -483,7 +483,7 @@ class Notice extends Memcached_DataObject
# return a wrapper of the array for use now
- return new NoticeWrapper(array_slice($notices, $offset, $limit));
+ return new ArrayWrapper(array_slice($notices, $offset, $limit));
}
}
@@ -514,7 +514,7 @@ class Notice extends Memcached_DataObject
# return a wrapper of the array for use now
- $wrapper = new NoticeWrapper(array_slice($notices, $offset, $limit));
+ $wrapper = new ArrayWrapper(array_slice($notices, $offset, $limit));
return $wrapper;
}
diff --git a/classes/NoticeWrapper.php b/classes/NoticeWrapper.php
deleted file mode 100644
index 233340ccd..000000000
--- a/classes/NoticeWrapper.php
+++ /dev/null
@@ -1,62 +0,0 @@
-<?php
-/*
- * Laconica - a distributed open-source microblogging tool
- * Copyright (C) 2008, Controlez-Vous, Inc.
- *
- * 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/>.
- */
-
-if (!defined('LACONICA')) { exit(1); }
-
-require_once(INSTALLDIR.'/classes/Notice.php');
-
-class NoticeWrapper extends Notice
-{
-
- public $id; // int(4) primary_key not_null
- public $profile_id; // int(4) not_null
- public $uri; // varchar(255) unique_key
- public $content; // varchar(140)
- public $rendered; // text()
- public $url; // varchar(255)
- public $created; // datetime() not_null
- public $modified; // timestamp() not_null default_CURRENT_TIMESTAMP
- public $reply_to; // int(4)
- public $is_local; // tinyint(1)
- public $source; // varchar(32)
-
- var $notices = null;
- var $i = -1;
-
- function __construct($arr)
- {
- $this->notices = $arr;
- }
-
- function fetch()
- {
- static $fields = array('id', 'profile_id', 'uri', 'content', 'rendered',
- 'url', 'created', 'modified', 'reply_to', 'is_local', 'source');
- $this->i++;
- if ($this->i >= count($this->notices)) {
- return false;
- } else {
- $n = $this->notices[$this->i];
- foreach ($fields as $f) {
- $this->$f = $n->$f;
- }
- return true;
- }
- }
-} \ No newline at end of file
diff --git a/lib/arraywrapper.php b/lib/arraywrapper.php
new file mode 100644
index 000000000..ef0eeffa5
--- /dev/null
+++ b/lib/arraywrapper.php
@@ -0,0 +1,79 @@
+<?php
+/*
+ * Laconica - a distributed open-source microblogging tool
+ * Copyright (C) 2008, Controlez-Vous, Inc.
+ *
+ * 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/>.
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+class ArrayWrapper
+{
+ var $_items = null;
+ var $_count = 0;
+ var $_i = -1;
+
+ function __construct($items)
+ {
+ $this->_items = $items;
+ $this->_count = count($this->_items);
+ }
+
+ function fetch()
+ {
+ if (!$this->_items) {
+ return false;
+ }
+ $this->_i++;
+ if ($this->_i < $this->_count) {
+ return true;
+ } else {
+ return false;
+ }
+ }
+
+ function __set($name, $value)
+ {
+ $item =& $this->_items[$this->_i];
+ $item->$name = $value;
+ return $item->$name;
+ }
+
+ function __get($name)
+ {
+ $item =& $this->_items[$this->_i];
+ return $item->$name;
+ }
+
+ function __isset($name)
+ {
+ $item =& $this->_items[$this->_i];
+ return isset($item->$name);
+ }
+
+ function __unset($name)
+ {
+ $item =& $this->_items[$this->_i];
+ unset($item->$name);
+ }
+
+ function __call($name, $args)
+ {
+ $item =& $this->_items[$this->_i];
+ return call_user_func_array(array($item, $name), $args);
+ }
+} \ No newline at end of file
diff --git a/lib/gallery.php b/lib/gallery.php
deleted file mode 100644
index 34b58518c..000000000
--- a/lib/gallery.php
+++ /dev/null
@@ -1,346 +0,0 @@
-<?php
-/**
- * Laconica - a distributed open-source microblogging tool
- * Copyright (C) 2008, Controlez-Vous, Inc.
- *
- * 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/>.
- */
-
-if (!defined('LACONICA')) {
- exit(1);
-}
-
-require_once INSTALLDIR.'/lib/profilelist.php';
-
-// 10x8
-
-define('AVATARS_PER_PAGE', 80);
-
-class GalleryAction extends Action
-{
- function is_readonly()
- {
- return true;
- }
-
- function handle($args)
- {
- parent::handle($args);
-
- // Post from the tag dropdown; redirect to a GET
-
- if ($_SERVER['REQUEST_METHOD'] == 'POST') {
- common_redirect($this->self_url(), 307);
- }
-
- $nickname = common_canonical_nickname($this->arg('nickname'));
-
- $user = User::staticGet('nickname', $nickname);
-
- if (!$user) {
- $this->no_such_user();
- return;
- }
-
- $profile = $user->getProfile();
-
- if (!$profile) {
- $this->server_error(_('User without matching profile in system.'));
- return;
- }
-
- $page = $this->arg('page');
-
- if (!$page) {
- $page = 1;
- }
-
- $display = $this->arg('display');
-
- if (!$display) {
- $display = 'list';
- }
-
- $tag = $this->arg('tag');
-
- common_show_header($profile->nickname . ": " . $this->gallery_type(),
- null, $profile,
- array($this, 'show_top'));
-
- $this->display_links($profile, $page, $display);
- $this->show_tags_dropdown($profile);
-
- $this->show_gallery($profile, $page, $display, $tag);
- common_show_footer();
- }
-
- function no_such_user()
- {
- $this->client_error(_('No such user.'));
- }
-
- function show_tags_dropdown($profile)
- {
- $tag = $this->trimmed('tag');
-
- list($lst, $usr) = $this->fields();
-
- $tags = $this->get_all_tags($profile, $lst, $usr);
-
- $content = array();
-
- foreach ($tags as $t) {
- $content[$t] = $t;
- }
- if ($tags) {
- common_element_start('dl', array('id'=>'filter_tags'));
- common_element('dt', null, _('Filter tags'));
- common_element_start('dd');
- common_element_start('ul');
- common_element_start('li', array('id' => 'filter_tags_all',
- 'class' => 'child_1'));
- common_element('a',
- array('href' =>
- common_local_url($this->trimmed('action'),
- array('nickname' =>
- $profile->nickname))),
- _('All'));
- common_element_end('li');
- common_element_start('li', array('id'=>'filter_tags_item'));
- common_element_start('form', array('name' => 'bytag',
- 'id' => 'bytag',
- 'method' => 'post'));
- common_dropdown('tag', _('Tag'), $content,
- _('Choose a tag to narrow list'), false, $tag);
- common_submit('go', _('Go'));
- common_element_end('form');
- common_element_end('li');
- common_element_end('ul');
- common_element_end('dd');
- common_element_end('dl');
- }
- }
-
- function show_top($profile)
- {
- common_element('div', 'instructions',
- $this->get_instructions($profile));
- $this->show_menu();
- }
-
- function show_menu()
- {
- // action => array('prompt', 'title', $args)
- $action = $this->trimmed('action');
- $nickname = $this->trimmed('nickname');
-
- $menu =
- array('subscriptions' =>
- array( _('Subscriptions'),
- _('Subscriptions'),
- array('nickname' => $nickname)),
- 'subscribers' =>
- array(
- _('Subscribers'),
- _('Subscribers'),
- array('nickname' => $nickname)),
- );
- $this->nav_menu($menu);
- }
-
- function show_gallery($profile, $page, $display='list', $tag=null)
- {
- $other = new Profile();
-
- list($lst, $usr) = $this->fields();
-
- $per_page = ($display == 'list') ? PROFILES_PER_PAGE : AVATARS_PER_PAGE;
-
- $offset = ($page-1)*$per_page;
- $limit = $per_page + 1;
-
- if (common_config('db', 'type') == 'pgsql') {
- $lim = ' LIMIT ' . $limit . ' OFFSET ' . $offset;
- } else {
- $lim = ' LIMIT ' . $offset . ', ' . $limit;
- }
-
- // XXX: memcached results
- // FIXME: SQL injection on $tag
-
- $other->query('SELECT profile.* ' .
- 'FROM profile JOIN subscription ' .
- 'ON profile.id = subscription.' . $lst . ' ' .
- (($tag) ? 'JOIN profile_tag ON (profile.id = profile_tag.tagged AND subscription.'.$usr.'= profile_tag.tagger) ' : '') .
- 'WHERE ' . $usr . ' = ' . $profile->id . ' ' .
- 'AND subscriber != subscribed ' .
- (($tag) ? 'AND profile_tag.tag= "' . $tag . '" ': '') .
- 'ORDER BY subscription.created DESC, profile.id DESC ' .
- $lim);
-
- if ($display == 'list') {
- $cls = $this->profile_list_class();
- $profile_list = new $cls($other, $profile, $this->trimmed('action'));
- $cnt = $profile_list->show_list();
- } else {
- $cnt = $this->icon_list($other);
- }
-
- // For building the pagination URLs
-
- $args = array('nickname' => $profile->nickname);
-
- if ($display != 'list') {
- $args['display'] = $display;
- }
-
- common_pagination($page > 1,
- $cnt > $per_page,
- $page,
- $this->trimmed('action'),
- $args);
- }
-
- function profile_list_class()
- {
- return 'ProfileList';
- }
-
- function icon_list($other)
- {
- common_element_start('ul', $this->div_class());
-
- $cnt = 0;
-
- while ($other->fetch()) {
-
- $cnt++;
-
- if ($cnt > AVATARS_PER_PAGE) {
- break;
- }
-
- common_element_start('li');
-
- common_element_start('a', array('title' => ($other->fullname) ?
- $other->fullname :
- $other->nickname,
- 'href' => $other->profileurl,
- 'class' => 'subscription'));
- $avatar = $other->getAvatar(AVATAR_STREAM_SIZE);
- common_element('img',
- array('src' =>
- (($avatar) ? common_avatar_display_url($avatar) :
- common_default_avatar(AVATAR_STREAM_SIZE)),
- 'width' => AVATAR_STREAM_SIZE,
- 'height' => AVATAR_STREAM_SIZE,
- 'class' => 'avatar stream',
- 'alt' => ($other->fullname) ?
- $other->fullname :
- $other->nickname));
- common_element_end('a');
-
- // XXX: subscribe form here
-
- common_element_end('li');
- }
-
- common_element_end('ul');
-
- return $cnt;
- }
-
- function gallery_type()
- {
- return null;
- }
-
- function get_instructions(&$profile)
- {
- return null;
- }
-
- function fields()
- {
- return null;
- }
-
- function div_class()
- {
- return '';
- }
-
- function display_links($profile, $page, $display)
- {
- $tag = $this->trimmed('tag');
-
- common_element_start('dl', array('id'=>'subscriptions_nav'));
- common_element('dt', null, _('Subscriptions navigation'));
- common_element_start('dd');
- common_element_start('ul', array('class'=>'nav'));
-
- switch ($display) {
- case 'list':
- common_element('li', array('class'=>'child_1'), _('List'));
- common_element_start('li');
- $url_args = array('display' => 'icons',
- 'nickname' => $profile->nickname,
- 'page' => 1 + floor((($page - 1) * PROFILES_PER_PAGE) / AVATARS_PER_PAGE));
- if ($tag) {
- $url_args['tag'] = $tag;
- }
- $url = common_local_url($this->trimmed('action'), $url_args);
- common_element('a', array('href' => $url),
- _('Icons'));
- common_element_end('li');
- break;
- default:
- common_element_start('li', array('class'=>'child_1'));
- $url_args = array('nickname' => $profile->nickname,
- 'page' => 1 + floor((($page - 1) * AVATARS_PER_PAGE) / PROFILES_PER_PAGE));
- if ($tag) {
- $url_args['tag'] = $tag;
- }
- $url = common_local_url($this->trimmed('action'), $url_args);
- common_element('a', array('href' => $url),
- _('List'));
- common_element_end('li');
- common_element('li', null, _('Icons'));
- break;
- }
-
- common_element_end('ul');
- common_element_end('dd');
- common_element_end('dl');
- }
-
- // Get list of tags we tagged other users with
-
- function get_all_tags($profile, $lst, $usr)
- {
- $profile_tag = new Notice_tag();
- $profile_tag->query('SELECT DISTINCT(tag) ' .
- 'FROM profile_tag, subscription ' .
- 'WHERE tagger = ' . $profile->id . ' ' .
- 'AND ' . $usr . ' = ' . $profile->id . ' ' .
- 'AND ' . $lst . ' = tagged ' .
- 'AND tagger != tagged');
- $tags = array();
- while ($profile_tag->fetch()) {
- $tags[] = $profile_tag->tag;
- }
- $profile_tag->free();
- return $tags;
- }
-} \ No newline at end of file
diff --git a/lib/galleryaction.php b/lib/galleryaction.php
new file mode 100644
index 000000000..a277762a6
--- /dev/null
+++ b/lib/galleryaction.php
@@ -0,0 +1,168 @@
+<?php
+/**
+ * Laconica - a distributed open-source microblogging tool
+ * Copyright (C) 2008, Controlez-Vous, Inc.
+ *
+ * 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/>.
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+require_once INSTALLDIR.'/lib/profilelist.php';
+
+// 10x8
+
+define('AVATARS_PER_PAGE', 80);
+
+class GalleryAction extends Action
+{
+ var $profile = null;
+ var $user = null;
+ var $page = null;
+
+ function prepare($args)
+ {
+ parent::prepare($args);
+
+ // FIXME very similar code below
+
+ $nickname_arg = $this->arg('nickname');
+ $nickname = common_canonical_nickname($nickname_arg);
+
+ // Permanent redirect on non-canonical nickname
+
+ if ($nickname_arg != $nickname) {
+ $args = array('nickname' => $nickname);
+ if ($this->arg('page') && $this->arg('page') != 1) {
+ $args['page'] = $this->arg['page'];
+ }
+ common_redirect(common_local_url('subscriptions', $args), 301);
+ return false;
+ }
+
+ $this->user = User::staticGet('nickname', $nickname);
+
+ if (!$this->user) {
+ $this->clientError(_('No such user.'), 404);
+ return false;
+ }
+
+ $this->profile = $this->user->getProfile();
+
+ if (!$this->profile) {
+ $this->serverError(_('User has no profile.'));
+ return false;
+ }
+
+ $this->page = ($this->arg('page')) ? ($this->arg('page')+0) : 1;
+
+ return true;
+ }
+
+ function isReadOnly()
+ {
+ return true;
+ }
+
+ function handle($args)
+ {
+ parent::handle($args);
+
+ # Post from the tag dropdown; redirect to a GET
+
+ if ($_SERVER['REQUEST_METHOD'] == 'POST') {
+ common_redirect($this->self_url(), 307);
+ return;
+ }
+
+ $this->showPage();
+ }
+
+ function showLocalNav()
+ {
+ $nav = new SubGroupNav($this, $this->user);
+ $nav->show();
+ }
+
+ function showContent()
+ {
+ $this->showTagsDropdown();
+ }
+
+ function showTagsDropdown()
+ {
+ $tag = $this->trimmed('tag');
+
+ $tags = $this->getAllTags();
+
+ $content = array();
+
+ foreach ($tags as $t) {
+ $content[$t] = $t;
+ }
+ if ($tags) {
+ $this->elementStart('dl', array('id'=>'filter_tags'));
+ $this->element('dt', null, _('Filter tags'));
+ $this->elementStart('dd');
+ $this->elementStart('ul');
+ $this->elementStart('li', array('id' => 'filter_tags_all',
+ 'class' => 'child_1'));
+ $this->element('a',
+ array('href' =>
+ common_local_url($this->trimmed('action'),
+ array('nickname' =>
+ $profile->nickname))),
+ _('All'));
+ $this->elementEnd('li');
+ $this->elementStart('li', array('id'=>'filter_tags_item'));
+ $this->elementStart('form', array('name' => 'bytag',
+ 'id' => 'bytag',
+ 'method' => 'post'));
+ $this->dropdown('tag', _('Tag'), $content,
+ _('Choose a tag to narrow list'), false, $tag);
+ $this->submit('go', _('Go'));
+ $this->elementEnd('form');
+ $this->elementEnd('li');
+ $this->elementEnd('ul');
+ $this->elementEnd('dd');
+ $this->elementEnd('dl');
+ }
+ }
+
+ // Get list of tags we tagged other users with
+
+ function getTags($lst, $usr)
+ {
+ $profile_tag = new Notice_tag();
+ $profile_tag->query('SELECT DISTINCT(tag) ' .
+ 'FROM profile_tag, subscription ' .
+ 'WHERE tagger = ' . $this->profile->id . ' ' .
+ 'AND ' . $usr . ' = ' . $this->profile->id . ' ' .
+ 'AND ' . $lst . ' = tagged ' .
+ 'AND tagger != tagged');
+ $tags = array();
+ while ($profile_tag->fetch()) {
+ $tags[] = $profile_tag->tag;
+ }
+ $profile_tag->free();
+ return $tags;
+ }
+
+ function getAllTags()
+ {
+ return array();
+ }
+} \ No newline at end of file
diff --git a/lib/grouplist.php b/lib/grouplist.php
index dd10a2753..869e44897 100644
--- a/lib/grouplist.php
+++ b/lib/grouplist.php
@@ -66,7 +66,7 @@ class GroupList extends Widget
function show()
{
- $this->out->elementStart('ul', 'groups');
+ $this->out->elementStart('ul', 'profiles groups xoxo');
$cnt = 0;
@@ -85,19 +85,19 @@ class GroupList extends Widget
function showGroup()
{
- $this->out->elementStart('li', array('class' => 'group',
+ $this->out->elementStart('li', array('class' => 'profile',
'id' => 'group-' . $this->group->id));
$user = common_current_user();
- $this->out->elementStart('div', array('id' => 'group_group',
- 'class' => 'vcard'));
+ $this->out->elementStart('div', 'entity_profile vcard');
$logo = ($this->group->stream_logo) ?
$this->group->stream_logo : User_group::defaultLogo(AVATAR_STREAM_SIZE);
$this->out->elementStart('a', array('href' => $this->group->homeUrl(),
- 'class' => 'url'));
+ 'class' => 'url',
+ 'rel' => 'group'));
$this->out->element('img', array('src' => $logo,
'class' => 'photo avatar',
'width' => AVATAR_STREAM_SIZE,
@@ -105,24 +105,24 @@ class GroupList extends Widget
'alt' =>
($this->group->fullname) ? $this->group->fullname :
$this->group->nickname));
- $hasFN = ($this->group->fullname) ? 'nickname' : 'fn nickname';
+ $hasFN = ($this->group->fullname) ? 'nickname url uid' : 'fn org nickname url uid';
$this->out->elementStart('span', $hasFN);
$this->out->raw($this->highlight($this->group->nickname));
$this->out->elementEnd('span');
$this->out->elementEnd('a');
if ($this->group->fullname) {
- $this->out->elementStart('dl', 'group_fn');
+ $this->out->elementStart('dl', 'entity_fn');
$this->out->element('dt', null, 'Full name');
$this->out->elementStart('dd');
- $this->out->elementStart('span', 'fn');
+ $this->out->elementStart('span', 'fn org');
$this->out->raw($this->highlight($this->group->fullname));
$this->out->elementEnd('span');
$this->out->elementEnd('dd');
$this->out->elementEnd('dl');
}
if ($this->group->location) {
- $this->out->elementStart('dl', 'group_location');
+ $this->out->elementStart('dl', 'entity_location');
$this->out->element('dt', null, _('Location'));
$this->out->elementStart('dd', 'location');
$this->out->raw($this->highlight($this->group->location));
@@ -130,7 +130,7 @@ class GroupList extends Widget
$this->out->elementEnd('dl');
}
if ($this->group->homepage) {
- $this->out->elementStart('dl', 'group_url');
+ $this->out->elementStart('dl', 'entity_url');
$this->out->element('dt', null, _('URL'));
$this->out->elementStart('dd');
$this->out->elementStart('a', array('href' => $this->group->homepage,
@@ -141,7 +141,7 @@ class GroupList extends Widget
$this->out->elementEnd('dl');
}
if ($this->group->description) {
- $this->out->elementStart('dl', 'group_note');
+ $this->out->elementStart('dl', 'entity_note');
$this->out->element('dt', null, _('Note'));
$this->out->elementStart('dd', 'note');
$this->out->raw($this->highlight($this->group->description));
diff --git a/lib/groupsbypostssection.php b/lib/groupsbypostssection.php
new file mode 100644
index 000000000..a5e33a93d
--- /dev/null
+++ b/lib/groupsbypostssection.php
@@ -0,0 +1,78 @@
+<?php
+/**
+ * Laconica, the distributed open-source microblogging tool
+ *
+ * Groups with the most posts section
+ *
+ * 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 Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @copyright 2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+/**
+ * Groups with the most posts section
+ *
+ * @category Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+class GroupsByPostsSection extends GroupSection
+{
+ function getGroups()
+ {
+ $qry = 'SELECT user_group.*, count(*) as value ' .
+ 'FROM user_group JOIN group_inbox '.
+ 'ON user_group.id = group_inbox.group_id ' .
+ 'GROUP BY user_group.id ' .
+ 'ORDER BY value DESC ';
+
+ $limit = GROUPS_PER_SECTION;
+ $offset = 0;
+
+ if (common_config('db','type') == 'pgsql') {
+ $qry .= ' LIMIT ' . $limit . ' OFFSET ' . $offset;
+ } else {
+ $qry .= ' LIMIT ' . $offset . ', ' . $limit;
+ }
+
+ $group = Memcached_DataObject::cachedQuery('User_group',
+ $qry,
+ 3600);
+ return $group;
+ }
+
+ function title()
+ {
+ return _('Groups with most posts');
+ }
+
+ function divId()
+ {
+ return 'top_groups_by_post';
+ }
+}
diff --git a/lib/groupsection.php b/lib/groupsection.php
new file mode 100644
index 000000000..bf26ab48c
--- /dev/null
+++ b/lib/groupsection.php
@@ -0,0 +1,103 @@
+<?php
+/**
+ * Laconica, the distributed open-source microblogging tool
+ *
+ * Base class for sections showing lists of groups
+ *
+ * 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 Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @copyright 2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+define('GROUPS_PER_SECTION', 6);
+
+/**
+ * Base class for sections
+ *
+ * These are the widgets that show interesting data about a person
+ * group, or site.
+ *
+ * @category Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+class GroupSection extends Section
+{
+ function showContent()
+ {
+ $profiles = $this->getGroups();
+
+ if (!$profiles) {
+ return false;
+ }
+
+ $cnt = 0;
+
+ $this->out->elementStart('ul', 'entities group xoxo');
+
+ while ($profiles->fetch() && ++$cnt <= GROUPS_PER_SECTION) {
+ $this->showGroup($profiles);
+ }
+
+ $this->out->elementEnd('ul');
+
+ return ($cnt > GROUPS_PER_SECTION);
+ }
+
+ function getGroups()
+ {
+ return null;
+ }
+
+ function showGroup($group)
+ {
+ $this->out->elementStart('li', 'vcard');
+ $this->out->elementStart('a', array('title' => ($group->fullname) ?
+ $group->fullname :
+ $group->nickname,
+ 'href' => $group->homeUrl(),
+ 'rel' => 'contact group',
+ 'class' => 'url'));
+ $logo = ($group->stream_logo) ?
+ $group->stream_logo : User_group::defaultLogo(AVATAR_STREAM_SIZE);
+
+ $this->out->element('img', array('src' => $logo,
+ 'width' => AVATAR_MINI_SIZE,
+ 'height' => AVATAR_MINI_SIZE,
+ 'class' => 'avatar photo',
+ 'alt' => ($group->fullname) ?
+ $group->fullname :
+ $group->nickname));
+ $this->out->element('span', 'fn org nickname', $group->nickname);
+ $this->out->elementEnd('a');
+ if ($group->value) {
+ $this->out->element('span', 'value', $group->value);
+ }
+ $this->out->elementEnd('li');
+ }
+}
diff --git a/lib/noticesection.php b/lib/noticesection.php
new file mode 100644
index 000000000..9d1079070
--- /dev/null
+++ b/lib/noticesection.php
@@ -0,0 +1,107 @@
+<?php
+/**
+ * Laconica, the distributed open-source microblogging tool
+ *
+ * Base class for sections showing lists of notices
+ *
+ * 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 Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @copyright 2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+define('NOTICES_PER_SECTION', 6);
+
+/**
+ * Base class for sections showing lists of notices
+ *
+ * These are the widgets that show interesting data about a person
+ * group, or site.
+ *
+ * @category Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+class NoticeSection extends Section
+{
+ function showContent()
+ {
+ $notices = $this->getNotices();
+
+ $cnt = 0;
+
+ $this->out->elementStart('table', 'notices');
+
+ while ($notices->fetch() && ++$cnt <= NOTICES_PER_SECTION) {
+ $this->showNotice($notices);
+ }
+
+ $this->out->elementEnd('table');
+
+ return ($cnt > NOTICES_PER_SECTION);
+ }
+
+ function getNotices()
+ {
+ return null;
+ }
+
+ function showNotice($notice)
+ {
+ $profile = $notice->getProfile();
+ $this->out->elementStart('tr');
+ $this->out->elementStart('td');
+ $avatar = $profile->getAvatar(AVATAR_MINI_SIZE);
+ $this->out->element('img', array('src' => (($avatar) ? common_avatar_display_url($avatar) : common_default_avatar(AVATAR_MINI_SIZE)),
+ 'width' => AVATAR_MINI_SIZE,
+ 'height' => AVATAR_MINI_SIZE,
+ 'class' => 'avatar photo',
+ 'alt' => ($profile->fullname) ?
+ $profile->fullname :
+ $profile->nickname));
+ $this->out->elementEnd('a');
+ $this->out->elementEnd('td');
+ $this->out->elementStart('td');
+ $this->out->elementStart('a', array('title' => ($profile->fullname) ?
+ $profile->fullname :
+ $profile->nickname,
+ 'href' => $profile->noticeurl,
+ 'rel' => 'contact member',
+ 'class' => 'url'));
+ $this->out->element('span', 'fn nickname', $profile->nickname);
+ $this->out->elementEnd('td');
+ $this->out->elementStart('td');
+ $this->out->raw($notice->rendered);
+ $this->out->elementEnd('td');
+ if ($notice->value) {
+ $this->out->elementStart('td');
+ $this->out->text($notice->value);
+ $this->out->elementEnd('td');
+ }
+ $this->out->elementEnd('tr');
+ }
+}
diff --git a/lib/popularnoticesection.php b/lib/popularnoticesection.php
new file mode 100644
index 000000000..89daaa563
--- /dev/null
+++ b/lib/popularnoticesection.php
@@ -0,0 +1,83 @@
+<?php
+/**
+ * Laconica, the distributed open-source microblogging tool
+ *
+ * Base class for sections showing lists of notices
+ *
+ * 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 Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @copyright 2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+define('NOTICES_PER_SECTION', 6);
+
+/**
+ * Base class for sections showing lists of notices
+ *
+ * These are the widgets that show interesting data about a person
+ * group, or site.
+ *
+ * @category Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+class PopularNoticeSection extends NoticeSection
+{
+ function getNotices()
+ {
+ $qry = 'SELECT notice.*, '.
+ 'sum(exp(-(now() - fave.modified) / %s)) as weight ' .
+ 'FROM notice JOIN fave ON notice.id = fave.notice_id ' .
+ 'GROUP BY fave.notice_id ' .
+ 'ORDER BY weight DESC';
+
+ $offset = 0;
+ $limit = NOTICES_PER_SECTION + 1;
+
+ if (common_config('db', 'type') == 'pgsql') {
+ $qry .= ' LIMIT ' . $limit . ' OFFSET ' . $offset;
+ } else {
+ $qry .= ' LIMIT ' . $offset . ', ' . $limit;
+ }
+
+ $notice = Memcached_DataObject::cachedQuery('Notice',
+ sprintf($qry, common_config('popular', 'dropoff')),
+ 1200);
+ return $notice;
+ }
+
+ function title()
+ {
+ return _('Popular notices');
+ }
+
+ function divId()
+ {
+ return 'popular_notices';
+ }
+}
diff --git a/lib/profilelist.php b/lib/profilelist.php
index 73c129efe..a510c518c 100644
--- a/lib/profilelist.php
+++ b/lib/profilelist.php
@@ -92,25 +92,24 @@ class ProfileList extends Widget
$user = common_current_user();
-
$this->out->elementStart('div', 'entity_profile vcard');
$avatar = $this->profile->getAvatar(AVATAR_STREAM_SIZE);
$this->out->elementStart('a', array('href' => $this->profile->profileurl,
'class' => 'url'));
$this->out->element('img', array('src' => ($avatar) ? common_avatar_display_url($avatar) : common_default_avatar(AVATAR_STREAM_SIZE),
- 'class' => 'photo avatar',
- 'width' => AVATAR_STREAM_SIZE,
- 'height' => AVATAR_STREAM_SIZE,
- 'alt' =>
- ($this->profile->fullname) ? $this->profile->fullname :
- $this->profile->nickname));
+ 'class' => 'photo avatar',
+ 'width' => AVATAR_STREAM_SIZE,
+ 'height' => AVATAR_STREAM_SIZE,
+ 'alt' =>
+ ($this->profile->fullname) ? $this->profile->fullname :
+ $this->profile->nickname));
$hasFN = ($this->profile->fullname) ? 'nickname' : 'fn nickname';
$this->out->elementStart('span', $hasFN);
$this->out->raw($this->highlight($this->profile->nickname));
$this->out->elementEnd('span');
$this->out->elementEnd('a');
-
+
if ($this->profile->fullname) {
$this->out->elementStart('dl', 'entity_fn');
$this->out->element('dt', null, 'Full name');
@@ -159,8 +158,8 @@ class ProfileList extends Widget
$this->out->elementStart('dt');
if ($user->id == $this->owner->id) {
$this->out->element('a', array('href' => common_local_url('tagother',
- array('id' => $this->profile->id))),
- _('Tags'));
+ array('id' => $this->profile->id))),
+ _('Tags'));
} else {
$this->out->text(_('Tags'));
}
@@ -172,10 +171,10 @@ class ProfileList extends Widget
$this->out->elementStart('li');
$this->element('span', 'mark_hash', '#');
$this->out->element('a', array('rel' => 'tag',
- 'href' => common_local_url($this->action,
- array('nickname' => $this->owner->nickname,
- 'tag' => $tag))),
- $tag);
+ 'href' => common_local_url($this->action,
+ array('nickname' => $this->owner->nickname,
+ 'tag' => $tag))),
+ $tag);
$this->out->elementEnd('li');
}
$this->out->elementEnd('ul');
diff --git a/lib/profilesection.php b/lib/profilesection.php
new file mode 100644
index 000000000..14068b082
--- /dev/null
+++ b/lib/profilesection.php
@@ -0,0 +1,101 @@
+<?php
+/**
+ * Laconica, the distributed open-source microblogging tool
+ *
+ * Base class for sections showing lists of people
+ *
+ * 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 Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @copyright 2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+define('PROFILES_PER_SECTION', 6);
+
+/**
+ * Base class for sections
+ *
+ * These are the widgets that show interesting data about a person
+ * group, or site.
+ *
+ * @category Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+class ProfileSection extends Section
+{
+ function showContent()
+ {
+ $profiles = $this->getProfiles();
+
+ if (!$profiles) {
+ return false;
+ }
+
+ $cnt = 0;
+
+ $this->out->elementStart('ul', 'entities users xoxo');
+
+ while ($profiles->fetch() && ++$cnt <= PROFILES_PER_SECTION) {
+ $this->showProfile($profiles);
+ }
+
+ $this->out->elementEnd('ul');
+
+ return ($cnt > PROFILES_PER_SECTION);
+ }
+
+ function getProfiles()
+ {
+ return null;
+ }
+
+ function showProfile($profile)
+ {
+ $this->out->elementStart('li', 'vcard');
+ $this->out->elementStart('a', array('title' => ($profile->fullname) ?
+ $profile->fullname :
+ $profile->nickname,
+ 'href' => $profile->profileurl,
+ 'rel' => 'contact member',
+ 'class' => 'url'));
+ $avatar = $profile->getAvatar(AVATAR_MINI_SIZE);
+ $this->out->element('img', array('src' => (($avatar) ? common_avatar_display_url($avatar) : common_default_avatar(AVATAR_MINI_SIZE)),
+ 'width' => AVATAR_MINI_SIZE,
+ 'height' => AVATAR_MINI_SIZE,
+ 'class' => 'avatar photo',
+ 'alt' => ($profile->fullname) ?
+ $profile->fullname :
+ $profile->nickname));
+ $this->out->element('span', 'fn nickname', $profile->nickname);
+ $this->out->elementEnd('a');
+ if ($profile->value) {
+ $this->out->element('span', 'value', $profile->value);
+ }
+ $this->out->elementEnd('li');
+ }
+}
diff --git a/lib/section.php b/lib/section.php
new file mode 100644
index 000000000..0c32ddcf8
--- /dev/null
+++ b/lib/section.php
@@ -0,0 +1,108 @@
+<?php
+/**
+ * Laconica, the distributed open-source microblogging tool
+ *
+ * Base class for sections (sidebar widgets)
+ *
+ * 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 Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @copyright 2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+require_once INSTALLDIR.'/lib/widget.php';
+
+/**
+ * Base class for sections
+ *
+ * These are the widgets that show interesting data about a person
+ * group, or site.
+ *
+ * @category Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+class Section extends Widget
+{
+ /**
+ * Show the form
+ *
+ * Uses a recipe to output the form.
+ *
+ * @return void
+ * @see Widget::show()
+ */
+
+ function show()
+ {
+ $this->out->elementStart('div',
+ array('id' => $this->divId(),
+ 'class' => 'section'));
+
+ $this->out->element('h2', null,
+ $this->title());
+
+ $have_more = $this->showContent();
+
+ if ($have_more) {
+ $this->out->elementStart('p');
+ $this->out->element('a', array('href' => $this->moreUrl(),
+ 'class' => 'more'),
+ $this->moreTitle());
+ $this->out->elementEnd('p');
+ }
+
+ $this->out->elementEnd('div');
+ }
+
+ function divId()
+ {
+ return 'generic_section';
+ }
+
+ function title()
+ {
+ return _('Untitled section');
+ }
+
+ function showContent()
+ {
+ $this->out->element('p', null,
+ _('(None)'));
+ return false;
+ }
+
+ function moreUrl()
+ {
+ return null;
+ }
+
+ function moreTitle()
+ {
+ return null;
+ }
+}
diff --git a/lib/subgroupnav.php b/lib/subgroupnav.php
new file mode 100644
index 000000000..5fd8a72a2
--- /dev/null
+++ b/lib/subgroupnav.php
@@ -0,0 +1,111 @@
+<?php
+/**
+ * Laconica, the distributed open-source microblogging tool
+ *
+ * Local navigation for subscriptions group of pages
+ *
+ * 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 Subs
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @copyright 2008-2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+require_once INSTALLDIR.'/lib/widget.php';
+
+/**
+ * Local nav menu for subscriptions, subscribers
+ *
+ * @category Subs
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+class SubGroupNav extends Widget
+{
+ var $action = null;
+ var $user = null;
+
+ /**
+ * Construction
+ *
+ * @param Action $action current action, used for output
+ */
+
+ function __construct($action=null, $user=null)
+ {
+ parent::__construct($action);
+ $this->action = $action;
+ $this->user = $user;
+ }
+
+ /**
+ * Show the menu
+ *
+ * @return void
+ */
+
+ function show()
+ {
+ $cur = common_current_user();
+ $action = $this->action->trimmed('action');
+
+ $this->out->elementStart('ul', array('class' => 'nav'));
+
+ $this->out->menuItem(common_local_url('subscriptions',
+ array('nickname' =>
+ $this->user->nickname)),
+ _('Subscriptions'),
+ sprintf(_('People %s subscribes to'),
+ $this->user->nickname),
+ $action == 'subscriptions',
+ 'nav_subscriptions');
+ $this->out->menuItem(common_local_url('subscribers',
+ array('nickname' =>
+ $this->user->nickname)),
+ _('Subscribers'),
+ sprintf(_('People subscribed to %s'),
+ $this->user->nickname),
+ $action == 'subscribers',
+ 'nav_subscribers');
+ $this->out->menuItem(common_local_url('usergroups',
+ array('nickname' =>
+ $this->user->nickname)),
+ _('Groups'),
+ sprintf(_('Groups %s is a member of'),
+ $this->user->nickname),
+ $action == 'usergroups',
+ 'nav_usergroups');
+ if ($this->user->id == $cur->id) {
+ $this->out->menuItem(common_local_url('invite'),
+ _('Invite'),
+ sprintf(_('Invite friends and colleagues to join you on %s'),
+ common_config('site', 'name')),
+ $action == 'invite',
+ 'nav_invite');
+ }
+ $this->out->elementEnd('ul');
+ }
+}
diff --git a/lib/topposterssection.php b/lib/topposterssection.php
new file mode 100644
index 000000000..7a3d46aa5
--- /dev/null
+++ b/lib/topposterssection.php
@@ -0,0 +1,106 @@
+<?php
+/**
+ * Laconica, the distributed open-source microblogging tool
+ *
+ * Base class for sections showing lists of people
+ *
+ * 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 Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @copyright 2009 Control Yourself, Inc.
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+if (!defined('LACONICA')) {
+ exit(1);
+}
+
+/**
+ * Base class for sections
+ *
+ * These are the widgets that show interesting data about a person
+ * group, or site.
+ *
+ * @category Widget
+ * @package Laconica
+ * @author Evan Prodromou <evan@controlyourself.ca>
+ * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
+ * @link http://laconi.ca/
+ */
+
+class TopPostersSection extends ProfileSection
+{
+ function getProfiles()
+ {
+ $qry = 'SELECT profile.*, count(*) as value ' .
+ 'FROM profile JOIN notice ON profile.id = notice.profile_id ' .
+ (common_config('public', 'localonly') ? 'WHERE is_local = 1 ' : '') .
+ 'GROUP BY profile.id ' .
+ 'ORDER BY value DESC ';
+
+ $limit = PROFILES_PER_SECTION;
+ $offset = 0;
+
+ if (common_config('db','type') == 'pgsql') {
+ $qry .= ' LIMIT ' . $limit . ' OFFSET ' . $offset;
+ } else {
+ $qry .= ' LIMIT ' . $offset . ', ' . $limit;
+ }
+
+ $profile = Memcached_DataObject::cachedQuery('Profile',
+ $qry,
+ 6 * 3600);
+ return $profile;
+ }
+
+ function showProfile($profile)
+ {
+ $this->out->elementStart('li', 'vcard');
+ $this->out->elementStart('a', array('title' => ($profile->fullname) ?
+ $profile->fullname :
+ $profile->nickname,
+ 'href' => $profile->profileurl,
+ 'rel' => 'contact member',
+ 'class' => 'url'));
+ $avatar = $profile->getAvatar(AVATAR_MINI_SIZE);
+ $this->out->element('img', array('src' => (($avatar) ? common_avatar_display_url($avatar) : common_default_avatar(AVATAR_MINI_SIZE)),
+ 'width' => AVATAR_MINI_SIZE,
+ 'height' => AVATAR_MINI_SIZE,
+ 'class' => 'avatar photo',
+ 'alt' => ($profile->fullname) ?
+ $profile->fullname :
+ $profile->nickname));
+ $this->out->element('span', 'fn nickname', $profile->nickname);
+ $this->out->elementEnd('a');
+ if ($profile->value) {
+ $this->out->element('span', 'value', $profile->value);
+ }
+ $this->out->elementEnd('li');
+ }
+
+ function title()
+ {
+ return _('Top posters');
+ }
+
+ function divId()
+ {
+ return 'top_posters';
+ }
+}
diff --git a/theme/base/css/display.css b/theme/base/css/display.css
index 91789f4f1..44dbff5da 100644
--- a/theme/base/css/display.css
+++ b/theme/base/css/display.css
@@ -36,7 +36,7 @@ font-weight:bold;
}
legend {
-font-weight:bold;
+font-weight:bold;
font-size:1.3em;
text-transform:uppercase;
}
@@ -99,9 +99,6 @@ display:none;
margin-left:11px;
}
-
-
-
/* FORM SETTINGS */
.form_settings fieldset {
margin-bottom:29px;
@@ -111,7 +108,6 @@ margin-bottom:29px;
font-style:italic;
}
-
.form_settings .form_data li {
width:100%;
float:left;
@@ -179,7 +175,6 @@ width:90%;
margin-left:0;
}
-
.form_settings .form_note {
border-radius:4px;
-moz-border-radius:4px;
@@ -187,13 +182,8 @@ border-radius:4px;
padding:0 7px;
}
-
-
/* FORM SETTINGS */
-
-
-
address {
float:left;
margin-bottom:18px;
@@ -206,7 +196,6 @@ address .fn {
font-weight:bold;
}
-
#header {
width:100%;
position:relative;
@@ -225,7 +214,6 @@ display:inline;
margin-left:11px;
}
-
.system_notice dt {
font-weight:bold;
text-transform:uppercase;
@@ -244,17 +232,12 @@ clear:both;
margin-bottom:18px;
}
-
-
-
#footer {
float:left;
width:64%;
padding:18px;
}
-
-
#site_nav_local_views {
width:100%;
float:left;
@@ -286,7 +269,6 @@ float:left;
width:100%;
}
-
#site_nav_global_primary dt,
#site_nav_global_secondary dt {
display:none;
@@ -315,8 +297,6 @@ padding-left:30px;
padding-left:28px;
}
-
-
#export_data ul {
display:inline;
}
@@ -329,8 +309,6 @@ margin-left:11px;
margin-left:0;
}
-
-
#licenses {
font-size:0.9em;
}
@@ -353,7 +331,6 @@ vertical-align:top;
margin-right:4px;
}
-
#wrap {
float:left;
margin:0 auto;
@@ -361,7 +338,6 @@ margin-right:4px;
width:71.714em;
}
-
#core {
position:relative;
width:100%;
@@ -396,9 +372,6 @@ border-radius:7px;
-webkit-border-radius:7px;
}
-
-
-
/*Start: FORM NOTICE*/
#form_notice {
width:384px;
@@ -501,10 +474,6 @@ float:left;
/*end FORM NOTICE*/
-
-
-
-
/* entity_profile */
.entity_profile {
position:relative;
@@ -575,10 +544,6 @@ display:none;
/* entity_profile */
-
-
-
-
/*entity_actions*/
.entity_actions {
float:right;
@@ -617,8 +582,6 @@ font-weight:bold;
display:block;
}
-
-
.form_user_block input.submit,
.form_user_unblock input.submit,
#entity_send-a-message a,
@@ -633,8 +596,6 @@ padding-left:20px;
padding:4px 4px 4px 23px;
}
-
-
.entity_tags ul {
list-style-type:none;
}
@@ -644,8 +605,6 @@ margin-right:1em;
float:left;
}
-
-
.aside .section {
margin-bottom:29px;
clear:both;
@@ -665,7 +624,6 @@ display:inline;
content: ":";
}
-
#user_subscriptions,
#user_subscribers,
#user_groups, {
@@ -692,23 +650,25 @@ display:none;
clear:both;
}
-
-
.profile .entity_profile {
margin-bottom:0;
min-height:60px;
}
+
+.profile .form_group_join,
+.profile .form_group_leave,
.profile .form_user_subscribe,
.profile .form_user_unsubscribe {
float:right;
}
+.profile .form_group_join legend,
+.profile .form_group_leave legend,
.profile .form_user_subscribe legend,
.profile .form_user_unsubscribe legend {
display:none;
}
-
.profiles {
list-style-type:none;
}
@@ -723,12 +683,25 @@ display:inline;
float:none;
}
.profile .entity_profile .entity_note,
-.profile .entity_profile .entity_url {
+.profile .entity_profile .entity_url,
+.profile .entity_profile .entity_tags,
+.profile .entity_profile .form_subcription_edit {
margin-left:59px;
clear:none;
display:block;
width:auto;
}
+.profile .entity_profile .entity_tags dt {
+display:inline;
+margin-right:11px;
+}
+
+
+.profile .entity_profile .form_subcription_edit label {
+font-weight:normal;
+margin-right:11px;
+}
+
@@ -756,7 +729,6 @@ border-radius:4px;
-webkit-border-radius:4px;
}
-
/* NOTICES */
#notices_primary {
float:left;
@@ -773,7 +745,6 @@ display:block;
padding-left:28px;
}
-
.notice .author {
margin-right:11px;
}
@@ -783,6 +754,14 @@ margin-right:11px;
/*content:":";*/
}
+.notice .author .fn {
+font-weight:bold;
+}
+
+.notice .author .photo {
+margin-bottom:0;
+}
+
.vcard .photo {
display:inline;
margin-right:11px;
@@ -802,7 +781,6 @@ font-style:italic;
}
-
.notice .entry-title {
float:left;
width:100%;
@@ -836,7 +814,7 @@ border-radius:4px;
/*border:1px solid blue;*/
clear:left;
float:left;
-width:48%;
+/*width:48%;*/
font-size:0.95em;
}
.notice div.entry-content a,
@@ -859,6 +837,10 @@ font-size:1.025em;
.notice div.entry-content dd {
display:inline;
}
+.notice div.entry-content .timestamp {
+margin-left:59px;
+}
+
.notice div.entry-content .timestamp dt,
.notice div.entry-content .response dt {
display:none;
@@ -877,6 +859,9 @@ text-transform:lowercase;
+
+
+
.notice-data {
position:absolute;
top:18px;
@@ -899,6 +884,8 @@ float:left;
width:50%;
position:relative;
font-size:0.95em;
+width:12.5%;
+float:right;
}
.notice-options a {
@@ -966,10 +953,6 @@ padding:0;
/*END: NOTICES */
-
-
-
-
.pagination dt {
font-weight:bold;
display:none;
@@ -1010,11 +993,8 @@ padding-right:20px;
border-right:0;
}
-
/* END: NOTICE */
-
-
/*If there is hentry on #content_inner, then this doesn't need to be specific to #doc or any other section */
.hentry .entry-content p {
margin-bottom:18px;
@@ -1030,7 +1010,6 @@ margin-bottom:18px;
margin-left:18px;
}
-
/*START: LOAD ALONG WITH JS*/
.notice .in-reply-to {
width:98%;
@@ -1046,8 +1025,6 @@ background-color:#E4E9F0;
background-color:#D1D9E4;
}
-
-
.entity_actions #user_subscribe .form_note,
.entity_actions #user_subscribe .form_data,
.entity_actions #user_subscribe .form_actions label {
@@ -1079,8 +1056,6 @@ background-color:#fff;
/*END: LOAD ALONG WITH JS*/
-
-
/* TOP_POSTERS */
#top-posters caption {
text-align:left;
@@ -1106,8 +1081,6 @@ height:24px;
width:24px;
}
-
-
/* tagcloud */
.tag-cloud {
list-style-type:none;
@@ -1150,20 +1123,15 @@ font-size:4em;
display:none;
}
-
-
-
#form_settings_photo .form_data {
clear:both;
}
-
-
#form_settings_avatar li {
width:auto;
}
#form_settings_avatar input {
-margin-left:0;
+margin-left:0;
}
#avatar_original,
#avatar_preview {
@@ -1188,8 +1156,6 @@ clear:both;
margin-bottom:0;
}
-
-
#groups_related ul,
#users_featured ul {
list-style-type:none;
@@ -1223,8 +1189,6 @@ display:none;
font-weight:bold;
}
-
-
#home.logged_out h1 {
display:none;
}
@@ -1267,7 +1231,6 @@ font-weight:bold;
text-align:center;
}
-
#testimonials {
clear:both;
}
diff --git a/theme/identica/css/display.css b/theme/identica/css/display.css
index 1df14a7f1..5d07cfd9a 100644
--- a/theme/identica/css/display.css
+++ b/theme/identica/css/display.css
@@ -58,8 +58,7 @@ background-color:#fcfffc;
#aside_primary,
#entity_subscribe a,
-#TB_window input.submit,
-.form_user_subscribe input.submit {
+#TB_window input.submit {
background-color:#CEE1E9;
}
@@ -215,9 +214,11 @@ background:transparent url(../images/icons/twotone/green/disfavourite.gif) no-re
background:transparent url(../images/icons/twotone/green/trash.gif) no-repeat 0 45%;
}
+div.entry-content,
div.notice-options {
opacity:0.3;
}
+.notices li.hover div.entry-content,
.notices li.hover div.notice-options {
opacity:1;
}