Vue Computed Not Updating - vue.js

I have a computed function that I'd like to utilize but I keep getting "Computed property was assigned to but it has no setter". I am simply trying to remove all the forward slashes and the 'SYG' at the end of this: 99/KRFS/010572//SYG when it's pasted into a v-model input to achieve this: 99KRFS010572.
Here is my setup function
<input v-model="policyMapName" />
policy-map <span>{{ policyMapName }}</span>
setup() {
const circuitID = ref('99/KRFS/010572//SYG');
const policyMapName = computed(() => {
const cID = circuitID.value;
return cID.replace(/[/]/g, '').slice(0, -3);
});
}

You should add a setter to your computed property with a getter :
<input v-model="policyMapName" />
policy-map <span>{{ policyMapName }}</span>
setup() {
const circuitID = ref('99/KRFS/010572//SYG');
const policyMapName = computed({
get: () => {
const cID = circuitID.value;
return cID.replace(/[/]/g, '').slice(0, -3);
},
set:(newval)=>{
circuitID.value =newval
}
});
}

Related

Can't get events from emits in my composable

I'm trying to generate a confirmation modal when calling my composable, my component instance is mounting well, but I can't access the emits via the : onCancel
The goal is to call the dialer every time I need to interact with a confirmation
useModalConfirm.ts
function confirm(props: ModalConfirmProps) {
const container = document.createElement('div');
document.body.appendChild(container);
const component = createVNode(ModalConfirm, {
...props,
// not working here :(
onCancel: () => {
console.log('canceled')
}
});
render(component, container);
return component.component;
}
ModalConfirm.vue
<script lang="ts" setup>
import {NButton} from "naive-ui";
const emits = defineEmits(["onConfirm", "onCancel"]);
export type ModalConfirmProps = {
title: string;
message: string;
confirmButtonText: string,
cancelButtonText: string,
};
const props = defineProps<ModalConfirmProps>();
const confirm = () => {
emits("onConfirm");
};
const cancel = () => {
emits("onCancel");
};
</script>
<template>
<div class="ModalConfirm">
<div class="ModalConfirmContent">
{{ props.title }}
{{ props.message }}
<NButton #click="cancel" type="error">{{ props.cancelButtonText }}</NButton>
<NButton #click="confirm" type="success">{{ props.confirmButtonText }}</NButton>
</div>
</div>
</template>
any ideas ?

Pagination vue 3

I have a local array with data which i pass to a table.
I created variables: page,limit,totalPage and function changePage.
In hook onMounted I calculate the number of pages based on the given limit.
Through the v-for loop, displayed the pages and made them clickable on the function, but I don't understand how to set the limit of displayed table elements, and how to make pagination work.
<template>
<div class="containerApp">
<Dialog
:showForm="showForm"
#changeToggle="changeToggle"
#hideDialog="hideDialog"
>
<PostForm #create="createPost" />
</Dialog>
<Select v-model="selectedSort" :options="sortOptions" />
</div>
<input type="text" v-model="searchQuery" />
<Table :tableData="searchAndSort" />
<div class="page_wrapper">
<div
v-for="(pageNumber, i) in totalPage"
:key="i"
#click="changePage(pageNumber)"
class="page"
:class="{ 'current-page': page === pageNumber }"
>
{{ pageNumber }}
</div>
</div>
</template>
<script>
import Table from "./components/Table.vue";
import Dialog from "./components/UI/Dialog.vue";
import PostForm from "./components/PostForm.vue";
import Select from "./components/UI/Select.vue";
import { ref } from "#vue/reactivity";
import { computed, onMounted } from "#vue/runtime-core";
export default {
name: "App",
components: {
...
},
setup() {
const showForm = ref(false);
const searchQuery = ref("");
const tableData = ref([
...
]);
const selectedSort = ref("");
const sortOptions = ref([
...
]);
const page = ref(1);
const limit = ref(5);
const totalPage = ref(0);
onMounted(() => {
totalPage.value = Math.ceil(tableData.value.length / limit.value);
});
const changePage = (pageNumber) => {
page.value = pageNumber;
};
const createPost = (post) => {
tableData.value.push(post);
showForm.value = false;
};
const changeToggle = (toggle) => {
showForm.value = !toggle;
};
const hideDialog = (val) => {
showForm.value = val;
};
const sortedPosts = computed(() => {
...
});
const searchAndSort = computed(() => {
...
});
return {
showForm,
tableData,
selectedSort,
sortOptions,
searchQuery,
changeToggle,
hideDialog,
createPost,
sortedPosts,
searchAndSort,
page,
limit,
totalPage,
changePage,
};
},
};
</script>
In order to show a portion of an array you can use Array.slice. (docs)
in your searchAndSort you should do something like this:
const searchAndSort = computed(() => {
const start = (page.value - 1) * limit.value;
const end = (page.value * limit.value)+1;
return tableData.value.slice(start, end);
});
page - 1 * limit - initially this will result in 0 * 5 meaning start from 0 (-1 is used because the page start from 1. When you go to the second page this will result in 1 * 5, etc.
end is defined by the limit itself multiplying the page.
So on the first page, you will slice the array from 0 to 5, on the second page - from 5 to 10, etc.
const end = (page.value * limit.value)+1; - +1 will give you the 5th element because Array.slice exlude the item at the end index)
You should add some checks as well.

How to deal with relations in Vuex/Pinia stores and keep them in sync?

I'm using Vue3 with the composition API. For this example, I will use Pinia but this applies to Vuex as well.
Imagine you have to take care of entity A holding entities of type entity B. For this example I will use a single todo holding multiple labels.
The sample store for the labels
export type Label = {
title: string; // "unique"
};
export const useLabelsStore = defineStore("labels", () => {
const labels = ref<Label[]>([]);
function addLabel(title: string): void {
if (!labels.value.some((label) => label.title === title)) {
labels.value.push({ title });
}
}
function deleteLabel(title: string): void {
labels.value = labels.value.filter((label) => label.title !== title);
}
return { labels, addLabel, deleteLabel };
});
and for the todos
export type Todo = {
title: string; // "unique"
labels: Label[];
};
export const useTodosStore = defineStore("todos", () => {
const todos = ref<Todo[]>([{ title: "foo", labels: [] }]);
function applyLabelToTodo(todoTitle: string, label: Label) {
const todo = todos.value.find((todo) => todo.title === todoTitle);
if (
todo &&
!todo.labels.some((todoLabel) => todoLabel.title === label.title)
) {
todo.labels.push(label);
}
}
return { todos, applyLabelToTodo };
});
You can create todos, labels and apply labels to todos. But after applying a label to a todo and deleting it afterwards the todo still holds it, the code has no sync mechanism yet, so the todos store is no "invalid" / "outdated".
This shows the App.vue file for testing purposes
<template>
<div>
<div>Labels: {{ labelsStore.labels }}</div>
<br />
<div v-for="todo in todosStore.todos" :key="todo.title">
<div>Todo: {{ todo.title }}</div>
<div>Applied labels: {{ todo.labels }}</div>
</div>
<br />
<button #click="addLabel">Add new label</button>
<br />
<br />
<select v-model="selectedLabelTitle">
<option
v-for="label in labelsStore.labels"
:value="label.title"
:key="label.title"
>
{{ label.title }}
</option>
</select>
<button #click="labelsStore.deleteLabel(selectedLabelTitle)">
Delete label "{{ selectedLabelTitle }}"
</button>
<button #click="applyLabelToTodo">
Apply label "{{ selectedLabelTitle }}" to todo "{{
todosStore.todos[0].title
}}"
</button>
</div>
</template>
<script lang="ts">
import { defineComponent, ref } from "vue";
import { useTodosStore } from "./todosStore";
import { useLabelsStore } from "./labelsStore";
export default defineComponent({
setup() {
const todosStore = useTodosStore();
const labelsStore = useLabelsStore();
const selectedLabelTitle = ref<string>();
function addLabel(): void {
labelsStore.addLabel(`Label ${labelsStore.labels.length + 1}`);
}
function applyLabelToTodo(): void {
const label = labelsStore.labels.find(
(label) => label.title === selectedLabelTitle.value
);
if (label) {
todosStore.applyLabelToTodo(todosStore.todos[0].title, label);
}
}
return {
todosStore,
labelsStore,
addLabel,
selectedLabelTitle,
applyLabelToTodo,
};
},
});
</script>
The todos store already has a todo
Click on "Add new label"
Select it in the select box
Apply that label to the todo
Delete that label
You will see that this todo still holds the deleted label.
What is the common way to synchronize the stores? My current solution based on subscribing to actions is extending the todos store
export const useTodosStore = defineStore("todos", () => {
// ...
const labelsStore = useLabelsStore();
labelsStore.$onAction((context) => {
if (context.name === "deleteLabel") {
context.after(() => {
todos.value = todos.value.map((todo) => {
todo.labels = todo.labels.filter(
(label) => label.title !== context.args[0]
);
return todo;
});
});
}
});
// ...
});
Do I have to listen for all the actions or are there any better ways?
Factory function that defines a store and allows to define it with Vue reactive API directly is advanced feature that is generally not needed. A store is commonly defined with an object that is similar to Vuex API.
Values that are supposed to derive from other values are supposed to go to getters, very similarly to Vue computed values (computed is a part of Vue reactive API and used by Pinia internally).
There are many ways to do this, for instance, todo labels can be filtered against up-to-date labels from another store
const todoStore = defineStore({
state: () => ({ _todos: [] })
getters: {
todos() {
const labelStore = useLabelStore();
return this._todos
.map((todo) => ({
...todo,
labels: todo.labels.filter(label => labelStore.labels.includes(label))
});
}
}
});

VueJS test-utils can't find element inside child component

I'm trying to use findComponent with find method to find a child component's element and set it's value. But every time I run test, it gives me Cannot call setValue on an empty DOMWrapper. error.
Test file
import { mount } from '#vue/test-utils';
import Create from './Create.vue';
// import State from '#/components/State.vue';
describe('it tests Create component', () => {
test('it emits create event and resets the form when form is valid and create button is clicked', async () => {
const div = document.createElement('div');
div.id = 'root';
document.body.append(div);
const expectedNameValue = 'TODO_NAME';
const expectedStateValue = 'Pending';
const wrapper = mount(Create, {
attachTo: '#root',
});
await wrapper.find(`input`).setValue(expectedNameValue);
await wrapper.findComponent({ ref: 'state-component' }).find('select').setValue(expectedStateValue);
await wrapper.find(`form`).trigger('submit');
expect(wrapper.emitted().create).toBeTruthy();
expect(wrapper.emitted().create[0]).toEqual([expectedNameValue]);
expect(wrapper.emitted().create[1]).toEqual(['Pending']);
expect(wrapper.find(`input[name='name']`).element.value).toEqual('');
expect(wrapper.find(`input[name='state']`).element.value).toEqual('Pending');
});
});
Create component
<template>
<form #submit.prevent="createTodo" class="flex gap-2 w-full">
<input class="flex-1 shadow rounded-md p-2 focus:ring-2 focus:ring-blue-900 focus:outline-none" type="text" placeholder="Todo Name" name="name" required/>
<State ref="state-component"/>
<button type="submit" class="rounded-md shadow text-white bg-blue-700 py-2 px-6">Create</button>
</form>
</template>
<script>
import State from '#/components/State.vue';
export default {
components: { State },
emits: ['create'],
methods: {
createTodo(event) {
const elems = event.target.elements;
const todo = { name: elems.name.value, state: elems.state.value };
this.$emit('create', todo);
elems.name.value = '';
elems.state.value = 'Pending';
}
}
}
</script>
<style scoped>
</style>
State component
<template>
<select id="state-select" class="rounded-md bg-green-200 text-white" name="state">
<option
v-for="(state, index) in states"
:selected="isSelected(state)"
:key="index"
>
{{ state }}
</option>
</select>
</template>
<script>
export default {
props: ["todo", "index"],
data() {
return {
name: "",
state: "",
states: ["Pending", "In Progress", "Done"],
};
},
created() {
if(!this.todo) return true;
this.state = this.todo.state;
this.name = this.todo.name;
},
methods: {
isSelected(equivalent){
return equivalent === this.state;
}
}
};
</script>
<style scoped></style>
I'm fairly new to VueJS so I'm open to all tips and tricks, thanks.
Some issues to fix:
You don't need to attach the component to the document, so remove that:
// ❌
// const div = document.createElement('div');
// div.id = 'root';
// document.body.append(div);
// const wrapper = mount(Create, { attachTo: '#root' });
// ✅
const wrapper = mount(Create);
The template ref to the State component would be the component's root element, so no need to find() the <select>:
// ❌
// await wrapper.findComponent({ ref: 'state-component' }).find('select').setValue(expectedStateValue);
^^^^^^^^^^^^^^^
// ✅
await wrapper.findComponent({ ref: 'state-component' }).setValue(expectedStateValue);
The emitted() object key is the event name, and the value is an array of of arrays, containing emitted data. You can verify the first create-event data contains another object with toMatchObject(object):
// ❌
// expect(wrapper.emitted().create[0]).toEqual([expectedNameValue]);
// expect(wrapper.emitted().create[1]).toEqual(['Pending']);
// ✅
expect(wrapper.emitted().create[0][0]).toMatchObject({ name: expectedNameValue, state: 'Pending' });
The last assertion tries to find input[name='state'], but that's actually a <select>, not an <input>:
// ❌
// expect(wrapper.find(`input[name='state']`).element.value).toEqual('Pending')
^^^^^
// ✅
expect(wrapper.find(`select[name='state']`).element.value).toEqual('Pending')
demo

how to select a default value in q-select from quasar?

I need my q-select to select the "name" depending on the previously assigned "id".
Currently the input shows me the "id" in number and not the name to which it belongs.
<q-select
class="text-uppercase"
v-model="model"
outlined
dense
use-input
input-debounce="0"
label="Marcas"
:options="options"
option-label="name"
option-value="id"
emit-value
map-options
#filter="filterFn"
#update:model-value="test()"
>
<template v-slot:no-option>
<q-item>
<q-item-section class="text-grey"> No results </q-item-section>
</q-item>
</template>
</q-select>
Example I would like the name that has the id: 12 to be shown loaded in the q-select.
const model = ref(12);
const options = ref([]);
const filterFn = (val, update) => {
if (val === "") {
update(() => {
options.value = tableData.value;
});
return;
}
update(() => {
const needle = val.toLowerCase();
options.value = tableData.value.filter((v) =>
v.name.toLowerCase().includes(needle)
);
});
};
I'm running into the same issue and couldn't find a proper way to set default value of object type.
I ended up use find() to look for the default value in the options and assign it on page created event.
created() {
this.model = this.options.find(
(o) => o.value == this.model
);
}