left-sidebar.spec.ts 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. // Copyright (C) 2012-2024 Zammad Foundation, https://zammad-foundation.org/
  2. import {
  3. fireEvent,
  4. getAllByRole,
  5. getByLabelText,
  6. getByRole,
  7. } from '@testing-library/vue'
  8. import { visitView } from '#tests/support/components/visitView.ts'
  9. import { mockPermissions } from '#tests/support/mock-permissions.ts'
  10. import { mockUserCurrent } from '#tests/support/mock-userCurrent.ts'
  11. import { mockLogoutMutation } from '#shared/graphql/mutations/logout.mocks.ts'
  12. describe('Left sidebar', () => {
  13. beforeEach(() => {
  14. mockUserCurrent({
  15. id: 'gid://zammad/User/999',
  16. firstname: 'Nicole',
  17. lastname: 'Braun',
  18. fullname: 'Nicole Braun',
  19. preferences: {},
  20. })
  21. })
  22. afterEach(() => {
  23. localStorage.clear()
  24. })
  25. describe('width handling', () => {
  26. it('renders initially with the default width', async () => {
  27. const view = await visitView('/')
  28. const aside = view.getByRole('complementary')
  29. expect(aside.parentElement).toHaveStyle({
  30. gridTemplateColumns: '260px 1fr',
  31. })
  32. })
  33. it('restores stored width', async () => {
  34. localStorage.setItem('gid://zammad/User/999-left-sidebar-width', '216')
  35. const view = await visitView('/')
  36. const aside = view.getByRole('complementary')
  37. expect(aside.parentElement).toHaveStyle({
  38. gridTemplateColumns: '216px 1fr',
  39. })
  40. })
  41. it('supports collapsing/expanding', async () => {
  42. const view = await visitView('/')
  43. const aside = view.getByRole('complementary')
  44. const collapseButton = getByRole(aside, 'button', {
  45. name: 'Collapse this element',
  46. })
  47. await view.events.click(collapseButton)
  48. expect(aside.parentElement).toHaveStyle({
  49. gridTemplateColumns: '48px 1fr',
  50. })
  51. const expandButton = getByRole(aside, 'button', {
  52. name: 'Expand this element',
  53. })
  54. await view.events.click(expandButton)
  55. expect(aside.parentElement).toHaveStyle({
  56. gridTemplateColumns: '260px 1fr',
  57. })
  58. })
  59. it('restores collapsed state width', async () => {
  60. localStorage.setItem(
  61. 'gid://zammad/User/999-left-sidebar-collapsed',
  62. 'true',
  63. )
  64. const view = await visitView('/')
  65. const aside = view.getByRole('complementary')
  66. expect(aside.parentElement).toHaveStyle({
  67. gridTemplateColumns: '48px 1fr',
  68. })
  69. })
  70. it('supports resizing', async () => {
  71. const view = await visitView('/')
  72. const aside = view.getByRole('complementary')
  73. const resizeHandle = getByLabelText(aside, 'Resize sidebar')
  74. await fireEvent.mouseDown(resizeHandle, { clientX: 260 })
  75. await fireEvent.mouseMove(document, { clientX: 216 })
  76. await fireEvent.mouseUp(document, { clientX: 216 })
  77. expect(aside.parentElement).toHaveStyle({
  78. gridTemplateColumns: '216px 1fr',
  79. })
  80. })
  81. it('supports resetting', async () => {
  82. localStorage.setItem('gid://zammad/User/999-left-sidebar-width', '216')
  83. const view = await visitView('/')
  84. const aside = view.getByRole('complementary')
  85. const resizeHandle = getByLabelText(aside, 'Resize sidebar')
  86. await view.events.dblClick(resizeHandle)
  87. expect(aside.parentElement).toHaveStyle({
  88. gridTemplateColumns: '260px 1fr',
  89. })
  90. })
  91. })
  92. describe('User menu', () => {
  93. afterEach(() => {
  94. vi.clearAllMocks()
  95. })
  96. it.each([{ collapsed: false }, { collapsed: true }])(
  97. 'shows menu popover on click (collapsed: $collapsed)',
  98. async ({ collapsed }) => {
  99. localStorage.setItem(
  100. 'gid://zammad/User/999-left-sidebar-collapsed',
  101. String(collapsed),
  102. )
  103. const view = await visitView('/')
  104. const aside = view.getByRole('complementary')
  105. const avatarButton = getByRole(aside, 'button', { name: 'User menu' })
  106. expect(avatarButton).toHaveTextContent('NB')
  107. await view.events.click(avatarButton)
  108. const popover = view.getByRole('region', { name: 'User menu' })
  109. expect(popover).toHaveTextContent('Nicole Braun')
  110. const menu = getByRole(popover, 'menu')
  111. const menuItems = getAllByRole(menu, 'menuitem')
  112. expect(menuItems).toHaveLength(4)
  113. },
  114. )
  115. it('supports cycling appearance state', async () => {
  116. mockPermissions(['user_preferences.appearance'])
  117. const view = await visitView('/')
  118. const aside = view.getByRole('complementary')
  119. const avatarButton = getByRole(aside, 'button', { name: 'User menu' })
  120. await view.events.click(avatarButton)
  121. const appearanceButton = view.getByRole('button', { name: 'Appearance' })
  122. const appearanceSwitch = view.getByRole('checkbox', { name: 'Dark Mode' })
  123. expect(appearanceSwitch).toBePartiallyChecked()
  124. await view.events.click(appearanceSwitch)
  125. expect(appearanceSwitch).toBeChecked()
  126. await view.events.click(appearanceButton)
  127. expect(appearanceSwitch).not.toBeChecked()
  128. await view.events.click(appearanceSwitch)
  129. expect(appearanceSwitch).toBePartiallyChecked()
  130. })
  131. it('supports navigating to playground', async () => {
  132. const view = await visitView('/')
  133. const aside = view.getByRole('complementary')
  134. const avatarButton = getByRole(aside, 'button', { name: 'User menu' })
  135. await view.events.click(avatarButton)
  136. const playgroundLink = view.getByRole('link', {
  137. name: 'Playground',
  138. })
  139. await view.events.click(playgroundLink)
  140. await vi.waitFor(() => {
  141. expect(view, 'correctly redirects to playground page').toHaveCurrentUrl(
  142. '/playground',
  143. )
  144. })
  145. expect(
  146. view.queryByRole('region', { name: 'User menu' }),
  147. ).not.toBeInTheDocument()
  148. })
  149. // TODO: Cover keyboard shortcuts menu item when ready.
  150. it('supports navigating to personal settings', async () => {
  151. const view = await visitView('/')
  152. const aside = view.getByRole('complementary')
  153. const avatarButton = getByRole(aside, 'button', { name: 'User menu' })
  154. await view.events.click(avatarButton)
  155. const personalSettingsLink = view.getByRole('link', {
  156. name: 'Profile settings',
  157. })
  158. await view.events.click(personalSettingsLink)
  159. await vi.waitFor(() => {
  160. expect(
  161. view,
  162. 'correctly redirects to personal settings page',
  163. ).toHaveCurrentUrl('/personal-setting')
  164. })
  165. expect(
  166. view.queryByRole('region', { name: 'User menu' }),
  167. ).not.toBeInTheDocument()
  168. })
  169. it('supports signing out', async () => {
  170. const view = await visitView('/')
  171. const aside = view.getByRole('complementary')
  172. const avatarButton = getByRole(aside, 'button', { name: 'User menu' })
  173. await view.events.click(avatarButton)
  174. const logoutLink = view.getByRole('link', { name: 'Sign out' })
  175. mockLogoutMutation({
  176. logout: {
  177. success: true,
  178. externalLogoutUrl: null,
  179. },
  180. })
  181. await view.events.click(logoutLink)
  182. await vi.waitFor(() => {
  183. expect(view, 'correctly redirects to login page').toHaveCurrentUrl(
  184. '/login',
  185. )
  186. })
  187. expect(
  188. view.queryByRole('region', { name: 'User menu' }),
  189. ).not.toBeInTheDocument()
  190. })
  191. })
  192. })