UX: puts back share-panel as floating pane on post actions (#7066)

This commit is contained in:
Joffrey JAFFEUX 2019-02-26 14:15:25 +01:00 committed by GitHub
parent 6ea9f5c9c5
commit d04c4bf8e7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
14 changed files with 367 additions and 186 deletions

View File

@ -1,15 +1,12 @@
import { escapeExpression } from "discourse/lib/utilities"; import { escapeExpression } from "discourse/lib/utilities";
import { longDateNoYear } from "discourse/lib/formatter";
import { default as computed } from "ember-addons/ember-computed-decorators"; import { default as computed } from "ember-addons/ember-computed-decorators";
import Sharing from "discourse/lib/sharing"; import Sharing from "discourse/lib/sharing";
export default Ember.Component.extend({ export default Ember.Component.extend({
tagName: null, tagName: null,
date: Ember.computed.alias("panel.model.date"),
type: Ember.computed.alias("panel.model.type"), type: Ember.computed.alias("panel.model.type"),
postNumber: Ember.computed.alias("panel.model.postNumber"),
postId: Ember.computed.alias("panel.model.postId"),
topic: Ember.computed.alias("panel.model.topic"), topic: Ember.computed.alias("panel.model.topic"),
@computed @computed
@ -17,21 +14,9 @@ export default Ember.Component.extend({
return Sharing.activeSources(this.siteSettings.share_links); return Sharing.activeSources(this.siteSettings.share_links);
}, },
@computed("date") @computed("type", "topic.title")
postDate(date) { shareTitle(type, topicTitle) {
return date ? longDateNoYear(new Date(date)) : null;
},
@computed("type", "postNumber", "postDate", "topic.title")
shareTitle(type, postNumber, postDate, topicTitle) {
topicTitle = escapeExpression(topicTitle); topicTitle = escapeExpression(topicTitle);
if (type === "topic") {
return I18n.t("share.topic_html", { topicTitle });
}
if (postNumber) {
return I18n.t("share.post_html", { postNumber, postDate });
}
return I18n.t("share.topic_html", { topicTitle }); return I18n.t("share.topic_html", { topicTitle });
}, },
@ -81,27 +66,10 @@ export default Ember.Component.extend({
actions: { actions: {
share(source) { share(source) {
const url = source.generateUrl( Sharing.shareSource(source, {
this.get("shareUrl"), url: this.get("shareUrl"),
this.get("topic.title") title: this.get("topic.title")
); });
const options = {
menubar: "no",
toolbar: "no",
resizable: "yes",
scrollbars: "yes",
width: 600,
height: source.popupHeight || 315
};
const stringOptions = Object.keys(options)
.map(k => `${k}=${options[k]}`)
.join(",");
if (source.shouldOpenInPopup) {
window.open(url, "", stringOptions);
} else {
window.open(url, "_blank");
}
} }
} }
}); });

View File

@ -0,0 +1,173 @@
import { wantsNewWindow } from "discourse/lib/intercept-click";
import { longDateNoYear } from "discourse/lib/formatter";
import computed from "ember-addons/ember-computed-decorators";
import Sharing from "discourse/lib/sharing";
import { nativeShare } from "discourse/lib/pwa-utils";
export default Ember.Component.extend({
elementId: "share-link",
classNameBindings: ["visible"],
link: null,
visible: null,
@computed
sources() {
return Sharing.activeSources(this.siteSettings.share_links);
},
@computed("type", "postNumber")
shareTitle(type, postNumber) {
if (type === "topic") {
return I18n.t("share.topic");
}
if (postNumber) {
return I18n.t("share.post", { postNumber });
}
return I18n.t("share.topic");
},
@computed("date")
displayDate(date) {
return longDateNoYear(new Date(date));
},
_focusUrl() {
const link = this.get("link");
if (!this.capabilities.touch) {
const $linkInput = $("#share-link input");
$linkInput.val(link);
// Wait for the fade-in transition to finish before selecting the link:
window.setTimeout(() => $linkInput.select().focus(), 160);
} else {
const $linkForTouch = $("#share-link .share-for-touch a");
$linkForTouch.attr("href", link);
$linkForTouch.text(link);
const range = window.document.createRange();
range.selectNode($linkForTouch[0]);
window.getSelection().addRange(range);
}
},
_showUrl($target, url) {
const $currentTargetOffset = $target.offset();
const $this = this.$();
if (Ember.isEmpty(url)) {
return;
}
// Relative urls
if (url.indexOf("/") === 0) {
url = window.location.protocol + "//" + window.location.host + url;
}
const shareLinkWidth = $this.width();
let x = $currentTargetOffset.left - shareLinkWidth / 2;
if (x < 25) {
x = 25;
}
if (x + shareLinkWidth > $(window).width()) {
x -= shareLinkWidth / 2;
}
const header = $(".d-header");
let y = $currentTargetOffset.top - ($this.height() + 20);
if (y < header.offset().top + header.height()) {
y = $currentTargetOffset.top + 10;
}
$this.css({ top: "" + y + "px" });
if (!this.site.mobileView) {
$this.css({ left: "" + x + "px" });
}
this.set("link", encodeURI(url));
this.set("visible", true);
Ember.run.scheduleOnce("afterRender", this, this._focusUrl);
},
didInsertElement() {
this._super(...arguments);
const $html = $("html");
$html.on("mousedown.outside-share-link", e => {
// Use mousedown instead of click so this event is handled before routing occurs when a
// link is clicked (which is a click event) while the share dialog is showing.
if (this.$().has(e.target).length !== 0) {
return;
}
this.send("close");
return true;
});
$html.on(
"click.discourse-share-link",
"button[data-share-url], .post-info .post-date[data-share-url]",
e => {
// if they want to open in a new tab, let it so
if (wantsNewWindow(e)) {
return true;
}
e.preventDefault();
const $currentTarget = $(e.currentTarget);
const url = $currentTarget.data("share-url");
const postNumber = $currentTarget.data("post-number");
const postId = $currentTarget.closest("article").data("post-id");
const date = $currentTarget.children().data("time");
this.setProperties({ postNumber, date, postId });
// use native webshare only when the user clicks on the "chain" icon
if (!$currentTarget.hasClass("post-date")) {
nativeShare({ url }).then(null, () =>
this._showUrl($currentTarget, url)
);
} else {
this._showUrl($currentTarget, url);
}
return false;
}
);
$html.on("keydown.share-view", e => {
if (e.keyCode === 27) {
this.send("close");
}
});
this.appEvents.on("share:url", (url, $target) =>
this._showUrl($target, url)
);
},
willDestroyElement() {
this._super(...arguments);
$("html")
.off("click.discourse-share-link")
.off("mousedown.outside-share-link")
.off("keydown.share-view");
},
actions: {
close() {
this.setProperties({
link: null,
postNumber: null,
postId: null,
visible: false
});
},
share(source) {
Sharing.shareSource(source, {
url: this.get("link"),
title: this.get("topic.title")
});
}
}
});

View File

@ -47,6 +47,27 @@ export default {
_sources[source.id] = source; _sources[source.id] = source;
}, },
shareSource(source, data) {
const url = source.generateUrl(data.url, data.title);
const options = {
menubar: "no",
toolbar: "no",
resizable: "yes",
scrollbars: "yes",
width: 600,
height: source.popupHeight || 315
};
const stringOptions = Object.keys(options)
.map(k => `${k}=${options[k]}`)
.join(",");
if (source.shouldOpenInPopup) {
window.open(url, "", stringOptions);
} else {
window.open(url, "_blank");
}
},
activeSources(linksSetting = "") { activeSources(linksSetting = "") {
return linksSetting return linksSetting
.split("|") .split("|")

View File

@ -0,0 +1,24 @@
<div class="title">
<h3>{{{shareTitle}}}</h3>
{{#if date}}
<span class="date">{{displayDate}}</span>
{{/if}}
</div>
<div>
<input type="text">
<div class="share-for-touch"><div class="overflow-ellipsis"><a></a></div></div>
</div>
<div class="actions">
{{#each sources as |s|}}
{{share-source source=s title=model.title action=(action "share")}}
{{/each}}
<div class="link">
<a href {{action "close"}} class="close-share" aria-label={{i18n "share.close"}} title={{i18n "share.close"}}>
{{d-icon "times"}}
</a>
</div>
</div>

View File

@ -318,6 +318,8 @@
</div> </div>
{{/if}} {{/if}}
{{share-popup topic=model}}
{{#if embedQuoteButton}} {{#if embedQuoteButton}}
{{quote-button quoteState=quoteState selectText=(action "selectText")}} {{quote-button quoteState=quoteState selectText=(action "selectText")}}
{{/if}} {{/if}}

View File

@ -2,7 +2,6 @@ import PostCooked from "discourse/widgets/post-cooked";
import DecoratorHelper from "discourse/widgets/decorator-helper"; import DecoratorHelper from "discourse/widgets/decorator-helper";
import { createWidget, applyDecorators } from "discourse/widgets/widget"; import { createWidget, applyDecorators } from "discourse/widgets/widget";
import { iconNode } from "discourse-common/lib/icon-library"; import { iconNode } from "discourse-common/lib/icon-library";
import { nativeShare } from "discourse/lib/pwa-utils";
import { transformBasicPost } from "discourse/lib/transform-post"; import { transformBasicPost } from "discourse/lib/transform-post";
import { postTransformCallbacks } from "discourse/widgets/post-stream"; import { postTransformCallbacks } from "discourse/widgets/post-stream";
import { h } from "virtual-dom"; import { h } from "virtual-dom";
@ -14,7 +13,6 @@ import {
formatUsername formatUsername
} from "discourse/lib/utilities"; } from "discourse/lib/utilities";
import hbs from "discourse/widgets/hbs-compiler"; import hbs from "discourse/widgets/hbs-compiler";
import showModal from "discourse/lib/show-modal";
function transformWithCallbacks(post) { function transformWithCallbacks(post) {
let transformed = transformBasicPost(post); let transformed = transformBasicPost(post);
@ -221,72 +219,6 @@ function showReplyTab(attrs, siteSettings) {
); );
} }
createWidget("post-date", {
tagName: "div.post-info.post-date",
buildClasses(attrs) {
let classes = "post-date";
const lastWikiEdit =
attrs.wiki && attrs.lastWikiEdit && new Date(attrs.lastWikiEdit);
if (lastWikiEdit) {
classes = `${classes} last-wiki-edit`;
}
return classes;
},
html(attrs) {
return h(
"a",
{
attributes: {
class: "post-date",
href: attrs.shareUrl,
"data-share-url": attrs.shareUrl,
"data-post-number": attrs.post_number
}
},
dateNode(this._date(attrs))
);
},
_date(attrs) {
const lastWikiEdit =
attrs.wiki && attrs.lastWikiEdit && new Date(attrs.lastWikiEdit);
const createdAt = new Date(attrs.created_at);
return lastWikiEdit ? lastWikiEdit : createdAt;
},
click(event) {
event.preventDefault();
const post = this.findAncestorModel();
const modalFallback = () => {
showModal("share-and-invite", {
modalClass: "share-and-invite",
panels: [
{
id: "share",
title: "topic.share.extended_title",
model: {
postNumber: this.attrs.post_number,
shareUrl: this.attrs.shareUrl,
date: this._date(this.attrs),
postId: post.get("id"),
topic: post.get("topic")
}
}
]
});
};
nativeShare({ url: this.attrs.shareUrl }).then(null, modalFallback);
}
});
createWidget("post-meta-data", { createWidget("post-meta-data", {
tagName: "div.topic-meta-data", tagName: "div.topic-meta-data",
@ -309,6 +241,21 @@ createWidget("post-meta-data", {
); );
} }
const lastWikiEdit =
attrs.wiki && attrs.lastWikiEdit && new Date(attrs.lastWikiEdit);
const createdAt = new Date(attrs.created_at);
const date = lastWikiEdit ? dateNode(lastWikiEdit) : dateNode(createdAt);
const attributes = {
class: "post-date",
href: attrs.shareUrl,
"data-share-url": attrs.shareUrl,
"data-post-number": attrs.post_number
};
if (lastWikiEdit) {
attributes["class"] += " last-wiki-edit";
}
if (attrs.via_email) { if (attrs.via_email) {
postInfo.push(this.attach("post-email-indicator", attrs)); postInfo.push(this.attach("post-email-indicator", attrs));
} }
@ -329,7 +276,7 @@ createWidget("post-meta-data", {
postInfo.push(this.attach("reply-to-tab", attrs)); postInfo.push(this.attach("reply-to-tab", attrs));
} }
postInfo.push(this.attach("post-date", attrs)); postInfo.push(h("div.post-info.post-date", h("a", { attributes }, date)));
postInfo.push( postInfo.push(
h( h(
@ -451,31 +398,6 @@ createWidget("post-contents", {
return lastWikiEdit ? lastWikiEdit : createdAt; return lastWikiEdit ? lastWikiEdit : createdAt;
}, },
share() {
const post = this.findAncestorModel();
const modalFallback = () => {
showModal("share-and-invite", {
modalClass: "share-and-invite",
panels: [
{
id: "share",
title: "topic.share.extended_title",
model: {
postNumber: this.attrs.post_number,
shareUrl: this.attrs.shareUrl,
date: this._date(this.attrs),
postId: post.get("id"),
topic: post.get("topic")
}
}
]
});
};
nativeShare({ url: this.attrs.shareUrl }).then(null, modalFallback);
},
toggleRepliesBelow(goToPost = "false") { toggleRepliesBelow(goToPost = "false") {
if (this.state.repliesBelow.length) { if (this.state.repliesBelow.length) {
this.state.repliesBelow = []; this.state.repliesBelow = [];

View File

@ -646,7 +646,9 @@
background: $danger; background: $danger;
&.single-tab { &.single-tab {
display: none; background: none;
color: $primary;
padding: s(1 0);
} }
} }
} }

View File

@ -0,0 +1,87 @@
// styles that apply to the "share" popup when sharing a link to a post or topic
#share-link {
position: absolute;
left: 20px;
z-index: z("dropdown");
box-shadow: shadow("card");
background-color: $secondary;
padding: s(2 2 1 2);
width: 300px;
display: none;
&.visible {
display: block;
}
input[type="text"] {
width: 100%;
}
.share-for-touch .overflow-ellipsis {
clear: both;
}
.share-for-touch {
margin: 14px 0;
}
.title {
margin-bottom: s(1);
align-items: center;
display: flex;
justify-content: space-between;
h3 {
font-size: $font-0;
margin: 0;
}
.date {
font-weight: normal;
color: dark-light-choose($primary-medium, $secondary-medium);
}
}
.copy-text {
display: inline-block;
position: absolute;
margin: 5px 5px 5px 15px;
color: $success;
opacity: 1;
transition: opacity 0.25s;
font-size: $font-0;
&:not(.success) {
opacity: 0;
}
}
.social-link {
margin-right: s(2);
font-size: $font-up-4;
}
.link {
font-size: $font-up-3;
a {
color: dark-light-choose($primary-medium, $secondary-medium);
}
}
input[type="text"] {
font-size: $font-up-1;
margin-bottom: 0;
}
.actions {
display: flex;
align-items: center;
margin-top: s(2);
.link {
margin-left: auto;
}
}
}
.discourse-no-touch #share-link .share-for-touch {
display: none;
}
.discourse-touch #share-link input[type="text"] {
display: none;
}

View File

@ -136,7 +136,7 @@ en:
placeholder: date placeholder: date
share: share:
topic_html: 'Topic: <span class="topic-title">%{topicTitle}</span>' topic_html: 'Topic: <span class="topic-title">%{topicTitle}</span>'
post_html: '<span class="post-number">Post #%{postNumber}</span>, <span class="post-date">%{postDate}</span>' post: "post #%{postNumber}"
close: "close" close: "close"
twitter: "Share this link on Twitter" twitter: "Share this link on Twitter"
facebook: "Share this link on Facebook" facebook: "Share this link on Facebook"

View File

@ -68,34 +68,5 @@ QUnit.test("Post date link", async assert => {
await visit("/t/internationalization-localization/280"); await visit("/t/internationalization-localization/280");
await click("#post_2 .post-info.post-date a"); await click("#post_2 .post-info.post-date a");
assert.ok(exists(".share-and-invite.modal"), "it shows the modal"); assert.ok(exists("#share-link"), "it shows the share modal");
assert.ok(
exists(".share-and-invite.modal .modal-tab.share"),
"it shows the share tab"
);
assert.ok(
exists(".share-and-invite.modal .modal-tab.share.single-tab"),
"it shows only one tab"
);
assert.ok(
!exists(".share-and-invite.modal .modal-tab.invite"),
"it doesnt show the invite tab"
);
assert.ok(
find(".share-and-invite.modal .modal-panel.share .title")
.text()
.includes("Post #2, Feb"),
"it shows the post number with the date"
);
assert.ok(
find(".share-and-invite.modal .modal-panel.share .topic-share-url")
.val()
.includes("/t/internationalization-localization/280/2?u=eviltrout"),
"it shows the topic sharing url including the post number"
);
}); });

View File

@ -57,20 +57,5 @@ QUnit.test("Post date link", async assert => {
await visit("/t/internationalization-localization/280"); await visit("/t/internationalization-localization/280");
await click("#post_2 .post-info.post-date a"); await click("#post_2 .post-info.post-date a");
assert.ok(exists(".share-and-invite.modal"), "it shows the modal"); assert.ok(exists("#share-link"), "it shows the share modal");
assert.ok(
exists(".share-and-invite.modal .modal-tab.share"),
"it shows the share tab"
);
assert.ok(
exists(".share-and-invite.modal .modal-tab.share.single-tab"),
"it shows only one tab"
);
assert.ok(
!exists(".share-and-invite.modal .modal-tab.invite"),
"it doesnt show the invite tab"
);
}); });

View File

@ -24,12 +24,9 @@ acceptance("Topic", {
QUnit.test("Share Modal", async assert => { QUnit.test("Share Modal", async assert => {
await visit("/t/internationalization-localization/280"); await visit("/t/internationalization-localization/280");
await click(".topic-post:first-child button.share"); await click(".topic-post:first-child button.share");
assert.ok(
exists(".modal.share-and-invite"), assert.ok(exists("#share-link"), "it shows the share modal");
"it shows the share and invite modal"
);
}); });
QUnit.test("Showing and hiding the edit controls", async assert => { QUnit.test("Showing and hiding the edit controls", async assert => {

View File

@ -0,0 +1,16 @@
import componentTest from "helpers/component-test";
moduleForComponent("share-button", { integration: true });
componentTest("share button", {
template: '{{share-button url="https://eviltrout.com"}}',
test(assert) {
assert.ok(find(`button.share`).length, "it has all the classes");
assert.ok(
find('button[data-share-url="https://eviltrout.com"]').length,
"it has the data attribute for sharing"
);
}
});

View File

@ -0,0 +1,13 @@
import Button from "discourse/components/d-button";
export default Button.extend({
classNames: ["btn-default", "share"],
icon: "link",
title: "topic.share.help",
label: "topic.share.title",
attributeBindings: ["url:data-share-url"],
click() {
return true;
}
});