UX: On mobile use a dropdown for topic controls instead of many buttons
This commit is contained in:
parent
4a21c29961
commit
b2c73e7045
|
@ -76,13 +76,12 @@ export default Ember.Component.extend({
|
||||||
$elem.select2({formatResult: this.comboTemplate, minimumResultsForSearch, width: 'resolve'});
|
$elem.select2({formatResult: this.comboTemplate, minimumResultsForSearch, width: 'resolve'});
|
||||||
|
|
||||||
const castInteger = this.get('castInteger');
|
const castInteger = this.get('castInteger');
|
||||||
const self = this;
|
$elem.on("change", e => {
|
||||||
$elem.on("change", function (e) {
|
|
||||||
let val = $(e.target).val();
|
let val = $(e.target).val();
|
||||||
if (val && val.length && castInteger) {
|
if (val && val.length && castInteger) {
|
||||||
val = parseInt(val, 10);
|
val = parseInt(val, 10);
|
||||||
}
|
}
|
||||||
self.set('value', val);
|
this.set('value', val);
|
||||||
});
|
});
|
||||||
$elem.trigger('change');
|
$elem.trigger('change');
|
||||||
}.on('didInsertElement'),
|
}.on('didInsertElement'),
|
||||||
|
|
|
@ -0,0 +1,60 @@
|
||||||
|
import Combobox from 'discourse/components/combo-box';
|
||||||
|
import { on, observes } from 'ember-addons/ember-computed-decorators';
|
||||||
|
|
||||||
|
export default Combobox.extend({
|
||||||
|
none: "topic.controls",
|
||||||
|
|
||||||
|
@on('init')
|
||||||
|
_createContent() {
|
||||||
|
const content = [];
|
||||||
|
const topic = this.get('topic');
|
||||||
|
const details = topic.get('details');
|
||||||
|
|
||||||
|
if (details.get('can_invite_to')) {
|
||||||
|
content.push({ id: 'invite', name: I18n.t('topic.invite_reply.title') });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (topic.get('bookmarked')) {
|
||||||
|
content.push({ id: 'bookmark', name: I18n.t('bookmarked.clear_bookmarks') });
|
||||||
|
} else {
|
||||||
|
content.push({ id: 'bookmark', name: I18n.t('bookmarked.title') });
|
||||||
|
}
|
||||||
|
content.push({ id: 'share', name: I18n.t('topic.share.title') });
|
||||||
|
|
||||||
|
if (details.get('can_flag_topic')) {
|
||||||
|
content.push({ id: 'flag', name: I18n.t('topic.flag_topic.title') });
|
||||||
|
}
|
||||||
|
|
||||||
|
this.set('content', content);
|
||||||
|
},
|
||||||
|
|
||||||
|
@observes('value')
|
||||||
|
_valueChanged() {
|
||||||
|
const value = this.get('value');
|
||||||
|
const controller = this.get('parentView.controller');
|
||||||
|
const topic = this.get('topic');
|
||||||
|
|
||||||
|
const refresh = () => {
|
||||||
|
this._createContent();
|
||||||
|
this.set('value', null);
|
||||||
|
};
|
||||||
|
|
||||||
|
switch(value) {
|
||||||
|
case 'invite':
|
||||||
|
controller.send('showInvite');
|
||||||
|
refresh();
|
||||||
|
break;
|
||||||
|
case 'bookmark':
|
||||||
|
topic.toggleBookmark().then(() => refresh());
|
||||||
|
break;
|
||||||
|
case 'share':
|
||||||
|
this.appEvents.trigger('share:url', topic.get('shareUrl'), $('#topic-footer-buttons'));
|
||||||
|
refresh();
|
||||||
|
break;
|
||||||
|
case 'flag':
|
||||||
|
controller.send('showFlagTopic', topic);
|
||||||
|
refresh();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
|
@ -1,9 +1,4 @@
|
||||||
/**
|
// An object that is responsible for logic related to mobile devices.
|
||||||
An object that is responsible for logic related to mobile devices.
|
|
||||||
|
|
||||||
@namespace Discourse
|
|
||||||
@module Mobile
|
|
||||||
**/
|
|
||||||
Discourse.Mobile = {
|
Discourse.Mobile = {
|
||||||
isMobileDevice: false,
|
isMobileDevice: false,
|
||||||
mobileView: false,
|
mobileView: false,
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import ButtonView from 'discourse/views/button';
|
import ButtonView from 'discourse/views/button';
|
||||||
|
import { iconHTML } from 'discourse/helpers/fa-icon';
|
||||||
|
|
||||||
export default ButtonView.extend({
|
export default ButtonView.extend({
|
||||||
classNames: ['bookmark'],
|
classNames: ['bookmark'],
|
||||||
|
@ -16,12 +17,12 @@ export default ButtonView.extend({
|
||||||
return this.get("bookmarked") ? "bookmarked.help.unbookmark" : "bookmarked.help.bookmark";
|
return this.get("bookmarked") ? "bookmarked.help.unbookmark" : "bookmarked.help.bookmark";
|
||||||
}.property("bookmarked"),
|
}.property("bookmarked"),
|
||||||
|
|
||||||
click: function() {
|
click() {
|
||||||
this.get('controller').send('toggleBookmark');
|
this.get('controller').send('toggleBookmark');
|
||||||
},
|
},
|
||||||
|
|
||||||
renderIcon: function(buffer) {
|
renderIcon(buffer) {
|
||||||
var className = this.get("bookmarked") ? "bookmarked" : "";
|
const className = this.get("bookmarked") ? "bookmarked" : "";
|
||||||
buffer.push("<i class='fa fa-bookmark " + className + "'></i>");
|
buffer.push(iconHTML('bookmark', { class: className }));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,15 +1,16 @@
|
||||||
import ButtonView from 'discourse/views/button';
|
import ButtonView from 'discourse/views/button';
|
||||||
|
import { iconHTML } from 'discourse/helpers/fa-icon';
|
||||||
|
|
||||||
export default ButtonView.extend({
|
export default ButtonView.extend({
|
||||||
classNames: ['flag-topic'],
|
classNames: ['flag-topic'],
|
||||||
textKey: 'topic.flag_topic.title',
|
textKey: 'topic.flag_topic.title',
|
||||||
helpKey: 'topic.flag_topic.help',
|
helpKey: 'topic.flag_topic.help',
|
||||||
|
|
||||||
click: function() {
|
click() {
|
||||||
this.get('controller').send('showFlagTopic', this.get('controller.content'));
|
this.get('controller').send('showFlagTopic', this.get('controller.content'));
|
||||||
},
|
},
|
||||||
|
|
||||||
renderIcon: function(buffer) {
|
renderIcon(buffer) {
|
||||||
buffer.push("<i class='fa fa-flag'></i>");
|
buffer.push(iconHTML('flag'));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import ButtonView from 'discourse/views/button';
|
import ButtonView from 'discourse/views/button';
|
||||||
|
import { iconHTML } from 'discourse/helpers/fa-icon';
|
||||||
|
|
||||||
export default ButtonView.extend({
|
export default ButtonView.extend({
|
||||||
textKey: 'topic.invite_reply.title',
|
textKey: 'topic.invite_reply.title',
|
||||||
|
@ -7,7 +8,7 @@ export default ButtonView.extend({
|
||||||
disabled: Em.computed.or('controller.model.archived', 'controller.model.closed', 'controller.model.deleted'),
|
disabled: Em.computed.or('controller.model.archived', 'controller.model.closed', 'controller.model.deleted'),
|
||||||
|
|
||||||
renderIcon(buffer) {
|
renderIcon(buffer) {
|
||||||
buffer.push("<i class='fa fa-users'></i>");
|
buffer.push(iconHTML('users'));
|
||||||
},
|
},
|
||||||
|
|
||||||
click() {
|
click() {
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import ButtonView from 'discourse/views/button';
|
import ButtonView from 'discourse/views/button';
|
||||||
|
import { iconHTML } from 'discourse/helpers/fa-icon';
|
||||||
|
|
||||||
export default ButtonView.extend({
|
export default ButtonView.extend({
|
||||||
classNames: ['share'],
|
classNames: ['share'],
|
||||||
|
@ -7,8 +8,7 @@ export default ButtonView.extend({
|
||||||
'data-share-url': Em.computed.alias('topic.shareUrl'),
|
'data-share-url': Em.computed.alias('topic.shareUrl'),
|
||||||
topic: Em.computed.alias('controller.model'),
|
topic: Em.computed.alias('controller.model'),
|
||||||
|
|
||||||
renderIcon: function(buffer) {
|
renderIcon(buffer) {
|
||||||
buffer.push("<i class='fa fa-link'></i>");
|
buffer.push(iconHTML("link"));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -56,26 +56,17 @@ export default Ember.View.extend({
|
||||||
return true;
|
return true;
|
||||||
});
|
});
|
||||||
|
|
||||||
$html.on('click.discoure-share-link', '[data-share-url]', function(e) {
|
function showPanel($target, url, postNumber, date) {
|
||||||
// if they want to open in a new tab, let it so
|
const $currentTargetOffset = $target.offset();
|
||||||
if (e.shiftKey || e.metaKey || e.ctrlKey || e.which === 2) { return true; }
|
const $shareLink = $('#share-link');
|
||||||
|
|
||||||
e.preventDefault();
|
|
||||||
|
|
||||||
var $currentTarget = $(e.currentTarget),
|
|
||||||
$currentTargetOffset = $currentTarget.offset(),
|
|
||||||
$shareLink = $('#share-link'),
|
|
||||||
url = $currentTarget.data('share-url'),
|
|
||||||
postNumber = $currentTarget.data('post-number'),
|
|
||||||
date = $currentTarget.children().data('time');
|
|
||||||
|
|
||||||
// Relative urls
|
// Relative urls
|
||||||
if (url.indexOf("/") === 0) {
|
if (url.indexOf("/") === 0) {
|
||||||
url = window.location.protocol + "//" + window.location.host + url;
|
url = window.location.protocol + "//" + window.location.host + url;
|
||||||
}
|
}
|
||||||
|
|
||||||
var shareLinkWidth = $shareLink.width();
|
const shareLinkWidth = $shareLink.width();
|
||||||
var x = $currentTargetOffset.left - (shareLinkWidth / 2);
|
let x = $currentTargetOffset.left - (shareLinkWidth / 2);
|
||||||
if (x < 25) {
|
if (x < 25) {
|
||||||
x = 25;
|
x = 25;
|
||||||
}
|
}
|
||||||
|
@ -83,8 +74,8 @@ export default Ember.View.extend({
|
||||||
x -= shareLinkWidth / 2;
|
x -= shareLinkWidth / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
var header = $('.d-header');
|
const header = $('.d-header');
|
||||||
var y = $currentTargetOffset.top - ($shareLink.height() + 20);
|
let y = $currentTargetOffset.top - ($shareLink.height() + 20);
|
||||||
if (y < header.offset().top + header.height()) {
|
if (y < header.offset().top + header.height()) {
|
||||||
y = $currentTargetOffset.top + 10;
|
y = $currentTargetOffset.top + 10;
|
||||||
}
|
}
|
||||||
|
@ -98,7 +89,21 @@ export default Ember.View.extend({
|
||||||
self.set('controller.link', url);
|
self.set('controller.link', url);
|
||||||
self.set('controller.postNumber', postNumber);
|
self.set('controller.postNumber', postNumber);
|
||||||
self.set('controller.date', date);
|
self.set('controller.date', date);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.appEvents.on('share:url', (url, $target) => showPanel($target, url));
|
||||||
|
|
||||||
|
$html.on('click.discoure-share-link', '[data-share-url]', function(e) {
|
||||||
|
// if they want to open in a new tab, let it so
|
||||||
|
if (e.shiftKey || e.metaKey || e.ctrlKey || e.which === 2) { return true; }
|
||||||
|
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
const $currentTarget = $(e.currentTarget),
|
||||||
|
url = $currentTarget.data('share-url'),
|
||||||
|
postNumber = $currentTarget.data('post-number'),
|
||||||
|
date = $currentTarget.children().data('time');
|
||||||
|
showPanel($currentTarget, url, postNumber, date);
|
||||||
return false;
|
return false;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -6,13 +6,19 @@ export default ContainerView.extend({
|
||||||
|
|
||||||
@on('init')
|
@on('init')
|
||||||
createButtons() {
|
createButtons() {
|
||||||
if (this.currentUser.get('staff')) {
|
const mobileView = Discourse.Mobile.mobileView;
|
||||||
|
|
||||||
|
if (!mobileView && this.currentUser.get('staff')) {
|
||||||
const viewArgs = {action: 'showTopicAdminMenu', title: 'topic_admin_menu', icon: 'wrench', position: 'absolute'};
|
const viewArgs = {action: 'showTopicAdminMenu', title: 'topic_admin_menu', icon: 'wrench', position: 'absolute'};
|
||||||
this.attachViewWithArgs(viewArgs, 'show-popup-button');
|
this.attachViewWithArgs(viewArgs, 'show-popup-button');
|
||||||
}
|
}
|
||||||
|
|
||||||
const topic = this.get('topic');
|
const topic = this.get('topic');
|
||||||
if (!topic.get('isPrivateMessage')) {
|
if (!topic.get('isPrivateMessage')) {
|
||||||
|
|
||||||
|
if (mobileView) {
|
||||||
|
this.attachViewWithArgs({ topic }, 'topic-footer-mobile-dropdown');
|
||||||
|
} else {
|
||||||
// We hide some controls from private messages
|
// We hide some controls from private messages
|
||||||
if (this.get('topic.details.can_invite_to')) {
|
if (this.get('topic.details.can_invite_to')) {
|
||||||
this.attachViewClass('invite-reply-button');
|
this.attachViewClass('invite-reply-button');
|
||||||
|
@ -23,6 +29,8 @@ export default ContainerView.extend({
|
||||||
this.attachViewClass('flag-topic-button');
|
this.attachViewClass('flag-topic-button');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
if (this.get('topic.details.can_create_post')) {
|
if (this.get('topic.details.can_create_post')) {
|
||||||
this.attachViewClass('reply-button');
|
this.attachViewClass('reply-button');
|
||||||
}
|
}
|
||||||
|
|
|
@ -300,6 +300,13 @@ a.star {
|
||||||
border-top: 1px solid dark-light-diff($primary, $secondary, 90%, -60%);
|
border-top: 1px solid dark-light-diff($primary, $secondary, 90%, -60%);
|
||||||
padding: 20px 0 0 0;
|
padding: 20px 0 0 0;
|
||||||
.fa-bookmark.bookmarked { color: $tertiary; }
|
.fa-bookmark.bookmarked { color: $tertiary; }
|
||||||
|
|
||||||
|
.combobox {
|
||||||
|
float: left;
|
||||||
|
margin-right: 1em;
|
||||||
|
width: 160px;
|
||||||
|
margin-bottom: 0.5em;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* this is to force the drop-down notification state description para below the button */
|
/* this is to force the drop-down notification state description para below the button */
|
||||||
|
|
|
@ -1275,6 +1275,8 @@ en:
|
||||||
error: "Sorry, there was an error inviting that user."
|
error: "Sorry, there was an error inviting that user."
|
||||||
group_name: "group name"
|
group_name: "group name"
|
||||||
|
|
||||||
|
controls: "Topic Controls"
|
||||||
|
|
||||||
invite_reply:
|
invite_reply:
|
||||||
title: 'Invite'
|
title: 'Invite'
|
||||||
username_placeholder: "username"
|
username_placeholder: "username"
|
||||||
|
|
Loading…
Reference in New Issue