Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/pngwn/svelte-adapter
Use Svelte components with Vue and React
https://github.com/pngwn/svelte-adapter
Last synced: 4 days ago
JSON representation
Use Svelte components with Vue and React
- Host: GitHub
- URL: https://github.com/pngwn/svelte-adapter
- Owner: pngwn
- Created: 2019-06-23T11:27:40.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2023-03-06T11:26:11.000Z (almost 2 years ago)
- Last Synced: 2024-10-14T13:03:16.289Z (3 months ago)
- Language: JavaScript
- Size: 2.49 MB
- Stars: 298
- Watchers: 6
- Forks: 11
- Open Issues: 19
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
- awesome-list - svelte-adapter
- awesome-svelte - Svelte Adapter - Utilitário para utilizar Svelte em projetos Vue e React. (Utilitários / Comunidade Global)
README
# svelte-adapter
A simple utility that allows you to use [Svelte](https://svelte.dev/) components inside [React](https://reactjs.org/) or [Vue](https://vuejs.org/) components.
There is an adapter each for Vue and React which allows you to pass props and respond to events in a way that makes sense for that library.
This isn't a perfect solution, there are some limitations.
---
- [Install](#install)
- [Use it](#use-it)
- [Examples](#examples)
- [React](#react)
- [Vue](#vue)
- [Limitations](#limitations)# Install
With [npm](https://www.npmjs.com/):
```bash
npm install svelte-adapter
```Or with [yarn](https://yarnpkg.com/lang/en/):
```bash
yarn add svelte-adapter
```# Use it
Each 'adapter' is a simple function that takes a svelte component and a few options and returns a Vue or React component that can be used in Vue templates or JSX as you would expect. The returned components will always have a wrapper element, by default this is a `` but it can be customised.
The adapters both have the same signature:
```ts
adapter(Component: SvelteComponent, styleObject?: object, wrapperElement?: string) : Component
```- `Component` should be a _compiled_ svelte component, either precompiled or compiled as part of your build step using `rollup-plugin-svelte` for rollup or `svelte-loader` from webpack.
- `styleObject` (optional) should be an object of styles that will be applied to the base elemement. This should be a valid JavaScript object with camelCased css property names. This defaults to an empty object.
- `wrapperElement` (optional). All component have a base 'wrapper' element, by default this is a `` but you can pass in a string to customise this behaviour (eg: `'div'`, `'li'`, etc.) If need a specific wrapper element but don't care about styles, you can simply pass an empty object as the `styleObject`.In the examples below, the Svelte component we will be using is a simple component that accepts a prop that will be rendered and emits an event upon clicking a button.
```svelte
import { createEventDispatcher } from 'svelte';
export let number = 0;
const dispatch = createEventDispatcher();{ number }
dispatch('magicalclick') }>
Click Me```
## Examples
### React
[Try it on codesandbox](https://codesandbox.io/s/svelte-adapterreact-8s33k)
The React-Svelte adapter is the default export from `svelte-adpater/react`.
The implementation of the React adapter uses [hooks](https://reactjs.org/docs/hooks-intro.html) so you will need a recent version of React for this to work. If necessary I could add a class-based version at some stage.
Using the adaptor is very straight-forward, the adapter is a Higher Order Component that takes a Svelte component and returns a React component. The below example assumes your are compiling Svelte components as they are imported using webpack or rollup. If not, just import the compiled javascript file instead.
Svelte components can emit events which doesn't quite make sense in React. Any events emitted by the svelte component can be passed callbacks via an `on*` prop containing a function, this function will fire when the event is emitted. Any prop that starts with `on` followed by a capital letter is assumed to be an event and will be used to add a listener. `onClick` will fire the provided callback when `'click'` events are emitted, `onSomethingRandom` will do the same for `'somethingRandom'` events. This does not interfere with props that have the same naming convention, they will all be passed regardless.
Some Svelte component's allow you to `bind` to internal data which doesn't make too much sense outside of Svelte yet they often form an important part of the API. Instead I have added the option to use a `watch*` prop (similar to the `on*` prop). This also takes a callback function and recieves the value you wish to watch as its only argument. `watchNumber={ n => setCount(n) }` would watch the internal value `number`, when `number` changes the callback you passed to it would be executed receiving the new `number` value as its only argument.
This may seem strange but many Svelte components are written to make use of this `bind` syntax, without it there is often a hole in the API leaving you unable to respond to internal state changes. You will probably want to control your state with React, this `watch*` prop is an escape hatch that allows you to pull out those internal values to use however you wish.
Normal props behave as you would expect.
```jsx
import React, { useState } from "react";
import toReact from "svelte-adapter/react";import SvelteApp from "../App.svelte";
const baseStyle = {
width: "50%"
};const SvelteInReact = toReact(SvelteApp, baseStyle, "div");
const App = () => {
const [count, setCount] = useState(10);const handleClick = () => setCount(prevCount => prevCount + 1);
return (
setCount(n)}
/>
Increment - {count}
);
};
```### Vue
[Try it on codesandbox](https://codesandbox.io/s/svelte-adaptervue-40uwg)
The Vue-Svelte adapter is the default export from `svelte-adpater/vue`.
Using the adapter is very straight-forward, the adapter is a Higher Order Component that takes a Svelte component and returns a Vue component. The below example assumes your are compiling Svelte components as they are imported using webpack or rollup. If not, just import the compiled javascript file instead.
Since Vue has an event mechanism similar to Svelte, event directives can be used as expected. `v-on:*` or `@*` directives will listen for the matching events on the Svelte component. `v-on:click` or `@click` will fire the provided callback when `'click'` events are emitted, `v-on:somethingRandom` or `@somethingrandom` will do the same for `'somethingRandom'` events. Other props behave as you would expect.
Some Svelte component's allow you to `bind` to internal data which doesn't make too much sense outside of Svelte yet they often form an important part of the API. I have added the option to use a `@watch:*` prop (similar to the `@:*` prop). This also takes a callback function and recieves the value you wish to watch as its only argument. `@watch:number="setCount"` would watch the internal value `number`, when `number` changes the callback you passed to it would be executed receiving the new `number` value as its only argument.
This may seem strange but many Svelte components are written to make use of this `bind` syntax, without it there is often a hole in the API leaving you unable to respond to internal state changes. You will probably want to control your state in a Vue component, this `@watch:*` prop is an escape hatch that allows you to pull out those internal values to use however you wish.
Normal props behave as expected.
```vue
Increment - {{ count }}
import SvelteApp from "../App.svelte";
import toVue from "svelte-adapter/vue";const baseStyle = {
width: "50%"
};export default {
components: {
SvelteInVue: toVue(SvelteApp, baseStyle, "div")
},
data() {
return {
count: 0
};
},
methods: {
handleClick() {
this.count += 1;
},
setCount(n) {
this.count = n;
}
}
};```
## Limitations
While everything should work in most situations, it is not currently possible to pass children or slots to Svelte components with these adapters.
This won't work with any of the adpaters:
```html
Hello
```
There may be more limitations that I am unaware of, this package is really just intended a simple way to use Svelte and should work most of the time. The code is short and simple, if you have specific needs you will probably be better off writing something custom for your application.