Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/paoloricciuti/sveltekit-search-params
The easiest way to read and WRITE from query parameters in sveltekit.
https://github.com/paoloricciuti/sveltekit-search-params
hacktoberfest query-params search-params svelte svelte-store svelte-stores sveltekit
Last synced: 6 days ago
JSON representation
The easiest way to read and WRITE from query parameters in sveltekit.
- Host: GitHub
- URL: https://github.com/paoloricciuti/sveltekit-search-params
- Owner: paoloricciuti
- License: mit
- Created: 2022-10-11T15:04:28.000Z (about 2 years ago)
- Default Branch: next
- Last Pushed: 2024-10-29T23:48:35.000Z (about 2 months ago)
- Last Synced: 2024-10-29T23:48:42.052Z (about 2 months ago)
- Topics: hacktoberfest, query-params, search-params, svelte, svelte-store, svelte-stores, sveltekit
- Language: TypeScript
- Homepage: https://sveltekit-search-params.netlify.app
- Size: 813 KB
- Stars: 533
- Watchers: 4
- Forks: 17
- Open Issues: 18
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Funding: .github/FUNDING.yml
- License: LICENSE
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
# sveltekit-search-params
The fastest way to read **AND WRITE** from query search params in [sveltekit](https://github.com/sveltejs/kit).
> **Warning**
>
> This package is meant to be used with Svelte-Kit as the name suggest. Because it uses api that are **only** present in Svelte-Kit it will not work in your normal svelte project.[![MIT License](https://img.shields.io/badge/License-MIT-green.svg)](https://choosealicense.com/licenses/mit/)
![npm](https://img.shields.io/npm/v/sveltekit-search-params)
![npm](https://img.shields.io/npm/dt/sveltekit-search-params)
![GitHub last commit](https://img.shields.io/github/last-commit/paoloricciuti/sveltekit-search-params)
## Contributing
Contributions are always welcome!
For the moment there's no contributing guidelines but if you found a problem or have an idea feel free to [open an issue](https://github.com/paoloricciuti/sveltekit-search-params/issues/new)
Function changes intended for release should [include a changeset](https://github.com/changesets/changesets/blob/main/docs/adding-a-changeset.md)
If you want the fastest way to open a PR try out Codeflow
[![Open in Codeflow](https://developer.stackblitz.com/img/open_in_codeflow.svg)](https://pr.new/paoloricciuti/sveltekit-search-params/)
## Authors
- [@paoloricciuti](https://www.github.com/paoloricciuti)
## Installation
Install sveltekit-search-params with npm
```bash
npm install sveltekit-search-params@latest -D
```## Usage/Examples
### Simple case
To start using the library you can import the method `queryParameters`. You can invoke this method without arguments to get an object containing all the present search params.
```svelte
import { queryParameters } from 'sveltekit-search-params';
const params = queryParameters();
{JSON.stringify(params, null, 2)}
```assuming the page is `/?framework=svelte&isCool=true` the above code will show
```json
{
"framework": "svelte",
"isCool": "true"
}
```by default all query parameters are string.
### Writing to the store
Reading query parameters is cool but you know what is even cooler? Writing query parameters! With this library you can treat your query parameters just like normal state in svelte. To update the state and consequently the url you can just do this
```svelte
import { queryParameters } from 'sveltekit-search-params';
const params = queryParameters();
{JSON.stringify(params, null, 2)}
{
params.username = e.target.value;
}}
/>
```### Expecting some parameters
Most of the times if you need to read from query parameters you are expecting some parameters to be present. You can define the parameters you are expecting during the initial invocation and those will be merged with the actual query parameters despite the fact that they are present or not.
```svelte
import { queryParameters } from 'sveltekit-search-params';
const params = queryParameters({
username: true,
});
{JSON.stringify(params, null, 2)}
```assuming the page is `/?framework=svelte&isCool=true` the above code will show
```json
{
"framework": "svelte",
"isCool": "true",
"username": null
}
```if we add username to the URL like this `/?framework=svelte&isCool=true&username=paoloricciuti` we will get
```json
{
"framework": "svelte",
"isCool": "true",
"username": "paoloricciuti"
}
```### Encoding and Decoding
By default query parameters are strings but more often than not though we are not working with strings. We are dealing with numbers, boolean, arrays and complex objects. During the initialization of the state you can specify an object containing an encode and a decode property that will be used to transform your data from and to the type you need.
```svelte
import { queryParameters } from 'sveltekit-search-params';
const params = queryParameters({
username: true,
isCool: {
encode: (booleanValue) => booleanValue.toString(),
decode: (stringValue) =>
stringValue !== null && stringValue !== 'false',
},
});
{JSON.stringify(params, null, 2)}
```assuming the page is `/?framework=svelte&isCool=true&username=paoloricciuti` the above code will show
```json
{
"framework": "svelte",
"isCool": true,
"username": null
}
```notice that this time isCool it's a boolean and not a string anymore. With this particular transformation we've assured that if the url is the following `/?framework=svelte&isCool=false&username=paoloricciuti` or if the isCool parameter is completely missing like this `/?framework=svelte&username=paoloricciuti` we will get
```json
{
"framework": "svelte",
"isCool": false,
"username": null
}
```### Default values
Sometimes when we want to create a new variable we like to pass a default value. You can do this by passing a third field, `defaultValue` in the object you pass to `queryParameters`.
```svelte
import { queryParameters } from 'sveltekit-search-params';
const params = queryParameters({
username: true,
count: {
encode: (value: number) => value.toString(),
decode: (value: string | null) => (value ? parseInt(value) : null),
defaultValue: 10,
},
});
{JSON.stringify(params, null, 2)}
```this will make the query parameter have the default value and the URL change as soon as the page is rendered on the browser (the query parameter will change only if it's not already present and only the first time the application render).
> **Warning**
>
> You can't run `goto` on the server so if the page is server side rendered it will still have the default value but it will not actually navigate until the client bundle loads.### Helpers encodings and decodings
Write an encode and decode function may seem trivial but it's tedious for sure. `sveltekit-search-params` provide with a set of helpers for better readability and to avoid the hassle of writing common transforms. You can find those helpers exported in a ssp variable from the same package.
```svelte
import { ssp, queryParameters } from 'sveltekit-search-params';
const params = queryParameters({
username: true,
isCool: ssp.boolean(),
});
{JSON.stringify(params, null, 2)}
```this code will produce the same output as the code written above but far more readable and easier to read. You can find all the exports documented in the section [ssp - Helpers](#ssp---helpers).
You can also pass a default value to the function that will be the defaultValue of the object.
```svelte
import { ssp, queryParameters } from 'sveltekit-search-params';
const params = queryParameters({
username: true,
isCool: ssp.count(10),
});
{JSON.stringify(params, null, 2)}
```## ssp - Helpers
There are six helpers all exported as functions on the object ssp. To each one of them you can pass a parameter that will be the default value for that query param.
#### object
To map from a query parameter to an object. An url like this `/?obj={"isComplex":%20true,%20"nested":%20{"field":%20"value"}}` will be mapped to
```typescript
params.obj.isComplex; //true
params.obj.nested; // {field: "value"}
params.obj.nested.value; // "value"
```#### array
To map from a query parameter to an array. An url like this `/?arr=[1,2,3,4]` will be mapped to
```typescript
params.arr[0]; //1
params.arr[1]; //2
params.arr[2]; //3
params.arr[3]; //4
```#### number
To map from a query parameter to a number. An url like this `/?num=1` will be mapped to
```typescript
params.num; //1
```#### boolean
To map from a query parameter to a boolean. An url like this `/?bool=true` will be mapped to
```typescript
params.bool; //true
```as we've seen an url like this `/?bool=false` will be mapped to
```typescript
params.bool; //false
```just like an url like this `/`
#### string
This is exported mainly for readability since all query parameters are already strings.
#### lz
To map any JSON serializable state to his lz-string representation. This is a common way to store state in query parameters that will prevent the link to directly show the state.
An url like this `/?state=N4IgbghgNgrgpiAXCAsgTwAQGMD2OoYCO8ATpgA4QkQC2cALnCSAL5A` will map to
```typescript
params.state.value; //My cool query parameter
```### equalityFn
While this is not a problem for primitive values if your state has a complex object or an array as a value even if the reference is the same svelte will trigger reactivity for it. To provide you with optimistic updates there's the possibility that the state will change multiple times during a single navigation. To fix this problem by default we check if the value of the param is the same by using `JSON.stringify` so that if the overall shape of your param is the same we avoid triggering the reactivity.
This is fine for most cases and you will likely never touch this option but if you have some use case not covered by `JSON.stringify` you can specify the option `equalityFn`. This option is a function that takes the `current` value and the `next` value as parameters and need to return a `boolean`. You should return `true` from this function when the value of the params is unchanged (according to your own logic). This will not trigger reactivity (note that the navigation will still happen).
You can specify this function as another argument in the encode and decode options object.
```svelte
import { queryParameters } from 'sveltekit-search-params';
const params = queryParameters({
obj: {
encode: (value: { value: string }) => JSON.stringify(value),
decode: (value: string | null) =>
value ? JSON.parse(value) : null,
equalityFn(current, next) {
return current?.value === next?.value;
},
},
});
{JSON.stringify(params, null, 2)}
```NOTE: the equality function will not be used on primitive values, hence you can't pass the equality function to parameters that have a primitive type.
To specify the equality function if you are using the `ssp` helpers you can invoke the return value of the helper with the equality function.
```ts
ssp.number()((current, next) => {
// your equality function
});// with default value
ssp.number(10)((current, next) => {
// your equality function
});
```the helper make use of a technique called currying so that, if you don't need to pass the equality function you can just invoke the function (with or without a default parameter), and if you want to pass an equality function you can avoid passing an `undefined` as the first parameter.
## Options
The `queryParameters` function accept a configuration object that contains the following properties:
### debounceHistory
The number of milliseconds to delay the writing of the history when the state changes. This is to avoid cluttering the history of the user especially when some parameter is bound to an input text (every keystroke would cause a new history entry). It defaults to 0. If set a new entry in the history will be added only after `debounceHistory` seconds of "inactivity".
### pushHistory
A boolean defining if the history have to be written at all. If set to false no new history entries will be written to the history stack (the URL will still update but the user will not be able to go back with the browser).
### sort
Whenever you interact with the returned object, it navigates for you. By default the search params are sorted to allow for better cache-ability. You can disable this behavior by passing `false` to this option. Keep in mind that this is a per-object settings. This mean that if you interact with an object that has this option set to `false` and than interact with one that has this option set to `true` (the default) the resulting URL will still have the search params sorted.
### showDefaults
If you specify a default value for the search param and it's not present by default the library will immediately navigate to the url that contains the default search param. For example if you have this code
```svelte
import { queryParameters, ssp } from 'sveltekit-search-params';
const params = queryParameters({
pageNum: ssp.number(0),
});```
and you navigate to `/` as soon as the page load it will be redirected to `/?pageNum=0`. If you prefer not to show the defaults in the URL you can set the `showDefaults` option to false.
```svelte
import { queryParameters, ssp } from 'sveltekit-search-params';
const params = queryParameters(
{
pageNum: ssp.number(0),
},
{
showDefaults: false,
},
);```
By doing so `params` will still have a value of 0 if the search param is not present but the user will not be redirected to `/?pageNum=0`.
### How to use it
To set the configuration object you can pass it as a second parameter
```svelte
import { ssp, queryParameters } from 'sveltekit-search-params';
const params = queryParameters(
{
username: true,
isCool: ssp.boolean(true),
},
{
debounceHistory: 500, // url will change after 500ms
pushHistory: false, // no new history entries for this object
},
);```
## Pre 3.0.0 documentation
Version 4 of this library got a major rewrite to use the runes syntax released in svelte 5. If you are looking at the pre 4.0 documentation that uses a store you can find it in the following docs.
Open the v3.0 documentation
### Simple case (single parameter)
The simplest and most effective way to use this library is by importing the method `queryParam`. You can invoke this method with a string that represent the search parameters you are looking for in the URL.
```svelte
import { queryParam } from 'sveltekit-search-params';
const username = queryParam('username');
Your username is {$username}
```the function returns a store so make sure to use it with the `$` prepended to handle auto-subscriprion. In case there's not a query parameter with the chosen name it will simply be null.
### Writing to the store (single parameter)
Reading query parameters is cool but you know what is even cooler? Writing query parameters! With this library you can treat your store just like normal state in svelte. To update the state and conseguentely the url you can just do this
```svelte
import { queryParam } from 'sveltekit-search-params';
const username = queryParam('username');
Your username is {$username}
```
or if you prefer
```svelte
import { queryParam } from 'sveltekit-search-params';
const username = queryParam('username');
Your username is {$username}
{
$username = e.target.value;
}}
/>
```### Encoding and decoding
By default query parameters are strings but more often than not though we are not working with strings. We are dealing with numbers, boolean, arrays and complex objects. During the creation of the store you can specify an object containing an encode and a decode property that will be used to transform your data from and to the type you need.
```svelte
import { queryParam } from 'sveltekit-search-params';
const count = queryParam('count', {
encode: (value: number) => value.toString(),
decode: (value: string | null) => (value ? parseInt(value) : null),
});The count is {$count}
```
this time $count would be of type number and the deconding function it's what's used to update the url when you write to the store.
### Default values
Sometimes when we want to create a new variable we like to pass a default value. You can do this by passing a third field, `defaultValue`, in the second argument of the query param object.
```svelte
import { queryParam } from 'sveltekit-search-params';
const count = queryParam('count', {
encode: (value: number) => value.toString(),
decode: (value: string | null) => (value ? parseInt(value) : null),
defaultValue: 10,
});The count is {$count}
```
this will make the query parameter change as soon as the page is rendered on the browser (the query parameter will change only if it's not already present and only the first time the application render).
> **Warning**
>
> You can't run `goto` on the server so if the page is server side rendered it will still have the null value (this is to say don't relay on the assumption that the store will always be not null).### Helpers encodings and decodings
Write an encode and decode function may seem trivial but it's tedious for sure. `sveltekit-search-params` provide with a set of helpers for better readability and to avoid the hassle of writing common transforms. You can find those helpers exported in a ssp variable from the same package.
```svelte
import { ssp, queryParam } from 'sveltekit-search-params';
const count = queryParam('count', ssp.number());
The count is {$count}
```
this code will produce the same output as the code written above but far more readable and easier to read. You can find all the exports documented in the section [ssp - Helpers](#ssp---helpers).
You can also pass a default value to the function that will be the defaultValue of the object.
```svelte
import { ssp, queryParam } from 'sveltekit-search-params';
const count = queryParam('count', ssp.number(10));
The count is {$count}
```
### Simple case (all parameters)
You can use the function queryParameters to get an object containing all the present search params.
```svelte
import { queryParameters } from 'sveltekit-search-params';
const store = queryParameters();
{JSON.stringify($store, null, 2)}
```assuming the page is `/?framework=svelte&isCool=true` the above code will show
```json
{
"framework": "svelte",
"isCool": "true"
}
```by default all query parameters are string.
### Writing to the store (all parameters)
Just like with the single parameter case you can just update the store and the URL at the same time by doing this
```svelte
import { queryParameters } from 'sveltekit-search-params';
const store = queryParameters();
{JSON.stringify($store, null, 2)}
{
$store.username = e.target.value;
}}
/>
```writing in the input will update the state and the URL at the same time.
### Expecting some parameters
Most of the times if you need to read from query parameters you are expecting some parameters to be present. You can define the parameters you are expecting during the store creation and those will be merged with the actual query parameters despite the fact that they are present or not.
```svelte
import { queryParameters } from 'sveltekit-search-params';
const store = queryParameters({
username: true,
});
{JSON.stringify($store, null, 2)}
```assuming the page is `/?framework=svelte&isCool=true` the above code will show
```json
{
"framework": "svelte",
"isCool": "true",
"username": null
}
```if we add username to the URL like this `/?framework=svelte&isCool=true&username=paoloricciuti` we will get
```json
{
"framework": "svelte",
"isCool": "true",
"username": "paoloricciuti"
}
```### Encoding and Decoding
The parameter passed to `queryParameters` can also be used to specify the encoding and decoding just like the `queryParam` method.
```svelte
import { queryParameters } from 'sveltekit-search-params';
const store = queryParameters({
username: true,
isCool: {
encode: (booleanValue) => booleanValue.toString(),
decode: (stringValue) =>
stringValue !== null && stringValue !== 'false',
defaultValue: true,
},
});
{JSON.stringify($store, null, 2)}
```assuming the page is `/?framework=svelte&isCool=true&username=paoloricciuti` the above code will show
```json
{
"framework": "svelte",
"isCool": true,
"username": null
}
```notice that this time isCool it's a boolean and not a string anymore. With this particular transformation we've assured that if the url is the following `/?framework=svelte&isCool=false&username=paoloricciuti` or if the isCool parameter is completely missing like this `/?framework=svelte&username=paoloricciuti` we will get
```json
{
"framework": "svelte",
"isCool": false,
"username": null
}
```### Helpers encodings and decodings
Obviously also in this case you can use the helpers functions provided inside `ssp`.
```svelte
import { ssp, queryParameters } from 'sveltekit-search-params';
const store = queryParameters({
username: true,
isCool: ssp.boolean(true),
});
{JSON.stringify($store, null, 2)}
```## ssp - Helpers
There are six helpers all exported as functions on the object ssp. To each one of them you can pass a parameter that will be the default value for that query param.
#### object
To map from a query parameter to an object. An url like this `/?obj={"isComplex":%20true,%20"nested":%20{"field":%20"value"}}` will be mapped to
```typescript
$store.obj.isComplex; //true
$store.obj.nested; // {field: "value"}
$store.obj.nested.value; // "value"
```#### array
To map from a query parameter to an array. An url like this `/?arr=[1,2,3,4]` will be mapped to
```typescript
$store.arr[0]; //1
$store.arr[1]; //2
$store.arr[2]; //3
$store.arr[3]; //4
```#### number
To map from a query parameter to a number. An url like this `/?num=1` will be mapped to
```typescript
$store.num; //1
```#### boolean
To map from a query parameter to a boolean. An url like this `/?bool=true` will be mapped to
```typescript
$store.bool; //true
```as we've seen an url like this `/?bool=false` will be mapped to
```typescript
$store.bool; //false
```just like an url like this `/`
#### string
This is exported mainly for readability since all query parameters are already strings.
#### lz
To map any JSON serializable state to his lz-string representation. This is a common way to store state in query parameters that will prevent the link to directly show the state.
An url like this `/?state=N4IgbghgNgrgpiAXCAsgTwAQGMD2OoYCO8ATpgA4QkQC2cALnCSAL5A` will map to
```typescript
$store.state.value; //My cool query parameter
```## Store options
Both functions accept a configuration object that contains the following properties:
### debounceHistory
The number of milliseconds to delay the writing of the history when the state changes. This is to avoid cluttering the history of the user especially when a store is bound to an input text (every keystroke would cause a new history entry). It defaults to 0. If set a new entry in the history will be added only after `debounceHistory` seconds of "inactivity".
### pushHistory
A boolean defining if the history have to be written at all. If set to false no new history entries will be written to the history stack (the URL will still update but the user will not be able to go back with the browser).
### sort
Whenever you interact with a store, it navigates for you. By default the search params are sorted to allow for better cache-ability. You can disable this behavior by passing `false` to this option. Keep in mind that this is a per-store settings. This mean that if you interact with a store that has this option set to `false` and than interact with one that has this option set to `true` (the default) the resulting URL will still have the search params sorted.
### showDefaults
If you specify a default value for the search param and it's not present by default the library will immediately navigate to the url that contains the default search param. For example if you have this code
```svelte
import { queryParam, ssp } from 'sveltekit-search-params';
const pageNum = queryParam('pageNum', ssp.number(0));
```
and you navigate to `/` as soon as the page load it will be redirected to `/?pageNum=0`. If you prefer not to show the defaults in the URL you can set the `showDefaults` option to false.
```svelte
import { queryParam, ssp } from 'sveltekit-search-params';
const pageNum = queryParam('pageNum', ssp.number(0), {
showDefaults: false,
});```
By doing so the store will still have a value of 0 if the search param is not present but the user will not be redirected to `/?pageNum=0`.
### equalityFn
While this is not a problem for primitive values if your store has a complex object or an array (or if you are using `queryParameters`) as a value even if the reference is the same svelte will trigger reactivity for it. To provide you with optimistic updates there's the possibility that the store will change multiple times during a single navigation. To fix this problem by default we check if the value of the store is the same by using `JSON.stringify` so that if the overall shape of your store is the same we avoid triggering the reactivity.
This is fine for most cases and you will likely never touch this option but if you have some use case not covered by `JSON.stringify` you can specify the option `equalityFn`. This option is a function that takes the `current` value and the `next` value as parameters and need to return a `boolean`. You should return `true` from this function when the value of the store is unchanged (according to your own logic). This will not trigger reactivity (note that the navigation will still happen).
For `queryParameters` the equality function applies to the entire store (so make sure to check that every query parameter is exactly the same before returning `true`).
```svelte
import { queryParam, ssp } from 'sveltekit-search-params';
const pageNum = queryParam('pageNum', ssp.object<{ num: number }>(), {
equalityFn(current, next) {
return current?.num === next?.num;
},
});```
NOTE: the equality function will not be used on primitive values, hence you can't pass the equality function to stores that have a primitive type.
### How to use it
To set the configuration object you can pass it as a third parameter in case of `queryParam` or the second in case of `queryParameters`.
```svelte
import { ssp, queryParameters, queryParam } from 'sveltekit-search-params';
const name = queryParam('name', ssp.string(), {
debounceHistory: 500, //a new history entry will be created after 500ms of this store not changing
});
const count = queryParam('count', ssp.number(), {
debounceHistory: 1500, //a new history entry will be created after 1500ms of this store not changing
});
const store = queryParameters(
{
username: true,
isCool: ssp.boolean(true),
},
{
pushHistory: false, //no new history entries for this store
},
);```
## Vite dependecies error
If you ran into issues with vite you need to update your `vite.config.ts` or `vite.config.js` file to include the plugin exported from `sveltekit-search-params/plugin`. It's as simple as
```javascript
import { sveltekit } from '@sveltejs/kit/vite';
import { ssp } from 'sveltekit-search-params/plugin';/** @type {import('vite').UserConfig} */
const config = {
plugins: [ssp(), sveltekit()],
};export default config;
```> **Warning**
>
> This step is required if you are running on an old version of vite/sveltekit