allow user display names to contain custom emoji (#448)
* allow user display names to contain custom emoji fixes #445 * fix tests * fix focus issue
This commit is contained in:
parent
c660c7d3a3
commit
350667e5df
|
@ -0,0 +1,7 @@
|
|||
import { WRITE_TIMEOUT, patch } from '../_utils/ajax'
|
||||
import { auth, basename } from './utils'
|
||||
|
||||
export async function updateCredentials (instanceName, accessToken, accountData) {
|
||||
let url = `${basename(instanceName)}/api/v1/accounts/update_credentials`
|
||||
return patch(url, accountData, auth(accessToken), {timeout: WRITE_TIMEOUT})
|
||||
}
|
|
@ -4,7 +4,7 @@
|
|||
<Avatar account={verifyCredentials} size="small"/>
|
||||
</a>
|
||||
<a class="compose-box-display-name" href="/accounts/{verifyCredentials.id}">
|
||||
{verifyCredentials.display_name || verifyCredentials.acct}
|
||||
<AccountDisplayName account={verifyCredentials} />
|
||||
</a>
|
||||
<span class="compose-box-handle">
|
||||
{'@' + verifyCredentials.acct}
|
||||
|
@ -51,9 +51,12 @@
|
|||
<script>
|
||||
import Avatar from '../Avatar.html'
|
||||
import { store } from '../../_store/store'
|
||||
import AccountDisplayName from '../profile/AccountDisplayName.html'
|
||||
|
||||
export default {
|
||||
components: {
|
||||
Avatar
|
||||
Avatar,
|
||||
AccountDisplayName
|
||||
},
|
||||
store: () => store,
|
||||
computed: {
|
||||
|
|
|
@ -12,7 +12,7 @@
|
|||
account={item}
|
||||
/>
|
||||
<span class="compose-autosuggest-list-display-name">
|
||||
{item.display_name || item.acct}
|
||||
<AccountDisplayName account={item} />
|
||||
</span>
|
||||
<span class="compose-autosuggest-list-username">
|
||||
{'@' + item.acct}
|
||||
|
@ -99,6 +99,7 @@
|
|||
<script>
|
||||
import Avatar from '../Avatar.html'
|
||||
import { store } from '../../_store/store'
|
||||
import AccountDisplayName from '../profile/AccountDisplayName.html'
|
||||
|
||||
export default {
|
||||
store: () => store,
|
||||
|
@ -110,7 +111,8 @@
|
|||
}
|
||||
},
|
||||
components: {
|
||||
Avatar
|
||||
Avatar,
|
||||
AccountDisplayName
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
|
|
@ -0,0 +1,23 @@
|
|||
<span class="account-display-name">{@html massagedAccountName }</span>
|
||||
<style>
|
||||
.account-display-name {
|
||||
pointer-events: none; /* allows focus to work correctly, focus on the parent only */
|
||||
}
|
||||
</style>
|
||||
<script>
|
||||
import { emojifyText } from '../../_utils/emojifyText'
|
||||
import { store } from '../../_store/store'
|
||||
import escapeHtml from 'escape-html'
|
||||
|
||||
export default {
|
||||
store: () => store,
|
||||
computed: {
|
||||
emojis: ({ account }) => (account.emojis || []),
|
||||
accountName: ({ account }) => (account.display_name || account.username),
|
||||
massagedAccountName: ({ accountName, emojis, $autoplayGifs }) => {
|
||||
accountName = escapeHtml(accountName)
|
||||
return emojifyText(accountName, emojis, $autoplayGifs)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
|
@ -7,7 +7,7 @@
|
|||
normalIconColor="true"
|
||||
ariaLabel="{account.display_name || account.acct} (opens in new window)"
|
||||
>
|
||||
{account.display_name || account.acct}
|
||||
<AccountDisplayName {account} />
|
||||
</ExternalLink>
|
||||
</div>
|
||||
<div class="account-profile-username">
|
||||
|
@ -80,11 +80,13 @@
|
|||
<script>
|
||||
import Avatar from '../Avatar.html'
|
||||
import ExternalLink from '../ExternalLink.html'
|
||||
import AccountDisplayName from '../profile/AccountDisplayName.html'
|
||||
|
||||
export default {
|
||||
components: {
|
||||
Avatar,
|
||||
ExternalLink
|
||||
ExternalLink,
|
||||
AccountDisplayName
|
||||
}
|
||||
}
|
||||
</script>
|
|
@ -2,7 +2,7 @@
|
|||
<div class="search-result-account">
|
||||
<Avatar {account} size="small" className="search-result-account-avatar"/>
|
||||
<div class="search-result-account-name">
|
||||
{account.display_name || account.acct}
|
||||
<AccountDisplayName {account} />
|
||||
</div>
|
||||
<div class="search-result-account-username">
|
||||
{'@' + account.acct}
|
||||
|
@ -71,6 +71,7 @@
|
|||
import Avatar from '../Avatar.html'
|
||||
import SearchResult from './SearchResult.html'
|
||||
import IconButton from '../IconButton.html'
|
||||
import AccountDisplayName from '../profile/AccountDisplayName.html'
|
||||
|
||||
export default {
|
||||
data: () => ({
|
||||
|
@ -89,7 +90,8 @@
|
|||
components: {
|
||||
Avatar,
|
||||
SearchResult,
|
||||
IconButton
|
||||
IconButton,
|
||||
AccountDisplayName
|
||||
}
|
||||
}
|
||||
</script>
|
|
@ -3,7 +3,7 @@
|
|||
title="{'@' + originalAccount.acct}"
|
||||
focus-key={focusKey}
|
||||
>
|
||||
{originalAccount.display_name || originalAccount.username}
|
||||
<AccountDisplayName account={originalAccount} />
|
||||
</a>
|
||||
<style>
|
||||
.status-author-name {
|
||||
|
@ -34,9 +34,14 @@
|
|||
|
||||
</style>
|
||||
<script>
|
||||
import AccountDisplayName from '../profile/AccountDisplayName.html'
|
||||
|
||||
export default {
|
||||
computed: {
|
||||
focusKey: ({ uuid }) => `status-author-name-${uuid}`
|
||||
},
|
||||
components: {
|
||||
AccountDisplayName
|
||||
}
|
||||
}
|
||||
</script>
|
|
@ -21,14 +21,6 @@
|
|||
display: block;
|
||||
}
|
||||
|
||||
:global(.status-content .status-emoji) {
|
||||
width: 1.4em;
|
||||
height: 1.4em;
|
||||
margin: -0.1em 0;
|
||||
object-fit: contain;
|
||||
vertical-align: middle;
|
||||
}
|
||||
|
||||
:global(.status-content p) {
|
||||
margin: 0 0 20px;
|
||||
}
|
||||
|
|
|
@ -16,7 +16,7 @@
|
|||
class="status-header-author"
|
||||
title="{'@' + account.acct}"
|
||||
focus-key={focusKey} >
|
||||
{account.display_name || account.username}
|
||||
<AccountDisplayName {account} />
|
||||
</a>
|
||||
{/if}
|
||||
|
||||
|
@ -103,10 +103,12 @@
|
|||
</style>
|
||||
<script>
|
||||
import Avatar from '../Avatar.html'
|
||||
import AccountDisplayName from '../profile/AccountDisplayName.html'
|
||||
|
||||
export default {
|
||||
components: {
|
||||
Avatar
|
||||
Avatar,
|
||||
AccountDisplayName
|
||||
},
|
||||
computed: {
|
||||
focusKey: ({ uuid }) => `status-header-${uuid}`,
|
||||
|
|
|
@ -16,14 +16,6 @@
|
|||
margin: 10px 5px;
|
||||
}
|
||||
|
||||
:global(.status-spoiler .status-emoji) {
|
||||
width: 1.4em;
|
||||
height: 1.4em;
|
||||
margin: -0.1em 0;
|
||||
object-fit: contain;
|
||||
vertical-align: middle;
|
||||
}
|
||||
|
||||
.status-spoiler.status-in-own-thread {
|
||||
font-size: 1.3em;
|
||||
margin: 20px 5px 10px;
|
||||
|
|
|
@ -9,7 +9,9 @@
|
|||
href={verifyCredentials.url}>
|
||||
{'@' + verifyCredentials.acct}
|
||||
</ExternalLink>
|
||||
<span class="acct-display-name">{verifyCredentials.display_name || verifyCredentials.acct}</span>
|
||||
<span class="acct-display-name">
|
||||
<AccountDisplayName account={verifyCredentials} />
|
||||
</span>
|
||||
</div>
|
||||
<h2>Theme:</h2>
|
||||
<form class="theme-chooser" aria-label="Choose a theme">
|
||||
|
@ -103,6 +105,7 @@
|
|||
updateVerifyCredentialsForInstance
|
||||
} from '../../../_actions/instances'
|
||||
import { themes } from '../../../_static/themes'
|
||||
import AccountDisplayName from '../../../_components/profile/AccountDisplayName.html'
|
||||
|
||||
export default {
|
||||
async oncreate () {
|
||||
|
@ -148,7 +151,8 @@
|
|||
components: {
|
||||
SettingsLayout,
|
||||
ExternalLink,
|
||||
Avatar
|
||||
Avatar,
|
||||
AccountDisplayName
|
||||
}
|
||||
}
|
||||
</script>
|
|
@ -39,7 +39,7 @@ async function _fetch (url, fetchOptions, options) {
|
|||
return throwErrorIfInvalidResponse(response)
|
||||
}
|
||||
|
||||
async function _putOrPost (method, url, body, headers, options) {
|
||||
async function _putOrPostOrPatch (method, url, body, headers, options) {
|
||||
let fetchOptions = makeFetchOptions(method, headers)
|
||||
if (body) {
|
||||
if (body instanceof FormData) {
|
||||
|
@ -53,11 +53,15 @@ async function _putOrPost (method, url, body, headers, options) {
|
|||
}
|
||||
|
||||
export async function put (url, body, headers, options) {
|
||||
return _putOrPost('PUT', url, body, headers, options)
|
||||
return _putOrPostOrPatch('PUT', url, body, headers, options)
|
||||
}
|
||||
|
||||
export async function post (url, body, headers, options) {
|
||||
return _putOrPost('POST', url, body, headers, options)
|
||||
return _putOrPostOrPatch('POST', url, body, headers, options)
|
||||
}
|
||||
|
||||
export async function patch (url, body, headers, options) {
|
||||
return _putOrPostOrPatch('PATCH', url, body, headers, options)
|
||||
}
|
||||
|
||||
export async function get (url, headers, options) {
|
||||
|
|
|
@ -8,7 +8,7 @@ export function emojifyText (text, emojis, autoplayGifs) {
|
|||
text = replaceAll(
|
||||
text,
|
||||
shortcodeWithColons,
|
||||
`<img class="status-emoji" draggable="false" src="${urlToUse}"
|
||||
`<img class="inline-custom-emoji" draggable="false" src="${urlToUse}"
|
||||
alt="${shortcodeWithColons}" title="${shortcodeWithColons}" />`
|
||||
)
|
||||
}
|
||||
|
|
|
@ -198,3 +198,12 @@ textarea {
|
|||
clip: rect(0, 0, 0, 0);
|
||||
border: 0;
|
||||
}
|
||||
|
||||
/* this gets injected as raw HTML, so it's easiest to just define it in global.scss */
|
||||
.inline-custom-emoji {
|
||||
width: 1.4em;
|
||||
height: 1.4em;
|
||||
margin: -0.1em 0;
|
||||
object-fit: contain;
|
||||
vertical-align: middle;
|
||||
}
|
|
@ -17,7 +17,7 @@
|
|||
<style>
|
||||
/* auto-generated w/ build-sass.js */
|
||||
body{--button-primary-bg: #6081e6;--button-primary-text: #fff;--button-primary-border: #132c76;--button-primary-bg-active: #456ce2;--button-primary-bg-hover: #6988e7;--button-bg: #e6e6e6;--button-text: #333;--button-border: #a7a7a7;--button-bg-active: #bfbfbf;--button-bg-hover: #f2f2f2;--input-border: #dadada;--anchor-text: #4169e1;--main-bg: #fff;--body-bg: #e8edfb;--body-text-color: #333;--main-border: #dadada;--svg-fill: #4169e1;--form-bg: #f7f7f7;--form-border: #c1c1c1;--nav-bg: #4169e1;--nav-border: #214cce;--nav-a-border: #4169e1;--nav-a-selected-border: #fff;--nav-a-selected-bg: #6d8ce8;--nav-svg-fill: #fff;--nav-text-color: #fff;--nav-a-selected-border-hover: #fff;--nav-a-selected-bg-hover: #839deb;--nav-a-bg-hover: #577ae4;--nav-a-border-hover: #4169e1;--nav-svg-fill-hover: #fff;--nav-text-color-hover: #fff;--action-button-fill-color: #90a8ee;--action-button-fill-color-hover: #a2b6f0;--action-button-fill-color-active: #577ae4;--action-button-fill-color-pressed: #2351dc;--action-button-fill-color-pressed-hover: #3862e0;--action-button-fill-color-pressed-active: #1d44b8;--action-button-deemphasized-fill-color: #666;--action-button-deemphasized-fill-color-hover: #9e9e9e;--action-button-deemphasized-fill-color-active: #737373;--action-button-deemphasized-fill-color-pressed: #545454;--action-button-deemphasized-fill-color-pressed-hover: #616161;--action-button-deemphasized-fill-color-pressed-active: #404040;--settings-list-item-bg: #fff;--settings-list-item-text: #4169e1;--settings-list-item-text-hover: #4169e1;--settings-list-item-border: #dadada;--settings-list-item-bg-active: #e6e6e6;--settings-list-item-bg-hover: #fafafa;--toast-bg: #333;--toast-border: #fafafa;--toast-text: #fff;--mask-bg: #333;--mask-svg-fill: #fff;--mask-opaque-bg: rgba(51,51,51,0.8);--loading-bg: #ededed;--account-profile-bg-backdrop-filter: rgba(255,255,255,0.7);--account-profile-bg: rgba(255,255,255,0.9);--deemphasized-text-color: #666;--focus-outline: #c5d1f6;--very-deemphasized-link-color: rgba(65,105,225,0.6);--very-deemphasized-text-color: rgba(102,102,102,0.6);--status-direct-background: #d2dcf8;--main-theme-color: #4169e1;--warning-color: #e01f19;--alt-input-bg: rgba(255,255,255,0.7);--muted-modal-bg: transparent;--muted-modal-focus: #999;--muted-modal-hover: rgba(255,255,255,0.2);--compose-autosuggest-item-hover: #ced8f7;--compose-autosuggest-item-active: #b8c7f4;--compose-autosuggest-outline: #dbe3f9;--compose-button-halo: rgba(255,255,255,0.1)}
|
||||
body{margin:0;font-family:system-ui, -apple-system, BlinkMacSystemFont, Segoe UI, Roboto, Oxygen, Ubuntu, Cantarell, Fira Sans, Droid Sans, Helvetica Neue, sans-serif;font-size:14px;line-height:1.4;color:var(--body-text-color);background:var(--body-bg);-webkit-tap-highlight-color:transparent}.container{overflow-y:auto;overflow-x:hidden;-webkit-overflow-scrolling:touch;will-change:transform;position:absolute;top:42px;left:0;right:0;bottom:0}@media (max-width: 991px){.container{top:52px}}@media (max-width: 767px){.container{top:62px}}main{position:relative;width:602px;max-width:100vw;padding:0;box-sizing:border-box;margin:30px auto 15px;background:var(--main-bg);border:1px solid var(--main-border);border-radius:1px;min-height:70vh}@media (max-width: 767px){main{margin:5px auto 15px}}footer{width:602px;max-width:100vw;box-sizing:border-box;margin:15px auto;border-radius:1px;background:var(--main-bg);font-size:0.9em;padding:20px;border:1px solid var(--main-border)}h1,h2,h3,h4,h5,h6{margin:0 0 0.5em 0;font-weight:400;line-height:1.2}h1{font-size:2em}a{color:var(--anchor-text);text-decoration:none}a:visited{color:var(--anchor-text)}a:hover{text-decoration:underline}input{border:1px solid var(--input-border);padding:5px;box-sizing:border-box}button,.button{font-size:1.2em;background:var(--button-bg);border-radius:2px;padding:10px 15px;border:1px solid var(--button-border);cursor:pointer;color:var(--button-text)}button:hover,.button:hover{background:var(--button-bg-hover);text-decoration:none}button:active,.button:active{background:var(--button-bg-active)}button[disabled],.button[disabled]{opacity:0.35;pointer-events:none;cursor:not-allowed}button.primary,.button.primary{border:1px solid var(--button-primary-border);background:var(--button-primary-bg);color:var(--button-primary-text)}button.primary:hover,.button.primary:hover{background:var(--button-primary-bg-hover)}button.primary:active,.button.primary:active{background:var(--button-primary-bg-active)}p,label,input{font-size:1.3em}ul,li,p{padding:0;margin:0}.hidden{opacity:0}*:focus{outline:2px solid var(--focus-outline)}.container:focus{outline:none}button::-moz-focus-inner{border:0}input:required,input:invalid{box-shadow:none}textarea{font-family:inherit;font-size:inherit;box-sizing:border-box}@keyframes spin{0%{transform:rotate(0deg)}25%{transform:rotate(90deg)}50%{transform:rotate(180deg)}75%{transform:rotate(270deg)}100%{transform:rotate(360deg)}}.spin{animation:spin 1.5s infinite linear}.ellipsis::after{content:"\2026"}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0, 0, 0, 0);border:0}
|
||||
body{margin:0;font-family:system-ui, -apple-system, BlinkMacSystemFont, Segoe UI, Roboto, Oxygen, Ubuntu, Cantarell, Fira Sans, Droid Sans, Helvetica Neue, sans-serif;font-size:14px;line-height:1.4;color:var(--body-text-color);background:var(--body-bg);-webkit-tap-highlight-color:transparent}.container{overflow-y:auto;overflow-x:hidden;-webkit-overflow-scrolling:touch;will-change:transform;position:absolute;top:42px;left:0;right:0;bottom:0}@media (max-width: 991px){.container{top:52px}}@media (max-width: 767px){.container{top:62px}}main{position:relative;width:602px;max-width:100vw;padding:0;box-sizing:border-box;margin:30px auto 15px;background:var(--main-bg);border:1px solid var(--main-border);border-radius:1px;min-height:70vh}@media (max-width: 767px){main{margin:5px auto 15px}}footer{width:602px;max-width:100vw;box-sizing:border-box;margin:15px auto;border-radius:1px;background:var(--main-bg);font-size:0.9em;padding:20px;border:1px solid var(--main-border)}h1,h2,h3,h4,h5,h6{margin:0 0 0.5em 0;font-weight:400;line-height:1.2}h1{font-size:2em}a{color:var(--anchor-text);text-decoration:none}a:visited{color:var(--anchor-text)}a:hover{text-decoration:underline}input{border:1px solid var(--input-border);padding:5px;box-sizing:border-box}button,.button{font-size:1.2em;background:var(--button-bg);border-radius:2px;padding:10px 15px;border:1px solid var(--button-border);cursor:pointer;color:var(--button-text)}button:hover,.button:hover{background:var(--button-bg-hover);text-decoration:none}button:active,.button:active{background:var(--button-bg-active)}button[disabled],.button[disabled]{opacity:0.35;pointer-events:none;cursor:not-allowed}button.primary,.button.primary{border:1px solid var(--button-primary-border);background:var(--button-primary-bg);color:var(--button-primary-text)}button.primary:hover,.button.primary:hover{background:var(--button-primary-bg-hover)}button.primary:active,.button.primary:active{background:var(--button-primary-bg-active)}p,label,input{font-size:1.3em}ul,li,p{padding:0;margin:0}.hidden{opacity:0}*:focus{outline:2px solid var(--focus-outline)}.container:focus{outline:none}button::-moz-focus-inner{border:0}input:required,input:invalid{box-shadow:none}textarea{font-family:inherit;font-size:inherit;box-sizing:border-box}@keyframes spin{0%{transform:rotate(0deg)}25%{transform:rotate(90deg)}50%{transform:rotate(180deg)}75%{transform:rotate(270deg)}100%{transform:rotate(360deg)}}.spin{animation:spin 1.5s infinite linear}.ellipsis::after{content:"\2026"}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0, 0, 0, 0);border:0}.inline-custom-emoji{width:1.4em;height:1.4em;margin:-0.1em 0;object-fit:contain;vertical-align:middle}
|
||||
body.offline,body.theme-hotpants.offline,body.theme-majesty.offline,body.theme-oaken.offline,body.theme-scarlet.offline,body.theme-seafoam.offline,body.theme-gecko.offline,body.theme-ozark.offline,body.theme-cobalt.offline,body.theme-sorcery.offline{--button-primary-bg: #ababab;--button-primary-text: #fff;--button-primary-border: #4d4d4d;--button-primary-bg-active: #9c9c9c;--button-primary-bg-hover: #b0b0b0;--button-bg: #e6e6e6;--button-text: #333;--button-border: #a7a7a7;--button-bg-active: #bfbfbf;--button-bg-hover: #f2f2f2;--input-border: #dadada;--anchor-text: #999;--main-bg: #fff;--body-bg: #fafafa;--body-text-color: #333;--main-border: #dadada;--svg-fill: #999;--form-bg: #f7f7f7;--form-border: #c1c1c1;--nav-bg: #999;--nav-border: gray;--nav-a-border: #999;--nav-a-selected-border: #fff;--nav-a-selected-bg: #b3b3b3;--nav-svg-fill: #fff;--nav-text-color: #fff;--nav-a-selected-border-hover: #fff;--nav-a-selected-bg-hover: #bfbfbf;--nav-a-bg-hover: #a6a6a6;--nav-a-border-hover: #999;--nav-svg-fill-hover: #fff;--nav-text-color-hover: #fff;--action-button-fill-color: #c7c7c7;--action-button-fill-color-hover: #d1d1d1;--action-button-fill-color-active: #a6a6a6;--action-button-fill-color-pressed: #878787;--action-button-fill-color-pressed-hover: #949494;--action-button-fill-color-pressed-active: #737373;--action-button-deemphasized-fill-color: #666;--action-button-deemphasized-fill-color-hover: #9e9e9e;--action-button-deemphasized-fill-color-active: #737373;--action-button-deemphasized-fill-color-pressed: #545454;--action-button-deemphasized-fill-color-pressed-hover: #616161;--action-button-deemphasized-fill-color-pressed-active: #404040;--settings-list-item-bg: #fff;--settings-list-item-text: #999;--settings-list-item-text-hover: #999;--settings-list-item-border: #dadada;--settings-list-item-bg-active: #e6e6e6;--settings-list-item-bg-hover: #fafafa;--toast-bg: #333;--toast-border: #fafafa;--toast-text: #fff;--mask-bg: #333;--mask-svg-fill: #fff;--mask-opaque-bg: rgba(51,51,51,0.8);--loading-bg: #ededed;--account-profile-bg-backdrop-filter: rgba(255,255,255,0.7);--account-profile-bg: rgba(255,255,255,0.9);--deemphasized-text-color: #666;--focus-outline: #bfbfbf;--very-deemphasized-link-color: rgba(153,153,153,0.6);--very-deemphasized-text-color: rgba(102,102,102,0.6);--status-direct-background: #ededed;--main-theme-color: #999;--warning-color: #e01f19;--alt-input-bg: rgba(255,255,255,0.7);--muted-modal-bg: transparent;--muted-modal-focus: #999;--muted-modal-hover: rgba(255,255,255,0.2);--compose-autosuggest-item-hover: #c4c4c4;--compose-autosuggest-item-active: #b8b8b8;--compose-autosuggest-outline: #ccc;--compose-button-halo: rgba(255,255,255,0.1)}
|
||||
|
||||
</style>
|
||||
|
|
|
@ -6,6 +6,7 @@ import { postStatus } from '../routes/_api/statuses'
|
|||
import { deleteStatus } from '../routes/_api/delete'
|
||||
import { authorizeFollowRequest, getFollowRequests } from '../routes/_actions/followRequests'
|
||||
import { followAccount, unfollowAccount } from '../routes/_api/follow'
|
||||
import { updateCredentials } from '../routes/_api/updateCredentials'
|
||||
|
||||
global.fetch = fetch
|
||||
global.File = FileApi.File
|
||||
|
@ -46,3 +47,7 @@ export async function followAs (username, userToFollow) {
|
|||
export async function unfollowAs (username, userToFollow) {
|
||||
return unfollowAccount(instanceName, users[username].accessToken, users[userToFollow].id)
|
||||
}
|
||||
|
||||
export async function updateUserDisplayNameAs (username, displayName) {
|
||||
return updateCredentials(instanceName, users[username].accessToken, {display_name: displayName})
|
||||
}
|
||||
|
|
|
@ -37,7 +37,7 @@ test('converts external links in profiles', async t => {
|
|||
.hover(getNthStatus(0))
|
||||
.navigateTo('/accounts/4')
|
||||
.expect(getUrl()).contains('/accounts/4')
|
||||
.expect($('.account-profile-name').innerText).eql('External Lonk')
|
||||
.expect($('.account-profile-name').innerText).contains('External Lonk')
|
||||
.expect($('.account-profile-name a').getAttribute('href')).eql('http://localhost:3000/@ExternalLinks')
|
||||
.expect($('.account-profile-name a').getAttribute('rel')).eql('nofollow noopener')
|
||||
.expect(getAnchorInProfile(0).getAttribute('href')).eql('https://joinmastodon.org')
|
||||
|
|
|
@ -44,9 +44,9 @@ test('content warnings can have emoji', async t => {
|
|||
.typeText(composeContentWarning, 'can you feel the :blobpats: tonight')
|
||||
.click(composeButton)
|
||||
.expect(getNthStatus(0).innerText).contains('can you feel the', {timeout: 30000})
|
||||
.expect($(`${getNthStatusSelector(0)} .status-spoiler img.status-emoji`).getAttribute('alt')).eql(':blobpats:')
|
||||
.expect($(`${getNthStatusSelector(0)} .status-spoiler img.inline-custom-emoji`).getAttribute('alt')).eql(':blobpats:')
|
||||
.click(getNthShowOrHideButton(0))
|
||||
.expect($(`${getNthStatusSelector(0)} .status-content img.status-emoji`).getAttribute('alt')).eql(':blobnom:')
|
||||
.expect($(`${getNthStatusSelector(0)} .status-content img.inline-custom-emoji`).getAttribute('alt')).eql(':blobnom:')
|
||||
})
|
||||
|
||||
test('no XSS in content warnings or text', async t => {
|
||||
|
|
|
@ -0,0 +1,27 @@
|
|||
import { loginAsFoobar } from '../roles'
|
||||
import { displayNameInComposeBox, getNthStatusSelector, getUrl, sleep } from '../utils'
|
||||
import { updateUserDisplayNameAs } from '../serverActions'
|
||||
import { Selector as $ } from 'testcafe'
|
||||
|
||||
fixture`118-display-name-custom-emoji.js`
|
||||
.page`http://localhost:4002`
|
||||
|
||||
test('Can put custom emoji in display name', async t => {
|
||||
await updateUserDisplayNameAs('foobar', 'foobar :blobpats:')
|
||||
await sleep(1000)
|
||||
await loginAsFoobar(t)
|
||||
await t
|
||||
.expect(displayNameInComposeBox.innerText).eql('foobar ')
|
||||
.expect($('.compose-box-display-name img').getAttribute('alt')).eql(':blobpats:')
|
||||
.click(displayNameInComposeBox)
|
||||
.expect(getUrl()).contains('/accounts/2')
|
||||
.expect($(`${getNthStatusSelector(0)} .status-author-name img`).getAttribute('alt')).eql(':blobpats:')
|
||||
})
|
||||
|
||||
test('Cannot XSS using display name HTML', async t => {
|
||||
await updateUserDisplayNameAs('foobar', '<script>alert("pwn")</script>')
|
||||
await sleep(1000)
|
||||
await loginAsFoobar(t)
|
||||
await t
|
||||
.expect(displayNameInComposeBox.innerText).eql('<script>alert("pwn")</script>')
|
||||
})
|
|
@ -40,6 +40,7 @@ export const mastodonLogInButton = $('button[type="submit"]')
|
|||
export const followsButton = $('.account-profile-details > *:nth-child(2)')
|
||||
export const followersButton = $('.account-profile-details > *:nth-child(3)')
|
||||
export const avatarInComposeBox = $('.compose-box-avatar')
|
||||
export const displayNameInComposeBox = $('.compose-box-display-name')
|
||||
|
||||
export const favoritesCountElement = $('.status-favs-reblogs:nth-child(3)').addCustomDOMProperties({
|
||||
innerCount: el => parseInt(el.innerText, 10)
|
||||
|
|
Loading…
Reference in New Issue