mirror of
https://github.com/discourse/discourse.git
synced 2025-01-19 05:43:16 +08:00
Merge pull request #4619 from tgxworld/group_page_design
UX: Redesign group page to follow user page.
This commit is contained in:
commit
96c70c74a1
|
@ -14,6 +14,7 @@ export default Ember.Component.extend({
|
|||
},
|
||||
|
||||
tagName: 'ul',
|
||||
selectedHtml: null,
|
||||
|
||||
classNames: ['mobile-nav'],
|
||||
|
||||
|
|
|
@ -0,0 +1,10 @@
|
|||
import computed from 'ember-addons/ember-computed-decorators';
|
||||
|
||||
export default Ember.Controller.extend({
|
||||
application: Ember.inject.controller(),
|
||||
|
||||
@computed('model.is_group_user')
|
||||
showGroupMessages(isGroupUser) {
|
||||
return isGroupUser || (this.currentUser && this.currentUser.admin);
|
||||
}
|
||||
});
|
|
@ -16,19 +16,15 @@ export default Ember.Controller.extend({
|
|||
application: Ember.inject.controller(),
|
||||
counts: null,
|
||||
showing: 'members',
|
||||
|
||||
tabs: [
|
||||
Tab.create({ name: 'members', active: true, 'location': 'group.index' }),
|
||||
Tab.create({ name: 'posts' }),
|
||||
Tab.create({ name: 'topics' }),
|
||||
Tab.create({ name: 'mentions' }),
|
||||
Tab.create({ name: 'messages', requiresMembership: true }),
|
||||
Tab.create({ name: 'members', 'location': 'group.index', icon: 'users' }),
|
||||
Tab.create({ name: 'activity' }),
|
||||
Tab.create({
|
||||
name: 'edit', i18nKey: 'edit.title',
|
||||
requiresMembership: true, requiresGroupAdmin: true
|
||||
name: 'edit', i18nKey: 'edit.title', icon: 'pencil', requiresGroupAdmin: true
|
||||
}),
|
||||
Tab.create({
|
||||
name: 'logs', i18nKey: 'logs.title',
|
||||
requiresMembership: true, requiresGroupAdmin: true
|
||||
name: 'logs', i18nKey: 'logs.title', icon: 'list-alt', requiresGroupAdmin: true
|
||||
})
|
||||
],
|
||||
|
||||
|
@ -57,31 +53,22 @@ export default Ember.Controller.extend({
|
|||
this.get('tabs')[0].set('count', this.get('model.user_count'));
|
||||
},
|
||||
|
||||
@observes('showing')
|
||||
showingChanged() {
|
||||
const showing = this.get('showing');
|
||||
|
||||
this.get('tabs').forEach(tab => {
|
||||
tab.set('active', showing === tab.get('name'));
|
||||
});
|
||||
},
|
||||
|
||||
@computed('model.is_group_user', 'model.is_group_owner', 'model.automatic')
|
||||
getTabs(isGroupUser, isGroupOwner, automatic) {
|
||||
return this.get('tabs').filter(t => {
|
||||
let isMember = false;
|
||||
let display = true;
|
||||
|
||||
if (this.currentUser && !automatic) {
|
||||
if (this.currentUser) {
|
||||
let admin = this.currentUser.admin;
|
||||
|
||||
if (t.get('requiresGroupAdmin')) {
|
||||
isMember = admin || isGroupOwner;
|
||||
if (automatic && t.get('requiresGroupAdmin')) {
|
||||
display = false;
|
||||
} else {
|
||||
isMember = admin || isGroupUser;
|
||||
display = admin || isGroupOwner;
|
||||
}
|
||||
}
|
||||
|
||||
return isMember || !t.get('requiresMembership');
|
||||
return display;
|
||||
});
|
||||
}
|
||||
});
|
||||
|
|
|
@ -49,10 +49,14 @@ export default function() {
|
|||
|
||||
this.route('group', { path: '/groups/:name', resetNamespace: true }, function() {
|
||||
this.route('members');
|
||||
this.route('posts');
|
||||
this.route('topics');
|
||||
this.route('mentions');
|
||||
this.route('messages');
|
||||
|
||||
this.route('activity', function() {
|
||||
this.route('posts');
|
||||
this.route('topics');
|
||||
this.route('mentions');
|
||||
this.route('messages');
|
||||
});
|
||||
|
||||
this.route('logs');
|
||||
this.route('edit');
|
||||
});
|
||||
|
|
|
@ -0,0 +1,3 @@
|
|||
import { buildGroupPage } from 'discourse/routes/group-activity-posts';
|
||||
|
||||
export default buildGroupPage('mentions');
|
|
@ -0,0 +1,3 @@
|
|||
import { buildGroupPage } from 'discourse/routes/group-activity-posts';
|
||||
|
||||
export default buildGroupPage('messages');
|
|
@ -11,12 +11,12 @@ export function buildGroupPage(type) {
|
|||
},
|
||||
|
||||
setupController(controller, model) {
|
||||
this.controllerFor('group-posts').setProperties({ model, type });
|
||||
this.controllerFor('group-activity-posts').setProperties({ model, type });
|
||||
this.controllerFor("group").set("showing", type);
|
||||
},
|
||||
|
||||
renderTemplate() {
|
||||
this.render('group-posts');
|
||||
this.render('group-activity-posts');
|
||||
},
|
||||
|
||||
actions: {
|
|
@ -0,0 +1,3 @@
|
|||
import { buildGroupPage } from 'discourse/routes/group-activity-posts';
|
||||
|
||||
export default buildGroupPage('topics');
|
|
@ -0,0 +1,5 @@
|
|||
export default Ember.Route.extend({
|
||||
beforeModel: function() {
|
||||
this.transitionTo("group.activity.posts");
|
||||
}
|
||||
});
|
|
@ -1,3 +0,0 @@
|
|||
import { buildGroupPage } from 'discourse/routes/group-posts';
|
||||
|
||||
export default buildGroupPage('mentions');
|
|
@ -1,3 +0,0 @@
|
|||
import { buildGroupPage } from 'discourse/routes/group-posts';
|
||||
|
||||
export default buildGroupPage('messages');
|
|
@ -1,3 +0,0 @@
|
|||
import { buildGroupPage } from 'discourse/routes/group-posts';
|
||||
|
||||
export default buildGroupPage('topics');
|
|
@ -1,6 +1,4 @@
|
|||
<div class="container group">
|
||||
{{#link-to "groups" class='group-breadcrumb'}}{{fa-icon 'arrow-left'}} {{i18n 'groups.index.title'}}{{/link-to}}
|
||||
|
||||
<div class='group-details-container'>
|
||||
<div class='group-info'>
|
||||
{{#if model.flair_url}}
|
||||
|
@ -31,10 +29,11 @@
|
|||
{{/if}}
|
||||
</div>
|
||||
|
||||
{{#mobile-nav class='group-nav' desktopClass="pull-left nav nav-stacked" currentPath=application.currentPath}}
|
||||
{{#mobile-nav class='group-nav' desktopClass="nav nav-pills" currentPath=application.currentPath}}
|
||||
{{#each getTabs as |tab|}}
|
||||
<li class="{{if tab.active 'active'}}">
|
||||
<li>
|
||||
{{#link-to tab.location model title=tab.message}}
|
||||
{{#if tab.icon}}{{fa-icon tab.icon}}{{/if}}
|
||||
{{tab.message}}
|
||||
{{#if tab.count}}<span class='count'>({{tab.count}})</span>{{/if}}
|
||||
{{/link-to}}
|
||||
|
@ -42,9 +41,7 @@
|
|||
{{/each}}
|
||||
{{/mobile-nav}}
|
||||
|
||||
<div class='pull-left group-outlet'>
|
||||
<div class='user-main'>
|
||||
{{outlet}}
|
||||
</div>
|
||||
<div class='group-outlet'>
|
||||
{{outlet}}
|
||||
</div>
|
||||
</div>
|
||||
|
|
|
@ -0,0 +1,27 @@
|
|||
<div class='group-activity container'>
|
||||
{{#mobile-nav class='group-activity-nav' desktopClass="pull-left nav nav-stacked" currentPath=application.currentPath}}
|
||||
<li>
|
||||
{{#link-to 'group.activity.posts'}}{{i18n 'groups.posts'}}{{/link-to}}
|
||||
</li>
|
||||
|
||||
<li>
|
||||
{{#link-to 'group.activity.topics'}}{{i18n 'groups.topics'}}{{/link-to}}
|
||||
</li>
|
||||
|
||||
<li>
|
||||
{{#link-to 'group.activity.mentions'}}{{i18n 'groups.mentions'}}{{/link-to}}
|
||||
</li>
|
||||
|
||||
{{#if showGroupMessages}}
|
||||
<li>
|
||||
{{#link-to 'group.activity.messages'}}{{i18n 'groups.messages'}}{{/link-to}}
|
||||
</li>
|
||||
{{/if}}
|
||||
{{/mobile-nav}}
|
||||
|
||||
<div class="{{if site.mobileView "" "pull-left"}} group-activity-outlet">
|
||||
<div class='user-main'>
|
||||
{{outlet}}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
|
@ -1,4 +1,6 @@
|
|||
<li><a {{action 'toggleExpanded'}}>{{{selectedHtml}}} <i class='fa fa-caret-down'></i></a></li>
|
||||
{{#if selectedHtml}}
|
||||
<li><a {{action 'toggleExpanded'}}>{{{selectedHtml}}} <i class='fa fa-caret-down'></i></a></li>
|
||||
{{/if}}
|
||||
<ul class='drop {{if expanded 'expanded'}}'>
|
||||
{{yield}}
|
||||
</ul>
|
||||
|
|
|
@ -1,12 +1,7 @@
|
|||
.group-details-container {
|
||||
background: rgba(230, 230, 230, 0.3);
|
||||
padding: 20px;
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.group-breadcrumb {
|
||||
display: block;
|
||||
margin-bottom: 10px;
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
.group-info {
|
||||
|
@ -160,9 +155,11 @@ table.group-members {
|
|||
}
|
||||
}
|
||||
|
||||
.group-edit .form-horizontal {
|
||||
label {
|
||||
font-weight: bold;
|
||||
.group-edit {
|
||||
.form-horizontal {
|
||||
label {
|
||||
font-weight: bold;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -1,8 +1,4 @@
|
|||
.group-members-input {
|
||||
.ac-wrap {
|
||||
width: 100% !important;
|
||||
}
|
||||
|
||||
.group-members-input-selector {
|
||||
margin-top: 10px;
|
||||
|
||||
|
|
|
@ -1,12 +1,59 @@
|
|||
.group-outlet {
|
||||
width: 75%;
|
||||
}
|
||||
|
||||
.group-nav {
|
||||
width: 20%;
|
||||
margin-right: 30px;
|
||||
li {
|
||||
float: left;
|
||||
|
||||
a, i {
|
||||
color: dark-light-choose(scale-color($primary, $lightness: 40%), scale-color($secondary, $lightness: 40%));
|
||||
}
|
||||
|
||||
.active {
|
||||
a, i {
|
||||
color: $secondary;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.group-info {
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.group-activity-nav {
|
||||
width: 15%;
|
||||
background-color: transparent;
|
||||
|
||||
li {
|
||||
border: none;
|
||||
|
||||
a {
|
||||
padding: 8px 13px;
|
||||
}
|
||||
|
||||
a.active {
|
||||
background-color: transparent;
|
||||
font-weight: bold;
|
||||
color: $primary;
|
||||
}
|
||||
|
||||
a.active:after {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.group-activity-outlet {
|
||||
width: 85%;
|
||||
}
|
||||
|
||||
.group-edit {
|
||||
border: 1px solid dark-light-diff($primary, $secondary, 90%, -60%);
|
||||
padding: 10px;
|
||||
|
||||
.form-horizontal {
|
||||
button {
|
||||
float: none;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -10,7 +10,7 @@
|
|||
margin: 5px 0px 0px 0px;
|
||||
}
|
||||
|
||||
.group-nav, .group-outlet {
|
||||
.group-nav {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
|
@ -20,16 +20,20 @@
|
|||
|
||||
.group-nav.mobile-nav {
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
> li {
|
||||
a {
|
||||
color: white;
|
||||
.group-activity {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.fa { color: white; }
|
||||
}
|
||||
}
|
||||
.group-activity-nav.mobile-nav {
|
||||
position: absolute;
|
||||
right: 0px;
|
||||
top: -50px;
|
||||
}
|
||||
|
||||
background-color: $quaternary;
|
||||
.group-activity-outlet {
|
||||
float: none;
|
||||
}
|
||||
|
||||
.form-horizontal {
|
||||
|
@ -55,7 +59,7 @@ table.group-members {
|
|||
|
||||
tr {
|
||||
.user-info {
|
||||
width: 130px;
|
||||
width: auto;
|
||||
}
|
||||
|
||||
td {
|
||||
|
|
|
@ -432,6 +432,7 @@ en:
|
|||
title:
|
||||
one: "group"
|
||||
other: "groups"
|
||||
activity: "Activity"
|
||||
members: "Members"
|
||||
topics: "Topics"
|
||||
posts: "Posts"
|
||||
|
|
|
@ -401,6 +401,8 @@ Discourse::Application.routes.draw do
|
|||
get "posts.rss" => "groups#posts_feed", format: :rss
|
||||
get "mentions.rss" => "groups#mentions_feed", format: :rss
|
||||
|
||||
get 'activity' => "groups#show"
|
||||
get 'activity/:filter' => "groups#show"
|
||||
get 'members'
|
||||
get 'posts'
|
||||
get 'topics'
|
||||
|
|
|
@ -24,24 +24,31 @@ test("Viewing Group", () => {
|
|||
ok(count('.group-members tr') > 0, "it lists group members");
|
||||
});
|
||||
|
||||
visit("/groups/discourse/posts");
|
||||
click(".nav-pills li a[title='Activity']");
|
||||
|
||||
andThen(() => {
|
||||
ok(count('.user-stream .item') > 0, "it lists stream items");
|
||||
});
|
||||
|
||||
visit("/groups/discourse/topics");
|
||||
click(".group-activity-nav li a[href='/groups/discourse/activity/topics']");
|
||||
|
||||
andThen(() => {
|
||||
ok(count('.user-stream .item') > 0, "it lists stream items");
|
||||
});
|
||||
|
||||
visit("/groups/discourse/mentions");
|
||||
click(".group-activity-nav li a[href='/groups/discourse/activity/mentions']");
|
||||
|
||||
andThen(() => {
|
||||
ok(count('.user-stream .item') > 0, "it lists stream items");
|
||||
});
|
||||
|
||||
visit("/groups/discourse/messages");
|
||||
andThen(() => {
|
||||
ok(find(".nav-stacked li a[title='Messages']").length === 0, 'it should not show messages tab if user is not admin');
|
||||
equal(
|
||||
find(".group-activity li a[href='/groups/discourse/activity/messages']").length,
|
||||
0,
|
||||
'it should not show messages tab if user is not a group user or admin'
|
||||
);
|
||||
|
||||
ok(find(".nav-stacked li a[title='Edit Group']").length === 0, 'it should not show messages tab if user is not admin');
|
||||
ok(find(".nav-stacked li a[title='Logs']").length === 0, 'it should not show Logs tab if user is not admin');
|
||||
ok(count('.user-stream .item') > 0, "it lists stream items");
|
||||
|
@ -55,10 +62,19 @@ test("Admin Viewing Group", () => {
|
|||
visit("/groups/discourse");
|
||||
|
||||
andThen(() => {
|
||||
ok(find(".nav-stacked li a[title='Messages']").length === 1, 'it should show messages tab if user is admin');
|
||||
ok(find(".nav-stacked li a[title='Edit Group']").length === 1, 'it should show edit group tab if user is admin');
|
||||
ok(find(".nav-stacked li a[title='Logs']").length === 1, 'it should show Logs tab if user is admin');
|
||||
ok(find(".nav-pills li a[title='Edit Group']").length === 1, 'it should show edit group tab if user is admin');
|
||||
ok(find(".nav-pills li a[title='Logs']").length === 1, 'it should show Logs tab if user is admin');
|
||||
|
||||
equal(find('.group-info-name').text(), 'Awesome Team', 'it should display the group name');
|
||||
});
|
||||
|
||||
click(".nav-pills li a[title='Activity']");
|
||||
|
||||
andThen(() => {
|
||||
equal(
|
||||
find(".group-activity li a[href='/groups/discourse/activity/messages']").length,
|
||||
1,
|
||||
'it should show messages tab if user is admin'
|
||||
);
|
||||
});
|
||||
});
|
||||
|
|
Loading…
Reference in New Issue
Block a user