/** * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors * SPDX-License-Identifier: AGPL-3.0-or-later */ import type { User } from "@nextcloud/cypress" export const getRowForFileId = (fileid: number) => cy.get(`[data-cy-files-list-row-fileid="${fileid}"]`) export const getRowForFile = (filename: string) => cy.get(`[data-cy-files-list-row-name="${CSS.escape(filename)}"]`) export const getActionsForFileId = (fileid: number) => getRowForFileId(fileid).find('[data-cy-files-list-row-actions]') export const getActionsForFile = (filename: string) => getRowForFile(filename).find('[data-cy-files-list-row-actions]') export const getActionButtonForFileId = (fileid: number) => getActionsForFileId(fileid).findByRole('button', { name: 'Actions' }) export const getActionButtonForFile = (filename: string) => getActionsForFile(filename).findByRole('button', { name: 'Actions' }) export const triggerActionForFileId = (fileid: number, actionId: string) => { getActionButtonForFileId(fileid).click({ force: true }) // Getting the last button to avoid the one from popup fading out cy.get(`[data-cy-files-list-row-action="${CSS.escape(actionId)}"] > button`).last() .should('exist').click({ force: true }) } export const triggerActionForFile = (filename: string, actionId: string) => { getActionButtonForFile(filename).click({ force: true }) // Getting the last button to avoid the one from popup fading out cy.get(`[data-cy-files-list-row-action="${CSS.escape(actionId)}"] > button`).last() .should('exist').click({ force: true }) } export const triggerInlineActionForFileId = (fileid: number, actionId: string) => { getActionsForFileId(fileid).find(`button[data-cy-files-list-row-action="${CSS.escape(actionId)}"]`).should('exist').click() } export const triggerInlineActionForFile = (filename: string, actionId: string) => { getActionsForFile(filename).get(`button[data-cy-files-list-row-action="${CSS.escape(actionId)}"]`).should('exist').click() } export const selectAllFiles = () => { cy.get('[data-cy-files-list-selection-checkbox]') .findByRole('checkbox', { checked: false }) .click({ force: true }) } export const deselectAllFiles = () => { cy.get('[data-cy-files-list-selection-checkbox]') .findByRole('checkbox', { checked: true }) .click({ force: true }) } export const selectRowForFile = (filename: string, options: Partial = {}) => { getRowForFile(filename) .find('[data-cy-files-list-row-checkbox]') .findByRole('checkbox') // don't use click to avoid triggering side effects events .trigger('change', { ...options, force: true }) .should('be.checked') cy.get('[data-cy-files-list-selection-checkbox]').findByRole('checkbox').should('satisfy', (elements) => { return elements.length === 1 && (elements[0].checked === true || elements[0].indeterminate === true) }) } export const triggerSelectionAction = (actionId: string) => { cy.get(`button[data-cy-files-list-selection-action="${CSS.escape(actionId)}"]`).should('exist').click() } export const moveFile = (fileName: string, dirPath: string) => { getRowForFile(fileName).should('be.visible') triggerActionForFile(fileName, 'move-copy') cy.get('.file-picker').within(() => { // intercept the copy so we can wait for it cy.intercept('MOVE', /\/(remote|public)\.php\/dav\/files\//).as('moveFile') if (dirPath === '/') { // select home folder cy.get('button[title="Home"]').should('be.visible').click() // click move cy.contains('button', 'Move').should('be.visible').click() } else if (dirPath === '.') { // click move cy.contains('button', 'Copy').should('be.visible').click() } else { const directories = dirPath.split('/') directories.forEach((directory) => { // select the folder cy.get(`[data-filename="${directory}"]`).should('be.visible').click() }) // click move cy.contains('button', `Move to ${directories.at(-1)}`).should('be.visible').click() } cy.wait('@moveFile') }) } export const copyFile = (fileName: string, dirPath: string) => { getRowForFile(fileName).should('be.visible') triggerActionForFile(fileName, 'move-copy') cy.get('.file-picker').within(() => { // intercept the copy so we can wait for it cy.intercept('COPY', /\/(remote|public)\.php\/dav\/files\//).as('copyFile') if (dirPath === '/') { // select home folder cy.get('button[title="Home"]').should('be.visible').click() // click copy cy.contains('button', 'Copy').should('be.visible').click() } else if (dirPath === '.') { // click copy cy.contains('button', 'Copy').should('be.visible').click() } else { const directories = dirPath.split('/') directories.forEach((directory) => { // select the folder cy.get(`[data-filename="${CSS.escape(directory)}"]`).should('be.visible').click() }) // click copy cy.contains('button', `Copy to ${directories.at(-1)}`).should('be.visible').click() } cy.wait('@copyFile') }) } export const renameFile = (fileName: string, newFileName: string) => { getRowForFile(fileName) triggerActionForFile(fileName, 'rename') // intercept the move so we can wait for it cy.intercept('MOVE', /\/(remote|public)\.php\/dav\/files\//).as('moveFile') getRowForFile(fileName).find('[data-cy-files-list-row-name] input').clear() getRowForFile(fileName).find('[data-cy-files-list-row-name] input').type(`${newFileName}{enter}`) cy.wait('@moveFile') } export const navigateToFolder = (dirPath: string) => { const directories = dirPath.split('/') directories.forEach((directory) => { getRowForFile(directory).should('be.visible').find('[data-cy-files-list-row-name-link]').click() }) } export const closeSidebar = () => { // {force: true} as it might be hidden behind toasts cy.get('[data-cy-sidebar] .app-sidebar__close').click({ force: true }) } export const clickOnBreadcrumbs = (label: string) => { cy.intercept('PROPFIND', /\/remote.php\/dav\//).as('propfind') cy.get('[data-cy-files-content-breadcrumbs]').contains(label).click() cy.wait('@propfind') } export const createFolder = (folderName: string) => { cy.intercept('MKCOL', /\/remote.php\/dav\/files\//).as('createFolder') // TODO: replace by proper data-cy selectors cy.get('[data-cy-upload-picker] .action-item__menutoggle').first().click() cy.get('[data-cy-upload-picker-menu-entry="newFolder"] button').click() cy.get('[data-cy-files-new-node-dialog]').should('be.visible') cy.get('[data-cy-files-new-node-dialog-input]').type(`{selectall}${folderName}`) cy.get('[data-cy-files-new-node-dialog-submit]').click() cy.wait('@createFolder') getRowForFile(folderName).should('be.visible') } /** * Check validity of an input element * @param validity The expected validity message (empty string means it is valid) * @example * ```js * cy.findByRole('textbox') * .should(haveValidity(/must not be empty/i)) * ``` */ export const haveValidity = (validity: string | RegExp) => { if (typeof validity === 'string') { return (el: JQuery) => expect((el.get(0) as HTMLInputElement).validationMessage).to.equal(validity) } return (el: JQuery) => expect((el.get(0) as HTMLInputElement).validationMessage).to.match(validity) } export const deleteFileWithRequest = (user: User, path: string) => { // Ensure path starts with a slash and has no double slashes path = `/${path}`.replace(/\/+/g, '/') cy.request('/csrftoken').then(({ body }) => { const requestToken = body.token cy.request({ method: 'DELETE', url: `${Cypress.env('baseUrl')}/remote.php/dav/files/${user.userId}` + path, headers: { requestToken, }, retryOnStatusCodeFailure: true, }) }) } export const triggerFileListAction = (actionId: string) => { cy.get(`button[data-cy-files-list-action="${CSS.escape(actionId)}"]`).last() .should('exist').click({ force: true }) }