improve link testing
This commit is contained in:
@@ -47,31 +47,23 @@ context('/api/marks/link', () => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
const validUrls = [
|
it('detects a pasted URL', () => {
|
||||||
'https://example.com',
|
cy.get('.ProseMirror').paste({ pastePayload: 'https://example.com', pasteType: 'text/plain' })
|
||||||
'https://example.com/with-path',
|
.find('a')
|
||||||
'http://example.com/with-http',
|
.should('contain', 'https://example.com')
|
||||||
'https://www.example.com/with-www',
|
.should('have.attr', 'href', 'https://example.com')
|
||||||
'https://www.example.com/with-numbers-123',
|
|
||||||
'https://www.example.com/with-parameters?var=true',
|
|
||||||
'https://www.example.com/with-multiple-parameters?var=true&foo=bar',
|
|
||||||
'https://www.example.com/with-spaces?var=true&foo=bar+3',
|
|
||||||
// TODO: 'https://www.example.com/with,comma',
|
|
||||||
// TODO: 'https://www.example.com/with(brackets)',
|
|
||||||
// TODO: 'https://www.example.com/with!exclamation!marks',
|
|
||||||
'http://thelongestdomainnameintheworldandthensomeandthensomemoreandmore.com/',
|
|
||||||
'https://example.longtopleveldomain',
|
|
||||||
'https://example-with-dashes.com',
|
|
||||||
]
|
|
||||||
|
|
||||||
validUrls.forEach(url => {
|
|
||||||
it(`url should be detected: ${url}`, () => {
|
|
||||||
cy.get('.ProseMirror').paste({ pastePayload: url, pasteType: 'text/plain' })
|
|
||||||
.find('a')
|
|
||||||
.should('contain', url)
|
|
||||||
.should('have.attr', 'href', url)
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
|
|
||||||
// TODO: Test invalid URLs
|
it('correctly detects multiple pasted URLs', () => {
|
||||||
|
cy.get('.ProseMirror').paste({ pastePayload: 'https://example1.com, https://example2.com/foobar, (http://example3.com/foobar)', pasteType: 'text/plain' })
|
||||||
|
|
||||||
|
cy.get('.ProseMirror').find('a[href="https://example1.com"]')
|
||||||
|
.should('contain', 'https://example1.com')
|
||||||
|
|
||||||
|
cy.get('.ProseMirror').find('a[href="https://example2.com/foobar"]')
|
||||||
|
.should('contain', 'https://example2.com/foobar')
|
||||||
|
|
||||||
|
cy.get('.ProseMirror').find('a[href="http://example3.com/foobar"]')
|
||||||
|
.should('contain', 'http://example3.com/foobar')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -61,3 +61,16 @@ export default {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss">
|
||||||
|
/* Basic editor styles */
|
||||||
|
.ProseMirror {
|
||||||
|
> * + * {
|
||||||
|
margin-top: 0.75em;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: #68CEF8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|||||||
@@ -8,7 +8,8 @@ export interface LinkOptions {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
export const pasteRegex = /https?:\/\/(?:www\.)?[-a-zA-Z0-9@:%._+~#=]{1,256}\.[a-zA-Z]{2,}\b(?:[-a-zA-Z0-9@:%._+~#=?!&/()]*)/gi
|
export const pasteRegex = /https?:\/\/(?:www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z]{2,}\b(?:[-a-zA-Z0-9@:%._+~#=?!&/]*)(?:[-a-zA-Z0-9@:%._+~#=?!&/]*)/gi
|
||||||
|
export const pasteRegexWithBrackets = /(?:\()https?:\/\/(?:www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z]{2,}\b(?:[-a-zA-Z0-9@:%._+~#=?!&/()]*)(?:\))/gi
|
||||||
|
|
||||||
const Link = Mark.create({
|
const Link = Mark.create({
|
||||||
name: 'link',
|
name: 'link',
|
||||||
@@ -70,6 +71,7 @@ const Link = Mark.create({
|
|||||||
addPasteRules() {
|
addPasteRules() {
|
||||||
return [
|
return [
|
||||||
markPasteRule(pasteRegex, this.type, (url: string) => ({ href: url })),
|
markPasteRule(pasteRegex, this.type, (url: string) => ({ href: url })),
|
||||||
|
markPasteRule(pasteRegexWithBrackets, this.type, (url: string) => ({ href: url })),
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@@ -2,10 +2,51 @@
|
|||||||
|
|
||||||
import { pasteRegex } from '@tiptap/extension-link'
|
import { pasteRegex } from '@tiptap/extension-link'
|
||||||
|
|
||||||
describe('link regex test', () => {
|
describe('link paste rules', () => {
|
||||||
|
const validUrls = [
|
||||||
|
'https://example.com',
|
||||||
|
'https://example.com/with-path',
|
||||||
|
'http://example.com/with-http',
|
||||||
|
'https://www.example.com/with-www',
|
||||||
|
'https://www.example.com/with-numbers-123',
|
||||||
|
'https://www.example.com/with-parameters?var=true',
|
||||||
|
'https://www.example.com/with-multiple-parameters?var=true&foo=bar',
|
||||||
|
'https://www.example.com/with-spaces?var=true&foo=bar+3',
|
||||||
|
'https://www.example.com/with,comma',
|
||||||
|
'https://www.example.com/with(brackets)',
|
||||||
|
'https://www.example.com/with!exclamation!marks',
|
||||||
|
'http://thelongestdomainnameintheworldandthensomeandthensomemoreandmore.com/',
|
||||||
|
'https://example.longtopleveldomain',
|
||||||
|
'https://example-with-dashes.com',
|
||||||
|
'https://example-with-dashes.com',
|
||||||
|
]
|
||||||
|
|
||||||
it('paste regex matches url', () => {
|
validUrls.forEach(url => {
|
||||||
expect('https://www.example.com/with-spaces?var=true&foo=bar+3').to.match(pasteRegex)
|
it(`paste regex matches url: ${url}`, {
|
||||||
|
// every second test fails, but the second try succeeds
|
||||||
|
retries: {
|
||||||
|
runMode: 2,
|
||||||
|
openMode: 2,
|
||||||
|
},
|
||||||
|
}, () => {
|
||||||
|
// TODO: Check the regex capture group to see *what* is matched
|
||||||
|
expect(url).to.match(pasteRegex)
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const invalidUrls = [
|
||||||
|
'ftp://www.example.com',
|
||||||
|
]
|
||||||
|
|
||||||
|
invalidUrls.forEach(url => {
|
||||||
|
it(`paste regex doesn’t match url: ${url}`, {
|
||||||
|
// every second test fails, but the second try succeeds
|
||||||
|
retries: {
|
||||||
|
runMode: 2,
|
||||||
|
openMode: 2,
|
||||||
|
},
|
||||||
|
}, () => {
|
||||||
|
expect(url).to.not.match(pasteRegex)
|
||||||
|
})
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
Reference in New Issue
Block a user