UNPKG

web-social-share

Version:

A Web Component to share urls and text on social networks

405 lines (389 loc) 19.2 kB
import { attachShadow, createEvent, h, proxyCustomElement } from '@stencil/core/internal/client'; export { setAssetPath, setPlatformOptions } from '@stencil/core/internal/client'; // Same implementation as in class @deckdeckgo/utils const isMobile = () => { return window === null || window === void 0 ? void 0 : window.matchMedia('(any-pointer:coarse)').matches; }; const staticOpenNewWindow = (urlString) => { if (window.self !== window.top) { window.open(urlString, '_blank'); } else { window.open(urlString, '_self'); } }; const shareEncodedUrl = (socialShareUrl) => { return encodeURIComponent(socialShareUrl || window.location.href); }; /** * Source: https://github.com/720kb/angular-socialshare/blob/master/dist/angular-socialshare.js */ const shareFacebook = async (attrs) => { let urlString; if (attrs.socialShareType && attrs.socialShareType === 'feed') { // if user specifies that they want to use the Facebook feed dialog //(https://developers.facebook.com/docs/sharing/reference/feed-dialog/v2.4) urlString = 'https://www.facebook.com/dialog/feed?'; if (attrs.socialShareVia) { urlString += '&app_id=' + encodeURIComponent(attrs.socialShareVia); } if (attrs.socialShareRedirectUri) { urlString += '&redirect_uri=' + encodeURIComponent(attrs.socialShareRedirectUri); } if (attrs.socialShareUrl) { urlString += '&link=' + shareEncodedUrl(attrs.socialShareUrl); } if (attrs.socialShareTo) { urlString += '&to=' + encodeURIComponent(attrs.socialShareTo); } if (attrs.socialShareDisplay) { urlString += '&display=' + encodeURIComponent(attrs.socialShareDisplay); } if (attrs.socialShareRef) { urlString += '&ref=' + encodeURIComponent(attrs.socialShareRef); } if (attrs.socialShareFrom) { urlString += '&from=' + encodeURIComponent(attrs.socialShareFrom); } if (attrs.socialShareSource) { urlString += '&source=' + encodeURIComponent(attrs.socialShareSource); } window.open(urlString, 'Facebook', 'toolbar=0,status=0,resizable=yes,width=' + attrs.socialSharePopupWidth + ',height=' + attrs.socialSharePopupHeight + ',top=' + (window.innerHeight - attrs.socialSharePopupHeight) / 2 + ',left=' + (window.innerWidth - attrs.socialSharePopupWidth) / 2); } else if (attrs.socialShareType && attrs.socialShareType === 'share') { // if user specifies that they want to use the Facebook share dialog //(https://developers.facebook.com/docs/sharing/reference/share-dialog) urlString = 'https://www.facebook.com/dialog/share?'; if (attrs.socialShareVia) { urlString += '&app_id=' + encodeURIComponent(attrs.socialShareVia); } if (attrs.socialShareRedirectUri) { urlString += '&redirect_uri=' + encodeURIComponent(attrs.socialShareRedirectUri); } if (attrs.socialShareUrl) { urlString += '&href=' + shareEncodedUrl(attrs.socialShareUrl); } if (attrs.socialShareQuote) { urlString += '&quote=' + encodeURIComponent(attrs.socialShareQuote); } if (attrs.socialShareDisplay) { urlString += '&display=' + encodeURIComponent(attrs.socialShareDisplay); } if (attrs.socialShareMobileiframe) { urlString += '&mobile_iframe=' + encodeURIComponent(attrs.socialShareMobileiframe); } if (attrs.socialShareHashtags) { urlString += '&hashtag=' + encodeURIComponent(attrs.socialShareHashtags); } window.open(urlString, 'Facebook', 'toolbar=0,status=0,resizable=yes,width=' + attrs.socialSharePopupWidth + ',height=' + attrs.socialSharePopupHeight + ',top=' + (window.innerHeight - attrs.socialSharePopupHeight) / 2 + ',left=' + (window.innerWidth - attrs.socialSharePopupWidth) / 2); } else if (attrs.socialShareType && attrs.socialShareType === 'send') { // if user specifies that they want to use the Facebook send dialog //(https://developers.facebook.com/docs/sharing/reference/send-dialog) urlString = 'https://www.facebook.com/dialog/send?'; if (attrs.socialShareVia) { urlString += '&app_id=' + encodeURIComponent(attrs.socialShareVia); } if (attrs.socialShareRedirectUri) { urlString += '&redirect_uri=' + encodeURIComponent(attrs.socialShareRedirectUri); } if (attrs.socialShareUrl) { urlString += '&link=' + shareEncodedUrl(attrs.socialShareUrl); } if (attrs.socialShareTo) { urlString += '&to=' + encodeURIComponent(attrs.socialShareTo); } if (attrs.socialShareDisplay) { urlString += '&display=' + encodeURIComponent(attrs.socialShareDisplay); } window.open(urlString, 'Facebook', 'toolbar=0,status=0,resizable=yes,width=' + attrs.socialSharePopupWidth + ',height=' + attrs.socialSharePopupHeight + ',top=' + (window.innerHeight - attrs.socialSharePopupHeight) / 2 + ',left=' + (window.innerWidth - attrs.socialSharePopupWidth) / 2); } else { //otherwise default to using sharer.php window.open('https://www.facebook.com/sharer/sharer.php?u=' + shareEncodedUrl(attrs.socialShareUrl), 'Facebook', 'toolbar=0,status=0,resizable=yes,width=' + attrs.socialSharePopupWidth + ',height=' + attrs.socialSharePopupHeight + ',top=' + (window.innerHeight - attrs.socialSharePopupHeight) / 2 + ',left=' + (window.innerWidth - attrs.socialSharePopupWidth) / 2); } }; const shareTwitter = async (attrs) => { let urlString = 'https://www.twitter.com/intent/tweet?'; if (attrs.socialShareText) { urlString += 'text=' + encodeURIComponent(attrs.socialShareText); } if (attrs.socialShareVia) { urlString += '&via=' + encodeURIComponent(attrs.socialShareVia); } if (attrs.socialShareHashtags) { urlString += '&hashtags=' + encodeURIComponent(attrs.socialShareHashtags); } //default to the current page if a URL isn't specified urlString += '&url=' + shareEncodedUrl(attrs.socialShareUrl); if (isMobile()) { staticOpenNewWindow(urlString); } else { window.open(urlString, 'Twitter', 'toolbar=0,status=0,resizable=yes,width=' + attrs.socialSharePopupWidth + ',height=' + attrs.socialSharePopupHeight + ',top=' + (window.innerHeight - attrs.socialSharePopupHeight) / 2 + ',left=' + (window.innerWidth - attrs.socialSharePopupWidth) / 2); } }; const email = async (attrs) => { let urlString = 'mailto:'; if (attrs.socialShareTo) { urlString += encodeURIComponent(attrs.socialShareTo); } urlString += '?'; if (attrs.socialShareBody) { urlString += 'body=' + encodeURIComponent(attrs.socialShareBody); } if (attrs.socialShareSubject) { urlString += '&subject=' + encodeURIComponent(attrs.socialShareSubject); } if (attrs.socialShareCc) { urlString += '&cc=' + encodeURIComponent(attrs.socialShareCc); } if (attrs.socialShareBcc) { urlString += '&bcc=' + encodeURIComponent(attrs.socialShareBcc); } staticOpenNewWindow(urlString); }; const linkedin = async (attrs) => { let urlString = 'https://www.linkedin.com/shareArticle?mini=true'; urlString += '&url=' + shareEncodedUrl(attrs.socialShareUrl); if (attrs.socialShareText) { urlString += '&title=' + encodeURIComponent(attrs.socialShareText); } if (attrs.socialShareDescription) { urlString += '&summary=' + encodeURIComponent(attrs.socialShareDescription); } if (attrs.socialShareSource) { urlString += '&source=' + encodeURIComponent(attrs.socialShareSource); } window.open(urlString, 'Linkedin', 'toolbar=0,status=0,resizable=yes,width=' + attrs.socialSharePopupWidth + ',height=' + attrs.socialSharePopupHeight + ',top=' + (window.innerHeight - attrs.socialSharePopupHeight) / 2 + ',left=' + (window.innerWidth - attrs.socialSharePopupWidth) / 2); }; const pinterest = async (attrs) => { window.open('https://www.pinterest.com/pin/create/button/?url=' + shareEncodedUrl(attrs.socialShareUrl) + '&media=' + encodeURIComponent(attrs.socialShareMedia) + '&description=' + encodeURIComponent(attrs.socialShareText), 'Pinterest', 'toolbar=0,status=0,resizable=yes,width=' + attrs.socialSharePopupWidth + ',height=' + attrs.socialSharePopupHeight + ',top=' + (window.innerHeight - attrs.socialSharePopupHeight) / 2 + ',left=' + (window.innerWidth - attrs.socialSharePopupWidth) / 2); }; const reddit = async (attrs) => { let urlString = 'https://www.reddit.com/'; if (attrs.socialShareSubreddit) { urlString += 'r/' + attrs.socialShareSubreddit + '/submit?url='; } else { urlString += 'submit?url='; } /*- * Reddit isn't responsive and at default width for our popups (500 x 500), everything is messed up. * So, overriding the width if it is less than 900 (played around to settle on this) and height if * it is less than 650px. */ if (attrs.socialSharePopupWidth < 900) { attrs.socialSharePopupWidth = 900; } if (attrs.socialSharePopupHeight < 650) { attrs.socialSharePopupHeight = 650; } window.open(urlString + shareEncodedUrl(attrs.socialShareUrl) + '&title=' + encodeURIComponent(attrs.socialShareText), 'Reddit', 'toolbar=0,status=0,resizable=yes,width=' + attrs.socialSharePopupWidth + ',height=' + attrs.socialSharePopupHeight + ',top=' + (window.innerHeight - attrs.socialSharePopupHeight) / 2 + ',left=' + (window.innerWidth - attrs.socialSharePopupWidth) / 2); }; const whatsapp = async (attrs) => { const mobile = isMobile(); let urlString = mobile ? 'https://api.whatsapp.com/send?text=' : 'https://web.whatsapp.com/send?text='; if (attrs.socialShareText) { urlString += encodeURIComponent(attrs.socialShareText) + '%0A'; } //default to the current page if a URL isn't specified urlString += shareEncodedUrl(attrs.socialShareUrl); if (mobile) { staticOpenNewWindow(urlString); } else { window.open(urlString, 'WhatsApp', 'toolbar=0,status=0,resizable=yes,width=' + attrs.socialSharePopupWidth + ',height=' + attrs.socialSharePopupHeight + ',top=' + (window.innerHeight - attrs.socialSharePopupHeight) / 2 + ',left=' + (window.innerWidth - attrs.socialSharePopupWidth) / 2); } }; const copy = async (attrs) => { try { await navigator.clipboard.writeText(attrs.socialShareUrl || window.location.href); } catch (err) { console.error('Well it seems that copy is not supported by this browser'); } }; const hackernews = async (attrs) => { let urlString = 'https://news.ycombinator.com/submitlink?u='; //default to the current page if a URL isn't specified urlString += shareEncodedUrl(attrs.socialShareUrl); if (attrs.socialShareText) { urlString += '&t=' + encodeURIComponent(attrs.socialShareText); } staticOpenNewWindow(urlString); }; const telegram = async (attrs) => { let urlString = `https://t.me/share/url?url=${shareEncodedUrl(attrs.socialShareUrl)}`; if (attrs.socialShareText) { urlString += `&text=${encodeURIComponent(attrs.socialShareText)}`; } staticOpenNewWindow(urlString); }; const webSocialShareCss = "div.web-social-share{visibility:hidden;opacity:0;cursor:pointer;touch-action:manipulation}div.web-social-share.web-social-share-open{visibility:visible;opacity:1}div.web-social-share.web-social-share-open div.web-social-share-backdrop{opacity:var(--web-social-share-backdrop-opacity, 0.25)}div.web-social-share.web-social-share-open div.web-social-share-action-sheet{opacity:1}div.web-social-share.web-social-share-open div.web-social-share-action-sheet div.web-social-share-action-sheet-container div.web-social-share-action-sheet-group{height:var(--web-social-share-height, 80px)}@media (max-width: 540px){div.web-social-share.web-social-share-open div.web-social-share-action-sheet div.web-social-share-action-sheet-container div.web-social-share-action-sheet-group{height:var(--web-social-share-height-small-device, 140px)}}div.web-social-share.web-social-share-open.web-social-share-transition-close div.web-social-share-backdrop{opacity:0}div.web-social-share.web-social-share-open.web-social-share-transition-close div.web-social-share-action-sheet div.web-social-share-action-sheet-container div.web-social-share-action-sheet-group{height:0}div.web-social-share div.web-social-share-backdrop{opacity:0;transition:opacity 0.1s linear;background-color:var(--web-social-share-backdrop-background, black);z-index:var(--web-social-share-zindex, 1000);transform:translate3d(0, 0, 2px);left:0;top:0;position:fixed;height:100%;width:100%}div.web-social-share div.web-social-share-action-sheet{left:0;right:0;top:0;bottom:0;margin:auto;position:fixed;z-index:calc(var(--web-social-share-zindex, 1000) + 1);transform:translate3d(0, 0, 3px);width:100%;max-width:540px}@media (min-width: 540px){div.web-social-share div.web-social-share-action-sheet div.web-social-share-action-sheet-container div.web-social-share-action-sheet-group{border-radius:var(--web-social-share-action-sheet-group-border-radius, 8px 8px 0 0)}}div.web-social-share div.web-social-share-action-sheet div.web-social-share-action-sheet-container{display:flex;flex-flow:column;justify-content:flex-end;height:100%;max-height:100%}div.web-social-share div.web-social-share-action-sheet div.web-social-share-action-sheet-container div.web-social-share-action-sheet-group{box-shadow:var(--web-social-share-action-sheet-group-box-shadow, 0 0 8px 4px rgba(0, 0, 0, 0.1));z-index:calc(var(--web-social-share-zindex, 1000) + 10);transform:translate3d(0, 0, 10px);background:var(--web-social-share-action-sheet-group-background, #fafafa);display:flex;justify-content:center;flex-wrap:wrap;height:0;transition-timing-function:cubic-bezier(0.36, 0.66, 0.04, 1);transition:height 0.2s ease-in}@media (max-width: 540px){div.web-social-share div.web-social-share-action-sheet div.web-social-share-action-sheet-container div.web-social-share-action-sheet-group{justify-content:flex-start}}div.web-social-share div.web-social-share-target{margin:auto;width:var(--web-social-share-target-width, 4rem);height:var(--web-social-share-target-height, 3rem);display:flex;flex-direction:column;align-items:center;justify-content:center}div.web-social-share div.web-social-share-target button{position:relative;cursor:pointer;border:0;background:transparent;width:var(--web-social-share-button-width, 100%);height:var(--web-social-share-button-height, 100%);font-size:var(--web-social-share-button-font-size)}div.web-social-share div.web-social-share-target p{margin:var(--web-social-share-brand-margin, 2px 0);color:var(--web-social-share-brand-color, inherit);font-size:var(--web-social-share-brand-font-size, 0.6rem)}div.web-social-share div.web-social-share-target div.web-social-share-button-icon{margin:0;display:flex;flex-direction:column;justify-content:center;align-items:center}::slotted([slot=facebook]),::slotted([slot=twitter]),::slotted([slot=email]),::slotted([slot=linkedin]),::slotted([slot=pinterest]),::slotted([slot=reddit]),::slotted([slot=whatsapp]),::slotted([slot=copy]),::slotted([slot=hackernews]){display:none}"; const WebSocialShare$1 = class extends HTMLElement { constructor() { super(); this.__registerHost(); attachShadow(this); this.closed = createEvent(this, "closed", 7); } hide() { if (this.refContainer) { this.refContainer.classList.add('web-social-share-transition-close'); setTimeout(() => { // Reflect css animation speed 400ms, see css this.show = false; this.refContainer.classList.remove('web-social-share-transition-close'); this.closed.emit(); }, 200); } else { // Well we don't find the action sheet, we could mark it as not displayed this.show = false; this.closed.emit(); } } async handleShare($event, attributes, action) { $event.stopPropagation(); await action(attributes); this.hide(); } render() { return (h("div", { class: this.show ? 'web-social-share web-social-share-open' : 'web-social-share web-social-share-close', ref: (el) => (this.refContainer = el) }, h("div", { class: "web-social-share-backdrop", onClick: () => this.hide() }), h("div", { class: "web-social-share-action-sheet", onClick: () => this.hide() }, h("div", { class: "web-social-share-action-sheet-container" }, h("div", { class: "web-social-share-action-sheet-group" }, this.renderTargets()))))); } renderTargets() { var _a; if (!((_a = this.share) === null || _a === void 0 ? void 0 : _a.config)) { return undefined; } return this.share.config.map((config) => (h("div", { class: "web-social-share-target" }, this.renderButtons(config)))); } renderButtons(share) { if (share.facebook) { return this.renderButton(share.facebook, 'facebook', shareFacebook, 'Facebook'); } else if (share.twitter) { return this.renderButton(share.twitter, 'twitter', shareTwitter, 'Twitter'); } else if (share.email) { return this.renderButton(share.email, 'email', email, 'Email'); } else if (share.linkedin) { return this.renderButton(share.linkedin, 'linkedin', linkedin, 'Linkedin'); } else if (share.pinterest) { return this.renderButton(share.pinterest, 'pinterest', pinterest, 'Pinterest'); } else if (share.reddit) { return this.renderButton(share.reddit, 'reddit', reddit, 'Reddit'); } else if (share.whatsapp) { return this.renderButton(share.whatsapp, 'whatsapp', whatsapp, 'WhatsApp'); } else if (share.copy) { return this.renderButton(share.copy, 'copy', copy, 'Copy'); } else if (share.hackernews) { return this.renderButton(share.hackernews, 'hackernews', hackernews, 'Hacker News'); } else if (share.telegram) { return this.renderButton(share.telegram, 'telegram', telegram, 'Telegram'); } return undefined; } renderButton(attributes, slotName, action, defaultBrandName) { return (h("button", { onClick: ($event) => this.handleShare($event, attributes, action), class: "web-social-share-button" }, h("div", { class: "web-social-share-button-icon" }, h("slot", { name: slotName })), this.renderName(attributes, defaultBrandName))); } renderName(displayAttributes, defaultBrandName) { if (this.share.displayNames) { return (h("p", null, displayAttributes && displayAttributes.brandName && displayAttributes.brandName !== '' ? displayAttributes.brandName : defaultBrandName)); } return undefined; } get el() { return this; } static get style() { return webSocialShareCss; } }; const WebSocialShare = /*@__PURE__*/proxyCustomElement(WebSocialShare$1, [1,"web-social-share",{"show":[1028],"share":[16]}]); const defineCustomElements = (opts) => { if (typeof customElements !== 'undefined') { [ WebSocialShare ].forEach(cmp => { if (!customElements.get(cmp.is)) { customElements.define(cmp.is, cmp, opts); } }); } }; export { WebSocialShare, defineCustomElements };