Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/LXSMNSYC/solid-styled

Reactive stylesheets for SolidJS
https://github.com/LXSMNSYC/solid-styled

css css-in-js scoped-css solid-js

Last synced: about 1 month ago
JSON representation

Reactive stylesheets for SolidJS

Awesome Lists containing this project

README

        

# solid-styled

> Reactive stylesheets for SolidJS

[![NPM](https://img.shields.io/npm/v/solid-styled.svg)](https://www.npmjs.com/package/solid-styled) [![JavaScript Style Guide](https://badgen.net/badge/code%20style/airbnb/ff5a5f?icon=airbnb)](https://github.com/airbnb/javascript)

## Install

```bash
npm i solid-styled
npm i -D postcss
```

```bash
yarn add solid-styled
yarn add -D postcss
```

```bash
pnpm add solid-styled
pnpm add -D postcss
```

## Features

- Render stylesheets only once
- Fine-grained reactive CSS properties
- Scoped stylesheets
- `:global` selector
- `@global` at-rule
- SSR
- Near zero-runtime
- ``

## Examples

- Vite - [![Open in CodeSandbox](https://img.shields.io/badge/Open%20in-CodeSandbox-blue?style=flat-square&logo=codesandbox)](https://codesandbox.io/s/github/LXSMNSYC/solid-styled/tree/main/examples/demo)
- Astro - [![Open in CodeSandbox](https://img.shields.io/badge/Open%20in-CodeSandbox-blue?style=flat-square&logo=codesandbox)](https://codesandbox.io/s/github/LXSMNSYC/solid-styled/tree/main/examples/astro-demo)

## Usage

### Integrations

- [Rollup](https://github.com/lxsmnsyc/solid-styled/tree/main/packages/rollup)
- [Vite](https://github.com/lxsmnsyc/solid-styled/tree/main/packages/vite)
- [Unplugin](https://github.com/lxsmnsyc/solid-styled/tree/main/packages/unplugin)

### `css`

Use the `css` tagged template for writing stylesheets.

```js
import { css } from 'solid-styled';

function Title() {
css`
h1 {
color: red;
}
`;

return <h1>Hello World</h1>;
}
```

The template is also reactive. It works by replacing all templating values with a CSS variable. This allows the stylesheet to be only rendered once and can be shared by multiple components of the same scope.

```js
import { css } from 'solid-styled';

function Button() {
const [color, setColor] = createSignal('red');
css`
button {
color: ${color()};
}
`;

return (
<button onClick={() => {
setColor((c) => (c === 'red' ? 'blue' : 'red'));
}}>
Current color: {color()}
</button>
);
}
```

By default, all styles are scoped to its component and cannot leak into another component. The scoping only applies to all DOM nodes that can be found in the component, including the children of the external components.

```js
import { css } from 'solid-styled';

function ForeignTitle() {
return <h1>This is not affected</h1>;
}

function Title() {
css`
h1 {
color: red;
}
`;

return (
<>
<h1>This is affected.</h1>
<ForeignTitle />
<Container>
<h1>This is also affected.</h1>
</Container>
</>
)
}
```

#### `:global`

Since all selectors in a given stylesheet are scoped by default, you can use the `:global` pseudo selector to opt-out of scoping:

```js
import { css } from 'solid-styled';

function Feed(props) {
css`
div > :global(* + *) {
margin-top: 0.5rem;
}
`;

return (
<div>
<For each={props.articles}>
{(item) => (
// This item is affected by the CSS of the Feed component.
<FeedArticle data={item} />
)}
</For>
</div>
);
}
```

#### `@global`

You can use `@global` instead of `:global` if you want to declare multiple global styles

```js
css`
/* this is global */
@global {
body {
background-color: black;
}

main {
padding: 0.5rem;
}
}

h1 {
color: white;
}
`;
```

### Forward scope

Since `solid-styled` scopes DOM elements and not components by default, it doesn't affect things like `<Dynamic>`. Using `use:solid-styled`, we can forward the current scope/sheet to the component.

```js
css`
* {
color: red;
}
`;

<Dynamic component={props.as} use:solid-styled>
{props.children}
</Dynamic>
```

which compiles into

```js
useSolidStyled('xxxx', '*[s\\:xxxx]{color:red}');

<Dynamic component={props.as} s:xxxx style={vars()}>
{props.children}
</Dynamic>
```

### `<style jsx>`

Inspired by [`styled-jsx`](https://github.com/vercel/styled-jsx).

Use `<style jsx>` instead of `css` for declaring styles in JSX expressions. Both `<style jsx>` and `css` functions the same.

```js
function Button() {
const [color, setColor] = createSignal('red');
return (
<>
<style jsx>
{`
button {
color: ${color()};
}
`}

{
setColor((c) => (c === 'red' ? 'blue' : 'red'));
}}>
Current color: {color()}

>
);
}
```

You can also use `` for declaring global styles.

The main motivation for writing an alternative way of declaring styles with `<style jsx>` is to facilitate the migration from `solid-styled-jsx` to `solid-styled`. Possibly, some developers may as well use `<style jsx>` because of their familiarity with adding the styles inside the JSX.

## SSR

### `<StyleRegistry>`

For SSR, you can pass an array to the `styles` prop of `<StyleRegistry>`. This array collects all of the "critical" (initial render) stylesheets, that which you can render as a string with `renderSheets`.

```js
import { renderSheets } from 'solid-styled';

const styles = [];

renderToString(() => (
<StyleRegistry styles={styles}>
<App />

));

// Render sheets
// You can use the resulting sheet by inserting
// it into an HTML template.
const styles = renderSheets(styles);
```

## CSS Processing

`solid-styled` uses [LightningCSS](https://lightningcss.dev/) to preprocess CSS templates as well as apply CSS scoping and transformations. By default, [CSS Nesting and Custom Media Queries](https://lightningcss.dev/transpilation.html#draft-syntax) are enabled by default.

## Limitations

- Scoping `css` can only be called directly on components. This is so that the Babel plugin can find and transform the JSX of the component. Global `css` (i.e. `:global` or `@global`) can be used inside other functions i.e. hooks, utilities.
- Dynamic styles are only limited to CSS properties.

## Sponsors

![Sponsors](https://github.com/lxsmnsyc/sponsors/blob/main/sponsors.svg?raw=true)

## License

MIT © [lxsmnsyc](https://github.com/lxsmnsyc)