Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Implement testing type switch promos #26894

Merged
merged 42 commits into from
Jun 6, 2023
Merged
Show file tree
Hide file tree
Changes from 34 commits
Commits
Show all changes
42 commits
Select commit Hold shift + click to select a range
9f753af
feat: Implement testing type switch promos
mike-plummer May 30, 2023
a1d6183
Add tests, changelog entry
mike-plummer May 31, 2023
f3f447c
Add tests
mike-plummer May 31, 2023
99bf726
Fix button styling
mike-plummer May 31, 2023
e7b8574
Styling fixes, add framework links
mike-plummer May 31, 2023
e0e6fd3
Add missing testId
mike-plummer May 31, 2023
a1d733f
run ci
mike-plummer May 31, 2023
6d6453b
Fix spec
mike-plummer May 31, 2023
aa45c66
Merge branch 'develop' into mikep/589-testing-type-switch
mike-plummer May 31, 2023
18ea97e
chore: updating v8 snapshot cache
Jun 1, 2023
8462979
Merge branch 'develop' into mikep/589-testing-type-switch
mike-plummer Jun 1, 2023
af2d7b8
chore: updating v8 snapshot cache
Jun 1, 2023
170adf7
chore: updating v8 snapshot cache
Jun 1, 2023
0c463c3
Fix styling issues
mike-plummer Jun 1, 2023
8b4e9b0
Merge branch 'develop' into mikep/589-testing-type-switch
mike-plummer Jun 1, 2023
66c1ae9
Resolve code review findings
mike-plummer Jun 1, 2023
4009213
Merge branch 'develop' into mikep/589-testing-type-switch
mike-plummer Jun 1, 2023
2d20d1a
Address review comments
mike-plummer Jun 1, 2023
1d1c410
reduce size of text box to match latest figma
marktnoonan Jun 4, 2023
754ac45
update button style to match figma
marktnoonan Jun 4, 2023
6f55226
increase width at which we collapse sidenav
marktnoonan Jun 5, 2023
4a531e1
add short versions of the headings
marktnoonan Jun 5, 2023
3f0a615
remove skeletons from header
marktnoonan Jun 5, 2023
7606860
avoid extra height
marktnoonan Jun 5, 2023
e5e103a
adjustments for column alignment
marktnoonan Jun 5, 2023
335982f
Merge branch 'develop' into mikep/589-testing-type-switch
marktnoonan Jun 5, 2023
29546a3
Merge branch 'develop' into mikep/589-testing-type-switch
marktnoonan Jun 5, 2023
33766bf
fix flaky test
marktnoonan Jun 5, 2023
32971d6
update tests for responsive text changes
marktnoonan Jun 5, 2023
aa4dde8
update changelog
marktnoonan Jun 5, 2023
ad808b8
restore spacing between header items
marktnoonan Jun 5, 2023
f5bbb2d
Merge branch 'develop' into mikep/589-testing-type-switch
marktnoonan Jun 5, 2023
b17c262
Merge branch 'develop' into mikep/589-testing-type-switch
marktnoonan Jun 5, 2023
54cb1a0
avoid occasional flash of promo on page load
marktnoonan Jun 5, 2023
22d6f5a
update text handling
marktnoonan Jun 5, 2023
c6b40b4
fix types and tests
astone123 Jun 5, 2023
890e021
Update packages/app/src/specs/SpecsList.vue
astone123 Jun 5, 2023
089df97
Merge branch 'develop' into mikep/589-testing-type-switch
astone123 Jun 5, 2023
19cf7a6
updated final e2e bullet
marktnoonan Jun 5, 2023
707350f
fix question mark icon flashing
astone123 Jun 5, 2023
82d0ace
text formatting
marktnoonan Jun 5, 2023
59df96a
remove superfluous snapshot [skip ci]
lmiller1990 Jun 6, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 5 additions & 1 deletion cli/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,8 +1,12 @@
<!-- See the ../guides/writing-the-cypress-changelog.md for details on writing the changelog. -->
## 12.13.1
## 12.14.0

_Released 06/06/2023 (PENDING)_

**Features:**

- A new testing type switcher has been added to the Spec Explorer to make it easier to move between E2E and Component Testing. An informational overview of each type is displayed if it isn't configured in your project to help newcomers to each testing type. Addresses [#26448](https://github.com/cypress-io/cypress/issues/26448), [#26836](https://github.com/cypress-io/cypress/issues/26836) and [#26837](https://github.com/cypress-io/cypress/issues/26837).

**Dependency Updates:**

- Upgraded [`find-process`](https://www.npmjs.com/package/find-process) from `1.4.1` to `1.4.7` to address this [Synk](https://security.snyk.io/vuln/SNYK-JS-FINDPROCESS-1090284) security vulnerability. Addressed in [#26631](https://github.com/cypress-io/cypress/pull/26906).
Expand Down
2 changes: 1 addition & 1 deletion packages/app/cypress/e2e/runs.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -331,7 +331,7 @@ describe('App: Runs', { viewportWidth: 1200 }, () => {
moveToRunsPage()
cy.findByText(defaultMessages.runs.connect.buttonProject).click()
cy.contains('button', defaultMessages.runs.connect.modal.selectProject.createProject).click()
cy.findByText(defaultMessages.runs.connectSuccessAlert.title, { timeout: 10000 }).should('be.visible')
cy.findByText(defaultMessages.runs.connectSuccessAlert.title, { timeout: 10000 }).scrollIntoView().should('be.visible')

cy.withCtx(async (ctx) => {
const config = await ctx.project.getConfig()
Expand Down
91 changes: 91 additions & 0 deletions packages/app/cypress/e2e/specs_list_switcher.cy.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
describe('App: Spec List Testing Type Switcher', () => {
mike-plummer marked this conversation as resolved.
Show resolved Hide resolved
context('ct unconfigured', () => {
beforeEach(() => {
cy.scaffoldProject('cypress-in-cypress')
cy.openProject('cypress-in-cypress')

cy.withCtx(async (ctx, o) => {
const config = await ctx.file.readFileInProject('cypress.config.js')
const newCypressConfig = config.replace(`component:`, `_component:`)

await ctx.actions.file.writeFileInProject('cypress.config.js', newCypressConfig)
})

cy.startAppServer('e2e')

cy.visitApp()
cy.contains('E2E specs').should('be.visible')
})

it('switches testing types', () => {
cy.findByTestId('testing-type-switch').within(() => {
cy.findByText('Component specs').click()
})

cy.contains('Component testing is not set up for this project')

cy.findByTestId('testing-type-setup-button').should('be.visible')
})
})

context('e2e unconfigured', () => {
beforeEach(() => {
cy.scaffoldProject('cypress-in-cypress')
cy.openProject('cypress-in-cypress')

cy.withCtx(async (ctx, o) => {
const config = await ctx.file.readFileInProject('cypress.config.js')
const newCypressConfig = config.replace(`e2e:`, `_e2e:`)

await ctx.actions.file.writeFileInProject('cypress.config.js', newCypressConfig)
})

cy.startAppServer('component')

cy.visitApp()
cy.contains('Component specs').should('be.visible')
})

it('switches testing types', () => {
cy.findByTestId('testing-type-switch').within(() => {
cy.findByText('E2E specs').click()
})

cy.contains('End-to-end testing is not set up for this project')

cy.findByTestId('testing-type-setup-button').should('be.visible')
})
})

context('both testing types configured', () => {
beforeEach(() => {
cy.scaffoldProject('cypress-in-cypress')
cy.findBrowsers()
cy.openProject('cypress-in-cypress')

cy.startAppServer('component')

cy.visitApp()
cy.contains('Component specs').should('be.visible')
})

it('displays expected switch content', () => {
cy.findByTestId('unconfigured-icon').should('not.exist')

cy.withCtx((ctx, o) => {
o.sinon.stub(ctx.actions.project, 'setAndLoadCurrentTestingType')
o.sinon.stub(ctx.actions.project, 'reconfigureProject').resolves()
})

cy.findByTestId('testing-type-switch').within(() => {
cy.findByText('E2E specs').click()
})

cy.withCtx((ctx) => {
expect(ctx.coreData.app.relaunchBrowser).eq(true)
expect(ctx.actions.project.setAndLoadCurrentTestingType).to.have.been.calledWith('e2e')
expect(ctx.actions.project.reconfigureProject).to.have.been.called
})
})
})
})
7 changes: 4 additions & 3 deletions packages/app/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -20,9 +20,10 @@
},
"dependencies": {},
"devDependencies": {
"@cypress-design/vue-button": "0.9.2",
"@cypress-design/vue-icon": "0.22.2",
"@cypress-design/vue-statusicon": "0.4.3",
"@cypress-design/vue-button": "^0.10.1",
"@cypress-design/vue-icon": "^0.23.1",
"@cypress-design/vue-statusicon": "^0.4.7",
"@cypress-design/vue-tabs": "^0.5.1",
"@graphql-typed-document-node/core": "^3.1.0",
"@headlessui/vue": "1.4.0",
"@iconify/iconify": "2.1.2",
Expand Down
3 changes: 2 additions & 1 deletion packages/app/src/components/promo/Promo.vue
Original file line number Diff line number Diff line change
Expand Up @@ -47,11 +47,12 @@ const props = defineProps<{
campaign: string
medium: string
cohort?: string
instanceId?: string
}>()

const promoSeenMutation = useMutation(Promo_PromoSeenDocument)

const promoInstanceId = nanoid()
const promoInstanceId = props.instanceId || nanoid()

useQuery({ query: PromoDocument })
.then((queryResult) => {
Expand Down
35 changes: 31 additions & 4 deletions packages/app/src/components/promo/PromoCard.vue
Original file line number Diff line number Diff line change
@@ -1,27 +1,54 @@
<template>
<div class="grid grid-cols-[480px] xl:grid-cols-[300px_470px] p-[40px] gap-y-[16px] gap-x-[100px]">
<div
class="p-[40px]"
:class="gridClasses"
>
<div>
<h2 class="text-xl font-semibold text-gray-900">
{{ title }}
</h2>
<p class="text-grey-700">
<p class="text-gray-700">
{{ body }}
</p>
<slot name="content" />
</div>
<div class="row-end-[span_2] xl:col-start-2">
<div
v-if="slots.image"
class="row-end-[span_2] xl:col-start-2"
>
<slot name="image" />
</div>
<div class="self-end">
<div
v-if="slots.action"
class="self-end"
>
<slot name="action" />
</div>
</div>
</template>

<script lang="ts" setup>
import { computed, useSlots } from 'vue'

defineProps<{
title: string
body: string
}>()

const slots = useSlots()

const gridClasses = computed(() => {
const classes = ['grid', 'grid-cols-[480px]', 'gap-y-[16px]']

// If `image` is defined then lay out side-by-side on xl viewport,
// otherwise it should flow vertically
if (slots.image) {
classes.push('xl:grid-cols-[300px_470px]', 'gap-x-[100px]')
} else {
classes.push('xl:grid-cols-[744px]')
}

return classes
})

</script>
119 changes: 119 additions & 0 deletions packages/app/src/composables/useTestingType.cy.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
import { UseTestingType_ActivateTestingTypeDocument } from '../generated/graphql'
import { useTestingType } from './useTestingType'
import type { FunctionalComponent } from 'vue'

describe('useTestingType', () => {
type ComposableWrapperProps<R> = { useComposable: () => R, callback: (result: R) => void }

const ComposableWrapper: FunctionalComponent<ComposableWrapperProps<unknown>> = <R, >({ useComposable, callback }: ComposableWrapperProps<R>) => {
const result = useComposable()

callback(result)

return <div>Composable</div>
}

const mountComposable = (composable: () => any) => {
const callback = cy.stub().as('callback')

cy.mount({
name: 'composable',
render () {
return (
<ComposableWrapper useComposable={composable} callback={callback} />
)
},
})

return cy.get('@callback').should('have.been.called').then((cb) => {
return cy.wrap(callback.getCall(callback.callCount - 1).args[0]).as('result')
})
}

beforeEach(() => {
const activateTestingTypeStub = cy.stub().as('activateTestingType')

cy.stubMutationResolver(UseTestingType_ActivateTestingTypeDocument, (defineResult, args) => {
activateTestingTypeStub()

return defineResult({ switchTestingTypeAndRelaunch: true })
})

cy.gqlStub.Query.currentProject = {
id: 'abc123',
currentTestingType: 'e2e',
isCTConfigured: false,
isE2EConfigured: false,
} as any
})

it('supplies expected query data', () => {
mountComposable(useTestingType).then((value) => {
const result = value as unknown as ReturnType<typeof useTestingType>

expect(result.activeTestingType.value).to.eql('e2e')
expect(result.isCTConfigured.value).to.eql(false)
expect(result.isE2EConfigured.value).to.eql(false)
expect(result.viewedTestingType.value).to.eql('e2e')
})
})

describe('viewTestingType', () => {
context('target mode is not configured', () => {
beforeEach(() => {
cy.gqlStub.Query.currentProject = {
id: 'abc123',
currentTestingType: 'e2e',
isCTConfigured: false,
} as any
})

it('should toggle viewed mode', () => {
mountComposable(useTestingType).then((value) => {
const result = value as unknown as ReturnType<typeof useTestingType>

expect(result.viewedTestingType.value).to.eql('e2e')

result.viewTestingType('component')

expect(result.viewedTestingType.value).to.eql('component')
})
})
})

context('target mode is configured', () => {
beforeEach(() => {
cy.gqlStub.Query.currentProject = {
id: 'abc123',
currentTestingType: 'e2e',
isE2EConfigured: true,
isCTConfigured: true,
} as any
})

it('should toggle active mode if not active mode', () => {
mountComposable(useTestingType).then((value) => {
const result = value as unknown as ReturnType<typeof useTestingType>

expect(result.viewedTestingType.value).to.eql('e2e')

result.viewTestingType('component')
})

cy.get('@activateTestingType').should('have.been.calledOnce')
})

it('should toggle viewed mode if active mode', () => {
mountComposable(useTestingType).then((value) => {
const result = value as unknown as ReturnType<typeof useTestingType>

expect(result.viewedTestingType.value).to.eql('e2e')

result.viewTestingType('e2e')
})

cy.get('@activateTestingType').should('not.have.been.called')
})
})
})
})
72 changes: 72 additions & 0 deletions packages/app/src/composables/useTestingType.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
import { useMutation, useQuery, gql } from '@urql/vue'
import { computed, ref, watchEffect } from 'vue'
import { UseTestingType_TestingTypeDocument, UseTestingType_ActivateTestingTypeDocument } from '../generated/graphql'

gql`
query UseTestingType_TestingType {
currentProject {
id
currentTestingType
isCTConfigured
isE2EConfigured
}
}
`

gql`
mutation UseTestingType_ActivateTestingType($testingType: TestingTypeEnum!) {
switchTestingTypeAndRelaunch(testingType: $testingType)
}
`

export function useTestingType () {
const query = useQuery({ query: UseTestingType_TestingTypeDocument })
const activateTestingTypeMutation = useMutation(UseTestingType_ActivateTestingTypeDocument)
const viewedTestingType = ref<'e2e' | 'component' | null>(null)

const activeTestingType = computed(() => query.data.value?.currentProject?.currentTestingType)
const isCTConfigured = computed(() => query.data.value?.currentProject?.isCTConfigured)
const isE2EConfigured = computed(() => query.data.value?.currentProject?.isE2EConfigured)

const showTestingTypePromo = computed(() => {
// confirm required data has resolved from other computed values
if (isE2EConfigured.value === undefined && isCTConfigured.value === undefined) {
return false
}

return (viewedTestingType.value === 'e2e' && isE2EConfigured.value === false) ||
astone123 marked this conversation as resolved.
Show resolved Hide resolved
(viewedTestingType.value === 'component' && isCTConfigured.value === false)
})

// Initialize 'viewed' testing type when query first resolves
watchEffect(() => {
if (!!activeTestingType.value && !viewedTestingType.value) {
viewedTestingType.value = activeTestingType.value
}
})

async function activateTestingType (testingType: 'e2e' | 'component') {
return await activateTestingTypeMutation.executeMutation({ testingType })
}

async function viewTestingType (testingType: 'e2e' | 'component') {
const switchingBackToActiveMode = testingType === activeTestingType.value
const targetModeIsConfigured = testingType === 'e2e' && isE2EConfigured.value || testingType === 'component' && isCTConfigured.value

if (!switchingBackToActiveMode && targetModeIsConfigured) {
await activateTestingType(testingType)
} else {
viewedTestingType.value = testingType
}
}

return {
activeTestingType,
viewedTestingType,
isCTConfigured,
isE2EConfigured,
showTestingTypePromo,
viewTestingType,
activateTestingType,
}
}
Loading