Fix activity rendering in the sidebar

Signed-off-by: Julius Härtl <jus@bitgrid.net>
This commit is contained in:
Julius Härtl
2018-09-07 15:04:42 +02:00
parent df4f8dda60
commit 28f58ee5f3
8 changed files with 79 additions and 23 deletions

View File

@@ -7,3 +7,17 @@
background-color: rgba(233, 50, 45, 0.2); background-color: rgba(233, 50, 45, 0.2);
text-decoration: none; text-decoration: none;
} }
.activityTabView .avatardiv-container {
display: inline-block;
bottom: -3px;
margin-left: 3px;
}
.activityTabView .avatar-name-wrapper {
font-weight: bold;
}
.activityTabView .activitysubject a {
font-weight: bold;
}

View File

@@ -48,7 +48,7 @@ class ActivityController {
} }
parseMessage(subject, parameters) { parseMessage(subject, parameters) {
OCA.Activity.RichObjectStringParser._userLocalTemplate = '<avatar ng-attr-contactsmenu ng-attr-tooltip ng-attr-user="{{ id }}" ng-attr-displayname="{{name}}"></avatar>'; OCA.Activity.RichObjectStringParser._userLocalTemplate = '<span class="avatar-name-wrapper"><avatar ng-attr-contactsmenu ng-attr-tooltip ng-attr-user="{{ id }}" ng-attr-displayname="{{name}}" ng-attr-size="16"></avatar> {{ name }}</span>';
return OCA.Activity.RichObjectStringParser.parseMessage(subject, parameters); return OCA.Activity.RichObjectStringParser.parseMessage(subject, parameters);
} }
@@ -57,20 +57,19 @@ class ActivityController {
let dataLengthBefore = self.getData(self.element.id).length; let dataLengthBefore = self.getData(self.element.id).length;
let _executeFetch = function() { let _executeFetch = function() {
let promise = self.activityservice.fetchMoreActivities(self.type, self.element.id); let promise = self.activityservice.fetchMoreActivities(self.type, self.element.id);
if (Promise.resolve(promise) === promise) { promise.then(function (data) {
promise.then(function (data) { let dataLengthAfter = self.getData(self.element.id).length;
let dataLengthAfter = self.getData(self.element.id).length; if (data !== null && (dataLengthAfter <= dataLengthBefore || dataLengthAfter < 5)) {
if (data !== null || dataLengthAfter <= dataLengthBefore || dataLengthAfter < 5) { _executeFetch();
_executeFetch(); } else {
} else {
self.loading = false;
self.$scope.$apply();
}
}, function () {
self.loading = false; self.loading = false;
self.$scope.$apply(); self.$scope.$apply();
}); }
} }, function () {
self.loading = false;
self.$scope.$apply();
});
}; };
_executeFetch(); _executeFetch();
} }

View File

@@ -4,20 +4,20 @@
* @author Julius Härtl <jus@bitgrid.net> * @author Julius Härtl <jus@bitgrid.net>
* *
* @license GNU AGPL version 3 or any later version * @license GNU AGPL version 3 or any later version
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as * it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the * published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version. * License, or (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of * but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details. * GNU Affero General Public License for more details.
* *
* You should have received a copy of the GNU Affero General Public License * 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/>. * along with this program. If not, see <http://www.gnu.org/licenses/>.
* *
*/ */
import app from '../app/App.js'; import app from '../app/App.js';
@@ -32,6 +32,10 @@ app.directive('avatar', function() {
link: function(scope, element, attr){ link: function(scope, element, attr){
scope.uid = attr.displayname; scope.uid = attr.displayname;
scope.displayname = attr.displayname; scope.displayname = attr.displayname;
scope.size = attr.size;
if (typeof scope.size === 'undefined') {
scope.size = 32;
}
var value = attr.user; var value = attr.user;
var avatardiv = $(element).find('.avatardiv'); var avatardiv = $(element).find('.avatardiv');
if(typeof attr.contactsmenu !== 'undefined' && attr.contactsmenu !== 'false') { if(typeof attr.contactsmenu !== 'undefined' && attr.contactsmenu !== 'false') {
@@ -44,8 +48,8 @@ app.directive('avatar', function() {
placement: 'top' placement: 'top'
}); });
} }
avatardiv.avatar(value, 32, false, false, false, attr.displayname); avatardiv.avatar(value, scope.size, false, false, false, attr.displayname);
}, },
controller: function () {} controller: function () {}
}; };
}); });

View File

@@ -0,0 +1,38 @@
/*
* @copyright Copyright (c) 2018 Julius Härtl <jus@bitgrid.net>
*
* @author Julius Härtl <jus@bitgrid.net>
*
* @license GNU AGPL version 3 or any later version
*
* 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/>.
*
*/
import app from '../app/App.js';
app.directive('bindHtmlCompile', function ($compile) {
'use strict';
return {
restrict: 'A',
link: function (scope, element, attrs) {
scope.$watch(function () {
return scope.$eval(attrs.bindHtmlCompile);
}, function (value) {
element.html(value);
$compile(element.contents())(scope);
});
}
};
});

View File

@@ -78,6 +78,7 @@ class ActivityService {
}); });
self.since[type][id].oldest = response.headers('X-Activity-Last-Given'); self.since[type][id].oldest = response.headers('X-Activity-Last-Given');
self.running = false; self.running = false;
return response;
}, function (error) { }, function (error) {
if (error.status === 304) { if (error.status === 304) {
self.since[type][id].finished = true; self.since[type][id].finished = true;

View File

@@ -150,7 +150,7 @@
</ul> </ul>
</div> </div>
<div id="board-detail-activity" class="tab" ng-if="params.tab==4"> <div id="board-detail-activity" class="tab activityTabView" ng-if="params.tab==4">
<activity-component ng-if="boardservice.getCurrent()" type="deck_board" element="boardservice.getCurrent()"></activity-component> <activity-component ng-if="boardservice.getCurrent()" type="deck_board" element="boardservice.getCurrent()"></activity-component>
</div> </div>

View File

@@ -1,11 +1,11 @@
<ul class="activities" infinite-scroll="$ctrl.page()" infinite-scroll-container="'#app-sidebar'" infinite-scroll-disabled="$ctrl.activityservice.running"> <ul class="activities" infinite-scroll="$ctrl.page()" infinite-scroll-container="'#app-sidebar'" infinite-scroll-disabled="$ctrl.activityservice.running" infinite-scroll-immediate-check="false">
<li ng-if="$ctrl.loadingNewer()"><div class="icon-loading-small"></div></li> <li ng-if="$ctrl.loadingNewer()"><div class="icon-loading-small"></div></li>
<li class="activity box" ng-repeat="activity in $ctrl.getData($ctrl.element.id) track by $index"> <li class="activity box" ng-repeat="activity in $ctrl.getData($ctrl.element.id) track by $index">
<div class="activity-icon"> <div class="activity-icon">
<img src="{{activity.icon}}" alt=""> <img src="{{activity.icon}}" alt="">
</div> </div>
<div class="activitysubject" <div class="activitysubject"
ng-bind-html="$ctrl.parseMessage(activity.subject_rich[0], activity.subject_rich[1])"></div> bind-html-compile="$ctrl.parseMessage(activity.subject_rich[0], activity.subject_rich[1])"></div>
<span class="activitytime has-tooltip live-relative-timestamp" <span class="activitytime has-tooltip live-relative-timestamp"
data-timestamp="{{ activity.timestamp }}">{{ activity.timestamp/1000 | relativeDateFilter }}</span> data-timestamp="{{ activity.timestamp }}">{{ activity.timestamp/1000 | relativeDateFilter }}</span>
<div class="activitymessage" ng-bind-html="activity.message"></div> <div class="activitymessage" ng-bind-html="activity.message"></div>

View File

@@ -131,7 +131,7 @@
</div> </div>
</div> </div>
<div class="section-content card-activity" ng-if="params.tab === 2"> <div class="section-content card-activity activityTabView" ng-if="params.tab === 2">
<activity-component type="deck_card" element="cardservice.getCurrent()"></activity-component> <activity-component type="deck_card" element="cardservice.getCurrent()"></activity-component>
</div> </div>