Skip to content

Commit 3b48dd4

Browse files
committed
Add Vue playground and add SSR
1 parent 69e8b58 commit 3b48dd4

File tree

27 files changed

+743
-466
lines changed

27 files changed

+743
-466
lines changed

.github/workflows/cd.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,6 @@ jobs:
2525
run: yarn build
2626

2727
- name: NPM PKG Publish 🎉
28-
run: yarn workspaces foreach -t --exclude '{embla-carousel-monorepo,embla-carousel-docs,embla-carousel-playground-vanilla,embla-carousel-playground-react,embla-carousel-playground-solid}' npm publish
28+
run: yarn workspaces foreach -t --exclude '{embla-carousel-monorepo,embla-carousel-docs,embla-carousel-playground-vanilla,embla-carousel-playground-react,embla-carousel-playground-solid,embla-carousel-playground-vue}' npm publish
2929
env:
3030
NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }}

.vscode/extenstions.json

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
{
2+
"recommendations": [
3+
"esbenp.prettier-vscode",
4+
"unifiedjs.vscode-mdx",
5+
"Vue.volar"
6+
]
7+
}

package.json

+8-4
Original file line numberDiff line numberDiff line change
@@ -28,26 +28,30 @@
2828
"packages/embla-carousel-reactive-utils",
2929
"playgrounds/embla-carousel-playground-vanilla",
3030
"playgrounds/embla-carousel-playground-react",
31-
"playgrounds/embla-carousel-playground-solid"
31+
"playgrounds/embla-carousel-playground-solid",
32+
"playgrounds/embla-carousel-playground-vue"
3233
],
3334
"scripts": {
34-
"eslint:report": "yarn workspaces foreach -v --exclude \"{embla-carousel-monorepo,embla-carousel-playground-vanilla,embla-carousel-playground-react}\" run eslint:report",
35+
"eslint:report": "yarn workspaces foreach -v --exclude \"{embla-carousel-monorepo,embla-carousel-playground-vanilla,embla-carousel-playground-react,embla-carousel-playground-solid,embla-carousel-playground-vue}\" run eslint:report",
3536
"prettier:report": "prettier \"**/*.{js,jsx,tsx,ts,scss,json}\" --check",
3637
"lint": "npm-run-all prettier:report eslint:report",
3738
"format": "prettier \"**/*.{js,jsx,tsx,ts,scss,json}\" --write",
38-
"test": "yarn workspaces foreach -v --exclude \"{embla-carousel-monorepo,embla-carousel-playground-vanilla,embla-carousel-playground-react}\" run test",
39+
"test": "yarn workspaces foreach -v --exclude \"{embla-carousel-monorepo,embla-carousel-playground-vanilla,embla-carousel-playground-react,embla-carousel-playground-solid,embla-carousel-playground-vue}\" run test",
3940
"watch:package-vanilla": "yarn workspace embla-carousel run start",
4041
"watch:package-react": "yarn workspace embla-carousel-react run start",
4142
"watch:package-solid": "yarn workspace embla-carousel-solid run start",
43+
"watch:package-vue": "yarn workspace embla-carousel-vue run start",
4244
"watch:playground-vanilla": "yarn workspace embla-carousel-playground-vanilla run dev",
4345
"watch:playground-react": "yarn workspace embla-carousel-playground-react run dev",
4446
"watch:playground-solid": "yarn workspace embla-carousel-playground-solid run dev",
47+
"watch:playground-vue": "yarn workspace embla-carousel-playground-vue run dev",
4548
"start": "yarn workspace embla-carousel-docs run start",
4649
"start:vanilla": "npm-run-all --parallel watch:package-vanilla watch:playground-vanilla",
4750
"start:react": "npm-run-all --parallel watch:package-vanilla watch:package-react watch:playground-react",
4851
"start:solid": "npm-run-all --parallel watch:package-vanilla watch:package-solid watch:playground-solid",
52+
"start:vue": "npm-run-all --parallel watch:package-vanilla watch:package-vue watch:playground-vue",
4953
"build": "npm-run-all build:packages build:package-readmes format",
50-
"build:packages": "yarn workspaces foreach -vt --exclude \"{embla-carousel-monorepo,embla-carousel-playground-vanilla,embla-carousel-playground-react}\" run build",
54+
"build:packages": "yarn workspaces foreach -vt --exclude \"{embla-carousel-monorepo,embla-carousel-playground-vanilla,embla-carousel-playground-react,embla-carousel-playground-solid,embla-carousel-playground-vue}\" run build",
5155
"build:package-readmes": "npx ts-node --project scripts/tsconfig.node.json scripts/create-readmes/index.ts --templatePath=scripts/create-readmes/readme-template.md",
5256
"build:docs": "yarn workspace embla-carousel-docs run predeploy",
5357
"version:patch": "yarn workspaces foreach version patch && yarn version:push",

packages/embla-carousel-react/src/components/useEmblaCarousel.ts

+8-8
Original file line numberDiff line numberDiff line change
@@ -6,12 +6,12 @@ import EmblaCarousel, {
66
EmblaPluginType
77
} from 'embla-carousel'
88

9-
export type EmblaViewportRefType = <ViewportElement extends HTMLElement>(
10-
instance: ViewportElement | null
9+
export type EmblaRootNodeRefType = <RootNode extends HTMLElement>(
10+
instance: RootNode | null
1111
) => void
1212

1313
export type UseEmblaCarouselType = [
14-
EmblaViewportRefType,
14+
EmblaRootNodeRefType,
1515
EmblaCarouselType | undefined,
1616
EmblaCarouselType
1717
]
@@ -27,7 +27,7 @@ function useEmblaCarousel(
2727

2828
const serverApi = useRef(EmblaCarousel(null, options, plugins))
2929
const [clientApi, setClientApi] = useState<EmblaCarouselType>()
30-
const [viewport, setViewport] = useState<HTMLElement>()
30+
const [rootNode, setRootNode] = useState<HTMLElement>()
3131

3232
const reInit = useCallback(() => {
3333
if (!clientApi) return
@@ -47,10 +47,10 @@ function useEmblaCarousel(
4747
}, [plugins, reInit])
4848

4949
useEffect(() => {
50-
if (viewport) {
50+
if (rootNode) {
5151
EmblaCarousel.globalOptions = useEmblaCarousel.globalOptions
5252
const newClientApi = EmblaCarousel(
53-
viewport,
53+
rootNode,
5454
storedOptions.current,
5555
storedPlugins.current
5656
)
@@ -59,9 +59,9 @@ function useEmblaCarousel(
5959
} else {
6060
setClientApi(undefined)
6161
}
62-
}, [viewport])
62+
}, [rootNode])
6363

64-
return [<EmblaViewportRefType>setViewport, clientApi, serverApi.current]
64+
return [<EmblaRootNodeRefType>setRootNode, clientApi, serverApi.current]
6565
}
6666

6767
declare namespace useEmblaCarousel {
+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
export {
22
UseEmblaCarouselType,
3-
EmblaViewportRefType
3+
EmblaRootNodeRefType
44
} from './components/useEmblaCarousel'
55
export { default } from './components/useEmblaCarousel'

packages/embla-carousel-solid/src/components/useEmblaCarousel.ts

+5-5
Original file line numberDiff line numberDiff line change
@@ -30,19 +30,19 @@ function useEmblaCarousel(
3030

3131
const serverApi = EmblaCarousel(null, storedOptions, storedPlugins)
3232
const [clientApi, setClientApi] = createSignal<EmblaCarouselType>()
33-
const [viewport, setViewport] = createSignal<HTMLElement>()
33+
const [rootNode, setRootNode] = createSignal<HTMLElement>()
3434

3535
function reInit(): void {
3636
const api = clientApi()
3737
if (api) api.reInit(storedOptions, storedPlugins)
3838
}
3939

4040
createEffect(
41-
on(viewport, (viewport) => {
42-
if (viewport) {
41+
on(rootNode, (rootNode) => {
42+
if (rootNode) {
4343
EmblaCarousel.globalOptions = useEmblaCarousel.globalOptions
4444
const newEmblaApi = EmblaCarousel(
45-
viewport,
45+
rootNode,
4646
storedOptions,
4747
storedPlugins
4848
)
@@ -68,7 +68,7 @@ function useEmblaCarousel(
6868
reInit()
6969
})
7070

71-
return [setViewport, clientApi, serverApi]
71+
return [setRootNode, clientApi, serverApi]
7272
}
7373

7474
declare namespace useEmblaCarousel {

packages/embla-carousel-vue/src/components/useEmblaCarousel.ts

+17-18
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,4 @@
1-
import {
2-
Ref,
3-
MaybeRef,
4-
isRef,
5-
watch,
6-
onMounted,
7-
onBeforeUnmount,
8-
shallowRef
9-
} from 'vue'
1+
import { Ref, MaybeRef, isRef, watch, onBeforeUnmount, shallowRef } from 'vue'
102
import { areOptionsEqual, arePluginsEqual } from 'embla-carousel-reactive-utils'
113
import EmblaCarousel, {
124
EmblaCarouselType,
@@ -41,15 +33,22 @@ function useEmblaCarousel(
4133
clientApi.value.reInit(storedOptions, storedPlugins)
4234
}
4335

44-
onMounted(() => {
45-
if (!rootNode.value) return
46-
EmblaCarousel.globalOptions = useEmblaCarousel.globalOptions
47-
clientApi.value = EmblaCarousel(
48-
rootNode.value,
49-
storedOptions,
50-
storedPlugins
51-
)
52-
})
36+
watch(
37+
() => rootNode.value,
38+
() => {
39+
if (rootNode.value) {
40+
EmblaCarousel.globalOptions = useEmblaCarousel.globalOptions
41+
clientApi.value = EmblaCarousel(
42+
rootNode.value,
43+
storedOptions,
44+
storedPlugins
45+
)
46+
} else {
47+
clientApi.value = undefined
48+
}
49+
},
50+
{ immediate: true }
51+
)
5352

5453
onBeforeUnmount(() => {
5554
if (clientApi.value) clientApi.value.destroy()

playgrounds/embla-carousel-playground-react/package.json

-6
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,6 @@
66
"scripts": {
77
"dev": "vite",
88
"build": "tsc -b && vite build",
9-
"lint": "eslint .",
109
"preview": "vite preview",
1110
"test": "echo \"Info: no tests specified\" && exit 0"
1211
},
@@ -17,16 +16,11 @@
1716
"react-router-dom": "^6.4.3"
1817
},
1918
"devDependencies": {
20-
"@eslint/js": "^9.17.0",
2119
"@types/react": "^18.3.18",
2220
"@types/react-dom": "^18.3.5",
2321
"@vitejs/plugin-react": "^4.3.4",
24-
"eslint": "^9.17.0",
25-
"eslint-plugin-react-hooks": "^5.0.0",
26-
"eslint-plugin-react-refresh": "^0.4.16",
2722
"globals": "^15.14.0",
2823
"typescript": "~5.6.2",
29-
"typescript-eslint": "^8.18.2",
3024
"vite": "^6.0.5"
3125
}
3226
}

playgrounds/embla-carousel-playground-react/src/Carousel/Buttons.tsx

+3-5
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,9 @@
1-
import React from 'react'
2-
31
type DotButtonPropType = {
42
selected: boolean
53
onClick: () => void
64
}
75

8-
export const DotButton: React.FC<DotButtonPropType> = (props) => {
6+
export const DotButton = (props: DotButtonPropType) => {
97
const { selected, onClick } = props
108

119
return (
@@ -22,7 +20,7 @@ type PrevNextButtonPropType = {
2220
onClick: () => void
2321
}
2422

25-
export const PrevButton: React.FC<PrevNextButtonPropType> = (props) => {
23+
export const PrevButton = (props: PrevNextButtonPropType) => {
2624
const { enabled, onClick } = props
2725

2826
return (
@@ -42,7 +40,7 @@ export const PrevButton: React.FC<PrevNextButtonPropType> = (props) => {
4240
)
4341
}
4442

45-
export const NextButton: React.FC<PrevNextButtonPropType> = (props) => {
43+
export const NextButton = (props: PrevNextButtonPropType) => {
4644
const { enabled, onClick } = props
4745

4846
return (

playgrounds/embla-carousel-playground-react/src/Carousel/Carousel.tsx

+8-5
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import React, { useState, useEffect, useCallback } from 'react'
1+
import { useState, useEffect, useCallback } from 'react'
22
import { EmblaCarouselType, EmblaOptionsType } from 'embla-carousel'
33
import useEmblaCarousel from 'embla-carousel-react'
44
import { DotButton, PrevButton, NextButton } from './Buttons'
@@ -9,7 +9,7 @@ type PropType = {
99
options?: EmblaOptionsType
1010
}
1111

12-
export const EmblaCarousel: React.FC<PropType> = (props) => {
12+
export const EmblaCarousel = (props: PropType) => {
1313
const { slides, options, isSsr } = props
1414
const [refAttached, setRefAttached] = useState(false)
1515
const [emblaRef, emblaApi, emblaServerApi] = useEmblaCarousel(options)
@@ -19,8 +19,11 @@ export const EmblaCarousel: React.FC<PropType> = (props) => {
1919
const [nextBtnEnabled, setNextBtnEnabled] = useState(
2020
emblaServerApi.canScrollNext()
2121
)
22-
const [selectedIndex, setSelectedIndex] = useState(0)
22+
const [selectedIndex, setSelectedIndex] = useState(
23+
emblaServerApi.selectedSnap()
24+
)
2325
const [scrollSnaps, setScrollSnaps] = useState<number[]>([])
26+
const showSsr = isSsr && !emblaApi
2427

2528
const scrollPrev = useCallback(
2629
() => emblaApi && emblaApi.scrollPrev(),
@@ -65,14 +68,14 @@ export const EmblaCarousel: React.FC<PropType> = (props) => {
6568

6669
return (
6770
<>
68-
{isSsr && !emblaApi && (
71+
{showSsr && (
6972
<style id="embla-ssr-styles">
7073
{emblaServerApi.ssrStyles('.embla__container', '.embla__slide')}
7174
</style>
7275
)}
7376

7477
<div className="playground__ssr-text">
75-
<strong>SSR:</strong> <span>{(isSsr && !emblaApi).toString()}</span>
78+
<strong>SSR:</strong> <span>{showSsr.toString()}</span>
7679
</div>
7780

7881
<div className="embla">

playgrounds/embla-carousel-playground-react/src/main.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -59,7 +59,7 @@ const injectBaseStyles = (): void => {
5959

6060
injectBaseStyles()
6161

62-
const App: React.FC = () => {
62+
const App = () => {
6363
return (
6464
<main className="playground">
6565
<h1 className="playground__h1">Playground - React</h1>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
# Logs
2+
logs
3+
*.log
4+
npm-debug.log*
5+
yarn-debug.log*
6+
yarn-error.log*
7+
pnpm-debug.log*
8+
lerna-debug.log*
9+
10+
node_modules
11+
dist
12+
dist-ssr
13+
*.local
14+
15+
# Editor directories and files
16+
.vscode/*
17+
!.vscode/extensions.json
18+
.idea
19+
.DS_Store
20+
*.suo
21+
*.ntvs*
22+
*.njsproj
23+
*.sln
24+
*.sw?
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
<!DOCTYPE html>
2+
<html lang="en">
3+
<head>
4+
<meta charset="UTF-8" />
5+
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
6+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
7+
<title>Vite + Vue + TS</title>
8+
</head>
9+
<body class="theme-dark">
10+
<div id="app"></div>
11+
<script type="module" src="/src/main.ts"></script>
12+
</body>
13+
</html>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
{
2+
"name": "embla-carousel-playground-vue",
3+
"private": true,
4+
"version": "8.5.2",
5+
"type": "module",
6+
"scripts": {
7+
"dev": "vite",
8+
"build": "vue-tsc -b && vite build",
9+
"preview": "vite preview",
10+
"test": "echo \"Info: no tests specified\" && exit 0"
11+
},
12+
"dependencies": {
13+
"embla-carousel-vue": "8.5.2",
14+
"vue": "^3.5.13"
15+
},
16+
"devDependencies": {
17+
"@vitejs/plugin-vue": "^5.2.1",
18+
"@vue/tsconfig": "^0.7.0",
19+
"typescript": "~5.6.2",
20+
"vite": "^6.0.5",
21+
"vue-tsc": "^2.2.0"
22+
}
23+
}

0 commit comments

Comments
 (0)