From 7a152fbdac38742a7ee9c73f77d8888e3a05e2f0 Mon Sep 17 00:00:00 2001
From: Nolan Lawson <nolan@nolanlawson.com>
Date: Mon, 18 Feb 2019 15:43:41 -0800
Subject: [PATCH] feat: allow muting notifications when muting (#1013)

fixes #738
---
 src/routes/_actions/mute.js                   |  4 +-
 src/routes/_actions/toggleMute.js             | 10 +++
 src/routes/_api/mute.js                       |  4 +-
 src/routes/_components/dialog/asyncDialogs.js |  8 +-
 .../AccountProfileOptionsDialog.html          |  6 +-
 .../dialog/components/ConfirmationDialog.html | 26 +++++--
 .../components/GenericConfirmationDialog.html | 75 +++++++++++++++++++
 .../dialog/components/MuteDialog.html         | 60 +++++++++++++++
 .../components/StatusOptionsDialog.html       |  6 +-
 .../components/TextConfirmationDialog.html    | 38 ++++++++++
 .../dialog/creators/showMuteDialog.js         | 16 ++++
 ...ialog.js => showTextConfirmationDialog.js} |  7 +-
 .../settings/instance/InstanceActions.html    | 13 ++--
 .../instance/PushNotificationSettings.html    | 13 ++--
 src/routes/_pages/muted.html                  |  5 +-
 tests/spec/002-login-spec.js                  |  6 +-
 tests/spec/114-mute-unmute.js                 | 19 ++++-
 tests/utils.js                                |  1 +
 18 files changed, 276 insertions(+), 41 deletions(-)
 create mode 100644 src/routes/_actions/toggleMute.js
 create mode 100644 src/routes/_components/dialog/components/GenericConfirmationDialog.html
 create mode 100644 src/routes/_components/dialog/components/MuteDialog.html
 create mode 100644 src/routes/_components/dialog/components/TextConfirmationDialog.html
 create mode 100644 src/routes/_components/dialog/creators/showMuteDialog.js
 rename src/routes/_components/dialog/creators/{showConfirmationDialog.js => showTextConfirmationDialog.js} (59%)

diff --git a/src/routes/_actions/mute.js b/src/routes/_actions/mute.js
index 5c38702..549bc06 100644
--- a/src/routes/_actions/mute.js
+++ b/src/routes/_actions/mute.js
@@ -4,12 +4,12 @@ import { toast } from '../_components/toast/toast'
 import { updateLocalRelationship } from './accounts'
 import { emit } from '../_utils/eventBus'
 
-export async function setAccountMuted (accountId, mute, toastOnSuccess) {
+export async function setAccountMuted (accountId, mute, notifications, toastOnSuccess) {
   let { currentInstance, accessToken } = store.get()
   try {
     let relationship
     if (mute) {
-      relationship = await muteAccount(currentInstance, accessToken, accountId)
+      relationship = await muteAccount(currentInstance, accessToken, accountId, notifications)
     } else {
       relationship = await unmuteAccount(currentInstance, accessToken, accountId)
     }
diff --git a/src/routes/_actions/toggleMute.js b/src/routes/_actions/toggleMute.js
new file mode 100644
index 0000000..4db5930
--- /dev/null
+++ b/src/routes/_actions/toggleMute.js
@@ -0,0 +1,10 @@
+import { importShowMuteDialog } from '../_components/dialog/asyncDialogs'
+import { setAccountMuted } from './mute'
+
+export async function toggleMute (account, mute) {
+  if (mute) {
+    (await importShowMuteDialog())(account)
+  } else {
+    await setAccountMuted(account.id, mute, /* notifications */ false, /* toastOnSuccess */ true)
+  }
+}
diff --git a/src/routes/_api/mute.js b/src/routes/_api/mute.js
index c090313..64d22e8 100644
--- a/src/routes/_api/mute.js
+++ b/src/routes/_api/mute.js
@@ -1,9 +1,9 @@
 import { auth, basename } from './utils'
 import { post, WRITE_TIMEOUT } from '../_utils/ajax'
 
-export async function muteAccount (instanceName, accessToken, accountId) {
+export async function muteAccount (instanceName, accessToken, accountId, notifications) {
   let url = `${basename(instanceName)}/api/v1/accounts/${accountId}/mute`
-  return post(url, null, auth(accessToken), { timeout: WRITE_TIMEOUT })
+  return post(url, { notifications }, auth(accessToken), { timeout: WRITE_TIMEOUT })
 }
 
 export async function unmuteAccount (instanceName, accessToken, accountId) {
diff --git a/src/routes/_components/dialog/asyncDialogs.js b/src/routes/_components/dialog/asyncDialogs.js
index eb1b950..0620010 100644
--- a/src/routes/_components/dialog/asyncDialogs.js
+++ b/src/routes/_components/dialog/asyncDialogs.js
@@ -8,8 +8,8 @@ export const importShowComposeDialog = () => import(
   /* webpackChunkName: 'showComposeDialog' */ './creators/showComposeDialog'
   ).then(getDefault)
 
-export const importShowConfirmationDialog = () => import(
-  /* webpackChunkName: 'showConfirmationDialog' */ './creators/showConfirmationDialog'
+export const importShowTextConfirmationDialog = () => import(
+  /* webpackChunkName: 'showTextConfirmationDialog' */ './creators/showTextConfirmationDialog'
   ).then(getDefault)
 
 export const importShowEmojiDialog = () => import(
@@ -35,3 +35,7 @@ export const importShowShortcutHelpDialog = () => import(
 export const importShowMediaDialog = () => import(
   /* webpackChunkName: 'showMediaDialog' */ './creators/showMediaDialog'
   ).then(getDefault)
+
+export const importShowMuteDialog = () => import(
+  /* webpackChunkName: 'showMuteDialog' */ './creators/showMuteDialog'
+  ).then(getDefault)
diff --git a/src/routes/_components/dialog/components/AccountProfileOptionsDialog.html b/src/routes/_components/dialog/components/AccountProfileOptionsDialog.html
index 0b449ce..fa4aa57 100644
--- a/src/routes/_components/dialog/components/AccountProfileOptionsDialog.html
+++ b/src/routes/_components/dialog/components/AccountProfileOptionsDialog.html
@@ -15,12 +15,12 @@ import { show } from '../helpers/showDialog'
 import { close } from '../helpers/closeDialog'
 import { oncreate } from '../helpers/onCreateDialog'
 import { setAccountBlocked } from '../../../_actions/block'
-import { setAccountMuted } from '../../../_actions/mute'
 import { setAccountFollowed } from '../../../_actions/follow'
 import { setShowReblogs } from '../../../_actions/setShowReblogs'
 import { setDomainBlocked } from '../../../_actions/setDomainBlocked'
 import { copyText } from '../../../_actions/copyText'
 import { composeNewStatusMentioning } from '../../../_actions/mention'
+import { toggleMute } from '../../../_actions/toggleMute'
 
 export default {
   oncreate,
@@ -155,9 +155,9 @@ export default {
       await setAccountBlocked(accountId, !blocking, true)
     },
     async onMuteClicked () {
-      let { accountId, muting } = this.get()
+      let { account, muting } = this.get()
       this.close()
-      await setAccountMuted(accountId, !muting, true)
+      await toggleMute(account, !muting)
     },
     async onShowReblogsClicked () {
       let { accountId, showingReblogs } = this.get()
diff --git a/src/routes/_components/dialog/components/ConfirmationDialog.html b/src/routes/_components/dialog/components/ConfirmationDialog.html
index 120485c..9fde2ff 100644
--- a/src/routes/_components/dialog/components/ConfirmationDialog.html
+++ b/src/routes/_components/dialog/components/ConfirmationDialog.html
@@ -1,18 +1,21 @@
 <ModalDialog
   {id}
   {label}
+  {title}
   background="var(--main-bg)"
 >
   <form class="confirmation-dialog-form">
-    <p>
-      {text}
-    </p>
+    {#if component}
+      <svelte:component this={component} {...componentOpts} />
+    {:else}
+      <p>{text}</p>
+    {/if}
     <div class="confirmation-dialog-form-flex">
       <button type="button" on:click="onPositive()">
-        {positiveText || 'OK'}
+        {positiveText}
       </button>
       <button type="button" on:click="onNegative()">
-        {negativeText || 'Cancel'}
+        {negativeText}
       </button>
     </div>
   </form>
@@ -44,6 +47,15 @@
       on('destroyDialog', this, this.onDestroyDialog)
       onCreateDialog.call(this)
     },
+    data: () => ({
+      component: void 0,
+      text: void 0,
+      onPositive: void 0,
+      onNegative: void 0,
+      title: '',
+      positiveText: 'OK',
+      negativeText: 'Cancel'
+    }),
     methods: {
       show,
       close,
@@ -58,10 +70,12 @@
           return
         }
         if (positiveResult) {
+          this.fire('positive')
           if (onPositive) {
             onPositive()
           }
         } else {
+          this.fire('negative')
           if (onNegative) {
             onNegative()
           }
@@ -79,4 +93,4 @@
       ModalDialog
     }
   }
-</script>
\ No newline at end of file
+</script>
diff --git a/src/routes/_components/dialog/components/GenericConfirmationDialog.html b/src/routes/_components/dialog/components/GenericConfirmationDialog.html
new file mode 100644
index 0000000..a090c71
--- /dev/null
+++ b/src/routes/_components/dialog/components/GenericConfirmationDialog.html
@@ -0,0 +1,75 @@
+<ModalDialog
+  {id}
+  {label}
+  {title}
+  background="var(--main-bg)"
+>
+  <form class="confirmation-dialog-form">
+    <slot></slot>
+    <div class="confirmation-dialog-form-flex">
+      <button type="button" on:click="onPositive()">
+        {positiveText || 'OK'}
+      </button>
+      <button type="button" on:click="onNegative()">
+        {negativeText || 'Cancel'}
+      </button>
+    </div>
+  </form>
+</ModalDialog>
+<style>
+  .confirmation-dialog-form-flex {
+    display: grid;
+    grid-template-columns: 1fr 1fr;
+    grid-gap: 10px;
+    padding: 10px 20px;
+  }
+  .confirmation-dialog-form-flex button {
+    min-width: 125px;
+  }
+</style>
+<script>
+  import ModalDialog from './ModalDialog.html'
+  import { show } from '../helpers/showDialog'
+  import { close } from '../helpers/closeDialog'
+  import { on } from '../../../_utils/eventBus'
+  import { oncreate as onCreateDialog } from '../helpers/onCreateDialog'
+
+  export default {
+    oncreate () {
+      on('destroyDialog', this, this.onDestroyDialog)
+      onCreateDialog.call(this)
+    },
+    data: () => ({
+      positiveText: void 0,
+      negativeText: void 0
+    }),
+    methods: {
+      show,
+      close,
+      onDestroyDialog (thisId) {
+        let {
+          id,
+          positiveResult
+        } = this.get()
+        if (thisId !== id) {
+          return
+        }
+        if (positiveResult) {
+          this.fire('positive')
+        } else {
+          this.fire('negative')
+        }
+      },
+      onPositive () {
+        this.set({ positiveResult: true })
+        this.close()
+      },
+      onNegative () {
+        this.close()
+      }
+    },
+    components: {
+      ModalDialog
+    }
+  }
+</script>
diff --git a/src/routes/_components/dialog/components/MuteDialog.html b/src/routes/_components/dialog/components/MuteDialog.html
new file mode 100644
index 0000000..5815935
--- /dev/null
+++ b/src/routes/_components/dialog/components/MuteDialog.html
@@ -0,0 +1,60 @@
+<GenericConfirmationDialog
+  {id}
+  {label}
+  {title}
+  {positiveText}
+  on:positive="doMute()"
+>
+  <div class="mute-dialog">
+    <p>
+      Mute @{account.acct} ?
+    </p>
+    <form class="mute-dialog-form">
+      <input type="checkbox"
+             id="mute-notifications"
+             name="mute-notifications"
+             bind:checked="muteNotifications">
+      <label for="mute-notifications">Mute notifications as well</label>
+    </form>
+  </div>
+</GenericConfirmationDialog>
+<style>
+  .mute-dialog {
+    padding: 20px;
+  }
+  .mute-dialog-form {
+    margin-top: 20px;
+  }
+</style>
+<script>
+  import GenericConfirmationDialog from './GenericConfirmationDialog.html'
+  import { show } from '../helpers/showDialog'
+  import { close } from '../helpers/closeDialog'
+  import { oncreate } from '../helpers/onCreateDialog'
+  import { setAccountMuted } from '../../../_actions/mute'
+
+  export default {
+    oncreate,
+    data: () => ({
+      positiveText: 'Mute',
+      title: '',
+      muteNotifications: true
+    }),
+    methods: {
+      show,
+      close,
+      async doMute () {
+        let { account, muteNotifications } = this.get()
+        this.close()
+        await setAccountMuted(
+          account.id,
+          /* mute */ true,
+          muteNotifications,
+          /* toastOnSuccess */ true)
+      }
+    },
+    components: {
+      GenericConfirmationDialog
+    }
+  }
+</script>
diff --git a/src/routes/_components/dialog/components/StatusOptionsDialog.html b/src/routes/_components/dialog/components/StatusOptionsDialog.html
index 7b389b4..e58901a 100644
--- a/src/routes/_components/dialog/components/StatusOptionsDialog.html
+++ b/src/routes/_components/dialog/components/StatusOptionsDialog.html
@@ -16,12 +16,12 @@ import { show } from '../helpers/showDialog'
 import { close } from '../helpers/closeDialog'
 import { oncreate } from '../helpers/onCreateDialog'
 import { setAccountBlocked } from '../../../_actions/block'
-import { setAccountMuted } from '../../../_actions/mute'
 import { setStatusPinnedOrUnpinned } from '../../../_actions/pin'
 import { setConversationMuted } from '../../../_actions/muteConversation'
 import { copyText } from '../../../_actions/copyText'
 import { deleteAndRedraft } from '../../../_actions/deleteAndRedraft'
 import { shareStatus } from '../../../_actions/share'
+import { toggleMute } from '../../../_actions/toggleMute'
 
 export default {
   oncreate,
@@ -183,9 +183,9 @@ export default {
       await setAccountBlocked(accountId, !blocking, true)
     },
     async onMuteClicked () {
-      let { accountId, muting } = this.get()
+      let { account, muting } = this.get()
       this.close()
-      await setAccountMuted(accountId, !muting, true)
+      await toggleMute(account, !muting)
     },
     async onMuteConversationClicked () {
       let { statusId, mutingConversation } = this.get()
diff --git a/src/routes/_components/dialog/components/TextConfirmationDialog.html b/src/routes/_components/dialog/components/TextConfirmationDialog.html
new file mode 100644
index 0000000..a5f78e3
--- /dev/null
+++ b/src/routes/_components/dialog/components/TextConfirmationDialog.html
@@ -0,0 +1,38 @@
+<GenericConfirmationDialog
+  {id}
+  {label}
+  {title}
+  {positiveText}
+  {negativeText}
+  on:positive
+  on:negative>
+  <p>{text}</p>
+</GenericConfirmationDialog>
+<style>
+  p {
+    font-size: 1.3em;
+    padding: 40px 20px;
+  }
+</style>
+<script>
+  import GenericConfirmationDialog from './GenericConfirmationDialog.html'
+  import { show } from '../helpers/showDialog'
+  import { close } from '../helpers/closeDialog'
+  import { oncreate } from '../helpers/onCreateDialog'
+
+  export default {
+    oncreate,
+    data: () => ({
+      title: void 0,
+      positiveText: void 0,
+      negativeText: void 0
+    }),
+    methods: {
+      show,
+      close
+    },
+    components: {
+      GenericConfirmationDialog
+    }
+  }
+</script>
diff --git a/src/routes/_components/dialog/creators/showMuteDialog.js b/src/routes/_components/dialog/creators/showMuteDialog.js
new file mode 100644
index 0000000..6911a9f
--- /dev/null
+++ b/src/routes/_components/dialog/creators/showMuteDialog.js
@@ -0,0 +1,16 @@
+import MuteDialog from '../components/MuteDialog.html'
+import { createDialogElement } from '../helpers/createDialogElement'
+import { createDialogId } from '../helpers/createDialogId'
+
+export default function showMuteDialog (account) {
+  let dialog = new MuteDialog({
+    target: createDialogElement(),
+    data: {
+      id: createDialogId(),
+      label: 'Mute dialog',
+      account
+    }
+  })
+  dialog.show()
+  return dialog
+}
diff --git a/src/routes/_components/dialog/creators/showConfirmationDialog.js b/src/routes/_components/dialog/creators/showTextConfirmationDialog.js
similarity index 59%
rename from src/routes/_components/dialog/creators/showConfirmationDialog.js
rename to src/routes/_components/dialog/creators/showTextConfirmationDialog.js
index 6b66011..7c0d6d2 100644
--- a/src/routes/_components/dialog/creators/showConfirmationDialog.js
+++ b/src/routes/_components/dialog/creators/showTextConfirmationDialog.js
@@ -1,9 +1,9 @@
-import ConfirmationDialog from '../components/ConfirmationDialog.html'
+import TextConfirmationDialog from '../components/TextConfirmationDialog.html'
 import { createDialogElement } from '../helpers/createDialogElement'
 import { createDialogId } from '../helpers/createDialogId'
 
-export default function showConfirmationDialog (options) {
-  let dialog = new ConfirmationDialog({
+export default function showTextConfirmationDialog (options) {
+  let dialog = new TextConfirmationDialog({
     target: createDialogElement(),
     data: Object.assign({
       id: createDialogId(),
@@ -11,4 +11,5 @@ export default function showConfirmationDialog (options) {
     }, options)
   })
   dialog.show()
+  return dialog
 }
diff --git a/src/routes/_components/settings/instance/InstanceActions.html b/src/routes/_components/settings/instance/InstanceActions.html
index d105e33..16a08e7 100644
--- a/src/routes/_components/settings/instance/InstanceActions.html
+++ b/src/routes/_components/settings/instance/InstanceActions.html
@@ -21,7 +21,7 @@
 </style>
 <script>
   import { store } from '../../../_store/store'
-  import { importShowConfirmationDialog } from '../../dialog/asyncDialogs'
+  import { importShowTextConfirmationDialog } from '../../dialog/asyncDialogs'
   import { switchToInstance, logOutOfInstance } from '../../../_actions/instances'
 
   export default {
@@ -36,12 +36,11 @@
         e.preventDefault()
         let { instanceName } = this.get()
 
-        let showConfirmationDialog = await importShowConfirmationDialog()
-        showConfirmationDialog({
-          text: `Log out of ${instanceName}?`,
-          onPositive () {
-            /* no await */ logOutOfInstance(instanceName)
-          }
+        let showTextConfirmationDialog = await importShowTextConfirmationDialog()
+        showTextConfirmationDialog({
+          text: `Log out of ${instanceName}?`
+        }).on('positive', () => {
+          /* no await */ logOutOfInstance(instanceName)
         })
       }
     }
diff --git a/src/routes/_components/settings/instance/PushNotificationSettings.html b/src/routes/_components/settings/instance/PushNotificationSettings.html
index 98aaf1a..98727f1 100644
--- a/src/routes/_components/settings/instance/PushNotificationSettings.html
+++ b/src/routes/_components/settings/instance/PushNotificationSettings.html
@@ -36,7 +36,7 @@
 </style>
 <script>
   import { store } from '../../../_store/store'
-  import { importShowConfirmationDialog } from '../../dialog/asyncDialogs'
+  import { importShowTextConfirmationDialog } from '../../dialog/asyncDialogs'
   import { logOutOfInstance } from '../../../_actions/instances'
   import { updatePushSubscriptionForInstance, updateAlerts } from '../../../_actions/pushSubscription'
   import { toast } from '../../toast/toast'
@@ -76,12 +76,11 @@
 
           // TODO: Better way to detect missing authorization scope
           if (err.message.startsWith('403:')) {
-            let showConfirmationDialog = await importShowConfirmationDialog()
-            showConfirmationDialog({
-              text: `You need to reauthenticate in order to enable push notification. Log out of ${instanceName}?`,
-              onPositive () {
-                logOutOfInstance(instanceName)
-              }
+            let showTextConfirmationDialog = await importShowTextConfirmationDialog()
+            showTextConfirmationDialog({
+              text: `You need to reauthenticate in order to enable push notification. Log out of ${instanceName}?`
+            }).on('positive', () => {
+              /* no await */ logOutOfInstance(instanceName)
             })
           } else {
             toast.say(`Failed to update push notification settings: ${err.message}`)
diff --git a/src/routes/_pages/muted.html b/src/routes/_pages/muted.html
index fdb0684..5e0c49e 100644
--- a/src/routes/_pages/muted.html
+++ b/src/routes/_pages/muted.html
@@ -15,7 +15,10 @@
         {
           icon: '#fa-volume-up',
           label: 'Unmute',
-          onclick: (accountId) => setAccountMuted(accountId, false, true)
+          onclick: (accountId) => setAccountMuted(accountId,
+            /* mute */ false,
+            /* notifications */ false,
+            /* toastOnSuccess */ true)
         }
       ]
     }),
diff --git a/tests/spec/002-login-spec.js b/tests/spec/002-login-spec.js
index 54cfd4d..b44ce11 100644
--- a/tests/spec/002-login-spec.js
+++ b/tests/spec/002-login-spec.js
@@ -1,7 +1,7 @@
 import { Selector as $ } from 'testcafe'
 import {
   addInstanceButton,
-  authorizeInput,
+  authorizeInput, confirmationDialogOKButton,
   emailInput,
   formError,
   getFirstVisibleStatus, getNthStatus, getOpacity,
@@ -62,7 +62,7 @@ test('Logs in and logs out of localhost:3000', async t => {
     .expect($('.acct-handle').innerText).eql('@foobar')
     .expect($('.acct-display-name').innerText).eql('foobar')
     .click($('button').withText('Log out'))
-    .click($('.modal-dialog button').withText('OK'))
+    .click(confirmationDialogOKButton)
     .expect($('.main-content').innerText).contains("You're not logged in to any instances")
     .click(homeNavButton)
     // check that the "hidden from SSR" content is visible
@@ -89,7 +89,7 @@ test('Logs in, refreshes, then logs out', async t => {
     .expect($('.acct-handle').innerText).eql('@foobar')
     .expect($('.acct-display-name').innerText).eql('foobar')
     .click($('button').withText('Log out'))
-    .click($('.modal-dialog button').withText('OK'))
+    .click(confirmationDialogOKButton)
     .expect($('.main-content').innerText).contains("You're not logged in to any instances")
     .click(homeNavButton)
     .expect(getOpacity('.hidden-from-ssr')()).eql('1')
diff --git a/tests/spec/114-mute-unmute.js b/tests/spec/114-mute-unmute.js
index 8f9addc..f709ec0 100644
--- a/tests/spec/114-mute-unmute.js
+++ b/tests/spec/114-mute-unmute.js
@@ -1,7 +1,15 @@
 import {
   accountProfileFollowButton,
-  accountProfileMoreOptionsButton, communityNavButton, getNthSearchResult,
-  getNthStatus, getNthStatusOptionsButton, getNthDialogOptionsOption, getUrl, modalDialog, closeDialogButton
+  accountProfileMoreOptionsButton,
+  communityNavButton,
+  getNthSearchResult,
+  getNthStatus,
+  getNthStatusOptionsButton,
+  getNthDialogOptionsOption,
+  getUrl,
+  modalDialog,
+  closeDialogButton,
+  confirmationDialogOKButton, sleep
 } from '../utils'
 import { Selector as $ } from 'testcafe'
 import { loginAsFoobar } from '../roles'
@@ -21,7 +29,12 @@ test('Can mute and unmute an account', async t => {
     .expect(getNthDialogOptionsOption(2).innerText).contains('Block @admin')
     .expect(getNthDialogOptionsOption(3).innerText).contains('Mute @admin')
     .click(getNthDialogOptionsOption(3))
+  await sleep(1000)
+  await t
+    .click(confirmationDialogOKButton)
     .expect(modalDialog.exists).notOk()
+  await sleep(1000)
+  await t
     .click(communityNavButton)
     .click($('a[href="/muted"]'))
     .expect(getNthSearchResult(1).innerText).contains('@admin')
@@ -33,6 +46,8 @@ test('Can mute and unmute an account', async t => {
     .expect(getNthDialogOptionsOption(3).innerText).contains('Block @admin')
     .expect(getNthDialogOptionsOption(4).innerText).contains('Unmute @admin')
     .click(getNthDialogOptionsOption(4))
+  await sleep(1000)
+  await t
     .click(accountProfileMoreOptionsButton)
     .expect(getNthDialogOptionsOption(1).innerText).contains('Mention @admin')
     .expect(getNthDialogOptionsOption(2).innerText).contains('Unfollow @admin')
diff --git a/tests/utils.js b/tests/utils.js
index 8357975..57c15a9 100644
--- a/tests/utils.js
+++ b/tests/utils.js
@@ -47,6 +47,7 @@ export const neverMarkMediaSensitiveInput = $('#choice-never-mark-media-sensitiv
 export const removeEmojiFromDisplayNamesInput = $('#choice-omit-emoji-in-display-names')
 export const dialogOptionsOption = $(`.modal-dialog button`)
 export const emojiSearchInput = $('.emoji-mart-search input')
+export const confirmationDialogOKButton = $('.confirmation-dialog-form-flex button:nth-child(1)')
 
 export const composeModalInput = $('.modal-dialog .compose-box-input')
 export const composeModalComposeButton = $('.modal-dialog .compose-box-button')