mirror of
https://github.com/humhub/humhub.git
synced 2025-01-17 22:28:51 +01:00
Transform the menu notification list in a widget and built-in a load more functionality and an "Mark all as seen" feature (#58)
This commit is contained in:
parent
55de7fce81
commit
8d22c5f410
@ -74,6 +74,16 @@ textarea {
|
|||||||
padding: 3px 10px;
|
padding: 3px 10px;
|
||||||
margin-bottom: 10px;
|
margin-bottom: 10px;
|
||||||
}
|
}
|
||||||
|
.topbar .dropdown-header .dropdown-header-link {
|
||||||
|
position: absolute;
|
||||||
|
top: 2px;
|
||||||
|
right: 10px;
|
||||||
|
}
|
||||||
|
.topbar .dropdown-header .dropdown-header-link a {
|
||||||
|
color: #428bca !important;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: normal;
|
||||||
|
}
|
||||||
#topbar-first {
|
#topbar-first {
|
||||||
top: 0;
|
top: 0;
|
||||||
background-color: #101010;
|
background-color: #101010;
|
||||||
@ -165,6 +175,10 @@ textarea {
|
|||||||
width: 350px;
|
width: 350px;
|
||||||
margin-left: -148px;
|
margin-left: -148px;
|
||||||
}
|
}
|
||||||
|
#topbar-first .notifications #dropdown-notifications ul.media-list {
|
||||||
|
max-height: 400px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
#topbar-first .notifications #dropdown-notifications li {
|
#topbar-first .notifications #dropdown-notifications li {
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
|
@ -55,7 +55,7 @@ hr {
|
|||||||
}
|
}
|
||||||
|
|
||||||
textarea {
|
textarea {
|
||||||
height:1.5em;
|
height: 1.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
@ -91,6 +91,18 @@ textarea {
|
|||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
padding: 3px 10px;
|
padding: 3px 10px;
|
||||||
margin-bottom: 10px;
|
margin-bottom: 10px;
|
||||||
|
|
||||||
|
.dropdown-header-link {
|
||||||
|
position: absolute;
|
||||||
|
top: 2px;
|
||||||
|
right: 10px;
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: #428bca !important;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: normal;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -189,7 +201,7 @@ textarea {
|
|||||||
left: 50%;
|
left: 50%;
|
||||||
margin-left: -18px;
|
margin-left: -18px;
|
||||||
border-top-width: 0;
|
border-top-width: 0;
|
||||||
border-bottom-color: rgba(0,0,0,0.15);
|
border-bottom-color: rgba(0, 0, 0, 0.15);
|
||||||
top: -16px;
|
top: -16px;
|
||||||
z-index: 1035;
|
z-index: 1035;
|
||||||
}
|
}
|
||||||
@ -198,6 +210,11 @@ textarea {
|
|||||||
width: 350px;
|
width: 350px;
|
||||||
margin-left: -148px;
|
margin-left: -148px;
|
||||||
|
|
||||||
|
ul.media-list {
|
||||||
|
max-height: 400px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
li {
|
li {
|
||||||
|
|
||||||
position: relative;
|
position: relative;
|
||||||
@ -654,7 +671,6 @@ input.placeholder, textarea.placeholder {
|
|||||||
color: #999999;
|
color: #999999;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// 10) Responsive modifications
|
// 10) Responsive modifications
|
||||||
// --------------------------------------------------
|
// --------------------------------------------------
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
<li class="<?php if (!$notification->seen) : ?>new<?php endif; ?>">
|
<li class="<?php if (!$notification->seen) : ?>new<?php endif; ?>" id="notification-<?php echo $notification->id; ?>">
|
||||||
<a href="<?php echo $notification->getUrl(); ?>">
|
<a href="<?php echo $notification->getUrl(); ?>">
|
||||||
<div class="media">
|
<div class="media">
|
||||||
|
|
||||||
|
@ -6,12 +6,14 @@
|
|||||||
* @package humhub.modules_core.notification.controllers
|
* @package humhub.modules_core.notification.controllers
|
||||||
* @since 0.5
|
* @since 0.5
|
||||||
*/
|
*/
|
||||||
class ListController extends Controller {
|
class ListController extends Controller
|
||||||
|
{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array action filters
|
* @return array action filters
|
||||||
*/
|
*/
|
||||||
public function filters() {
|
public function filters()
|
||||||
|
{
|
||||||
return array(
|
return array(
|
||||||
'accessControl', // perform access control for CRUD operations
|
'accessControl', // perform access control for CRUD operations
|
||||||
);
|
);
|
||||||
@ -22,7 +24,8 @@ class ListController extends Controller {
|
|||||||
* This method is used by the 'accessControl' filter.
|
* This method is used by the 'accessControl' filter.
|
||||||
* @return array access control rules
|
* @return array access control rules
|
||||||
*/
|
*/
|
||||||
public function accessRules() {
|
public function accessRules()
|
||||||
|
{
|
||||||
return array(
|
return array(
|
||||||
array('allow', // allow authenticated user to perform 'create' and 'update' actions
|
array('allow', // allow authenticated user to perform 'create' and 'update' actions
|
||||||
'users' => array('@'),
|
'users' => array('@'),
|
||||||
@ -34,19 +37,69 @@ class ListController extends Controller {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a List of all Comments belong to this Model
|
* Returns a List of all notifications for an user
|
||||||
*/
|
*/
|
||||||
public function actionIndex() {
|
public function actionIndex()
|
||||||
|
{
|
||||||
|
|
||||||
|
// the id from the last entry loaded
|
||||||
|
$lastEntryId = Yii::app()->request->getParam('from');
|
||||||
|
|
||||||
|
// create database query
|
||||||
$criteria = new CDbCriteria();
|
$criteria = new CDbCriteria();
|
||||||
|
if ($lastEntryId > 0) {
|
||||||
|
// start from last entry id loaded
|
||||||
|
$criteria->condition = 'id<' . $lastEntryId;
|
||||||
|
}
|
||||||
$criteria->order = 'seen ASC, created_at DESC';
|
$criteria->order = 'seen ASC, created_at DESC';
|
||||||
$criteria->limit = 6;
|
$criteria->limit = 6;
|
||||||
|
|
||||||
|
// safe query
|
||||||
$notifications = Notification::model()->findAllByAttributes(array('user_id' => Yii::app()->user->id), $criteria);
|
$notifications = Notification::model()->findAllByAttributes(array('user_id' => Yii::app()->user->id), $criteria);
|
||||||
|
|
||||||
$this->renderPartial('index', array(
|
// variable for notification list
|
||||||
'notifications' => $notifications
|
$output = "";
|
||||||
));
|
|
||||||
|
foreach ($notifications as $notification) {
|
||||||
|
// format and save all entries
|
||||||
|
$output .= $notification->getOut();
|
||||||
|
|
||||||
|
// get the id from the last entry
|
||||||
|
$lastEntryId = $notification->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
// build json array
|
||||||
|
$json = array();
|
||||||
|
$json['output'] = $output;
|
||||||
|
$json['lastEntryId'] = $lastEntryId;
|
||||||
|
$json['counter'] = count($notifications);
|
||||||
|
|
||||||
|
// return json
|
||||||
|
echo CJSON::encode($json);
|
||||||
|
|
||||||
|
// compete action
|
||||||
|
Yii::app()->end();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public function actionMarkAsSeen()
|
||||||
|
{
|
||||||
|
|
||||||
|
// build query
|
||||||
|
$criteria = new CDbCriteria();
|
||||||
|
$criteria->condition = 'seen=0';
|
||||||
|
|
||||||
|
// load all unseen notification for this user
|
||||||
|
$notifications = Notification::model()->findAllByAttributes(array('user_id' => Yii::app()->user->id), $criteria);
|
||||||
|
|
||||||
|
foreach ($notifications as $notification) {
|
||||||
|
// mark all unseen notification as seen
|
||||||
|
$notification->markAsSeen();
|
||||||
|
}
|
||||||
|
|
||||||
|
// compete action
|
||||||
|
Yii::app()->end();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* NotificationListWidget shows an stream of notifications for an user at the top menu.
|
||||||
|
*
|
||||||
|
* @author andystrobel
|
||||||
|
* @package humhub.modules_core.notification
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class NotificationListWidget extends HWidget {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Runs the notification widget
|
||||||
|
*/
|
||||||
|
public function run() {
|
||||||
|
|
||||||
|
$this->render('list', array());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
178
protected/modules_core/notification/widgets/views/list.php
Normal file
178
protected/modules_core/notification/widgets/views/list.php
Normal file
@ -0,0 +1,178 @@
|
|||||||
|
<div class="btn-group">
|
||||||
|
<a href="#" id="icon-notifications" data-toggle="dropdown">
|
||||||
|
<i class="fa fa-bell"></i>
|
||||||
|
</a>
|
||||||
|
<span id="badge-notifications" style="display:none;" class="label label-danger label-notifications">1</span>
|
||||||
|
|
||||||
|
<!-- container for ajax response -->
|
||||||
|
<ul id="dropdown-notifications" class="dropdown-menu">
|
||||||
|
<li class="dropdown-header">
|
||||||
|
<div class="arrow"></div><?php echo Yii::t('base', 'Notifications'); ?>
|
||||||
|
<div class="dropdown-header-link"><a
|
||||||
|
href="javascript:markNotificationsAsSeen();"><?php echo Yii::t('NotificationModule', 'Mark all as seen'); ?></a>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
<ul class="media-list"></ul>
|
||||||
|
<li id="loader_notifications">
|
||||||
|
<div class="loader"></div>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
|
||||||
|
// set niceScroll to notification list
|
||||||
|
$("#dropdown-notifications ul.media-list").niceScroll({
|
||||||
|
cursorwidth: "7",
|
||||||
|
cursorborder:"",
|
||||||
|
cursorcolor:"#555",
|
||||||
|
cursoropacitymax:"0.2",
|
||||||
|
railpadding:{top:0,right:3,left:0,bottom:0}
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
function markNotificationsAsSeen() {
|
||||||
|
|
||||||
|
//$('#dropdown-notifications').css({display: 'block'});
|
||||||
|
|
||||||
|
// call ajax request to mark all notifications as seen
|
||||||
|
jQuery.ajax({
|
||||||
|
'type': 'GET',
|
||||||
|
'url': '<?php echo $this->createUrl('//notification/list/markAsSeen', array('ajax' => 1)); ?>',
|
||||||
|
'cache': false,
|
||||||
|
'data': jQuery(this).parents("form").serialize(),
|
||||||
|
'success': function(html) {
|
||||||
|
// hide notification badge at the top menu
|
||||||
|
$('#badge-notifications').css('display', 'none');
|
||||||
|
}});
|
||||||
|
}
|
||||||
|
|
||||||
|
$(document).ready(function () {
|
||||||
|
|
||||||
|
// set the ID for the last loaded activity entry to 1
|
||||||
|
var notificationLastLoadedEntryId = 0;
|
||||||
|
|
||||||
|
// save if the last entries are already loaded
|
||||||
|
var notificationLastEntryReached = false;
|
||||||
|
|
||||||
|
// safe action url
|
||||||
|
var _notificationUrl = '<?php echo $this->createUrl('//notification/list/index', array('from' => 'lastEntryId', 'ajax' => 1)); ?>';
|
||||||
|
|
||||||
|
// Open the notification menu
|
||||||
|
$('#icon-notifications').click(function () {
|
||||||
|
|
||||||
|
// reset variables by dropdown reopening
|
||||||
|
notificationLastLoadedEntryId = 0;
|
||||||
|
notificationLastEntryReached = false;
|
||||||
|
|
||||||
|
// remove all notification entries from dropdown
|
||||||
|
$('#dropdown-notifications ul.media-list').find('li').remove();
|
||||||
|
|
||||||
|
// checking if ajax loading is necessary or the last entries are already loaded
|
||||||
|
if (notificationLastEntryReached == false) {
|
||||||
|
|
||||||
|
// load notifications
|
||||||
|
loadNotificationEntries();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
$('#dropdown-notifications ul.media-list').scroll(function () {
|
||||||
|
|
||||||
|
// save height of the overflow container
|
||||||
|
var _containerHeight = $("#dropdown-notifications ul.media-list").height();
|
||||||
|
|
||||||
|
// save scroll height
|
||||||
|
var _scrollHeight = $("#dropdown-notifications ul.media-list").prop("scrollHeight");
|
||||||
|
|
||||||
|
// save current scrollbar position
|
||||||
|
var _currentScrollPosition = $('#dropdown-notifications ul.media-list').scrollTop();
|
||||||
|
|
||||||
|
// load more activites if current scroll position is near scroll height
|
||||||
|
if (_currentScrollPosition >= (_scrollHeight - _containerHeight - 1)) {
|
||||||
|
|
||||||
|
// checking if ajax loading is necessary or the last entries are already loaded
|
||||||
|
if (notificationLastEntryReached == false) {
|
||||||
|
|
||||||
|
// load more notifications
|
||||||
|
loadNotificationEntries();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
function loadNotificationEntries() {
|
||||||
|
|
||||||
|
// replace placeholder name with the id from the last loaded entry
|
||||||
|
var _modifiedNotificationUrl = _notificationUrl.replace('lastEntryId', notificationLastLoadedEntryId)
|
||||||
|
|
||||||
|
// show loader
|
||||||
|
$("#loader_notifications .loader").show();
|
||||||
|
|
||||||
|
// send ajax request
|
||||||
|
jQuery.getJSON(_modifiedNotificationUrl, function (json) {
|
||||||
|
|
||||||
|
// hide loader
|
||||||
|
$("#loader_notifications .loader").hide();
|
||||||
|
|
||||||
|
// save id from the last entry for the next loading
|
||||||
|
notificationLastLoadedEntryId = json.lastEntryId;
|
||||||
|
|
||||||
|
if (json.counter < 6) {
|
||||||
|
// prevent the next ajax calls, if there are no more entries
|
||||||
|
notificationLastEntryReached = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// add new entries
|
||||||
|
$("#dropdown-notifications ul.media-list").append(json.output);
|
||||||
|
|
||||||
|
// format time
|
||||||
|
$('span.time').timeago();
|
||||||
|
|
||||||
|
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// load number of new notifications at page loading
|
||||||
|
getNotifications();
|
||||||
|
|
||||||
|
// load number of new notifications in a loop
|
||||||
|
setInterval(getNotifications, 60000);
|
||||||
|
|
||||||
|
|
||||||
|
// load and show new count of notifications
|
||||||
|
function getNotifications() {
|
||||||
|
|
||||||
|
var $newNotifications = parseInt(0);
|
||||||
|
|
||||||
|
// load data
|
||||||
|
jQuery.getJSON("<?php echo $this->createUrl('//dashboard/dashboard/GetFrontEndInfo'); ?>", function (json) {
|
||||||
|
|
||||||
|
// save numbers to variables
|
||||||
|
$newNotifications = parseInt(json.newNotifications);
|
||||||
|
|
||||||
|
// show or hide the badge for new notifications
|
||||||
|
if ($newNotifications == 0) {
|
||||||
|
$('#badge-notifications').css('display', 'none');
|
||||||
|
} else {
|
||||||
|
$('#badge-notifications').empty();
|
||||||
|
$('#badge-notifications').append($newNotifications);
|
||||||
|
$('#badge-notifications').fadeIn('fast');
|
||||||
|
}
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
</script>
|
@ -100,15 +100,7 @@
|
|||||||
<div class="notifications pull-right">
|
<div class="notifications pull-right">
|
||||||
|
|
||||||
<!-- global notifications dropdown -->
|
<!-- global notifications dropdown -->
|
||||||
<div class="btn-group">
|
<?php $this->widget('application.modules_core.notification.widgets.NotificationListWidget'); ?>
|
||||||
<a href="#" id="icon-notifications" data-toggle="dropdown">
|
|
||||||
<i class="fa fa-bell"></i>
|
|
||||||
</a>
|
|
||||||
<span id="badge-notifications" style="display:none;" class="label label-danger label-notifications">1</span>
|
|
||||||
|
|
||||||
<!-- container for ajax response -->
|
|
||||||
<ul id="dropdown-notifications" class="dropdown-menu"></ul>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Notification addon widget for modules -->
|
<!-- Notification addon widget for modules -->
|
||||||
<?php $this->widget('application.widgets.NotificationAddonWidget', array('widgets' => array())); ?>
|
<?php $this->widget('application.widgets.NotificationAddonWidget', array('widgets' => array())); ?>
|
||||||
@ -181,60 +173,6 @@
|
|||||||
|
|
||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Open the notification menu
|
|
||||||
$('#icon-notifications').click(function () {
|
|
||||||
|
|
||||||
// remove all <li> entries from dropdown
|
|
||||||
$('#dropdown-notifications').find('li').remove();
|
|
||||||
|
|
||||||
// append title and loader to dropdown
|
|
||||||
$('#dropdown-notifications').append('<li class="dropdown-header"><div class="arrow"></div><?php echo Yii::t('base', 'Notifications'); ?></li><li id="loader_notifications"><div class="loader"></div></li>');
|
|
||||||
|
|
||||||
// load newest notifications
|
|
||||||
$.ajax({
|
|
||||||
'type': 'GET',
|
|
||||||
'url': '<?php echo $this->createUrl('//notification/list', array('ajax' => 1)); ?>',
|
|
||||||
'cache': false,
|
|
||||||
'data': jQuery(this).parents("form").serialize(),
|
|
||||||
'success': function (html) {
|
|
||||||
$("#loader_notifications").replaceWith(html)
|
|
||||||
}});
|
|
||||||
|
|
||||||
})
|
|
||||||
|
|
||||||
// load number of new notifications at page loading
|
|
||||||
getNotifications();
|
|
||||||
|
|
||||||
// load number of new notifications in a loop
|
|
||||||
setInterval(getNotifications, 60000);
|
|
||||||
|
|
||||||
|
|
||||||
// load and show new count of notifications
|
|
||||||
function getNotifications() {
|
|
||||||
|
|
||||||
var $newNotifications = parseInt(0);
|
|
||||||
|
|
||||||
// load data
|
|
||||||
jQuery.getJSON("<?php echo $this->createUrl('//dashboard/dashboard/GetFrontEndInfo'); ?>", function (json) {
|
|
||||||
|
|
||||||
// save numbers to variables
|
|
||||||
$newNotifications = parseInt(json.newNotifications);
|
|
||||||
|
|
||||||
// show or hide the badge for new notifications
|
|
||||||
if ($newNotifications == 0) {
|
|
||||||
$('#badge-notifications').css('display', 'none');
|
|
||||||
} else {
|
|
||||||
$('#badge-notifications').empty();
|
|
||||||
$('#badge-notifications').append($newNotifications);
|
|
||||||
$('#badge-notifications').fadeIn('fast');
|
|
||||||
}
|
|
||||||
|
|
||||||
})
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Ensures after hide modal content is removed. */
|
/* Ensures after hide modal content is removed. */
|
||||||
$('#globalModal').on('hidden.bs.modal', function (e) {
|
$('#globalModal').on('hidden.bs.modal', function (e) {
|
||||||
$(this).removeData('bs.modal');
|
$(this).removeData('bs.modal');
|
||||||
@ -268,8 +206,6 @@
|
|||||||
|
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|
||||||
|
@ -23,6 +23,9 @@ h4 {
|
|||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
color: #bebebe;
|
color: #bebebe;
|
||||||
}
|
}
|
||||||
|
.topbar .dropdown-header .dropdown-header-link a {
|
||||||
|
color: #4cd9c0 !important;
|
||||||
|
}
|
||||||
#topbar-first {
|
#topbar-first {
|
||||||
background-color: #7191a8;
|
background-color: #7191a8;
|
||||||
color: white;
|
color: white;
|
||||||
|
@ -72,6 +72,10 @@ h4 {
|
|||||||
.dropdown-header {
|
.dropdown-header {
|
||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
color: @colorFont4;
|
color: @colorFont4;
|
||||||
|
|
||||||
|
.dropdown-header-link a {
|
||||||
|
color: #4cd9c0 !important;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user