StoreModule
é uma biblioteca para gerar stores genéricas de C.R.U.D. É compatível com Pinia e Vuex, foi desenvolvida para substituir a antiga VuexStoreModule, que faz basicamente a mesma coisa com algumas mudanças porém só é compatível com o Vuex
, uma vez que o Pinia
é o gerenciador de estado de aplicação recomendado pela equipe do Vue
.
Esta biblioteca faz o uso do StoreAdapter.
npm i @bildvitta/store-module
Para utilizar o StoreModule, você precisa inicializa-lo passando o adapter pinia
e o apiService axios
, exemplo:
- Crie um arquivo chamado
store-module.js
contendo:
import StoreModule from '@bildvitta/store-module'
import axios from 'axios'
const storeModule = new StoreModule({
apiService: axios,
adapter: 'pinia'
})
export default storeModule
- Após a inicialização, vamos criar uma store de usuários, para isto crie um arquivo
users.js
contendo:
import storeModule from 'caminho-do-arquivo-store-module.js' // arquivo que inicializamos o StoreModule
import { defineStore } from 'pinia'
export default defineStore('users', storeModule.createStoreModule('users'))
- Pronto, a store foi criada com sucesso! Agora vamos importar a store para ver como poderia utilizar ela.
import usersStore from 'caminho-do-arquivo-users.js' // arquivo que criamos nossa store de usuários.
const user = usersStore()
console.log(user) // user tem acesso a toda nossa store, com states, getters e actions.
Para utilizar esta biblioteca com Pinia e Asteroid, é necessário algumas configurações adicionais, uma vez que o asteroid faz o uso das stores através de variáveis globais, $store
quando é vuex (que é injetada automaticamente pelo próprio vuex), e $piniaStore
quando é pinia (que é injetado manualmente pelo DefineGlobalPiniaStore
).
- Instale a biblioteca
StoreAdapter
:
npm i @bildvitta/store-adapter
-
Crie um
boot
chamadostore-adapter
(ou algum outro nome de sua preferencia), importe este boot noquasar.config.js
. -
Importe o plugin
DefineGlobalPiniaStore
doStoreAdapter
dentro do boot. -
Após isto, importe todas as stores criadas na aplicação dentro deste boot e adicione elas ao plugin
DefineGlobalPiniaStore
.
Exemplo de como deveria ficar seu boot store-adapter.js
:
import { boot } from 'quasar/wrappers'
import { DefineGlobalPiniaStore } from '@bildvitta/store-adapter'
import users from 'caminho-do-arquivo-users.js'
export default boot(({ app }) => {
app.use(DefineGlobalPiniaStore, {
stores: {
users
}
})
})
- Como a biblioteca da compatibilidade tanto para pinia quanto para vuex, não existem
mutations
dentro de nossas stores pois não existe mutations no pinia, por isto, precisamos desativar o modostrict
do vuex, para isto, vá até a inicialização do vuex e desative passandostrict: false
, exemplo:
export default store(function (/* { ssrContext } */) {
const Store = createStore({
modules: {},
strict: false // ----------------------Aqui desativamos o strict!!!----------------------
})
return Store
})
Caso você não desative o strict, vai disparar vários erros avisando que existem states sendo mutado fora de mutations.
Error: [vuex] Do not mutate vuex store state outside mutation handlers
- Para utilizar o
StoreModule
com o vuex, você precisa inicializa-lo passando o adaptervuex
e o apiServiceaxios
, para isto crie um arquivo chamadostore-module.js
contendo:
import StoreModule from '@bildvitta/store-module'
import axios from 'axios'
const storeModule = new StoreModule({
apiService: axios,
adapter: 'vuex'
})
export default storeModule
- Após a inicialização, vamos criar uma store de usuários, para isto crie um arquivo
users.js
contendo:
import storeModule from 'caminho-do-arquivo-store-module.js'
export default storeModule.createStoreModule('users')
- Após, isto vamos no arquivo onde foi inicializado o
Vuex
, vamos importar ousers.js
e adicionar ele dentro demodules
:
import users from 'caminho-do-users.js'
export default store(function (/* { ssrContext } */) {
const Store = createStore({
modules: {
users // ----------------------Aqui adicionamos o modulo users!!!----------------------
},
strict: false
})
return Store
})
- Pronto, a store foi criada com sucesso! Agora vamos importar a store para ver como poderia utilizar ela.
import usersStore from 'caminho-do-arquivo-users.js' // arquivo que criamos nossa store de usuários.
import { mapActions, mapGetters, mapState } from 'vuex' // para utilizar o vuex é a forma convencional
Para utilizar esta biblioteca com Vuex e Asteroid, não é necessário nenhuma configuração adicional.
new StoreModule(options: StoreModuleOptions).createStoreModule(
resource: string,
options: ModuleOptions
): StoreModule
export interface StoreModuleOptions {
// adapter da store: pinia | vuex
adapter: StoreModuleAdapter
// instancia do axios
apiService: ApiService
// opção para alterar chave identificadora para ser usado como chave primaria
// configuração global, caso não seja configurado o padrão é "uuid"
idKey: string
// itens por listagem definidos na action "fetchList"
// configuração global, caso não seja configurado o padrão é 12
perPage: number
}
export interface ModuleOptions {
// actions do vuex ou pinia adicionais
actions: ExternalActions
// callback para alterar a url da action "destroy" que tem como parâmetro "{ id }"
destroyURL: RunCallbackFn<DestroyURL>
// opção para alterar a url da action "fetchFieldOptions"
fetchFieldOptionsURL: string
// opção para alterar a url da action "fetchFilters"
fetchFiltersURL: string
// opção para alterar a url da action "fetchList"
fetchListURL: string
// callback para alterar a url da action "fetchSingle" que tem como parâmetro "{ form, id }"
fetchSingleURL: RunCallbackFn<FetchSingleURL>
// getters do vuex ou pinia adicionais
getters: ExternalGetters
// opção para alterar chave identificadora para ser usado como chave primaria
// sobrescreve a configuração global, caso não seja configurado o padrão é
// configurado global StoreModuleOptions.idKey ou "uuid"
idKey: string
// callback para alterar a url da action "update" que tem como parâmetro "{ id }"
updateURL: RunCallbackFn<UpdateURL>
// callback para alterar a url da action "replace" que tem como parâmetro "{ id }"
replaceURL: RunCallbackFn<ReplaceURL>
// opção para alterar a url da action "create"
createURL: string
// state do vuex ou pinia adicionais
state: ExternalState
}
export interface StoreModule {
// se for vuex namespaced então tem o valor "true"
namespaced?: boolean
// state do pinia ou vuex
state: () => State
// getters do pinia ou vuex
getters: Getters
// actions do pinia ou vuex
actions: FactoryActions
}
state
├── list
├── filters
└── totalPages
getters
└── byId
actions
├── create
├── destroy
├── fetchFieldOptions
├── fetchFilters
├── fetchList
├── fetchSingle
├── update
└── replace
export interface State {
filters: ItemOfItem
list: ItemOfItem[]
totalPages: number
}
export interface Getters {
byId: (state: State) => (id: string | number) => Item | undefined
}
export interface Actions {
destroy: (payload: DestroyActionPayload) => Promise<AxiosResponse<DestroyApiResponse>>
fetchFieldOptions: (payload: FetchFieldOptionsActionPayload) => Promise<AxiosResponse<FetchFieldOptionsApiResponse>>
fetchFilters: (payload: FetchFiltersActionPayload) => Promise<AxiosResponse<FetchFiltersApiResponse>>
fetchList: (payload: FetchListActionPayload) => Promise<AxiosResponse<FetchListApiResponse>>
fetchSingle: (payload: FetchSingleActionPayload) => Promise<AxiosResponse<FetchSingleApiResponse>>
update: (payload: UpdateActionPayload) => Promise<AxiosResponse<UpdateApiResponse>>
replace: (payload: ReplaceActionPayload) => Promise<AxiosResponse<ReplaceApiResponse>>
create: (payload: CreateActionPayload) => Promise<AxiosResponse<CreateApiResponse>>
}
- Instalar eslint
- Testes unitários (vitest)